mirror of
https://github.com/danog/libtgvoip.git
synced 2025-01-07 21:28:23 +01:00
203 lines
6.3 KiB
C++
203 lines
6.3 KiB
C++
#pragma once
|
|
#include "../../../tools/Buffers.h"
|
|
#include "../../net/CongestionControl.h"
|
|
#include "../protocol/Extra.h"
|
|
#include "../protocol/Interface.h"
|
|
#include <memory>
|
|
#include <vector>
|
|
//#include "../net/PacketSender.h"
|
|
|
|
namespace tgvoip
|
|
{
|
|
class PacketSender;
|
|
class PacketManager;
|
|
struct PendingOutgoingPacket
|
|
{
|
|
PendingOutgoingPacket(std::shared_ptr<Buffer> &&_packet, CongestionControlPacket &&_pktInfo, int64_t _endpoint) : packet(std::move(_packet)),
|
|
pktInfo(std::move(_pktInfo)),
|
|
endpoint(_endpoint){};
|
|
//TGVOIP_MOVE_ONLY(PendingOutgoingPacket);
|
|
//TGVOIP_DISALLOW_COPY_AND_ASSIGN(PendingOutgoingPacket);
|
|
std::shared_ptr<Buffer> packet;
|
|
CongestionControlPacket pktInfo;
|
|
int64_t endpoint = 0;
|
|
};
|
|
struct ReliableOutgoingPacket
|
|
{
|
|
PendingOutgoingPacket pkt;
|
|
double retryInterval;
|
|
double timeout;
|
|
uint8_t tries;
|
|
|
|
double firstSentTime;
|
|
double lastSentTime;
|
|
};
|
|
|
|
struct RecentOutgoingPacket
|
|
{
|
|
RecentOutgoingPacket(const PendingOutgoingPacket &_pkt, double _sendTime) : pkt(_pkt.pktInfo),
|
|
size(_pkt.packet->Length()),
|
|
endpoint(_pkt.endpoint),
|
|
sendTime(_sendTime){};
|
|
CongestionControlPacket pkt;
|
|
size_t size;
|
|
|
|
int64_t endpoint;
|
|
|
|
uint16_t id; // for group calls only
|
|
|
|
double sendTime;
|
|
double ackTime = 0.0;
|
|
double rttTime = 0.0;
|
|
bool lost = false;
|
|
};
|
|
struct UnacknowledgedExtraData
|
|
{
|
|
UnacknowledgedExtraData(Wrapped<Extra> &&_data, int64_t _endpointId = 0)
|
|
: data(std::move(_data)),
|
|
endpointId(_endpointId)
|
|
{
|
|
}
|
|
TGVOIP_MOVE_ONLY(UnacknowledgedExtraData);
|
|
|
|
Wrapped<Extra> data;
|
|
HistoricBuffer<uint32_t, 16> seqs{};
|
|
int64_t endpointId = 0;
|
|
};
|
|
struct Packet : public Serializable, SingleChoice<Packet>
|
|
{
|
|
public:
|
|
bool parse(const BufferInputStream &in, const VersionInfo &ver) override;
|
|
void serialize(BufferOutputStream &out, const VersionInfo &ver) const override;
|
|
|
|
private:
|
|
bool parseLegacy(const BufferInputStream &in, const VersionInfo &ver);
|
|
bool parseLegacyLegacy(const BufferInputStream &in, unsigned char &type, uint32_t &ackId, uint32_t &pseq, uint32_t &acks, unsigned char &pflags, size_t &packetInnerLen, int peerVersion);
|
|
|
|
public:
|
|
void serializeLegacy(std::vector<std::pair<Buffer, bool>> &out, const VersionInfo &ver, const int state, const unsigned char *callID);
|
|
|
|
private:
|
|
void writePacketHeaderLegacy(BufferOutputStream &out, const VersionInfo &ver, const uint32_t seq, const uint32_t ackSeq, const uint32_t ackMask, const unsigned char type, const std::vector<Wrapped<Extra>> &extras);
|
|
void writePacketHeaderLegacyLegacy(BufferOutputStream &out, const VersionInfo &ver, const uint32_t pseq, const uint32_t ackSeq, const uint32_t ackMask, const unsigned char type, const uint32_t length, const std::vector<Wrapped<Extra>> &extras, const int state, const unsigned char *callID);
|
|
|
|
public:
|
|
void prepare(PacketManager &pm);
|
|
void prepare(PacketManager &pm, std::vector<UnacknowledgedExtraData> ¤tExtras, const int64_t &endpointId);
|
|
void prepare(PacketManager &pm, std::vector<UnacknowledgedExtraData> ¤tExtras, const int64_t &endpointId, PacketManager &legacyPm, const int peerVersion);
|
|
|
|
public:
|
|
enum Flags : uint8_t
|
|
{
|
|
Len16 = 1,
|
|
RecvTS = 2,
|
|
ExtraEC = 4,
|
|
ExtraSignaling = 8,
|
|
Reserved = 16
|
|
};
|
|
enum EFlags : uint8_t
|
|
{
|
|
Fragmented = 1,
|
|
Keyframe = 2
|
|
};
|
|
|
|
enum StreamId : uint8_t
|
|
{
|
|
Signaling = 0,
|
|
Audio = 1,
|
|
Video = 2,
|
|
Extended = 3
|
|
};
|
|
|
|
bool legacy = false;
|
|
uint32_t legacySeq = 0;
|
|
|
|
bool nopPacket = true;
|
|
|
|
uint32_t seq = 0;
|
|
uint32_t ackSeq = 0;
|
|
uint32_t ackMask = 0;
|
|
|
|
uint8_t streamId = 0;
|
|
uint8_t eFlags = 0;
|
|
|
|
uint8_t fragmentIndex = 0;
|
|
uint8_t fragmentCount = 1;
|
|
|
|
uint32_t recvTS = 0;
|
|
|
|
std::unique_ptr<Buffer> data;
|
|
|
|
Mask<Wrapped<Bytes>> extraEC;
|
|
Array<Wrapped<Extra>> extraSignaling;
|
|
|
|
// Ugly backwards compatibility hacks
|
|
std::vector<Packet> otherPackets; // parse
|
|
public:
|
|
operator bool()
|
|
{
|
|
return (data && data->Length()) || extraEC || extraSignaling || nopPacket;
|
|
}
|
|
std::string print() const override
|
|
{
|
|
std::stringstream res;
|
|
res << ((data && data->Length()) ? "Data packet" : extraEC ? "EC packet" : extraSignaling ? "Signaling packet" : nopPacket ? "NOP packet" : "Empty packet");
|
|
res << " (seq=" << seq << ", legacySeq=" << legacySeq << ", streamId=" << int(streamId) << ")";
|
|
if (extraEC)
|
|
res << "; extraEC";
|
|
if (extraSignaling)
|
|
res << "; signaling " << extraSignaling.print();
|
|
return res.str();
|
|
}
|
|
size_t getSize(const VersionInfo &ver) const override
|
|
{
|
|
if (!ver.isNew())
|
|
return 0; // Don't even try
|
|
|
|
return sizeof(seq) +
|
|
sizeof(ackSeq) +
|
|
sizeof(ackMask) +
|
|
sizeof(streamId) +
|
|
(streamId > StreamId::Extended ? sizeof(streamId) : 0) +
|
|
((data && data->Length() > 0xFF) || eFlags ? 2 : 1) + // Length
|
|
(recvTS ? sizeof(recvTS) : 0) +
|
|
(extraEC ? extraEC.getSize(ver) : 0) +
|
|
(extraSignaling ? extraSignaling.getSize(ver) : 0);
|
|
}
|
|
void clear()
|
|
{
|
|
seq = 0;
|
|
ackSeq = 0;
|
|
ackMask = 0;
|
|
streamId = 0;
|
|
data = nullptr;
|
|
recvTS = 0;
|
|
for (auto &v : extraEC) {
|
|
v.d = nullptr;
|
|
}
|
|
extraSignaling.v.clear();
|
|
}
|
|
};
|
|
|
|
using StreamId = Packet::StreamId;
|
|
struct OutgoingPacket
|
|
{
|
|
OutgoingPacket(Packet &&packet_, int64_t endpoint_ = 0)
|
|
: packet(std::move(packet_)),
|
|
endpoint(endpoint_)
|
|
{
|
|
}
|
|
TGVOIP_DISALLOW_COPY_AND_ASSIGN(OutgoingPacket);
|
|
Packet packet;
|
|
int64_t endpoint = 0;
|
|
};
|
|
|
|
/*
|
|
struct DebugLoggedPacket
|
|
{
|
|
int32_t seq;
|
|
double timestamp;
|
|
int32_t length;
|
|
};
|
|
*/
|
|
} // namespace tgvoip
|