1#include <vanetza/btp/data_indication.hpp>
2#include <vanetza/btp/data_request.hpp>
3#include <vanetza/common/its_aid.hpp>
4#include <vanetza/common/position_fix.hpp>
5#include <vanetza/common/runtime.hpp>
6#include <vanetza/dcc/data_request.hpp>
7#include <vanetza/dcc/interface.hpp>
8#include <vanetza/dcc/profile.hpp>
9#include <vanetza/net/mac_address.hpp>
10#include <vanetza/net/osi_layer.hpp>
11#include <vanetza/units/frequency.hpp>
12#include <vanetza/units/length.hpp>
13#include <vanetza/units/time.hpp>
14#include <vanetza/geonet/router.hpp>
15#include <vanetza/geonet/cbf_counter.hpp>
16#include <vanetza/geonet/data_confirm.hpp>
17#include <vanetza/geonet/dcc_field_generator.hpp>
18#include <vanetza/geonet/duplicate_packet_list.hpp>
19#include <vanetza/geonet/indication_context.hpp>
20#include <vanetza/geonet/loctex_g5.hpp>
21#include <vanetza/geonet/next_hop.hpp>
22#include <vanetza/geonet/pdu_conversion.hpp>
23#include <vanetza/geonet/repetition_dispatcher.hpp>
24#include <vanetza/geonet/transport_interface.hpp>
25#include <vanetza/geonet/extended_pdu.hpp>
26#include <vanetza/geonet/secured_pdu.hpp>
27#include <boost/units/cmath.hpp>
42 ControlInfo(
const DataRequest& request) :
43 communication_profile(request.communication_profile),
44 its_aid(request.its_aid),
45 permissions(request.permissions)
48 const CommunicationProfile communication_profile;
50 const ByteBuffer permissions;
57 PendingPacketBufferData(PendingPacket<PDU>&& packet) : m_packet(std::move(packet)) {}
59 std::size_t length()
const override
61 return m_packet.length();
64 Clock::duration reduce_lifetime(Clock::duration d)
override
66 return m_packet.reduce_lifetime(d);
75 PendingPacket<PDU> m_packet;
78dcc::RequestInterface* get_default_request_interface()
80 static dcc::NullRequestInterface null;
91auto create_forwarding_duplicate(
const PDU& pdu,
const UpPacket& packet) ->
92std::tuple<std::unique_ptr<ExtendedPdu<typename PDU::ExtendedHeader>>, std::unique_ptr<DownPacket>>
95 std::unique_ptr<pdu_type> pdu_dup {
new pdu_type { pdu }};
96 std::unique_ptr<DownPacket> packet_dup;
98 packet_dup.reset(
new DownPacket());
100 packet_dup = duplicate(packet);
102 return std::make_tuple(std::move(pdu_dup), std::move(packet_dup));
105template<
typename PDU>
106PDU& get_pdu(
const std::tuple<std::unique_ptr<PDU>, std::unique_ptr<DownPacket>>& packet)
108 PDU* pdu = std::get<0>(packet).get();
113std::unique_ptr<CbfCounter> create_cbf_counter(Runtime& rt,
const MIB& mib)
115 std::unique_ptr<CbfCounter> counter;
116 if (mib.vanetzaFadingCbfCounter) {
117 counter.reset(
new CbfCounterFading(rt, units::clock_cast(mib.vanetzaFadingCbfCounterLifetime)));
127using units::clock_cast;
130const access::EtherType ether_type = access::ethertype::GeoNetworking;
132Router::Router(Runtime& rt,
const MIB& mib) :
135 m_request_interface(get_default_request_interface()),
136 m_dcc_field_generator(get_default_dcc_field_generator()),
137 m_security_entity(nullptr),
138 m_location_table(mib, m_runtime),
139 m_bc_forward_buffer(mib.itsGnBcForwardingPacketBufferSize * 1024),
140 m_uc_forward_buffer(mib.itsGnUcForwardingPacketBufferSize * 1024),
141 m_cbf_buffer(m_runtime,
142 [](PendingPacketGbc&& packet) { packet.process(); },
143 create_cbf_counter(rt, mib),
144 mib.itsGnCbfPacketBufferSize * 1024),
145 m_local_sequence_number(0),
146 m_repeater(m_runtime,
147 std::bind(&Router::dispatch_repetition,
this, std::placeholders::_1, std::placeholders::_2)),
148 m_random_gen(mib.vanetzaDefaultSeed)
150 if (!m_mib.vanetzaDisableBeaconing) {
151 if (m_mib.vanetzaDeferInitialBeacon > Clock::duration::zero()) {
153 std::uniform_real_distribution<double> dist_jitter(0.0, 1.0);
154 const auto jitter = clock_cast(dist_jitter(m_random_gen) * m_mib.itsGnBeaconServiceMaxJitter);
155 reset_beacon_timer(m_mib.vanetzaDeferInitialBeacon + jitter);
158 reset_beacon_timer(Clock::duration::zero());
162 m_gbc_memory.capacity(m_mib.vanetzaGbcMemoryCapacity);
167 m_runtime.cancel(
this);
175 m_local_position_vector.timestamp = m_runtime.now();
176 m_local_position_vector.latitude =
static_cast<geo_angle_i32t
>(position_fix.latitude);
177 m_local_position_vector.longitude =
static_cast<geo_angle_i32t
>(position_fix.longitude);
178 if (m_mib.itsGnIsMobile) {
179 m_local_position_vector.speed =
static_cast<LongPositionVector::speed_u15t
>(position_fix.speed.value());
180 m_local_position_vector.heading =
static_cast<heading_u16t
>(position_fix.course.value() - units::TrueNorth::from_value(0.0));
182 m_local_position_vector.speed =
static_cast<LongPositionVector::speed_u15t
>(0);
183 m_local_position_vector.heading =
static_cast<heading_u16t
>(0);
186 m_local_position_vector.position_accuracy_indicator =
187 position_fix.confidence.semi_major * 2.0 < m_mib.itsGnPaiInterval;
192 m_transport_ifcs[proto] = ifc;
197 m_security_entity = entity;
202 m_request_interface = (ifc ==
nullptr ? get_default_request_interface() : ifc);
203 assert(m_request_interface !=
nullptr);
208 m_dcc_field_generator = (dcc ==
nullptr) ? get_default_dcc_field_generator() : dcc;
209 assert(m_dcc_field_generator !=
nullptr);
214 m_local_position_vector.gn_addr = addr;
219 m_random_gen.seed(seed);
225 result ^= validate_data_request(
request, m_mib);
226 result ^= validate_payload(payload, m_mib);
228 if (result.accepted()) {
234 m_repeater.add(
request, *payload);
239 pdu->common().payload = payload->size();
242 auto transmit = [
this, ctrl](PendingPacket::Packet&& packet) {
243 std::unique_ptr<ShbPdu> pdu;
244 std::unique_ptr<DownPacket> payload;
245 std::tie(pdu, payload) = std::move(packet);
248 pdu->extended().source_position = m_local_position_vector;
251 if (m_mib.itsGnSecurity) {
252 payload =
encap_packet(ctrl.its_aid, ctrl.permissions, *pdu, std::move(payload));
263 pass_down(cBroadcastMacAddress, std::move(pdu), std::move(payload));
269 PendingPacket packet(std::make_tuple(std::move(pdu), std::move(payload)), transmit);
272 if (
request.traffic_class.store_carry_forward() && !m_location_table.has_neighbours()) {
273 PacketBuffer::data_ptr data {
new PendingPacketBufferData<ShbPdu>(std::move(packet)) };
274 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
287 result ^= validate_data_request(
request, m_mib);
288 result ^= validate_payload(payload, m_mib);
290 if (!result.accepted())
299 m_repeater.add(
request, *payload);
303 using Packet = PendingPacket::Packet;
307 pdu->common().payload = payload->size();
310 auto transmit = [
this, ctrl](Packet&& packet,
const MacAddress& mac) {
311 std::unique_ptr<GbcPdu> pdu;
312 std::unique_ptr<DownPacket> payload;
313 std::tie(pdu, payload) = std::move(packet);
316 pdu->extended().source_position = m_local_position_vector;
319 if (m_mib.itsGnSecurity) {
320 assert(pdu->basic().next_header == NextHeaderBasic::Secured);
321 payload =
encap_packet(ctrl.its_aid, ctrl.permissions, *pdu, std::move(payload));
334 pass_down(mac, std::move(pdu), std::move(payload));
337 auto forwarding = [
this, transmit](Packet&& packet) {
342 std::move(nh).process();
345 PendingPacket packet(std::make_tuple(std::move(pdu), std::move(payload)), forwarding);
348 const bool scf =
request.traffic_class.store_carry_forward();
349 if (scf && !m_location_table.has_neighbours()) {
350 PacketBuffer::data_ptr data {
new PendingPacketBufferData<GbcPdu>(std::move(packet)) };
351 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
361 return DataConfirm(DataConfirm::ResultCode::Rejected_Unspecified);
364DataConfirm Router::request(
const GucDataRequest&, DownPacketPtr)
366 return DataConfirm(DataConfirm::ResultCode::Rejected_Unspecified);
369DataConfirm Router::request(
const TsbDataRequest&, DownPacketPtr)
371 return DataConfirm(DataConfirm::ResultCode::Rejected_Unspecified);
377 const auto size_limit = m_mib.itsGnMaxSduSize + m_mib.itsGnMaxGeoNetworkingHeaderSize;
378 if (size(*packet) <= size_limit) {
379 IndicationContext::LinkLayer link_layer;
380 link_layer.sender = sender;
381 link_layer.destination = destination;
383 if (
auto cohesive = boost::get<CohesivePacket>(packet.get())) {
387 }
else if (
auto chunk = boost::get<ChunkPacket>(packet.get())) {
404 }
else if (basic->version.raw() != m_mib.itsGnProtocolVersion) {
408 indication.remaining_packet_lifetime = basic->lifetime;
409 indication.remaining_hop_limit = basic->hop_limit;
411 if (basic->next_header == NextHeaderBasic::Secured) {
413 }
else if (basic->next_header == NextHeaderBasic::Common) {
414 if (!m_mib.itsGnSecurity || SecurityDecapHandling::Non_Strict == m_mib.itsGnSnDecapResultHandling) {
415 indication.security_report = boost::blank {};
429 }
else if (common->maximum_hop_limit < basic.hop_limit) {
434 indication.traffic_class = common->traffic_class;
435 switch (common->next_header)
437 case NextHeaderCommon::BTP_A:
438 indication.upper_protocol = UpperProtocol::BTP_A;
440 case NextHeaderCommon::BTP_B:
441 indication.upper_protocol = UpperProtocol::BTP_B;
443 case NextHeaderCommon::IPv6:
444 indication.upper_protocol = UpperProtocol::IPv6;
447 indication.upper_protocol = UpperProtocol::Unknown;
452 m_location_table.drop_expired();
469 struct secured_payload_visitor :
public boost::static_visitor<>
472 m_router(router), m_context(ctx), m_basic(basic)
479 m_router.indicate_common(ctx, m_basic);
485 m_router.indicate_common(ctx, m_basic);
493 auto secured_message = ctx.parse_secured();
494 if (!secured_message) {
496 }
else if (m_security_entity) {
498 using namespace vanetza::security;
500 ctx.service_primitive().security_report = decap_confirm.report;
501 ctx.service_primitive().its_aid = decap_confirm.its_aid;
502 ctx.service_primitive().permissions = decap_confirm.permissions;
503 secured_payload_visitor visitor(*
this, ctx, basic);
506 if (is_successful(decap_confirm.report)) {
507 boost::apply_visitor(visitor, decap_confirm.plaintext_payload);
508 }
else if (SecurityDecapHandling::Non_Strict == m_mib.itsGnSnDecapResultHandling) {
511 if (!decap_confirm.plaintext_payload.empty()) {
512 boost::apply_visitor(visitor, decap_confirm.plaintext_payload);
528 struct extended_header_visitor :
public boost::static_visitor<bool>
530 extended_header_visitor(Router& router,
IndicationContext& ctx,
const UpPacket& packet) :
531 m_router(router), m_context(ctx), m_packet(packet)
538 indication.transport_type = TransportType::SHB;
541 auto& pdu = m_context.pdu();
543 return m_router.process_extended(shb_pdu, m_packet, m_context.link_layer());
549 indication.transport_type = TransportType::TSB;
552 auto& pdu = m_context.pdu();
554 return m_router.process_extended(tsb_pdu, m_packet, m_context.link_layer());
560 indication.transport_type = TransportType::GBC;
562 indication.destination = gbc.destination(m_context.pdu().common().header_type);
564 auto& pdu = m_context.pdu();
566 return m_router.process_extended(gbc_pdu, m_packet, m_context.link_layer());
571 auto& pdu = m_context.pdu();
573 return m_router.process_extended(beacon_pdu, m_packet, m_context.link_layer());
578 const UpPacket& m_packet;
581 auto extended = ctx.parse_extended(common.header_type);
582 UpPacketPtr packet = ctx.
finish();
587 }
else if (common.payload != size(*packet, OsiLayer::Transport, max_osi_layer())) {
590 extended_header_visitor visitor(*
this, ctx, *packet);
591 if (boost::apply_visitor(visitor, *extended)) {
592 pass_up(ctx.service_primitive(), std::move(packet));
600 const Area& destination = packet.pdu().extended().destination(packet.pdu().common().header_type);
601 if (inside_or_at_border(destination, m_local_position_vector.position())) {
602 switch (m_mib.itsGnAreaForwardingAlgorithm) {
603 case BroadcastForwarding::Unspecified:
605 case BroadcastForwarding::SIMPLE:
607 nh.
transmit(std::move(packet), cBroadcastMacAddress);
609 case BroadcastForwarding::CBF:
612 case BroadcastForwarding::Advanced:
616 throw std::runtime_error(
"unhandled area forwarding algorithm");
621 const LongPositionVector* pv_se = ll ? m_location_table.get_position(ll->sender) :
nullptr;
622 if (pv_se && pv_se->position_accuracy_indicator && inside_or_at_border(destination, pv_se->position())) {
626 switch (m_mib.itsGnNonAreaForwardingAlgorithm) {
627 case UnicastForwarding::Unspecified:
629 case UnicastForwarding::Greedy:
632 case UnicastForwarding::CBF:
636 throw std::runtime_error(
"unhandled non-area forwarding algorithm");
647 switch (com_profile) {
648 case CommunicationProfile::ITS_G5:
651 case CommunicationProfile::Unspecified:
652 case CommunicationProfile::LTE_V2X:
656 throw std::runtime_error(
"Unhandled communication profile");
670 if (pdu->secured()) {
671 if (pdu->basic().next_header != NextHeaderBasic::Secured) {
672 throw std::runtime_error(
"PDU with secured message but Secured not set in basic header");
674 if (payload->size(OsiLayer::Transport, max_osi_layer()) > 0) {
675 throw std::runtime_error(
"PDU with secured message and illegal upper layer payload");
678 if (pdu->basic().next_header == NextHeaderBasic::Secured) {
679 throw std::runtime_error(
"PDU without secured message but Secured set in basic header");
684 assert(m_request_interface);
685 m_request_interface->request(
request, std::move(payload));
694 request.source = m_local_position_vector.gn_addr.mid();
695 request.dcc_profile = map_tc_onto_profile(pdu->common().traffic_class);
696 request.ether_type = geonet::ether_type;
697 request.lifetime = clock_cast(pdu->basic().lifetime.decode());
705 if (transport !=
nullptr) {
706 transport->indicate(ind, std::move(packet));
712 if (m_mib.vanetzaDisableBeaconing) {
718 DownPacketPtr payload {
new DownPacket() };
721 if (m_mib.itsGnSecurity) {
722 pdu->basic().next_header = NextHeaderBasic::Secured;
723 payload =
encap_packet(aid::GN_MGMT, ByteBuffer {}, *pdu, std::move(payload));
729 pdu->basic().next_header = NextHeaderBasic::Common;
733 pass_down(cBroadcastMacAddress, std::move(pdu), std::move(payload));
739 using duration_t =
decltype(m_mib.itsGnBeaconServiceRetransmitTimer);
740 using real_t = duration_t::value_type;
741 static_assert(std::is_floating_point<real_t>::value,
"floating point type expected");
743 std::uniform_real_distribution<real_t> dist_jitter(0.0, 1.0);
744 const auto jitter = dist_jitter(m_random_gen);
745 const duration_t next_beacon = m_mib.itsGnBeaconServiceRetransmitTimer +
746 jitter * m_mib.itsGnBeaconServiceMaxJitter;
752 m_runtime.cancel(
this);
753 m_runtime.schedule(next_beacon, [
this](Clock::time_point) {
761 boost::apply_visitor(dispatcher,
request);
768 const units::Length own = distance(dest, m_local_position_vector.position());
769 units::Length mfr_dist = own;
773 if (neighbour.has_position_vector()) {
774 const units::Length dist = distance(dest, neighbour.get_position_vector().position());
775 if (dist < mfr_dist) {
776 mfr_addr = neighbour.link_layer_address();
782 if (mfr_dist < own) {
783 nh.
transmit(std::move(packet), mfr_addr);
785 const bool scf = packet.pdu().common().traffic_class.store_carry_forward();
787 std::function<void(PendingPacketForwarding&&)> greedy_fwd = [
this](PendingPacketForwarding&& packet) {
789 std::move(nh).process();
792 PacketBuffer::data_ptr data {
new PendingPacketBufferData<GbcPdu>(std::move(greedy_packet)) };
793 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
796 nh.
transmit(std::move(packet), cBroadcastMacAddress);
807 const auto cbf_id = identifier(gbc.source_position.gn_addr, gbc.sequence_number);
811 nh.
transmit(std::move(packet), cBroadcastMacAddress);
812 }
else if (m_cbf_buffer.remove(cbf_id)) {
816 const HeaderType ht = packet.pdu().common().header_type;
817 const Area destination = gbc.destination(ht);
818 const auto& epv = m_local_position_vector;
819 const LongPositionVector* pv_se = sender ? m_location_table.get_position(*sender) :
nullptr;
821 if (pv_se && pv_se->position_accuracy_indicator) {
822 const auto& pv_p = destination.position;
823 const units::Length dist_sender = distance(pv_p, pv_se->position());
824 const units::Length dist_local = distance(pv_p, epv.position());
825 if (dist_sender > dist_local) {
826 CbfPacket cbf { std::move(packet), *sender };
827 const auto progress = dist_sender - dist_local;
828 m_cbf_buffer.add(std::move(cbf), clock_cast(
timeout_cbf(progress)));
835 CbfPacket cbf { std::move(packet), *sender };
836 const auto to_cbf_max = m_mib.itsGnCbfMaxTime;
837 m_cbf_buffer.add(std::move(cbf), clock_cast(to_cbf_max));
848 const auto cbf_id = identifier(gbc.source_position.gn_addr, gbc.sequence_number);
851 nh.
transmit(std::move(packet), cBroadcastMacAddress);
852 }
else if (m_cbf_buffer.remove(cbf_id) || m_cbf_buffer.counter(cbf_id) >= m_mib.vanetzaCbfMaxCounter) {
855 const units::Duration timeout =
timeout_cbf(*sender);
856 m_cbf_buffer.add(
CbfPacket { std::move(packet), *sender }, clock_cast(timeout));
865 const auto dist_max = m_mib.itsGnDefaultMaxCommunicationRange;
866 const auto to_cbf_min = m_mib.itsGnCbfMinTime;
867 const auto to_cbf_max = m_mib.itsGnCbfMaxTime;
869 if (prog > dist_max) {
871 }
else if (prog > 0.0 * units::si::meter) {
872 return to_cbf_max + (to_cbf_min - to_cbf_max) / dist_max * prog;
881 units::Duration timeout = m_mib.itsGnCbfMaxTime;
883 if (pv_se && pv_se->position_accuracy_indicator) {
884 units::Length dist = distance(pv_se->position(), m_local_position_vector.position());
895 nh.
transmit(std::move(packet), cBroadcastMacAddress);
898 const HeaderType ht = packet.pdu().common().header_type;
899 const Area destination_area = gbc.destination(ht);
900 const std::size_t max_counter = m_mib.vanetzaCbfMaxCounter;
901 const auto cbf_id = identifier(gbc.source_position.gn_addr, gbc.sequence_number);
902 const CbfPacket* cbf_packet = m_cbf_buffer.find(cbf_id);
906 if (m_cbf_buffer.counter(cbf_id) >= max_counter) {
908 m_cbf_buffer.remove(cbf_id);
914 m_cbf_buffer.remove(cbf_id);
917 m_cbf_buffer.update(cbf_id, clock_cast(
timeout_cbf(ll->sender)));
921 if (ll->destination == m_local_position_vector.gn_addr.mid()) {
925 if (nh.
valid() && nh.
mac() == cBroadcastMacAddress) {
927 static const PendingPacketForwarding::Function noop_fn =
928 [](PendingPacketForwarding::Packet&&,
const MacAddress&) {};
929 PendingPacketForwarding noop { std::move(packet).packet(), noop_fn };
930 CbfPacket cbf { std::move(noop), ll->sender };
931 m_cbf_buffer.add(std::move(cbf), clock_cast(m_mib.itsGnCbfMaxTime));
934 CbfPacket cbf { std::move(packet), ll->sender };
935 m_cbf_buffer.add(std::move(cbf), clock_cast(m_mib.itsGnCbfMaxTime));
941 CbfPacket cbf { std::move(packet), ll->sender };
942 m_cbf_buffer.add(std::move(cbf), clock_cast(
timeout_cbf(ll->sender)));
952 using units::si::meter;
953 auto position_sender = m_location_table.get_position(sender);
954 auto position_forwarder = m_location_table.get_position(forwarder);
958 if (position_sender && position_forwarder) {
959 auto dist_r = distance(position_sender->position(), m_local_position_vector.position());
960 auto dist_f = distance(position_forwarder->position(), position_sender->position());
961 const auto dist_max = m_mib.itsGnDefaultMaxCommunicationRange;
963 auto dist_rf = distance(position_forwarder->position(), m_local_position_vector.position());
964 auto angle_fsr = 0.0 * units::si::radians;
965 if (dist_r > 0.0 * meter && dist_f > 0.0 * meter) {
966 auto cos_fsr = (dist_rf * dist_rf - dist_r * dist_r - dist_f * dist_f) /
967 (-2.0 * dist_r * dist_f);
968 angle_fsr = boost::units::acos(cos_fsr);
970 const auto angle_th = m_mib.itsGnBroadcastCBFDefSectorAngle;
972 return !(dist_r < dist_f && dist_f < dist_max && angle_fsr < angle_th);
981 const Address& source_addr = shb.source_position.gn_addr;
987 auto& source_entry = m_location_table.update(shb.source_position);
989 assert(source_entry.has_position_vector() || !is_valid(shb.source_position));
992 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
993 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
996 source_entry.set_neighbour(
true, m_mib.vanetzaNeighbourFlagExpiry);
999 if (m_mib.itsGnIfType == InterfaceType::ITS_G5) {
1000 boost::optional<DccMcoField> dcc_mco = get_dcc_mco(shb.dcc);
1002 auto& loctex = source_entry.extensions.get<
LocTEX_G5>();
1003 loctex.local_update = m_runtime.now();
1004 loctex.source_update = shb.source_position.timestamp;
1005 loctex.dcc_mco = *dcc_mco;
1016 const Address& source_addr = tsb.source_position.gn_addr;
1019 const bool locte_exists = m_location_table.has_entry(source_addr);
1028 if (m_mib.vanetzaMultiHopDuplicateAddressDetection) {
1034 auto& source_entry = m_location_table.update(tsb.source_position);
1035 if (!locte_exists) {
1037 source_entry.set_neighbour(
false);
1041 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
1042 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
1051 if (pdu.basic().hop_limit <= 1) {
1055 }
else if (m_mib.itsGnMaxPacketDataRate < std::numeric_limits<
decltype(m_mib.itsGnMaxPacketDataRate)>::max()) {
1057 if (source_entry.get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1060 }
else if (
const auto* sender_entry = m_location_table.get_entry(ll.sender)) {
1061 if (sender_entry->get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1069 auto fwd_dup = create_forwarding_duplicate(pdu, packet);
1070 TsbPdu& fwd_pdu = get_pdu(fwd_dup);
1071 --fwd_pdu.basic().hop_limit;
1072 assert(fwd_pdu.basic().hop_limit + 1 == pdu.basic().hop_limit);
1074 auto transmit = [
this](PendingPacket<TsbPdu>::Packet&& packet) {
1079 std::unique_ptr<Pdu> pdu;
1080 std::unique_ptr<DownPacket> payload;
1081 std::tie(pdu, payload) = std::move(packet);
1084 request.destination = cBroadcastMacAddress;
1085 request.source = m_local_position_vector.gn_addr.mid();
1086 request.dcc_profile = dcc::Profile::DP3;
1087 request.ether_type = geonet::ether_type;
1088 request.lifetime = clock_cast(pdu->basic().lifetime.decode());
1095 const bool scf = pdu.common().traffic_class.store_carry_forward();
1096 if (scf && !m_location_table.has_neighbours()) {
1097 PacketBuffer::data_ptr data {
new PendingPacketBufferData<TsbPdu>(std::move(fwd_packet)) };
1098 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
1103 std::move(fwd_packet).process();
1112 const Address& source_addr = beacon.source_position.gn_addr;
1118 auto& source_entry = m_location_table.update(beacon.source_position);
1121 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
1122 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
1125 source_entry.set_neighbour(
true, m_mib.vanetzaNeighbourFlagExpiry);
1135 const Address& source_addr = gbc.source_position.gn_addr;
1136 const Area dest_area = gbc.destination(pdu.common().header_type);
1139 const bool locte_exists = m_location_table.has_entry(source_addr);
1142 const bool within_destination = inside_or_at_border(dest_area, m_local_position_vector.position());
1144 bool duplicate_packet =
false;
1145 if (!within_destination) {
1146 if (m_mib.itsGnNonAreaForwardingAlgorithm == UnicastForwarding::Unspecified ||
1147 m_mib.itsGnNonAreaForwardingAlgorithm == UnicastForwarding::Greedy) {
1152 if (m_mib.itsGnAreaForwardingAlgorithm == BroadcastForwarding::Unspecified ||
1153 m_mib.itsGnAreaForwardingAlgorithm == BroadcastForwarding::SIMPLE) {
1158 if (duplicate_packet) {
1164 if (m_mib.vanetzaMultiHopDuplicateAddressDetection) {
1170 const std::size_t packet_size = size(packet, OsiLayer::Network, OsiLayer::Application);
1171 auto& source_entry = m_location_table.update(gbc.source_position);
1172 source_entry.update_pdr(packet_size, m_mib.itsGnMaxPacketDataRateEmaBeta);
1173 if (!locte_exists) {
1175 source_entry.set_neighbour(
false);
1185 if (pdu.basic().hop_limit <= 1) {
1188 }
else if (m_mib.itsGnMaxPacketDataRate < std::numeric_limits<
decltype(m_mib.itsGnMaxPacketDataRate)>::max()) {
1190 if (source_entry.get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1193 }
else if (
const auto* sender_entry = m_location_table.get_entry(ll.sender)) {
1194 if (sender_entry->get_pdr() > m_mib.itsGnMaxPacketDataRate * 1000.0) {
1202 auto fwd_dup = create_forwarding_duplicate(pdu, packet);
1203 GbcPdu& fwd_pdu = get_pdu(fwd_dup);
1204 --fwd_pdu.basic().hop_limit;
1205 assert(fwd_pdu.basic().hop_limit + 1 == pdu.basic().hop_limit);
1207 using Packet = PendingPacketGbc::Packet;
1209 auto transmit = [
this](Packet&& packet,
const MacAddress& mac) {
1214 std::unique_ptr<Pdu> pdu;
1215 std::unique_ptr<DownPacket> payload;
1216 std::tie(pdu, payload) = std::move(packet);
1220 request.source = m_local_position_vector.gn_addr.mid();
1221 request.dcc_profile = dcc::Profile::DP3;
1222 request.ether_type = geonet::ether_type;
1223 request.lifetime = clock_cast(pdu->basic().lifetime.decode());
1228 auto forwarding = [
this, transmit, ll](Packet&& packet) {
1234 std::move(forwarding).process();
1237 PendingPacketGbc fwd_packet(std::move(fwd_dup), forwarding);
1240 const bool scf = pdu.common().traffic_class.store_carry_forward();
1241 if (scf && !m_location_table.has_neighbours()) {
1242 PacketBuffer::data_ptr data {
new PendingPacketBufferData<GbcPdu>(std::move(fwd_packet)) };
1243 m_bc_forward_buffer.push(std::move(data), m_runtime.now());
1245 fwd_packet.process();
1254 if (m_mib.vanetzaGbcMemoryCapacity == 0) {
1256 return within_destination;
1257 }
else if (within_destination) {
1259 return !m_gbc_memory.remember(std::make_tuple(gbc.source_position.gn_addr, gbc.sequence_number));
1267 m_bc_forward_buffer.flush(m_runtime.now());
1273 m_uc_forward_buffer.flush(m_runtime.now());
1279 if (m_mib.itsGnLocalAddrConfMethod == AddrConfMethod::Auto) {
1280 const Address& local = m_local_position_vector.gn_addr;
1281 if (source == local || sender == local.mid()) {
1283 std::uniform_int_distribution<unsigned> octet_dist;
1284 for (
auto& octet : random_mac_addr.octets) {
1285 octet = octet_dist(m_random_gen);
1288 m_local_position_vector.gn_addr.mid(random_mac_addr);
1295 bool is_duplicate =
false;
1296 ObjectContainer& so_ext = m_location_table.get_or_create_entry(addr_so).extensions;
1299 is_duplicate = dpl->
check(sn);
1301 std::unique_ptr<DuplicatePacketList> dpl {
new DuplicatePacketList(m_mib.itsGnDPLLength) };
1302 is_duplicate = dpl->check(sn);
1303 so_ext.insert(std::move(dpl));
1305 return is_duplicate;
1310 std::unique_ptr<ShbPdu> pdu {
new ShbPdu(
request, m_mib) };
1311 pdu->basic().hop_limit = 1;
1312 pdu->common().header_type = HeaderType::TSB_Single_Hop;
1313 pdu->common().maximum_hop_limit = 1;
1314 pdu->extended().source_position = m_local_position_vector;
1315 pdu->extended().dcc = m_dcc_field_generator->generate_dcc_field();
1321 std::unique_ptr<BeaconPdu> pdu {
new BeaconPdu(m_mib) };
1322 pdu->basic().hop_limit = 1;
1323 pdu->common().next_header = NextHeaderCommon::Any;
1324 pdu->common().header_type = HeaderType::Beacon;
1325 pdu->common().maximum_hop_limit = 1;
1327 pdu->extended().source_position = m_local_position_vector;
1333 std::unique_ptr<GbcPdu> pdu {
new GbcPdu(
request, m_mib) };
1334 pdu->common().header_type = gbc_header_type(
request.destination);
1335 pdu->extended().sequence_number = m_local_sequence_number++;
1336 pdu->extended().source_position = m_local_position_vector;
1337 pdu->extended().destination(
request.destination);
1343 if (m_security_entity) {
1344 DownPacket sec_payload;
1345 sec_payload[OsiLayer::Network] =
SecuredPdu(pdu);
1346 sec_payload.
merge(*packet, OsiLayer::Transport, max_osi_layer());
1349 sign_request.plain_message = std::move(sec_payload);
1350 sign_request.its_aid = its_aid;
1351 sign_request.permissions = std::move(ssp);
1355 struct Visitor : boost::static_visitor<DownPacketPtr>
1357 Visitor(DownPacketPtr packet,
Pdu& pdu) : m_packet(std::move(packet)), m_pdu(pdu)
1359 assert(size(*m_packet, OsiLayer::Transport, max_osi_layer()) == 0);
1360 assert(m_pdu.basic().next_header == NextHeaderBasic::Secured);
1363 DownPacketPtr operator() (security::SecuredMessage& msg)
1365 m_pdu.secured(std::move(msg));
1366 return std::move(m_packet);
1369 DownPacketPtr operator() (security::SignConfirmError signing_error)
1375 DownPacketPtr m_packet;
1379 Visitor visitor(std::move(packet), pdu);
1380 return boost::apply_visitor(visitor, confirm);
1389 std::string reason_string;
1393 case Router::PacketDropReason::Parse_Basic_Header:
1394 reason_string =
"Parse_Basic_Header";
1396 case Router::PacketDropReason::Parse_Common_Header:
1397 reason_string =
"Parse_Common_Header";
1399 case Router::PacketDropReason::Parse_Secured_Header:
1400 reason_string =
"Parse_Secured_Header";
1402 case Router::PacketDropReason::Parse_Extended_Header:
1403 reason_string =
"Parse_Extended_Header";
1405 case Router::PacketDropReason::ITS_Protocol_Version:
1406 reason_string =
"ITS_Protocol_Version";
1408 case Router::PacketDropReason::Decap_Unsuccessful_Non_Strict:
1409 reason_string =
"Decap_Unsuccessful_Non_Strict";
1411 case Router::PacketDropReason::Decap_Unsuccessful_Strict:
1412 reason_string =
"Decap_Unsuccessful_Strict";
1414 case Router::PacketDropReason::Hop_Limit:
1415 reason_string =
"Hop_Limit";
1417 case Router::PacketDropReason::Payload_Size:
1418 reason_string =
"Payload_Size";
1420 case Router::PacketDropReason::Security_Entity_Missing:
1421 reason_string =
"Security_Entity_Missing";
1424 reason_string =
"UNKNOWN";
1428 return reason_string;
ChunckPacket is a packet consisting of several memory chunks.
ChunkPacket & merge(ChunkPacket &packet, OsiLayer from, OsiLayer to)
const MacAddress & sender() const
bool check(SequenceNumber)
virtual UpPacketPtr finish()=0
const MacAddress & mac() const
void transmit(Packet &&packet, const MacAddress &destination)
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.
NextHop greedy_forwarding(PendingPacketForwarding &&)
Determine next hop for greedy forwarding. See EN 302 636-4-1 v1.3.1 Annex E.2.
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.
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.
Hook< PacketDropReason > packet_dropped
When a packet is dropped, this Hook is invoked.
Hook< ForwardingStopReason > forwarding_stopped
When packet forwarding is stopped, this Hook is invoked.
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....
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)
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....