2013-10-27 16:20:29 +01:00
|
|
|
#ifndef I2NP_PROTOCOL_H__
|
|
|
|
#define I2NP_PROTOCOL_H__
|
|
|
|
|
|
|
|
#include <inttypes.h>
|
2014-01-05 15:53:44 +01:00
|
|
|
#include <set>
|
2015-01-01 22:51:15 +01:00
|
|
|
#include <cryptopp/sha.h>
|
2013-11-11 00:23:26 +01:00
|
|
|
#include <string.h>
|
2014-02-07 17:26:00 +01:00
|
|
|
#include "I2PEndian.h"
|
2014-12-30 18:25:08 +01:00
|
|
|
#include "Identity.h"
|
2013-10-27 16:20:29 +01:00
|
|
|
#include "RouterInfo.h"
|
2014-07-29 20:31:55 +02:00
|
|
|
#include "LeaseSet.h"
|
2013-10-27 16:20:29 +01:00
|
|
|
|
|
|
|
namespace i2p
|
2015-01-01 22:51:15 +01:00
|
|
|
{
|
|
|
|
// I2NP header
|
|
|
|
const size_t I2NP_HEADER_TYPEID_OFFSET = 0;
|
|
|
|
const size_t I2NP_HEADER_MSGID_OFFSET = I2NP_HEADER_TYPEID_OFFSET + 1;
|
|
|
|
const size_t I2NP_HEADER_EXPIRATION_OFFSET = I2NP_HEADER_MSGID_OFFSET + 4;
|
|
|
|
const size_t I2NP_HEADER_SIZE_OFFSET = I2NP_HEADER_EXPIRATION_OFFSET + 8;
|
|
|
|
const size_t I2NP_HEADER_CHKS_OFFSET = I2NP_HEADER_SIZE_OFFSET + 2;
|
2015-01-02 00:53:44 +01:00
|
|
|
const size_t I2NP_HEADER_SIZE = I2NP_HEADER_CHKS_OFFSET + 1;
|
|
|
|
|
|
|
|
// Tunnel Gateway header
|
|
|
|
const size_t TUNNEL_GATEWAY_HEADER_TUNNELID_OFFSET = 0;
|
|
|
|
const size_t TUNNEL_GATEWAY_HEADER_LENGTH_OFFSET = TUNNEL_GATEWAY_HEADER_TUNNELID_OFFSET + 4;
|
|
|
|
const size_t TUNNEL_GATEWAY_HEADER_SIZE = TUNNEL_GATEWAY_HEADER_LENGTH_OFFSET + 2;
|
|
|
|
|
2013-10-27 16:20:29 +01:00
|
|
|
#pragma pack (1)
|
|
|
|
|
|
|
|
struct I2NPHeader
|
|
|
|
{
|
|
|
|
uint8_t typeID;
|
|
|
|
uint32_t msgID;
|
|
|
|
uint64_t expiration;
|
|
|
|
uint16_t size;
|
|
|
|
uint8_t chks;
|
|
|
|
};
|
|
|
|
|
2014-02-07 17:26:00 +01:00
|
|
|
struct I2NPHeaderShort
|
|
|
|
{
|
|
|
|
uint8_t typeID;
|
|
|
|
uint32_t shortExpiration;
|
|
|
|
};
|
|
|
|
|
2013-10-27 16:20:29 +01:00
|
|
|
struct I2NPDatabaseStoreMsg
|
|
|
|
{
|
|
|
|
uint8_t key[32];
|
|
|
|
uint8_t type;
|
2013-11-20 13:46:09 +01:00
|
|
|
uint32_t replyToken;
|
2013-10-27 16:20:29 +01:00
|
|
|
};
|
|
|
|
|
2014-01-17 14:12:57 +01:00
|
|
|
struct I2NPDeliveryStatusMsg
|
|
|
|
{
|
|
|
|
uint32_t msgID;
|
|
|
|
uint64_t timestamp;
|
|
|
|
};
|
2013-11-19 02:37:38 +01:00
|
|
|
|
2013-10-27 16:20:29 +01:00
|
|
|
struct I2NPBuildRequestRecordClearText
|
|
|
|
{
|
|
|
|
uint32_t receiveTunnel;
|
|
|
|
uint8_t ourIdent[32];
|
|
|
|
uint32_t nextTunnel;
|
|
|
|
uint8_t nextIdent[32];
|
|
|
|
uint8_t layerKey[32];
|
|
|
|
uint8_t ivKey[32];
|
|
|
|
uint8_t replyKey[32];
|
|
|
|
uint8_t replyIV[16];
|
|
|
|
uint8_t flag;
|
|
|
|
uint32_t requestTime;
|
|
|
|
uint32_t nextMessageID;
|
|
|
|
uint8_t filler[29];
|
|
|
|
};
|
|
|
|
|
|
|
|
struct I2NPBuildResponseRecord
|
|
|
|
{
|
|
|
|
uint8_t hash[32];
|
|
|
|
uint8_t padding[495];
|
|
|
|
uint8_t ret;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct I2NPBuildRequestRecordElGamalEncrypted
|
|
|
|
{
|
|
|
|
uint8_t toPeer[16];
|
|
|
|
uint8_t encrypted[512];
|
|
|
|
};
|
|
|
|
|
|
|
|
#pragma pack ()
|
|
|
|
|
|
|
|
enum I2NPMessageType
|
|
|
|
{
|
|
|
|
eI2NPDatabaseStore = 1,
|
|
|
|
eI2NPDatabaseLookup = 2,
|
|
|
|
eI2NPDatabaseSearchReply = 3,
|
|
|
|
eI2NPDeliveryStatus = 10,
|
|
|
|
eI2NPGarlic = 11,
|
|
|
|
eI2NPTunnelData = 18,
|
|
|
|
eI2NPTunnelGateway = 19,
|
|
|
|
eI2NPData = 20,
|
2014-04-30 20:08:57 +02:00
|
|
|
eI2NPTunnelBuild = 21,
|
|
|
|
eI2NPTunnelBuildReply = 22,
|
2013-10-27 16:20:29 +01:00
|
|
|
eI2NPVariableTunnelBuild = 23,
|
|
|
|
eI2NPVariableTunnelBuildReply = 24
|
|
|
|
};
|
|
|
|
|
2014-04-30 20:08:57 +02:00
|
|
|
const int NUM_TUNNEL_BUILD_RECORDS = 8;
|
|
|
|
|
2014-03-13 01:13:49 +01:00
|
|
|
namespace tunnel
|
|
|
|
{
|
|
|
|
class InboundTunnel;
|
2014-10-07 02:18:18 +02:00
|
|
|
class TunnelPool;
|
2014-03-13 01:13:49 +01:00
|
|
|
}
|
|
|
|
|
2014-07-04 18:35:02 +02:00
|
|
|
const size_t I2NP_MAX_MESSAGE_SIZE = 32768;
|
2014-07-30 22:52:35 +02:00
|
|
|
const size_t I2NP_MAX_SHORT_MESSAGE_SIZE = 2400;
|
2013-10-27 16:20:29 +01:00
|
|
|
struct I2NPMessage
|
|
|
|
{
|
2014-07-30 22:52:35 +02:00
|
|
|
uint8_t * buf;
|
2014-07-31 00:20:42 +02:00
|
|
|
size_t len, offset, maxLen;
|
2014-03-13 01:13:49 +01:00
|
|
|
i2p::tunnel::InboundTunnel * from;
|
2013-11-11 00:23:26 +01:00
|
|
|
|
2014-07-31 00:20:42 +02:00
|
|
|
I2NPMessage (): buf (nullptr),len (sizeof (I2NPHeader) + 2),
|
|
|
|
offset(2), maxLen (0), from (nullptr) {};
|
2014-07-30 22:52:35 +02:00
|
|
|
// reserve 2 bytes for NTCP header
|
2015-01-01 22:51:15 +01:00
|
|
|
I2NPHeader * GetHeader () { return (I2NPHeader *)GetBuffer (); }; // depricated
|
|
|
|
// header accessors
|
|
|
|
uint8_t * GetHeaderBuffer () { return GetBuffer (); };
|
|
|
|
const uint8_t * GetHeaderBuffer () const { return GetBuffer (); };
|
|
|
|
void SetTypeID (uint8_t typeID) { GetHeaderBuffer ()[I2NP_HEADER_TYPEID_OFFSET] = typeID; };
|
|
|
|
uint8_t GetTypeID () const { return GetHeaderBuffer ()[I2NP_HEADER_TYPEID_OFFSET]; };
|
|
|
|
void SetMsgID (uint32_t msgID) { htobe32buf (GetHeaderBuffer () + I2NP_HEADER_MSGID_OFFSET, msgID); };
|
|
|
|
uint32_t GetMsgID () const { return bufbe32toh (GetHeaderBuffer () + I2NP_HEADER_MSGID_OFFSET); };
|
|
|
|
void SetExpiration (uint64_t expiration) { htobe64buf (GetHeaderBuffer () + I2NP_HEADER_EXPIRATION_OFFSET, expiration); };
|
|
|
|
uint64_t GetExpiration () const { return bufbe64toh (GetHeaderBuffer () + I2NP_HEADER_EXPIRATION_OFFSET); };
|
|
|
|
uint16_t GetSize () const { return bufbe16toh (GetHeaderBuffer () + I2NP_HEADER_SIZE_OFFSET); };
|
|
|
|
void UpdateSize () { htobe16buf (GetHeaderBuffer () + I2NP_HEADER_SIZE_OFFSET, GetPayloadLength ()); };
|
|
|
|
void UpdateChks ()
|
|
|
|
{
|
|
|
|
uint8_t hash[32];
|
|
|
|
CryptoPP::SHA256().CalculateDigest(hash, GetPayload (), GetPayloadLength ());
|
|
|
|
GetHeaderBuffer ()[I2NP_HEADER_CHKS_OFFSET] = hash[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
// payload
|
2014-03-20 14:47:02 +01:00
|
|
|
uint8_t * GetPayload () { return GetBuffer () + sizeof(I2NPHeader); };
|
2013-10-27 16:20:29 +01:00
|
|
|
uint8_t * GetBuffer () { return buf + offset; };
|
2014-03-20 14:47:02 +01:00
|
|
|
const uint8_t * GetBuffer () const { return buf + offset; };
|
2015-01-01 22:51:15 +01:00
|
|
|
size_t GetLength () const { return len - offset; };
|
|
|
|
size_t GetPayloadLength () const { return GetLength () - sizeof(I2NPHeader); };
|
|
|
|
|
2014-11-26 17:04:49 +01:00
|
|
|
void Align (size_t alignment)
|
|
|
|
{
|
|
|
|
size_t rem = ((size_t)GetBuffer ()) % alignment;
|
|
|
|
if (rem)
|
2014-11-26 17:54:35 +01:00
|
|
|
{
|
2014-11-26 17:04:49 +01:00
|
|
|
offset += (alignment - rem);
|
2014-11-26 17:54:35 +01:00
|
|
|
len += (alignment - rem);
|
|
|
|
}
|
2014-11-26 17:04:49 +01:00
|
|
|
}
|
2013-11-11 00:23:26 +01:00
|
|
|
|
|
|
|
I2NPMessage& operator=(const I2NPMessage& other)
|
|
|
|
{
|
|
|
|
memcpy (buf + offset, other.buf + other.offset, other.GetLength ());
|
|
|
|
len = offset + other.GetLength ();
|
2014-03-13 01:13:49 +01:00
|
|
|
from = other.from;
|
2013-11-11 00:23:26 +01:00
|
|
|
return *this;
|
|
|
|
}
|
2014-02-07 17:26:00 +01:00
|
|
|
|
|
|
|
// for SSU only
|
|
|
|
uint8_t * GetSSUHeader () { return buf + offset + sizeof(I2NPHeader) - sizeof(I2NPHeaderShort); };
|
|
|
|
void FromSSU (uint32_t msgID) // we have received SSU message and convert it to regular
|
|
|
|
{
|
|
|
|
I2NPHeaderShort ssu = *(I2NPHeaderShort *)GetSSUHeader ();
|
|
|
|
I2NPHeader * header = GetHeader ();
|
|
|
|
header->typeID = ssu.typeID;
|
|
|
|
header->msgID = htobe32 (msgID);
|
|
|
|
header->expiration = htobe64 (be32toh (ssu.shortExpiration)*1000LL);
|
|
|
|
header->size = htobe16 (len - offset - sizeof (I2NPHeader));
|
|
|
|
header->chks = 0;
|
|
|
|
}
|
2014-02-10 00:28:34 +01:00
|
|
|
uint32_t ToSSU () // return msgID
|
|
|
|
{
|
|
|
|
I2NPHeader header = *GetHeader ();
|
|
|
|
I2NPHeaderShort * ssu = (I2NPHeaderShort *)GetSSUHeader ();
|
|
|
|
ssu->typeID = header.typeID;
|
|
|
|
ssu->shortExpiration = htobe32 (be64toh (header.expiration)/1000LL);
|
|
|
|
len = offset + sizeof (I2NPHeaderShort) + be16toh (header.size);
|
|
|
|
return be32toh (header.msgID);
|
|
|
|
}
|
2013-10-27 16:20:29 +01:00
|
|
|
};
|
2014-07-30 22:52:35 +02:00
|
|
|
|
|
|
|
template<int sz>
|
|
|
|
struct I2NPMessageBuffer: public I2NPMessage
|
|
|
|
{
|
2014-07-31 00:20:42 +02:00
|
|
|
I2NPMessageBuffer () { buf = m_Buffer; maxLen = sz; };
|
2014-07-30 22:52:35 +02:00
|
|
|
uint8_t m_Buffer[sz];
|
|
|
|
};
|
|
|
|
|
2013-10-27 16:20:29 +01:00
|
|
|
I2NPMessage * NewI2NPMessage ();
|
2014-07-30 22:52:35 +02:00
|
|
|
I2NPMessage * NewI2NPShortMessage ();
|
2014-07-31 00:20:42 +02:00
|
|
|
I2NPMessage * NewI2NPMessage (size_t len);
|
2013-10-27 16:20:29 +01:00
|
|
|
void DeleteI2NPMessage (I2NPMessage * msg);
|
|
|
|
void FillI2NPMessageHeader (I2NPMessage * msg, I2NPMessageType msgType, uint32_t replyMsgID = 0);
|
2014-03-20 17:48:03 +01:00
|
|
|
void RenewI2NPMessageHeader (I2NPMessage * msg);
|
2013-10-27 16:20:29 +01:00
|
|
|
I2NPMessage * CreateI2NPMessage (I2NPMessageType msgType, const uint8_t * buf, int len, uint32_t replyMsgID = 0);
|
2014-09-29 22:41:41 +02:00
|
|
|
I2NPMessage * CreateI2NPMessage (const uint8_t * buf, int len, i2p::tunnel::InboundTunnel * from = nullptr);
|
2013-10-27 16:20:29 +01:00
|
|
|
|
2014-01-09 04:47:22 +01:00
|
|
|
I2NPMessage * CreateDeliveryStatusMsg (uint32_t msgID);
|
2013-11-19 02:37:38 +01:00
|
|
|
I2NPMessage * CreateDatabaseLookupMsg (const uint8_t * key, const uint8_t * from,
|
2014-01-05 15:53:44 +01:00
|
|
|
uint32_t replyTunnelID, bool exploratory = false,
|
2014-10-07 02:18:18 +02:00
|
|
|
std::set<i2p::data::IdentHash> * excludedPeers = nullptr, bool encryption = false,
|
|
|
|
i2p::tunnel::TunnelPool * pool = nullptr);
|
2014-12-30 18:25:08 +01:00
|
|
|
I2NPMessage * CreateLeaseSetDatabaseLookupMsg (const i2p::data::IdentHash& dest,
|
|
|
|
const std::set<i2p::data::IdentHash>& excludedFloodfills,
|
|
|
|
const i2p::tunnel::InboundTunnel * replyTunnel, const uint8_t * replyKey, const uint8_t * replyTag);
|
2014-07-25 04:01:07 +02:00
|
|
|
I2NPMessage * CreateDatabaseSearchReply (const i2p::data::IdentHash& ident, const i2p::data::RouterInfo * floodfill);
|
2014-01-06 04:21:59 +01:00
|
|
|
|
2014-07-24 21:59:00 +02:00
|
|
|
I2NPMessage * CreateDatabaseStoreMsg (const i2p::data::RouterInfo * router = nullptr);
|
2014-08-20 17:12:53 +02:00
|
|
|
I2NPMessage * CreateDatabaseStoreMsg (const i2p::data::LeaseSet * leaseSet, uint32_t replyToken = 0);
|
2014-07-29 20:31:55 +02:00
|
|
|
|
2013-10-27 16:20:29 +01:00
|
|
|
I2NPBuildRequestRecordClearText CreateBuildRequestRecord (
|
|
|
|
const uint8_t * ourIdent, uint32_t receiveTunnelID,
|
|
|
|
const uint8_t * nextIdent, uint32_t nextTunnelID,
|
|
|
|
const uint8_t * layerKey,const uint8_t * ivKey,
|
|
|
|
const uint8_t * replyKey, const uint8_t * replyIV, uint32_t nextMessageID,
|
|
|
|
bool isGateway, bool isEndpoint);
|
|
|
|
void EncryptBuildRequestRecord (const i2p::data::RouterInfo& router,
|
|
|
|
const I2NPBuildRequestRecordClearText& clearText,
|
|
|
|
I2NPBuildRequestRecordElGamalEncrypted& record);
|
2014-04-30 20:08:57 +02:00
|
|
|
|
|
|
|
bool HandleBuildRequestRecords (int num, I2NPBuildRequestRecordElGamalEncrypted * records, I2NPBuildRequestRecordClearText& clearText);
|
2013-10-27 16:20:29 +01:00
|
|
|
void HandleVariableTunnelBuildMsg (uint32_t replyMsgID, uint8_t * buf, size_t len);
|
|
|
|
void HandleVariableTunnelBuildReplyMsg (uint32_t replyMsgID, uint8_t * buf, size_t len);
|
2014-04-30 20:08:57 +02:00
|
|
|
void HandleTunnelBuildMsg (uint8_t * buf, size_t len);
|
|
|
|
|
2013-10-27 16:20:29 +01:00
|
|
|
I2NPMessage * CreateTunnelDataMsg (const uint8_t * buf);
|
|
|
|
I2NPMessage * CreateTunnelDataMsg (uint32_t tunnelID, const uint8_t * payload);
|
|
|
|
|
2013-11-11 00:23:26 +01:00
|
|
|
void HandleTunnelGatewayMsg (I2NPMessage * msg);
|
|
|
|
I2NPMessage * CreateTunnelGatewayMsg (uint32_t tunnelID, const uint8_t * buf, size_t len);
|
|
|
|
I2NPMessage * CreateTunnelGatewayMsg (uint32_t tunnelID, I2NPMessageType msgType,
|
|
|
|
const uint8_t * buf, size_t len, uint32_t replyMsgID = 0);
|
|
|
|
I2NPMessage * CreateTunnelGatewayMsg (uint32_t tunnelID, I2NPMessage * msg);
|
2013-12-14 02:07:35 +01:00
|
|
|
|
2014-10-12 03:27:55 +02:00
|
|
|
size_t GetI2NPMessageLength (const uint8_t * msg);
|
2014-03-13 01:13:49 +01:00
|
|
|
void HandleI2NPMessage (uint8_t * msg, size_t len);
|
|
|
|
void HandleI2NPMessage (I2NPMessage * msg);
|
2013-10-27 16:20:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|