/* * Copyright (c) 2013-2026, 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 */ #include #include "Log.h" #include "util.h" #include "ClientContext.h" #include "I2PTunnel.h" // for GetLoopbackAddressFor #include "UDPTunnel.h" namespace i2p { namespace client { constexpr std::string_view UDP_SESSION_SEQN { "seqn" }; constexpr std::string_view UDP_SESSION_ACKED { "acked" }; constexpr std::string_view UDP_SESSION_FLAGS { "flags" }; constexpr uint8_t UDP_SESSION_FLAG_RESET_PATH = 0x01; constexpr uint8_t UDP_SESSION_FLAG_ACK_REQUESTED = 0x02; constexpr uint8_t UDP_SESSION_FLAG_RESET_SEQN = 0x04; void I2PUDPServerTunnel::HandleRecvFromI2P(const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len, const i2p::util::Mapping * options) { if (!m_LastSession || m_LastSession->Identity.GetLL()[0] != from.GetIdentHash ().GetLL()[0] || (fromPort && fromPort != m_LastSession->RemotePort)) m_LastSession = ObtainUDPSession(from, toPort, fromPort); boost::system::error_code ec; if (len > 0) m_LastSession->IPSocket.send_to(boost::asio::buffer(buf, len), m_RemoteEndpoint, 0, ec); if (!ec) m_LastSession->LastActivity = i2p::util::GetMillisecondsSinceEpoch(); else LogPrint (eLogInfo, "UDP Server: Send exception: ", ec.message (), " to ", m_RemoteEndpoint); if (options) { uint32_t seqn = 0; if (options->Get (UDP_SESSION_SEQN, seqn) && seqn > m_LastSession->m_LastReceivedPacketNum) m_LastSession->m_LastReceivedPacketNum = seqn; uint8_t flags = 0; if (options->Get (UDP_SESSION_FLAGS, flags)) { if (flags & UDP_SESSION_FLAG_RESET_PATH) m_LastSession->GetDatagramSession ()->DropSharedRoutingPath (); if ((flags & UDP_SESSION_FLAG_RESET_SEQN) && seqn) m_LastSession->m_LastReceivedPacketNum = seqn; if (flags & UDP_SESSION_FLAG_ACK_REQUESTED) { i2p::util::Mapping replyOptions; replyOptions.Put (UDP_SESSION_ACKED, m_LastSession->m_LastReceivedPacketNum); m_LastSession->m_Destination->SendDatagram(m_LastSession->GetDatagramSession (), nullptr, 0, m_LastSession->LocalPort, m_LastSession->RemotePort, &replyOptions); // Ack only, no payload m_LastSession->m_LastRepliableDatagramTime = i2p::util::GetMillisecondsSinceEpoch (); } } if (options->Get (UDP_SESSION_ACKED, seqn)) m_LastSession->Acked (seqn); } } void I2PUDPServerTunnel::HandleRecvFromI2PRaw (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len) { if (m_LastSession && (fromPort != m_LastSession->RemotePort || toPort != m_LastSession->LocalPort)) { std::lock_guard lock(m_SessionsMutex); auto it = m_Sessions.find (GetSessionIndex (fromPort, toPort)); if (it != m_Sessions.end ()) m_LastSession = it->second; else m_LastSession = nullptr; } if (m_LastSession) { boost::system::error_code ec; m_LastSession->IPSocket.send_to(boost::asio::buffer(buf, len), m_RemoteEndpoint, 0, ec); if (!ec) m_LastSession->LastActivity = i2p::util::GetMillisecondsSinceEpoch(); else LogPrint (eLogInfo, "UDP Server: Send exception: ", ec.message (), " to ", m_RemoteEndpoint); } } void I2PUDPServerTunnel::ExpireStale(const uint64_t delta) { std::lock_guard lock(m_SessionsMutex); uint64_t now = i2p::util::GetMillisecondsSinceEpoch(); auto itr = m_Sessions.begin(); while(itr != m_Sessions.end()) { if(now - itr->second->LastActivity >= delta ) itr = m_Sessions.erase(itr); else itr++; } } void I2PUDPClientTunnel::ExpireStale(const uint64_t delta) { std::lock_guard lock(m_SessionsMutex); uint64_t now = i2p::util::GetMillisecondsSinceEpoch(); std::vector removePorts; for (const auto & s : m_Sessions) { if (now - s.second->second >= delta) removePorts.push_back(s.first); } for(auto port : removePorts) { m_Sessions.erase(port); } } UDPSessionPtr I2PUDPServerTunnel::ObtainUDPSession(const i2p::data::IdentityEx& from, uint16_t localPort, uint16_t remotePort) { auto ih = from.GetIdentHash(); auto idx = GetSessionIndex (remotePort, localPort); { std::lock_guard lock(m_SessionsMutex); auto it = m_Sessions.find (idx); if (it != m_Sessions.end ()) { if (it->second->Identity.GetLL()[0] == ih.GetLL()[0]) { LogPrint(eLogDebug, "UDPServer: Found session ", it->second->IPSocket.local_endpoint(), " ", ih.ToBase32()); return it->second; } else { LogPrint(eLogWarning, "UDPServer: Session with from ", remotePort, " and to ", localPort, " ports already exists. But from different address. Removed"); m_Sessions.erase (it); } } } boost::asio::ip::address addr; /** create new udp session */ if(m_IsUniqueLocal && m_LocalAddress.is_loopback()) { auto ident = from.GetIdentHash(); addr = GetLoopbackAddressFor(ident); } else addr = m_LocalAddress; auto s = std::make_shared(boost::asio::ip::udp::endpoint(addr, 0), m_LocalDest, m_RemoteEndpoint, ih, localPort, remotePort); std::lock_guard lock(m_SessionsMutex); m_Sessions.emplace (idx, s); return s; } void UDPConnection::Stop () { m_AckTimer.cancel (); } void UDPConnection::Acked (uint32_t seqn) { m_IsFirstPacket = false; // first packet confirmed if (m_AckTimerSeqn) { if (seqn >= m_AckTimerSeqn) { m_AckTimerSeqn = 0; m_AckTimer.cancel (); } } else if (!m_UnackedDatagrams.empty ()) seqn = m_UnackedDatagrams.back ().first; // if we receive ack after path change, clear window and send new datagrams if (m_UnackedDatagrams.empty () || seqn < m_UnackedDatagrams.front ().first) return; bool acknowledged = false; auto it = m_UnackedDatagrams.begin (); while (it != m_UnackedDatagrams.end ()) { if (it->first > seqn) break; if (it->first == seqn && m_IsSendingAllowed) // ignore first ack after path change { auto rtt = i2p::util::GetMillisecondsSinceEpoch () - it->second; m_RTT = m_RTT ? (m_RTT + rtt)/2 : rtt; acknowledged = true; } it++; } m_UnackedDatagrams.erase (m_UnackedDatagrams.begin (), it); m_IsSendingAllowed = true; // if we recieve ack after path change, now can send new datagrams if (acknowledged && !m_UnackedDatagrams.empty ()) { m_AckTimer.cancel (); m_AckTimerSeqn = 0; ScheduleAckTimer (m_UnackedDatagrams.back ().first); } } void UDPConnection::ScheduleAckTimer (uint32_t seqn) { if (!m_AckTimerSeqn) { m_AckTimerSeqn = seqn; m_AckTimer.expires_after (std::chrono::milliseconds (m_RTT ? 2*m_RTT : I2P_UDP_MAX_UNACKED_DATAGRAM_TIME)); m_AckTimer.async_wait ([this](const boost::system::error_code& ecode) { if (ecode != boost::asio::error::operation_aborted) { LogPrint (eLogInfo, "UDP Connection: Packet ", m_AckTimerSeqn, " was not acked"); // DeleteExpiredUnackedDatagrams (); m_IsSendingAllowed = false; // stop sending datagrams m_AckTimerSeqn = 0; m_RTT = 0; if (!m_UnackedDatagrams.empty ()) ScheduleAckTimer (0); // try again if failed // send empty packet with reset path flag i2p::util::Mapping options; options.Put (UDP_SESSION_FLAGS, UDP_SESSION_FLAG_RESET_PATH | UDP_SESSION_FLAG_ACK_REQUESTED); auto session = GetDatagramSession (); session->DropSharedRoutingPath (); m_Destination->SendDatagram (session, nullptr, 0, 0, 0, &options); } }); } } void UDPConnection::DeleteExpiredUnackedDatagrams () { if (m_UnackedDatagrams.empty ()) return; auto expired = i2p::util::GetMillisecondsSinceEpoch () - (m_RTT ? 2*m_RTT : I2P_UDP_MAX_UNACKED_DATAGRAM_TIME); auto it = m_UnackedDatagrams.begin (); while (it != m_UnackedDatagrams.end ()) { if (it->second < expired) break; it++; } m_UnackedDatagrams.erase (m_UnackedDatagrams.begin (), it); } std::shared_ptr UDPConnection::GetDatagramSession () { auto session = m_LastDatagramSession.lock (); if (!session && isIdentity) { session = m_Destination->GetSession (Identity); m_LastDatagramSession = session; } return session; } UDPSession::UDPSession(boost::asio::ip::udp::endpoint localEndpoint, const std::shared_ptr & localDestination, const boost::asio::ip::udp::endpoint& endpoint, const i2p::data::IdentHash& to, uint16_t ourPort, uint16_t theirPort) : UDPConnection (localDestination->GetService(), localDestination->GetDatagramDestination()), IPSocket(localDestination->GetService(), localEndpoint), SendEndpoint(endpoint), LastActivity(i2p::util::GetMillisecondsSinceEpoch()), LocalPort(ourPort), RemotePort(theirPort) { SetIdentity (to); Start (); IPSocket.set_option (boost::asio::socket_base::receive_buffer_size (I2P_UDP_MAX_MTU )); IPSocket.non_blocking (true); Receive(); } void UDPSession::Receive() { LogPrint(eLogDebug, "UDPSession: Receive"); IPSocket.async_receive_from(boost::asio::buffer(m_Buffer, I2P_UDP_MAX_MTU), FromEndpoint, std::bind(&UDPSession::HandleReceived, this, std::placeholders::_1, std::placeholders::_2)); } void UDPSession::HandleReceived(const boost::system::error_code & ecode, std::size_t len) { if(!ecode) { if (!m_UnackedDatagrams.empty () && m_NextSendPacketNum > m_UnackedDatagrams.front ().first + I2P_UDP_MAX_NUM_UNACKED_DATAGRAMS) { // window is full, drop packet Receive (); return; } LogPrint(eLogDebug, "UDPSession: Forward ", len, "B from ", FromEndpoint); auto ts = i2p::util::GetMillisecondsSinceEpoch(); auto session = GetDatagramSession (); uint64_t repliableDatagramInterval = I2P_UDP_REPLIABLE_DATAGRAM_INTERVAL; if (m_RTT && m_RTT >= I2P_UDP_REPLIABLE_DATAGRAM_INTERVAL && m_RTT < I2P_UDP_REPLIABLE_DATAGRAM_INTERVAL*10) repliableDatagramInterval = m_RTT/10; // 10 - 100 ms if (ts > m_LastRepliableDatagramTime + repliableDatagramInterval) { if (session->GetVersion () == i2p::datagram::eDatagramV3) { uint8_t flags = 0; if (!m_RTT || !m_AckTimerSeqn || (!m_UnackedDatagrams.empty () && ts > m_UnackedDatagrams.back ().second + repliableDatagramInterval)) // last ack request { flags |= UDP_SESSION_FLAG_ACK_REQUESTED; m_UnackedDatagrams.push_back ({ m_NextSendPacketNum, ts }); ScheduleAckTimer (m_NextSendPacketNum); } if (m_IsFirstPacket) flags |= UDP_SESSION_FLAG_RESET_SEQN; i2p::util::Mapping options; options.Put (UDP_SESSION_SEQN, m_NextSendPacketNum); if (m_LastReceivedPacketNum > 0) options.Put (UDP_SESSION_ACKED, m_LastReceivedPacketNum); if (flags) options.Put (UDP_SESSION_FLAGS, flags); m_Destination->SendDatagram(session, m_Buffer, len, LocalPort, RemotePort, &options); ScheduleAckTimer (m_NextSendPacketNum); } else m_Destination->SendDatagram(session, m_Buffer, len, LocalPort, RemotePort); m_LastRepliableDatagramTime = ts; } else m_Destination->SendRawDatagram(session, m_Buffer, len, LocalPort, RemotePort); size_t numPackets = 0; while (numPackets < i2p::datagram::DATAGRAM_SEND_QUEUE_MAX_SIZE) { boost::system::error_code ec; size_t moreBytes = IPSocket.available(ec); if (ec || !moreBytes) break; len = IPSocket.receive_from (boost::asio::buffer (m_Buffer, I2P_UDP_MAX_MTU), FromEndpoint, 0, ec); m_Destination->SendRawDatagram (session, m_Buffer, len, LocalPort, RemotePort); numPackets++; } if (numPackets > 0) LogPrint(eLogDebug, "UDPSession: Forward more ", numPackets, "packets B from ", FromEndpoint); m_NextSendPacketNum += numPackets + 1; m_Destination->FlushSendQueue (session); LastActivity = ts; Receive(); } else LogPrint(eLogError, "UDPSession: ", ecode.message()); } I2PUDPServerTunnel::I2PUDPServerTunnel (const std::string & name, std::shared_ptr localDestination, const boost::asio::ip::address& localAddress, const boost::asio::ip::udp::endpoint& forwardTo, uint16_t inPort, bool gzip) : m_IsUniqueLocal (true), m_Name (name), m_LocalAddress (localAddress), m_RemoteEndpoint (forwardTo), m_LocalDest (localDestination), m_inPort(inPort), m_Gzip (gzip) { } I2PUDPServerTunnel::~I2PUDPServerTunnel () { Stop (); } void I2PUDPServerTunnel::Start () { m_LocalDest->Start (); auto dgram = m_LocalDest->CreateDatagramDestination (m_Gzip); dgram->SetReceiver ( std::bind (&I2PUDPServerTunnel::HandleRecvFromI2P, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4, std::placeholders::_5, std::placeholders::_6), m_inPort ); dgram->SetRawReceiver ( std::bind (&I2PUDPServerTunnel::HandleRecvFromI2PRaw, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4), m_inPort ); } void I2PUDPServerTunnel::Stop () { auto dgram = m_LocalDest->GetDatagramDestination (); if (dgram) { dgram->ResetReceiver (m_inPort); dgram->ResetRawReceiver (m_inPort); } m_Sessions.clear (); } std::vector > I2PUDPServerTunnel::GetSessions () { std::vector > sessions; std::lock_guard lock (m_SessionsMutex); for (const auto &it: m_Sessions) { auto s = it.second; if (!s->m_Destination) continue; auto info = s->m_Destination->GetInfoForRemote (s->Identity); if (!info) continue; auto sinfo = std::make_shared (); sinfo->Name = m_Name; sinfo->LocalIdent = std::make_shared (m_LocalDest->GetIdentHash ().data ()); sinfo->RemoteIdent = std::make_shared (s->Identity.data ()); sinfo->CurrentIBGW = info->IBGW; sinfo->CurrentOBEP = info->OBEP; sessions.push_back (sinfo); } return sessions; } I2PUDPClientTunnel::I2PUDPClientTunnel (const std::string & name, const std::string &remoteDest, const boost::asio::ip::udp::endpoint& localEndpoint, std::shared_ptr localDestination, uint16_t remotePort, bool gzip, i2p::datagram::DatagramVersion datagramVersion) : UDPConnection (localDestination->GetService (), localDestination->GetDatagramDestination ()), m_Name (name), m_RemoteDest (remoteDest), m_LocalDest (localDestination), m_LocalEndpoint (localEndpoint), m_ResolveThread (nullptr), m_LocalSocket (nullptr), RemotePort (remotePort), m_LastPort (0), m_cancel_resolve (false), m_Gzip (gzip), m_DatagramVersion (datagramVersion) { } I2PUDPClientTunnel::~I2PUDPClientTunnel () { Stop (); } void I2PUDPClientTunnel::Start () { UDPConnection::Start (); // Reset flag in case of tunnel reload if (m_cancel_resolve) m_cancel_resolve = false; m_LocalSocket.reset (new boost::asio::ip::udp::socket (m_LocalDest->GetService (), m_LocalEndpoint)); m_LocalSocket->set_option (boost::asio::socket_base::receive_buffer_size (I2P_UDP_MAX_MTU)); m_LocalSocket->set_option (boost::asio::socket_base::reuse_address (true)); m_LocalSocket->non_blocking (true); auto dgram = m_LocalDest->CreateDatagramDestination (m_Gzip, m_DatagramVersion); m_Destination = dgram; dgram->SetReceiver (std::bind (&I2PUDPClientTunnel::HandleRecvFromI2P, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4, std::placeholders::_5, std::placeholders::_6), RemotePort ); dgram->SetRawReceiver (std::bind (&I2PUDPClientTunnel::HandleRecvFromI2PRaw, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4), RemotePort ); m_LocalDest->Start (); if (m_ResolveThread == nullptr) m_ResolveThread = new std::thread (std::bind (&I2PUDPClientTunnel::TryResolving, this)); RecvFromLocal (); } void I2PUDPClientTunnel::Stop () { auto dgram = m_LocalDest->GetDatagramDestination (); if (dgram) { dgram->ResetReceiver (RemotePort); dgram->ResetRawReceiver (RemotePort); } m_cancel_resolve = true; m_Sessions.clear(); if(m_LocalSocket && m_LocalSocket->is_open ()) m_LocalSocket->close (); if(m_ResolveThread) { m_ResolveThread->join (); delete m_ResolveThread; m_ResolveThread = nullptr; } UDPConnection::Stop (); } void I2PUDPClientTunnel::RecvFromLocal () { m_LocalSocket->async_receive_from (boost::asio::buffer (m_RecvBuff, I2P_UDP_MAX_MTU), m_RecvEndpoint, std::bind (&I2PUDPClientTunnel::HandleRecvFromLocal, this, std::placeholders::_1, std::placeholders::_2)); } void I2PUDPClientTunnel::HandleRecvFromLocal (const boost::system::error_code & ec, std::size_t transferred) { if (m_cancel_resolve) { LogPrint (eLogDebug, "UDP Client: Ignoring incoming data: stopping"); return; } if (ec) { LogPrint (eLogError, "UDP Client: Reading from socket error: ", ec.message (), ". Restarting listener..."); RecvFromLocal (); // Restart listener and continue work return; } if (!isIdentity) { LogPrint (eLogWarning, "UDP Client: Remote endpoint not resolved yet"); RecvFromLocal (); return; // drop, remote not resolved } if ((!m_UnackedDatagrams.empty () && m_NextSendPacketNum > m_UnackedDatagrams.front ().first + I2P_UDP_MAX_NUM_UNACKED_DATAGRAMS) || !m_IsSendingAllowed) { // window is full, drop packet RecvFromLocal (); return; } auto remotePort = m_RecvEndpoint.port (); if (!m_LastPort || m_LastPort != remotePort) { auto itr = m_Sessions.find (remotePort); if (itr != m_Sessions.end ()) m_LastSession = itr->second; else { m_LastSession = std::make_shared (boost::asio::ip::udp::endpoint (m_RecvEndpoint), 0); m_Sessions.emplace (remotePort, m_LastSession); } m_LastPort = remotePort; } // send off to remote i2p destination auto ts = i2p::util::GetMillisecondsSinceEpoch (); LogPrint (eLogDebug, "UDP Client: Send ", transferred, " to ", Identity.ToBase32 (), ":", RemotePort); auto session = GetDatagramSession (); uint64_t repliableDatagramInterval = I2P_UDP_REPLIABLE_DATAGRAM_INTERVAL; if (m_RTT && m_RTT >= I2P_UDP_REPLIABLE_DATAGRAM_INTERVAL && m_RTT < I2P_UDP_REPLIABLE_DATAGRAM_INTERVAL*10) repliableDatagramInterval = m_RTT/10; // 10 - 100 ms if (ts > m_LastRepliableDatagramTime + repliableDatagramInterval) { if (m_DatagramVersion == i2p::datagram::eDatagramV3) { uint8_t flags = 0; if (!m_RTT || !m_AckTimerSeqn || (!m_UnackedDatagrams.empty () && ts > m_UnackedDatagrams.back ().second + repliableDatagramInterval)) // last ack request { flags |= UDP_SESSION_FLAG_ACK_REQUESTED; m_UnackedDatagrams.push_back ({ m_NextSendPacketNum, ts }); ScheduleAckTimer (m_NextSendPacketNum); } if (m_IsFirstPacket) flags |= UDP_SESSION_FLAG_RESET_SEQN; i2p::util::Mapping options; options.Put (UDP_SESSION_SEQN, m_NextSendPacketNum); if (m_LastReceivedPacketNum > 0) options.Put (UDP_SESSION_ACKED, m_LastReceivedPacketNum); if (flags) options.Put (UDP_SESSION_FLAGS, flags); m_Destination->SendDatagram (session, m_RecvBuff, transferred, remotePort, RemotePort, &options); if (m_IsFirstPacket) m_IsSendingAllowed = false; // send only one packet at the start and wait ack } else m_Destination->SendDatagram (session, m_RecvBuff, transferred, remotePort, RemotePort); m_LastRepliableDatagramTime = ts; } else m_Destination->SendRawDatagram (session, m_RecvBuff, transferred, remotePort, RemotePort); size_t numPackets = 0; while (numPackets < i2p::datagram::DATAGRAM_SEND_QUEUE_MAX_SIZE) { boost::system::error_code ec; size_t moreBytes = m_LocalSocket->available (ec); if (ec || !moreBytes) break; transferred = m_LocalSocket->receive_from (boost::asio::buffer (m_RecvBuff, I2P_UDP_MAX_MTU), m_RecvEndpoint, 0, ec); remotePort = m_RecvEndpoint.port (); // TODO: check remotePort m_Destination->SendRawDatagram (session, m_RecvBuff, transferred, remotePort, RemotePort); numPackets++; } if (numPackets) LogPrint (eLogDebug, "UDP Client: Sent ", numPackets, " more packets to ", Identity.ToBase32 ()); m_NextSendPacketNum += numPackets + 1; m_Destination->FlushSendQueue (session); // mark convo as active if (m_LastSession) m_LastSession->second = ts; RecvFromLocal (); } std::vector > I2PUDPClientTunnel::GetSessions () { // TODO: implement std::vector > infos; return infos; } void I2PUDPClientTunnel::TryResolving () { i2p::util::SetThreadName ("UDP Resolver"); LogPrint (eLogInfo, "UDP Tunnel: Trying to resolve ", m_RemoteDest); std::shared_ptr remoteAddr; while (!(remoteAddr = context.GetAddressBook().GetAddress(m_RemoteDest)) && !m_cancel_resolve) { LogPrint (eLogWarning, "UDP Tunnel: Failed to lookup ", m_RemoteDest); std::this_thread::sleep_for (std::chrono::seconds (1)); } if (m_cancel_resolve) { LogPrint(eLogError, "UDP Tunnel: Lookup of ", m_RemoteDest, " was cancelled"); return; } if (!remoteAddr) { LogPrint (eLogError, "UDP Tunnel: ", m_RemoteDest, " not found"); return; } if (!remoteAddr->IsIdentHash ()) // TODO: handle B33 { LogPrint (eLogError, "UDP Tunnel: ", m_RemoteDest, " resolved to invalid address type"); return; } LogPrint(eLogInfo, "UDP Tunnel: Resolved ", m_RemoteDest, " to ", remoteAddr->identHash.ToBase32 ()); SetIdentity (remoteAddr->identHash); } void I2PUDPClientTunnel::HandleRecvFromI2P (const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len, const i2p::util::Mapping * options) { if (isIdentity && from.GetIdentHash() == Identity) { if (options) { uint32_t seqn = 0; if (options->Get (UDP_SESSION_SEQN, seqn) && seqn > m_LastReceivedPacketNum) m_LastReceivedPacketNum = seqn; uint8_t flags = 0; if (options->Get (UDP_SESSION_FLAGS, flags)) { if ((flags & UDP_SESSION_FLAG_RESET_SEQN) && seqn) m_LastReceivedPacketNum = seqn; if (flags & UDP_SESSION_FLAG_ACK_REQUESTED) { i2p::util::Mapping replyOptions; replyOptions.Put (UDP_SESSION_ACKED, m_LastReceivedPacketNum); m_Destination->SendDatagram (GetDatagramSession (), nullptr, 0, m_LastPort, RemotePort, &replyOptions); // Ack only, no payload m_LastRepliableDatagramTime = i2p::util::GetMillisecondsSinceEpoch (); } } if (options->Get (UDP_SESSION_ACKED, seqn)) Acked (seqn); } if (len > 0) HandleRecvFromI2PRaw (fromPort, toPort, buf, len); } else LogPrint(eLogWarning, "UDP Client: Unwarranted traffic from ", from.GetIdentHash().ToBase32 ()); } void I2PUDPClientTunnel::HandleRecvFromI2PRaw (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len) { auto itr = m_Sessions.find (toPort); // found convo ? if (itr != m_Sessions.end ()) { // found convo if (len > 0) { LogPrint (eLogDebug, "UDP Client: Got ", len, "B from ", isIdentity ? Identity.ToBase32 () : ""); boost::system::error_code ec; m_LocalSocket->send_to (boost::asio::buffer (buf, len), itr->second->first, 0, ec); if (!ec) // mark convo as active itr->second->second = i2p::util::GetMillisecondsSinceEpoch (); else LogPrint (eLogInfo, "UDP Client: Send exception: ", ec.message (), " to ", itr->second->first); } } else LogPrint (eLogWarning, "UDP Client: Not tracking udp session using port ", (int) toPort); } } }