1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-25 02:54:15 +00:00
i2pd/libi2pd/SSUData.h

139 lines
4.0 KiB
C
Raw Normal View History

/*
* Copyright (c) 2013-2020, The PurpleI2P Project
*
* This file is part of Purple i2pd project and licensed under BSD3
*
* See full license text in LICENSE file at top of project tree
*/
2014-04-22 11:39:26 -04:00
#ifndef SSU_DATA_H__
#define SSU_DATA_H__
#include <inttypes.h>
2014-07-14 22:06:58 -04:00
#include <string.h>
#include <unordered_map>
2014-06-12 11:14:22 -04:00
#include <vector>
2016-08-07 16:27:36 -04:00
#include <unordered_set>
2015-01-30 20:30:39 -05:00
#include <memory>
2014-07-20 10:38:39 -04:00
#include <boost/asio.hpp>
2014-04-22 11:39:26 -04:00
#include "I2NPProtocol.h"
2014-09-18 14:37:29 -04:00
#include "Identity.h"
#include "RouterInfo.h"
2014-04-22 11:39:26 -04:00
namespace i2p
{
namespace transport
2014-04-22 11:39:26 -04:00
{
2014-10-29 11:17:30 -04:00
const size_t SSU_MTU_V4 = 1484;
2016-06-27 15:06:15 -04:00
#ifdef MESHNET
2016-06-27 15:28:26 -04:00
const size_t SSU_MTU_V6 = 1286;
2016-06-27 15:06:15 -04:00
#else
2017-02-12 10:12:12 -05:00
const size_t SSU_MTU_V6 = 1488;
2016-06-27 15:06:15 -04:00
#endif
2014-09-14 17:54:55 -04:00
const size_t IPV4_HEADER_SIZE = 20;
2018-01-06 11:48:51 +08:00
const size_t IPV6_HEADER_SIZE = 40;
2014-09-14 17:54:55 -04:00
const size_t UDP_HEADER_SIZE = 8;
2014-10-29 11:17:30 -04:00
const size_t SSU_V4_MAX_PACKET_SIZE = SSU_MTU_V4 - IPV4_HEADER_SIZE - UDP_HEADER_SIZE; // 1456
2017-02-12 10:12:12 -05:00
const size_t SSU_V6_MAX_PACKET_SIZE = SSU_MTU_V6 - IPV6_HEADER_SIZE - UDP_HEADER_SIZE; // 1440
2014-07-20 10:38:39 -04:00
const int RESEND_INTERVAL = 3; // in seconds
const int MAX_NUM_RESENDS = 5;
2015-02-08 08:50:05 -05:00
const int DECAY_INTERVAL = 20; // in seconds
const int INCOMPLETE_MESSAGES_CLEANUP_TIMEOUT = 30; // in seconds
2015-12-17 07:58:09 +00:00
const unsigned int MAX_NUM_RECEIVED_MESSAGES = 1000; // how many msgID we store for duplicates check
2016-06-26 17:03:04 -04:00
const int MAX_OUTGOING_WINDOW_SIZE = 200; // how many unacked message we can store
2014-04-22 11:39:26 -04:00
// data flags
const uint8_t DATA_FLAG_EXTENDED_DATA_INCLUDED = 0x02;
const uint8_t DATA_FLAG_WANT_REPLY = 0x04;
const uint8_t DATA_FLAG_REQUEST_PREVIOUS_ACKS = 0x08;
const uint8_t DATA_FLAG_EXPLICIT_CONGESTION_NOTIFICATION = 0x10;
const uint8_t DATA_FLAG_ACK_BITFIELDS_INCLUDED = 0x40;
2018-01-06 11:48:51 +08:00
const uint8_t DATA_FLAG_EXPLICIT_ACKS_INCLUDED = 0x80;
2014-04-22 11:39:26 -04:00
2014-07-14 22:06:58 -04:00
struct Fragment
{
2014-07-20 10:38:39 -04:00
int fragmentNum;
size_t len;
2014-07-14 22:06:58 -04:00
bool isLast;
2014-10-29 11:17:30 -04:00
uint8_t buf[SSU_V4_MAX_PACKET_SIZE + 18]; // use biggest
2014-07-14 22:06:58 -04:00
2014-07-20 10:38:39 -04:00
Fragment () = default;
2018-01-06 11:48:51 +08:00
Fragment (int n, const uint8_t * b, int l, bool last):
fragmentNum (n), len (l), isLast (last) { memcpy (buf, b, len); };
};
2014-07-14 22:06:58 -04:00
struct FragmentCmp
{
2015-02-05 12:13:37 -05:00
bool operator() (const std::unique_ptr<Fragment>& f1, const std::unique_ptr<Fragment>& f2) const
2018-01-06 12:01:44 +08:00
{
2018-01-06 11:48:51 +08:00
return f1->fragmentNum < f2->fragmentNum;
2014-07-14 22:06:58 -04:00
};
2018-01-06 11:48:51 +08:00
};
2014-07-14 22:06:58 -04:00
struct IncompleteMessage
{
std::shared_ptr<I2NPMessage> msg;
2018-01-06 11:48:51 +08:00
int nextFragmentNum;
2015-02-08 08:50:05 -05:00
uint32_t lastFragmentInsertTime; // in seconds
2015-02-05 12:13:37 -05:00
std::set<std::unique_ptr<Fragment>, FragmentCmp> savedFragments;
2018-01-06 11:48:51 +08:00
IncompleteMessage (std::shared_ptr<I2NPMessage> m): msg (m), nextFragmentNum (0), lastFragmentInsertTime (0) {};
2018-01-06 11:48:51 +08:00
void AttachNextFragment (const uint8_t * fragment, size_t fragmentSize);
2014-07-14 22:06:58 -04:00
};
2014-07-19 18:53:02 -04:00
struct SentMessage
{
2015-01-30 20:30:39 -05:00
std::vector<std::unique_ptr<Fragment> > fragments;
2014-07-19 18:53:02 -04:00
uint32_t nextResendTime; // in seconds
int numResends;
2018-01-06 11:48:51 +08:00
};
2014-04-22 11:39:26 -04:00
class SSUSession;
class SSUData
{
public:
2018-01-06 11:48:51 +08:00
SSUData (SSUSession& session);
2014-04-22 11:39:26 -04:00
~SSUData ();
2015-02-08 08:50:05 -05:00
void Start ();
2018-01-06 11:48:51 +08:00
void Stop ();
2014-04-22 11:39:26 -04:00
void ProcessMessage (uint8_t * buf, size_t len);
2015-02-15 14:17:55 -05:00
void FlushReceivedMessage ();
void Send (std::shared_ptr<i2p::I2NPMessage> msg);
2014-04-22 11:39:26 -04:00
2018-01-06 11:48:51 +08:00
void AdjustPacketSize (std::shared_ptr<const i2p::data::RouterInfo> remoteRouter);
2014-09-18 14:37:29 -04:00
void UpdatePacketSize (const i2p::data::IdentHash& remoteIdent);
2014-06-10 11:19:31 -04:00
private:
void SendMsgAck (uint32_t msgID);
void SendFragmentAck (uint32_t msgID, int fragmentNum);
2014-07-17 15:22:32 -04:00
void ProcessAcks (uint8_t *& buf, uint8_t flag);
void ProcessFragments (uint8_t * buf);
2018-01-06 11:48:51 +08:00
void ProcessSentMessageAck (uint32_t msgID);
2014-06-10 11:19:31 -04:00
2014-07-20 10:38:39 -04:00
void ScheduleResend ();
2018-01-06 11:48:51 +08:00
void HandleResendTimer (const boost::system::error_code& ecode);
2015-02-08 08:50:05 -05:00
void ScheduleIncompleteMessagesCleanup ();
2018-01-06 11:48:51 +08:00
void HandleIncompleteMessagesCleanupTimer (const boost::system::error_code& ecode);
private:
2014-04-22 11:39:26 -04:00
SSUSession& m_Session;
std::unordered_map<uint32_t, std::unique_ptr<IncompleteMessage> > m_IncompleteMessages;
std::unordered_map<uint32_t, std::unique_ptr<SentMessage> > m_SentMessages;
2016-08-07 16:27:36 -04:00
std::unordered_set<uint32_t> m_ReceivedMessages;
boost::asio::deadline_timer m_ResendTimer, m_IncompleteMessagesCleanupTimer;
2014-10-29 11:17:30 -04:00
int m_MaxPacketSize, m_PacketSize;
2015-01-29 22:35:57 -05:00
i2p::I2NPMessagesHandler m_Handler;
2016-08-07 16:27:36 -04:00
uint32_t m_LastMessageReceivedTime; // in second
2018-01-06 11:48:51 +08:00
};
2014-04-22 11:39:26 -04:00
}
}
#endif