1#include <vanetza/btp/data_indication.hpp>
2#include <vanetza/btp/data_request.hpp>
3#include <vanetza/common/annotation.hpp>
4#include <vanetza/common/its_aid.hpp>
5#include <vanetza/common/position_fix.hpp>
6#include <vanetza/common/runtime.hpp>
7#include <vanetza/dcc/data_request.hpp>
8#include <vanetza/dcc/interface.hpp>
9#include <vanetza/dcc/profile.hpp>
10#include <vanetza/net/mac_address.hpp>
11#include <vanetza/net/osi_layer.hpp>
12#include <vanetza/units/frequency.hpp>
13#include <vanetza/units/length.hpp>
14#include <vanetza/units/time.hpp>
15#include <vanetza/geonet/router.hpp>
16#include <vanetza/geonet/cbf_counter.hpp>
17#include <vanetza/geonet/data_confirm.hpp>
18#include <vanetza/geonet/dcc_field_generator.hpp>
19#include <vanetza/geonet/duplicate_packet_list.hpp>
20#include <vanetza/geonet/indication_context.hpp>
21#include <vanetza/geonet/loctex_g5.hpp>
22#include <vanetza/geonet/next_hop.hpp>
23#include <vanetza/geonet/pdu_conversion.hpp>
24#include <vanetza/geonet/repetition_dispatcher.hpp>
25#include <vanetza/geonet/transport_interface.hpp>
26#include <vanetza/geonet/extended_pdu.hpp>
27#include <vanetza/geonet/secured_pdu.hpp>
28#include <boost/units/cmath.hpp>
44 communication_profile(request.communication_profile),
45 its_aid(request.its_aid),
46 permissions(request.permissions)
49 const CommunicationProfile communication_profile;
51 const ByteBuffer permissions;
55class PendingPacketBufferData :
public packet_buffer::
Data
58 PendingPacketBufferData(
PendingPacket<PDU>&& packet) : m_packet(std::move(packet)) {}
60 std::size_t length()
const override
62 return m_packet.length();
65 Clock::duration reduce_lifetime(
Clock::duration d)
override
67 return m_packet.reduce_lifetime(d);
92auto create_forwarding_duplicate(
const PDU& pdu,
const UpPacket& packet) ->
93std::tuple<std::unique_ptr<ExtendedPdu<
typename PDU::ExtendedHeader>>, std::unique_ptr<DownPacket>>
95 using pdu_type = ExtendedPdu<
typename PDU::ExtendedHeader>;
96 std::unique_ptr<pdu_type> pdu_dup {
new pdu_type { pdu }};
97 std::unique_ptr<DownPacket> packet_dup;
99 packet_dup.reset(
new DownPacket());
101 packet_dup = duplicate(packet);
103 return std::make_tuple(std::move(pdu_dup), std::move(packet_dup));
106template<
typename PDU>
107PDU& get_pdu(
const std::tuple<std::unique_ptr<PDU>, std::unique_ptr<DownPacket>>& packet)
109 PDU* pdu = std::get<0>(packet).get();
114std::unique_ptr<CbfCounter> create_cbf_counter(Runtime& rt,
const MIB& mib)
116 std::unique_ptr<CbfCounter> counter;
117 if (mib.vanetzaFadingCbfCounter) {
118 counter.reset(
new CbfCounterFading(rt, units::clock_cast(mib.vanetzaFadingCbfCounterLifetime)));
120 counter.reset(
new CbfCounterContending());
128using units::clock_cast;
131const access::EtherType ether_type = access::ethertype::GeoNetworking;
136 m_request_interface(get_default_request_interface()),
137 m_dcc_field_generator(get_default_dcc_field_generator()),
138 m_security_entity(
nullptr),
139 m_location_table(mib, m_runtime),
140 m_bc_forward_buffer(mib.itsGnBcForwardingPacketBufferSize * 1024),
141 m_uc_forward_buffer(mib.itsGnUcForwardingPacketBufferSize * 1024),
142 m_cbf_buffer(m_runtime,
143 [](PendingPacketGbc&& packet) { packet.process(); },
144 create_cbf_counter(rt, mib),
145 mib.itsGnCbfPacketBufferSize * 1024),
146 m_local_sequence_number(0),
147 m_repeater(m_runtime,
148 std::bind(&Router::dispatch_repetition,
this, std::placeholders::_1, std::placeholders::_2)),
149 m_random_gen(mib.vanetzaDefaultSeed)
151 if (!m_mib.vanetzaDisableBeaconing) {
152 if (m_mib.vanetzaDeferInitialBeacon > Clock::duration::zero()) {
154 std::uniform_real_distribution<
double> dist_jitter(0.0, 1.0);
155 const auto jitter = clock_cast(dist_jitter(m_random_gen) * m_mib.itsGnBeaconServiceMaxJitter);
156 reset_beacon_timer(m_mib.vanetzaDeferInitialBeacon + jitter);
159 reset_beacon_timer(Clock::duration::zero());
163 m_gbc_memory.capacity(m_mib.vanetzaGbcMemoryCapacity);
176 m_local_position_vector.timestamp = m_runtime.now();
177 m_local_position_vector.latitude =
static_cast<geo_angle_i32t>(position_fix.latitude);
178 m_local_position_vector.longitude =
static_cast<geo_angle_i32t>(position_fix.longitude);
179 if (m_mib.itsGnIsMobile) {
180 m_local_position_vector.speed =
static_cast<LongPositionVector::speed_u15t>(position_fix.speed.value());
181 m_local_position_vector.heading =
static_cast<heading_u16t>(position_fix.course.value() - units::TrueNorth::from_value(0.0));
183 m_local_position_vector.speed =
static_cast<LongPositionVector::speed_u15t>(0);
184 m_local_position_vector.heading =
static_cast<heading_u16t>(0);
187 m_local_position_vector.position_accuracy_indicator =
188 position_fix.confidence.semi_major * 2.0 < m_mib.itsGnPaiInterval;
193 m_transport_ifcs[proto] = ifc;
198 m_security_entity = entity;
203 m_request_interface = (ifc ==
nullptr ? get_default_request_interface() : ifc);
204 assert(m_request_interface !=
nullptr);
209 m_dcc_field_generator = (dcc ==
nullptr) ? get_default_dcc_field_generator() : dcc;
210 assert(m_dcc_field_generator !=
nullptr);
215 m_local_position_vector.gn_addr = addr;
220 m_random_gen.seed(seed);
226 result ^= validate_data_request(request, m_mib);
227 result ^= validate_payload(payload, m_mib);
229 if (result.accepted()) {
233 if (request.repetition) {
235 m_repeater.add(request, *payload);
239 auto pdu = create_shb_pdu(request);
240 pdu->common().payload = payload->size();
242 ControlInfo ctrl(request);
243 auto transmit = [
this, ctrl](PendingPacket::Packet&& packet) {
244 std::unique_ptr<ShbPdu> pdu;
245 std::unique_ptr<DownPacket> payload;
246 std::tie(pdu, payload) = std::move(packet);
249 pdu->extended().source_position = m_local_position_vector;
252 if (m_mib.itsGnSecurity) {
253 payload =
encap_packet(ctrl.its_aid
, ctrl.permissions
, *pdu
, std::move(payload)
);
264 pass_down(cBroadcastMacAddress, std::move(pdu), std::move(payload));
270 PendingPacket packet(std::make_tuple(std::move(pdu), std::move(payload)), transmit);
273 if (request.traffic_class.store_carry_forward() && !m_location_table.has_neighbours()) {
274 PacketBuffer::data_ptr data {
new PendingPacketBufferData<ShbPdu>(std::move(packet)) };
275 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
288 result ^= validate_data_request(request, m_mib);
289 result ^= validate_payload(payload, m_mib);
291 if (!result.accepted())
299 if (request.repetition) {
300 m_repeater.add(request, *payload);
304 using Packet = PendingPacket::Packet;
307 auto pdu = create_gbc_pdu(request);
308 pdu->common().payload = payload->size();
310 ControlInfo ctrl(request);
311 auto transmit = [
this, ctrl](Packet&& packet,
const MacAddress& mac) {
312 std::unique_ptr<GbcPdu> pdu;
313 std::unique_ptr<DownPacket> payload;
314 std::tie(pdu, payload) = std::move(packet);
317 pdu->extended().source_position = m_local_position_vector;
320 if (m_mib.itsGnSecurity) {
321 assert(pdu->basic().next_header == NextHeaderBasic::Secured);
322 payload =
encap_packet(ctrl.its_aid
, ctrl.permissions
, *pdu
, std::move(payload)
);
335 pass_down(mac, std::move(pdu), std::move(payload));
338 auto forwarding = [
this, transmit](Packet&& packet) {
343 std::move(nh).process();
346 PendingPacket packet(std::make_tuple(std::move(pdu), std::move(payload)), forwarding);
349 const bool scf = request.traffic_class.store_carry_forward();
350 if (scf && !m_location_table.has_neighbours()) {
351 PacketBuffer::data_ptr data {
new PendingPacketBufferData<GbcPdu>(std::move(packet)) };
352 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
379 link_layer.sender = sender;
380 link_layer.destination = destination;
382 if (
auto cohesive = boost::get<CohesivePacket>(packet.get())) {
385 }
else if (
auto chunk = boost::get<ChunkPacket>(packet.get())) {
389 packet_dropped(PacketDropReason::Internal_Error);
397 packet_dropped(PacketDropReason::Parse_Basic_Header);
398 }
else if (basic->version.raw() != m_mib.itsGnProtocolVersion) {
399 packet_dropped(PacketDropReason::ITS_Protocol_Version);
402 indication.remaining_packet_lifetime = basic->lifetime;
403 indication.remaining_hop_limit = basic->hop_limit;
405 if (basic->next_header == NextHeaderBasic::Secured) {
407 }
else if (basic->next_header == NextHeaderBasic::Common) {
408 if (!m_mib.itsGnSecurity || SecurityDecapHandling::Non_Strict == m_mib.itsGnSnDecapResultHandling) {
409 indication.security_report = boost::blank {};
412 packet_dropped(PacketDropReason::Decap_Unsuccessful_Strict);
422 packet_dropped(PacketDropReason::Parse_Common_Header);
423 }
else if (common->maximum_hop_limit < basic.hop_limit) {
425 packet_dropped(PacketDropReason::Hop_Limit);
428 indication.traffic_class = common->traffic_class;
429 switch (common->next_header)
431 case NextHeaderCommon::BTP_A:
432 indication.upper_protocol = UpperProtocol::BTP_A;
434 case NextHeaderCommon::BTP_B:
435 indication.upper_protocol = UpperProtocol::BTP_B;
437 case NextHeaderCommon::IPv6:
438 indication.upper_protocol = UpperProtocol::IPv6;
441 indication.upper_protocol = UpperProtocol::Unknown;
446 m_location_table.drop_expired();
463 struct secured_payload_visitor :
public boost::static_visitor<>
466 m_router(router), m_context(ctx), m_basic(basic)
487 auto secured_message = ctx.parse_secured();
488 if (!secured_message) {
489 packet_dropped(PacketDropReason::Parse_Secured_Header);
490 }
else if (m_security_entity) {
492 using namespace vanetza::security;
494 ctx.service_primitive().security_report = decap_confirm.report;
495 ctx.service_primitive().its_aid = decap_confirm.its_aid;
496 ctx.service_primitive().permissions = decap_confirm.permissions;
497 secured_payload_visitor visitor(*
this, ctx, basic);
500 if (is_successful(decap_confirm.report)) {
501 boost::apply_visitor(visitor, decap_confirm.plaintext_payload);
502 }
else if (SecurityDecapHandling::Non_Strict == m_mib.itsGnSnDecapResultHandling) {
505 if (!decap_confirm.plaintext_payload.empty()) {
506 boost::apply_visitor(visitor, decap_confirm.plaintext_payload);
509 packet_dropped(PacketDropReason::Decap_Unsuccessful_Non_Strict);
513 packet_dropped(PacketDropReason::Decap_Unsuccessful_Strict);
516 packet_dropped(PacketDropReason::Security_Entity_Missing);
522 struct extended_header_visitor :
public boost::static_visitor<
bool>
525 m_router(router), m_context(ctx), m_packet(packet)
532 indication.transport_type = TransportType::SHB;
535 auto& pdu = m_context.pdu();
537 return m_router.process_extended(shb_pdu, m_packet, m_context.link_layer());
543 indication.transport_type = TransportType::TSB;
546 auto& pdu = m_context.pdu();
548 return m_router.process_extended(tsb_pdu, m_packet, m_context.link_layer());
554 indication.transport_type = TransportType::GBC;
556 indication.destination = gbc.destination(m_context.pdu().common().header_type);
558 auto& pdu = m_context.pdu();
560 return m_router.process_extended(gbc_pdu, m_packet, m_context.link_layer());
565 auto& pdu = m_context.pdu();
567 return m_router.process_extended(beacon_pdu, m_packet, m_context.link_layer());
572 const UpPacket& m_packet;
575 auto extended = ctx.parse_extended(common.header_type);
576 UpPacketPtr packet = ctx.finish();
580 packet_dropped(PacketDropReason::Parse_Extended_Header);
581 }
else if (common.payload != size(*packet, OsiLayer::Transport, max_osi_layer())) {
582 packet_dropped(PacketDropReason::Payload_Size);
584 extended_header_visitor visitor(*
this, ctx, *packet);
585 if (boost::apply_visitor(visitor, *extended)) {
586 pass_up(ctx.service_primitive()
, std::move(packet)
);
594 const Area& destination = packet.pdu().extended().destination(packet.pdu().common().header_type);
595 if (inside_or_at_border(destination, m_local_position_vector.position())) {
596 switch (m_mib.itsGnAreaForwardingAlgorithm) {
597 case BroadcastForwarding::Unspecified:
599 case BroadcastForwarding::SIMPLE:
603 case BroadcastForwarding::CBF:
606 case BroadcastForwarding::Advanced:
610 throw std::runtime_error(
"unhandled area forwarding algorithm");
615 const LongPositionVector* pv_se = ll ? m_location_table.get_position(ll->sender) :
nullptr;
616 if (pv_se && pv_se->position_accuracy_indicator && inside_or_at_border(destination, pv_se->position())) {
618 forwarding_stopped(ForwardingStopReason::Outside_Destination_Area);
620 switch (m_mib.itsGnNonAreaForwardingAlgorithm) {
621 case UnicastForwarding::Unspecified:
623 case UnicastForwarding::Greedy:
626 case UnicastForwarding::CBF:
630 throw std::runtime_error(
"unhandled non-area forwarding algorithm");
641 switch (com_profile) {
642 case CommunicationProfile::ITS_G5:
645 case CommunicationProfile::Unspecified:
646 case CommunicationProfile::LTE_V2X:
650 throw std::runtime_error(
"Unhandled communication profile");
664 if (pdu->secured()) {
665 if (pdu->basic().next_header != NextHeaderBasic::Secured) {
666 throw std::runtime_error(
"PDU with secured message but Secured not set in basic header");
668 if (payload->size(OsiLayer::Transport, max_osi_layer()) > 0) {
669 throw std::runtime_error(
"PDU with secured message and illegal upper layer payload");
672 if (pdu->basic().next_header == NextHeaderBasic::Secured) {
673 throw std::runtime_error(
"PDU without secured message but Secured set in basic header");
678 assert(m_request_interface);
679 m_request_interface->request(request, std::move(payload));
687 request.destination = addr;
688 request.source = m_local_position_vector.gn_addr.mid();
689 request.dcc_profile = map_tc_onto_profile(pdu->common().traffic_class);
690 request.ether_type = geonet::ether_type;
691 request.lifetime = clock_cast(pdu->basic().lifetime.decode());
693 pass_down(request, std::move(pdu), std::move(payload));
699 if (transport !=
nullptr) {
700 transport->indicate(ind, std::move(packet));
706 if (m_mib.vanetzaDisableBeaconing) {
713 auto pdu = create_beacon_pdu();
715 if (m_mib.itsGnSecurity) {
716 pdu->basic().next_header = NextHeaderBasic::Secured;
723 pdu->basic().next_header = NextHeaderBasic::Common;
727 pass_down(cBroadcastMacAddress, std::move(pdu), std::move(payload));
733 using duration_t =
decltype(m_mib.itsGnBeaconServiceRetransmitTimer);
734 using real_t = duration_t::value_type;
735 static_assert(std::is_floating_point<real_t>::value,
"floating point type expected");
737 std::uniform_real_distribution<real_t> dist_jitter(0.0, 1.0);
738 const auto jitter = dist_jitter(m_random_gen);
739 const duration_t next_beacon = m_mib.itsGnBeaconServiceRetransmitTimer +
740 jitter * m_mib.itsGnBeaconServiceMaxJitter;
741 reset_beacon_timer(clock_cast(next_beacon));
747 m_runtime.schedule(next_beacon, [
this](
Clock::time_point) {
755 boost::apply_visitor(dispatcher, request);
762 const units::Length own = distance(dest, m_local_position_vector.position());
763 units::Length mfr_dist = own;
766 for (
const LocationTableEntry& neighbour : m_location_table.neighbours()) {
767 if (neighbour.has_position_vector()) {
768 const units::Length dist = distance(dest, neighbour.get_position_vector().position());
769 if (dist < mfr_dist) {
770 mfr_addr = neighbour.link_layer_address();
776 if (mfr_dist < own) {
779 const bool scf = packet.pdu().common().traffic_class.store_carry_forward();
781 std::function<
void(PendingPacketForwarding&&)> greedy_fwd = [
this](PendingPacketForwarding&& packet) {
782 NextHop nh = greedy_forwarding(std::move(packet));
783 std::move(nh).process();
785 PendingPacket<GbcPdu> greedy_packet(std::move(packet), greedy_fwd);
786 PacketBuffer::data_ptr data {
new PendingPacketBufferData<GbcPdu>(std::move(greedy_packet)) };
787 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
801 const auto cbf_id = identifier(gbc.source_position.gn_addr, gbc.sequence_number);
806 }
else if (m_cbf_buffer.remove(cbf_id)) {
810 const HeaderType ht = packet.pdu().common().header_type;
811 const Area destination = gbc.destination(ht);
812 const auto& epv = m_local_position_vector;
813 const LongPositionVector* pv_se = sender ? m_location_table.get_position(*sender) :
nullptr;
815 if (pv_se && pv_se->position_accuracy_indicator) {
816 const auto& pv_p = destination.position;
817 const units::Length dist_sender = distance(pv_p, pv_se->position());
818 const units::Length dist_local = distance(pv_p, epv.position());
819 if (dist_sender > dist_local) {
820 CbfPacket cbf { std::move(packet), *sender };
821 const auto progress = dist_sender - dist_local;
822 m_cbf_buffer.add(std::move(cbf), clock_cast(timeout_cbf(progress)));
829 CbfPacket cbf { std::move(packet), *sender };
830 const auto to_cbf_max = m_mib.itsGnCbfMaxTime;
831 m_cbf_buffer.add(std::move(cbf), clock_cast(to_cbf_max));
842 const auto cbf_id = identifier(gbc.source_position.gn_addr, gbc.sequence_number);
846 }
else if (m_cbf_buffer.remove(cbf_id) || m_cbf_buffer.counter(cbf_id) >= m_mib.vanetzaCbfMaxCounter) {
850 m_cbf_buffer.add(CbfPacket { std::move(packet), *sender }, clock_cast(timeout));
859 const auto dist_max = m_mib.itsGnDefaultMaxCommunicationRange;
860 const auto to_cbf_min = m_mib.itsGnCbfMinTime;
861 const auto to_cbf_max = m_mib.itsGnCbfMaxTime;
863 if (prog > dist_max) {
865 }
else if (prog > 0.0 * units::si::meter) {
866 return to_cbf_max + (to_cbf_min - to_cbf_max) / dist_max * prog;
875 units::Duration timeout = m_mib.itsGnCbfMaxTime;
877 if (pv_se && pv_se->position_accuracy_indicator) {
878 units::Length dist = distance(pv_se->position(), m_local_position_vector.position());
892 const HeaderType ht = packet.pdu().common().header_type;
893 const Area destination_area = gbc.destination(ht);
894 const std::size_t max_counter = m_mib.vanetzaCbfMaxCounter;
895 const auto cbf_id = identifier(gbc.source_position.gn_addr, gbc.sequence_number);
896 const CbfPacket* cbf_packet = m_cbf_buffer.find(cbf_id);
900 if (m_cbf_buffer.counter(cbf_id) >= max_counter) {
902 m_cbf_buffer.remove(cbf_id);
908 m_cbf_buffer.remove(cbf_id);
911 m_cbf_buffer.update(cbf_id, clock_cast(timeout_cbf(ll->sender)));
915 if (ll->destination == m_local_position_vector.gn_addr.mid()) {
921 static const PendingPacketForwarding::Function noop_fn =
922 [](PendingPacketForwarding::Packet&&,
const MacAddress&) {};
923 PendingPacketForwarding noop { std::move(packet).packet(), noop_fn };
924 CbfPacket cbf { std::move(noop), ll->sender };
925 m_cbf_buffer.add(std::move(cbf), clock_cast(m_mib.itsGnCbfMaxTime));
928 CbfPacket cbf { std::move(packet), ll->sender };
929 m_cbf_buffer.add(std::move(cbf), clock_cast(m_mib.itsGnCbfMaxTime));
935 CbfPacket cbf { std::move(packet), ll->sender };
936 m_cbf_buffer.add(std::move(cbf), clock_cast(timeout_cbf(ll->sender)));
946 using units::si::meter;
947 auto position_sender = m_location_table.get_position(sender);
948 auto position_forwarder = m_location_table.get_position(forwarder);
952 if (position_sender && position_forwarder) {
953 auto dist_r = distance(position_sender->position(), m_local_position_vector.position());
954 auto dist_f = distance(position_forwarder->position(), position_sender->position());
955 const auto dist_max = m_mib.itsGnDefaultMaxCommunicationRange;
957 auto dist_rf = distance(position_forwarder->position(), m_local_position_vector.position());
958 auto angle_fsr = 0.0 * units::si::radians;
959 if (dist_r > 0.0 * meter && dist_f > 0.0 * meter) {
960 auto cos_fsr = (dist_rf * dist_rf - dist_r * dist_r - dist_f * dist_f) /
961 (-2.0 * dist_r * dist_f);
962 angle_fsr = boost::units::acos(cos_fsr);
964 const auto angle_th = m_mib.itsGnBroadcastCBFDefSectorAngle;
966 return !(dist_r < dist_f && dist_f < dist_max && angle_fsr < angle_th);
975 const Address& source_addr = shb.source_position.gn_addr;
981 auto& source_entry = m_location_table.update(shb.source_position);
983 assert(source_entry.has_position_vector() || !is_valid(shb.source_position));
986 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
987 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
990 source_entry.set_neighbour(
true, m_mib.vanetzaNeighbourFlagExpiry);
993 if (m_mib.itsGnIfType == InterfaceType::ITS_G5) {
994 boost::optional<
DccMcoField> dcc_mco = get_dcc_mco(shb.dcc);
996 auto& loctex = source_entry.extensions.get<
LocTEX_G5>();
997 loctex.local_update = m_runtime
.now();
998 loctex.source_update = shb.source_position.timestamp;
999 loctex.dcc_mco = *dcc_mco;
1010 const Address& source_addr = tsb.source_position.gn_addr;
1013 const bool locte_exists = m_location_table.has_entry(source_addr);
1022 if (m_mib.vanetzaMultiHopDuplicateAddressDetection) {
1028 auto& source_entry = m_location_table.update(tsb.source_position);
1029 if (!locte_exists) {
1031 source_entry.set_neighbour(
false);
1035 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
1036 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
1045 if (pdu.basic().hop_limit <= 1) {
1047 forwarding_stopped(ForwardingStopReason::Hop_Limit);
1049 }
else if (m_mib.itsGnMaxPacketDataRate < std::numeric_limits<
decltype(m_mib.itsGnMaxPacketDataRate)>::max()) {
1051 if (source_entry.get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1052 forwarding_stopped(ForwardingStopReason::Source_PDR);
1054 }
else if (
const auto* sender_entry = m_location_table.get_entry(ll.sender)) {
1055 if (sender_entry->get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1056 forwarding_stopped(ForwardingStopReason::Sender_PDR);
1063 auto fwd_dup = create_forwarding_duplicate(pdu, packet);
1064 TsbPdu& fwd_pdu = get_pdu(fwd_dup);
1065 --fwd_pdu.basic().hop_limit;
1066 assert(fwd_pdu.basic().hop_limit + 1 == pdu.basic().hop_limit);
1068 auto transmit = [
this](PendingPacket<TsbPdu>::Packet&& packet) {
1073 std::unique_ptr<Pdu> pdu;
1074 std::unique_ptr<DownPacket> payload;
1075 std::tie(pdu, payload) = std::move(packet);
1078 request.destination = cBroadcastMacAddress;
1079 request.source = m_local_position_vector.gn_addr.mid();
1080 request.dcc_profile = dcc::Profile::DP3;
1081 request.ether_type = geonet::ether_type;
1082 request.lifetime = clock_cast(pdu->basic().lifetime.decode());
1083 pass_down(request, std::move(pdu), std::move(payload));
1086 PendingPacket<TsbPdu> fwd_packet(std::move(fwd_dup), transmit);
1089 const bool scf = pdu.common().traffic_class.store_carry_forward();
1090 if (scf && !m_location_table.has_neighbours()) {
1091 PacketBuffer::data_ptr data {
new PendingPacketBufferData<TsbPdu>(std::move(fwd_packet)) };
1092 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
1097 std::move(fwd_packet).process();
1106 const Address& source_addr = beacon.source_position.gn_addr;
1112 auto& source_entry = m_location_table.update(beacon.source_position);
1115 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
1116 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
1119 source_entry.set_neighbour(
true, m_mib.vanetzaNeighbourFlagExpiry);
1129 const Address& source_addr = gbc.source_position.gn_addr;
1130 const Area dest_area = gbc.destination(pdu.common().header_type);
1133 const bool locte_exists = m_location_table.has_entry(source_addr);
1136 const bool within_destination = inside_or_at_border(dest_area, m_local_position_vector.position());
1138 bool duplicate_packet =
false;
1139 if (!within_destination) {
1140 if (m_mib.itsGnNonAreaForwardingAlgorithm == UnicastForwarding::Unspecified ||
1141 m_mib.itsGnNonAreaForwardingAlgorithm == UnicastForwarding::Greedy) {
1146 if (m_mib.itsGnAreaForwardingAlgorithm == BroadcastForwarding::Unspecified ||
1147 m_mib.itsGnAreaForwardingAlgorithm == BroadcastForwarding::SIMPLE) {
1152 if (duplicate_packet) {
1158 if (m_mib.vanetzaMultiHopDuplicateAddressDetection) {
1164 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
1165 auto& source_entry = m_location_table.update(gbc.source_position);
1166 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
1167 if (!locte_exists) {
1169 source_entry.set_neighbour(
false);
1179 if (pdu.basic().hop_limit <= 1) {
1180 forwarding_stopped(ForwardingStopReason::Hop_Limit);
1182 }
else if (m_mib.itsGnMaxPacketDataRate < std::numeric_limits<
decltype(m_mib.itsGnMaxPacketDataRate)>::max()) {
1184 if (source_entry.get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1185 forwarding_stopped(ForwardingStopReason::Source_PDR);
1187 }
else if (
const auto* sender_entry = m_location_table.get_entry(ll.sender)) {
1188 if (sender_entry->get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1189 forwarding_stopped(ForwardingStopReason::Sender_PDR);
1196 auto fwd_dup = create_forwarding_duplicate(pdu, packet);
1197 GbcPdu& fwd_pdu = get_pdu(fwd_dup);
1198 --fwd_pdu.basic().hop_limit;
1199 assert(fwd_pdu.basic().hop_limit + 1 == pdu.basic().hop_limit);
1201 using Packet = PendingPacketGbc::Packet;
1203 auto transmit = [
this](Packet&& packet,
const MacAddress& mac) {
1208 std::unique_ptr<Pdu> pdu;
1209 std::unique_ptr<DownPacket> payload;
1210 std::tie(pdu, payload) = std::move(packet);
1213 request.destination = mac;
1214 request.source = m_local_position_vector.gn_addr.mid();
1215 request.dcc_profile = dcc::Profile::DP3;
1216 request.ether_type = geonet::ether_type;
1217 request.lifetime = clock_cast(pdu->basic().lifetime.decode());
1219 pass_down(request, std::move(pdu), std::move(payload));
1222 auto forwarding = [
this, transmit, ll](Packet&& packet) {
1228 std::move(forwarding).process();
1231 PendingPacketGbc fwd_packet(std::move(fwd_dup), forwarding);
1234 const bool scf = pdu.common().traffic_class.store_carry_forward();
1235 if (scf && !m_location_table.has_neighbours()) {
1236 PacketBuffer::data_ptr data {
new PendingPacketBufferData<GbcPdu>(std::move(fwd_packet)) };
1237 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
1239 fwd_packet.process();
1249 const bool accept = within_destination || m_mib.vanetzaGbcPassUpOutsideDestination;
1251 if (m_mib.vanetzaGbcMemoryCapacity == 0) {
1254 }
else if (accept) {
1256 return !m_gbc_memory.remember(std::make_tuple(gbc.source_position.gn_addr, gbc.sequence_number));
1264 m_bc_forward_buffer.flush(m_runtime.now());
1270 mark_unused(source);
1271 m_uc_forward_buffer.flush(m_runtime.now());
1277 if (m_mib.itsGnLocalAddrConfMethod == AddrConfMethod::Auto) {
1278 const Address& local = m_local_position_vector.gn_addr;
1279 if (source == local || sender == local.mid()) {
1281 std::uniform_int_distribution<
unsigned> octet_dist;
1282 for (
auto& octet : random_mac_addr.octets) {
1283 octet = octet_dist(m_random_gen);
1286 m_local_position_vector.gn_addr.mid(random_mac_addr);
1293 bool is_duplicate =
false;
1294 ObjectContainer& so_ext = m_location_table.get_or_create_entry(addr_so).extensions;
1299 std::unique_ptr<DuplicatePacketList> dpl {
new DuplicatePacketList(m_mib.itsGnDPLLength) };
1300 is_duplicate = dpl->check(sn);
1301 so_ext.insert(std::move(dpl));
1303 return is_duplicate;
1308 std::unique_ptr<ShbPdu> pdu {
new ShbPdu(request, m_mib) };
1309 pdu->basic().hop_limit = 1;
1310 pdu->common().header_type = HeaderType::TSB_Single_Hop;
1311 pdu->common().maximum_hop_limit = 1;
1312 pdu->extended().source_position = m_local_position_vector;
1313 pdu->extended().dcc = m_dcc_field_generator->generate_dcc_field();
1319 std::unique_ptr<BeaconPdu> pdu {
new BeaconPdu(m_mib) };
1320 pdu->basic().hop_limit = 1;
1321 pdu->common().next_header = NextHeaderCommon::Any;
1322 pdu->common().header_type = HeaderType::Beacon;
1323 pdu->common().maximum_hop_limit = 1;
1325 pdu->extended().source_position = m_local_position_vector;
1331 std::unique_ptr<GbcPdu> pdu {
new GbcPdu(request, m_mib) };
1332 pdu->common().header_type = gbc_header_type(request.destination);
1333 pdu->extended().sequence_number = m_local_sequence_number++;
1334 pdu->extended().source_position = m_local_position_vector;
1335 pdu->extended().destination(request.destination);
1341 if (m_security_entity) {
1344 sec_payload
.merge(*packet
, OsiLayer::Transport
, max_osi_layer()
);
1347 sign_request.plain_message = std::move(sec_payload);
1348 sign_request.its_aid = its_aid;
1349 sign_request.permissions = std::move(ssp);
1353 struct Visitor : boost::static_visitor<DownPacketPtr>
1355 Visitor(DownPacketPtr packet,
Pdu& pdu) : m_packet(std::move(packet)), m_pdu(pdu)
1357 assert(size(*m_packet, OsiLayer::Transport, max_osi_layer()) == 0);
1358 assert(m_pdu.basic().next_header == NextHeaderBasic::Secured);
1361 DownPacketPtr operator() (security::SecuredMessage& msg)
1363 m_pdu.secured(std::move(msg));
1364 return std::move(m_packet);
1367 DownPacketPtr operator() (
const security::SignConfirmError&)
1373 DownPacketPtr m_packet;
1377 Visitor visitor(std::move(packet), pdu);
1378 return boost::apply_visitor(visitor, confirm);
1387 std::string reason_string;
1392 reason_string =
"Parse_Basic_Header";
1395 reason_string =
"Parse_Common_Header";
1398 reason_string =
"Parse_Secured_Header";
1401 reason_string =
"Parse_Extended_Header";
1404 reason_string =
"ITS_Protocol_Version";
1407 reason_string =
"Decap_Unsuccessful_Non_Strict";
1410 reason_string =
"Decap_Unsuccessful_Strict";
1413 reason_string =
"Hop_Limit";
1416 reason_string =
"Payload_Size";
1419 reason_string =
"Security_Entity_Missing";
1422 reason_string =
"UNKNOWN";
1426 return reason_string;
ChunckPacket is a packet consisting of several memory chunks.
ByteBufferConvertible & operator[](OsiLayer ol)
ChunkPacket & merge(ChunkPacket &packet, OsiLayer from, OsiLayer to)
virtual void cancel(const void *scope)=0
virtual Clock::time_point now() const =0
const MacAddress & sender() const
bool check(SequenceNumber)
const MacAddress & mac() const
void transmit(Packet &&packet, const MacAddress &destination)
void reset_beacon_timer(Clock::duration next)
Reschedule timer for next Beacon transmission.
NextHop area_advanced_forwarding(PendingPacketForwarding &&, const LinkLayer *sender)
Determine next hop for area advanced forwarding See EN 302 636-4-1 v1.3.1 Annex F....
void flush_broadcast_forwarding_buffer()
Send all packets in the broadcast forwarding buffer with expired waiting time.
void indicate_extended(IndicationContext &, const CommonHeader &)
Process ExtendedHeader at packet indication.
bool detect_duplicate_packet(const Address &source, SequenceNumber sn)
Detect duplicate packets See EN 302 636-4-1 v1.3.1 Annex A.2.
void pass_down(const dcc::DataRequest &, PduPtr, DownPacketPtr)
Send packet using the information in the DataRequest. The packet is formed using the data in PDU and ...
NextHop greedy_forwarding(PendingPacketForwarding &&)
Determine next hop for greedy forwarding. See EN 302 636-4-1 v1.3.1 Annex E.2.
bool process_extended(const ExtendedPduConstRefs< ShbHeader > &, const UpPacket &, const LinkLayer &ll)
Process ExtendedHeader information. Update router's LocationTable and neighbour relationship....
void pass_up(const DataIndication &, UpPacketPtr)
Pass packet up to the transport layer.
units::Duration timeout_cbf(units::Length distance) const
Determine CBF buffering time for a packet. Complies to EN 302 636-4-1 v1.3.1 Annex E....
void on_beacon_timer_expired()
Send Beacon packet to all neighbours with updated position vector. Only to be called when the beacon ...
void indicate_common(IndicationContext &, const BasicHeader &)
Process CommonHeader at packet indication.
DownPacketPtr encap_packet(ItsAid aid, ByteBuffer ssp, Pdu &pdu, DownPacketPtr packet)
Encaspulate a packet according to security profile.
bool decide_pass_up(bool within_destination, const GeoBroadcastHeader &gbc)
Decide if GBC packet shall be passed up to transport layer.
void set_access_interface(dcc::RequestInterface *ifc)
Register access layer interface.
void set_random_seed(std::uint_fast32_t seed)
Set seed for internal random number generator (RNG) RNG is used e.g. for random Beacon jitter.
units::Duration timeout_cbf(const MacAddress &sender) const
Determine (area) CBF buffering time for a packet from a sender.
NextHop area_contention_based_forwarding(PendingPacketForwarding &&, const MacAddress *sender)
Determine next hop for area contention-based forwarding See EN 302 636-4-1 v1.3.1 Annex F....
std::unique_ptr< ShbPdu > create_shb_pdu(const ShbDataRequest &)
Create an initialized Single-Hop-Broadcast PDU.
void flush_unicast_forwarding_buffer(const Address &addr)
Send all matching packets in the unicast forwarding buffer with expired waiting time.
void detect_duplicate_address(const Address &source, const MacAddress &sender)
Helper method to handle duplicate addresses. If own address collides with the address of a received p...
std::unique_ptr< BeaconPdu > create_beacon_pdu()
Create an initialzed Beacon PDU.
void indicate_basic(IndicationContextBasic &)
Process BasicHeader at packet indication.
void set_dcc_field_generator(DccFieldGenerator *dcc)
Register generator for DCC-MCO fields.
bool process_extended(const ExtendedPduConstRefs< TsbHeader > &, const UpPacket &, const LinkLayer &ll)
packet handling of received TSB packet
PacketDropReason
Reason for packet drop used by drop hook.
DataConfirm request(const ShbDataRequest &, DownPacketPtr)
Request to send payload per single hop broadcast (SHB). If security is enabled, the message gets enca...
void pass_down(const MacAddress &, PduPtr, DownPacketPtr)
Pass down the packet to the access layer.
void dispatch_repetition(const DataRequestVariant &, DownPacketPtr)
Callback function for dispatching a packet repetition. Invoked by Repeater when a scheduled repetitio...
NextHop non_area_contention_based_forwarding(PendingPacketForwarding &&, const MacAddress *sender)
Determine next hop for non-area contention-based forwarding See EN 302 636-4-1 v1....
bool process_extended(const ExtendedPduConstRefs< GeoBroadcastHeader > &, const UpPacket &, const LinkLayer &ll)
Process ExtendedHeader information. Update router's LocationTable and neighbour relationship....
void indicate(UpPacketPtr, const MacAddress &sender, const MacAddress &destination)
Handle the received packet on network layer. Packet handling involves these steps:
void set_transport_handler(UpperProtocol proto, TransportInterface *ifc)
Register a transport protocol handler.
void reset_beacon_timer()
Reschedule timer for next Beacon transmission Timer will be scheduled according to MIB's Beacon timer...
void update_position(const PositionFix &)
Update router's local position vector.
void execute_media_procedures(CommunicationProfile)
Executes media specific functionalities Details are described in TS 102 636-4-2.
void set_security_entity(security::SecurityEntity *entity)
Register security entity used when itsGnSecurity is enabled.
std::unique_ptr< GbcPdu > create_gbc_pdu(const GbcDataRequest &)
Create an initialized GeoBroadcast PDU.
void set_address(const Address &)
Set Router's own GeoNetworking address.
NextHop forwarding_algorithm_selection(PendingPacketForwarding &&, const LinkLayer *ll=nullptr)
DataConfirm request(const GbcDataRequest &, DownPacketPtr)
Request to send payload per GeoBroadcast (GBC). If security is enabled, the message gets encapsulated...
void execute_itsg5_procedures()
Executes ITS-G5 media specific procedures Details are described in TS 102 636-4-2.
bool process_extended(const ExtendedPduConstRefs< BeaconHeader > &, const UpPacket &, const LinkLayer &ll)
Process ExtendedHeader information. Update router's LocationTable and neighbour relationship.
void indicate_secured(IndicationContextBasic &, const BasicHeader &)
Process SecuredMessage at packet indication.
bool outside_sectorial_contention_area(const MacAddress &sender, const MacAddress &forwarder) const
Check if router is outside the sectorial contention area See TS 102 636-4-1 v1.2.3 section E....
virtual EncapConfirm encapsulate_packet(EncapRequest &&request)=0
Creates a security envelope covering the given payload.
virtual DecapConfirm decapsulate_packet(DecapRequest &&request)=0
Decapsulates the payload within a SecuredMessage.