b7e825d6e7
Modelled after UDP, add limited TCP support to the boot net stack. The socket works by queuing received packets as well as sent packets that have not yet been ACK'ed. Some known issues are documented, especially there's only limited congestion control. I.e., we send immediately and in unlimited quantity, thus its use should be restricted to local networks, and due to a fixed window size there is potential for our socket being overrun with data packets before they are read. Some corner cases like wrapping sequence numbers may cause a timeout. The TCP implementation is based on Andrew S. Tanenbaum's "Computer Networks", 4th ed., as well as lecture notes from Prof. W. Effelsberg, the relevant RFCs and Wikipedia. The pseudo-random number Galois LFSR used for the sequence number was suggested by Endre Varga. Since the code is unlikely to get much smaller, better merge it now so that subsequent changes get easier to review. No platform actively uses TCP sockets yet, and the receiving code has been reviewed for endianness issues and should terminate okay after verifying the checksum if no sockets are open. Based on a version tested with custom code (#5240) as well as with iSCSI. Compile-tested boot_loader_openfirmware, pxehaiku-loader with gcc4 and haiku_loader with gcc2. Closes ticket #5240. Changes from #5240 proposed patch: * Various bug fixes related to queuing, some memory leaks fixed. * Never bump the sequence number when dequeuing a packet. It's done afterwards. * Don't bump the sequence number again when resending the queue or ACK'ing. * Aggressively ACK while waiting for packets. * Don't queue sent ACK-only packets. * More trace output, esp. for queue inspection. * Adapted use of TCP header flags to r38434. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@38472 a95241bf-73f2-0310-859d-f6bbb57e9c96
138 lines
3.3 KiB
C++
138 lines
3.3 KiB
C++
/*
|
|
* Copyright 2010 Andreas Färber <andreas.faerber@web.de>
|
|
* All rights reserved. Distributed under the terms of the MIT License.
|
|
*/
|
|
#ifndef BOOT_NET_TCP_H
|
|
#define BOOT_NET_TCP_H
|
|
|
|
|
|
#include <boot/net/IP.h>
|
|
|
|
|
|
class TCPPacket {
|
|
public:
|
|
TCPPacket();
|
|
~TCPPacket();
|
|
|
|
status_t SetTo(const void* data, size_t size, ip_addr_t sourceAddress,
|
|
uint16 sourcePort, ip_addr_t destinationAddress,
|
|
uint16 destinationPort, uint32 sequenceNumber,
|
|
uint32 acknowledgmentNumber, uint8 flags);
|
|
|
|
ip_addr_t SourceAddress() const;
|
|
ip_addr_t DestinationAddress() const;
|
|
uint16 SourcePort() const;
|
|
uint16 DestinationPort() const;
|
|
uint32 SequenceNumber() const;
|
|
uint32 AcknowledgmentNumber() const;
|
|
const void* Data() const { return fData; }
|
|
size_t DataSize() const { return fSize; }
|
|
uint8 Flags() const { return fFlags; }
|
|
|
|
bool ProvidesSequenceNumber(uint32 sequenceNo) const;
|
|
|
|
TCPPacket* Next() const;
|
|
void SetNext(TCPPacket* packet);
|
|
|
|
private:
|
|
ip_addr_t fSourceAddress;
|
|
ip_addr_t fDestinationAddress;
|
|
uint16 fSourcePort;
|
|
uint16 fDestinationPort;
|
|
uint32 fSequenceNumber;
|
|
uint32 fAcknowledgmentNumber;
|
|
void* fData;
|
|
size_t fSize;
|
|
uint8 fFlags;
|
|
TCPPacket* fNext;
|
|
};
|
|
|
|
class TCPService;
|
|
|
|
enum TCPSocketState {
|
|
TCP_SOCKET_STATE_INITIAL,
|
|
TCP_SOCKET_STATE_SYN_SENT,
|
|
TCP_SOCKET_STATE_SYN_RECEIVED,
|
|
TCP_SOCKET_STATE_OPEN,
|
|
TCP_SOCKET_STATE_FIN_SENT,
|
|
TCP_SOCKET_STATE_CLOSED
|
|
};
|
|
|
|
class TCPSocket {
|
|
public:
|
|
TCPSocket();
|
|
~TCPSocket();
|
|
|
|
ip_addr_t Address() const { return fAddress; }
|
|
uint16 Port() const { return fPort; }
|
|
|
|
status_t Connect(ip_addr_t address, uint16 port);
|
|
status_t Close();
|
|
status_t Read(void* buffer, size_t bufferSize, size_t* bytesRead, bigtime_t timeout = 0);
|
|
status_t Write(const void* buffer, size_t bufferSize);
|
|
|
|
void Acknowledge(uint32 number);
|
|
void ProcessPacket(TCPPacket* packet);
|
|
|
|
private:
|
|
TCPPacket* _PeekPacket();
|
|
TCPPacket* _DequeuePacket();
|
|
status_t _Send(TCPPacket* packet, bool enqueue = true);
|
|
status_t _ResendQueue();
|
|
void _EnqueueOutgoingPacket(TCPPacket* packet);
|
|
inline void _DumpQueue();
|
|
status_t _WaitForState(TCPSocketState state, bigtime_t timeout = 0);
|
|
status_t _Ack();
|
|
|
|
TCPService* fTCPService;
|
|
ip_addr_t fAddress;
|
|
uint16 fPort;
|
|
ip_addr_t fRemoteAddress;
|
|
uint16 fRemotePort;
|
|
uint32 fSequenceNumber;
|
|
uint32 fAcknowledgeNumber;
|
|
uint32 fNextSequence;
|
|
TCPPacket* fFirstPacket;
|
|
TCPPacket* fLastPacket;
|
|
TCPPacket* fFirstSentPacket;
|
|
TCPPacket* fLastSentPacket;
|
|
TCPSocketState fState;
|
|
TCPSocketState fRemoteState;
|
|
};
|
|
|
|
class TCPService : public IPSubService {
|
|
public:
|
|
TCPService(IPService* ipService);
|
|
virtual ~TCPService();
|
|
|
|
status_t Init();
|
|
|
|
virtual uint8 IPProtocol() const;
|
|
|
|
virtual void HandleIPPacket(IPService* ipService, ip_addr_t sourceIP,
|
|
ip_addr_t destinationIP, const void* data, size_t size);
|
|
|
|
status_t Send(uint16 sourcePort, ip_addr_t destinationAddress,
|
|
uint16 destinationPort, uint32 sequenceNumber,
|
|
uint32 acknowledgmentNumber, uint8 flags, ChainBuffer* buffer);
|
|
|
|
void ProcessIncomingPackets();
|
|
|
|
status_t BindSocket(TCPSocket* socket);
|
|
void UnbindSocket(TCPSocket* socket);
|
|
|
|
private:
|
|
uint16 _ChecksumBuffer(ChainBuffer* buffer, ip_addr_t source,
|
|
ip_addr_t destination, uint16 length);
|
|
uint16 _ChecksumData(const void* data, uint16 length, ip_addr_t source,
|
|
ip_addr_t destination);
|
|
|
|
TCPSocket* _FindSocket(ip_addr_t address, uint16 port);
|
|
|
|
IPService* fIPService;
|
|
Vector<TCPSocket*> fSockets;
|
|
};
|
|
|
|
|
|
#endif
|