1#ifndef PENDING_PACKET_HPP_ZHSCP1UI
2#define PENDING_PACKET_HPP_ZHSCP1UI
4#include <vanetza/common/clock.hpp>
5#include <vanetza/geonet/lifetime.hpp>
6#include <vanetza/geonet/packet.hpp>
7#include <vanetza/units/time.hpp>
21template<
typename PDU,
typename... Args>
25 using Packet = std::tuple<std::unique_ptr<PDU>, std::unique_ptr<DownPacket>>;
26 using Function = std::function<void(Packet&&, Args&&...)>;
31 m_packet(std::move(packet)), m_function(fn) {}
33 template<
typename... OtherArgs>
35 m_packet(std::move(other).packet())
37 auto other_fn = other.action();
38 m_function = [fn, other_fn](Packet&& packet, Args&&... args) {
43 template<
typename... OtherArgs,
typename... T>
45 m_packet(std::move(other).packet())
47 typename PendingPacket<PDU, OtherArgs...>::Function other_action = other.action();
48 std::function<void(Packet&&)> bound = std::bind(other_action, std::placeholders::_1, std::forward<T>(ts)...);
49 m_function = [bound](Packet&& packet, Args&&... args) {
50 bound(std::move(packet), std::forward<Args>(args)...);
54 void process(Args&&... args)
56 if (std::get<0>(m_packet) && std::get<1>(m_packet)) {
57 m_function(std::move(m_packet), std::forward<Args>(args)...);
61 std::size_t length()
const
63 const PDU* pdu = std::get<0>(m_packet).get();
64 const DownPacket* payload = std::get<1>(m_packet).get();
65 return (pdu ? get_length(*pdu) : 0) + (payload ? payload->
size(OsiLayer::Transport, max_osi_layer()) : 0);
68 Clock::duration reduce_lifetime(Clock::duration queuing_time)
70 Clock::duration remaining = Clock::duration::zero();
71 PDU* pdu_ptr = std::get<0>(m_packet).get();
73 using vanetza::units::clock_cast;
74 Clock::duration packet_lifetime = clock_cast(pdu_ptr->basic().lifetime.decode());
75 if (queuing_time <= Clock::duration::zero()) {
76 remaining = packet_lifetime;
77 }
else if (queuing_time < packet_lifetime) {
78 remaining = packet_lifetime - queuing_time;
79 pdu_ptr->basic().lifetime.encode(clock_cast(remaining));
81 pdu_ptr->basic().lifetime = Lifetime::zero();
87 const PDU& pdu()
const
89 const PDU* ptr = std::get<0>(m_packet).get();
96 const DownPacket* ptr = std::get<1>(m_packet).get();
101 Function action()
const {
return m_function; }
103 Packet packet() && {
return std::move(m_packet); }
107 const PDU* pdu_ptr = std::get<0>(m_packet).get();
108 std::unique_ptr<PDU> pdu_dup { pdu_ptr ?
new PDU(*pdu_ptr) :
nullptr };
109 std::unique_ptr<DownPacket> payload_dup;
110 if (!pdu_ptr || pdu_ptr->secured()) {
112 }
else if (
const DownPacket* payload_ptr = std::get<1>(m_packet).get()) {
113 payload_dup = vanetza::duplicate(*payload_ptr);
115 return PendingPacket(std::make_tuple(std::move(pdu_dup), std::move(payload_dup)), m_function);
ChunckPacket is a packet consisting of several memory chunks.