0
0
mirror of https://github.com/OpenVPN/openvpn3.git synced 2024-09-20 04:02:15 +02:00
openvpn3/openvpn/reliable/relrecv.hpp
James Yonan 990231b226 First successful negotiation with AS.
Still need to parse and apply PUSH_REPLY options.
2011-12-16 10:02:15 +00:00

75 lines
1.8 KiB
C++

#ifndef OPENVPN_RELIABLE_RELRECV_H
#define OPENVPN_RELIABLE_RELRECV_H
#include <openvpn/common/types.hpp>
#include <openvpn/common/exception.hpp>
#include <openvpn/common/msgwin.hpp>
#include <openvpn/reliable/relcommon.hpp>
namespace openvpn {
template <typename PACKET>
class ReliableRecvTemplate
{
public:
OPENVPN_SIMPLE_EXCEPTION(rel_next_sequenced_not_ready);
typedef reliable::id_t id_t;
class Message : public ReliableMessageBase<PACKET>
{
friend class ReliableRecvTemplate;
};
ReliableRecvTemplate() {}
ReliableRecvTemplate(const id_t span) { init(span); }
void init(const id_t span)
{
window_.init(0, span);
}
// Call with unsequenced packet off of the wire.
// Will return receive flags below.
enum {
ACK_TO_SENDER = (1<<0), // ACK for this packet should be returned to sender
IN_WINDOW = (1<<1) // Packet is in-window (otherwise, packet is dropped)
};
unsigned int receive(const PACKET& packet, const id_t id)
{
if (window_.in_window(id))
{
Message& m = window_.ref_by_id(id);
m.id_ = id;
m.packet = packet;
return ACK_TO_SENDER|IN_WINDOW;
}
else
return window_.pre_window(id) ? ACK_TO_SENDER : 0;
}
// Return true if next_sequenced() is ready to return next message
bool ready() const { return window_.head_defined(); }
// Return next message in sequence.
// Requires that ready() returns true.
Message& next_sequenced()
{
return window_.ref_head();
}
// Call after message returned by receive is ready to
// be disposed of.
void advance()
{
window_.rm_head_nocheck();
}
private:
MessageWindow<Message, id_t> window_;
};
} // namespace openvpn
#endif // OPENVPN_RELIABLE_RELRECV_H