|  |  |  | /*
 | 
					
						
							|  |  |  |              LUFA Library | 
					
						
							|  |  |  |      Copyright (C) Dean Camera, 2011. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dean [at] fourwalledcubicle [dot] com | 
					
						
							|  |  |  |            www.lufa-lib.org | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |   Copyright 2011  Dean Camera (dean [at] fourwalledcubicle [dot] com) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Permission to use, copy, modify, distribute, and sell this | 
					
						
							|  |  |  |   software and its documentation for any purpose is hereby granted | 
					
						
							|  |  |  |   without fee, provided that the above copyright notice appear in | 
					
						
							|  |  |  |   all copies and that both that the copyright notice and this | 
					
						
							|  |  |  |   permission notice and warranty disclaimer appear in supporting | 
					
						
							|  |  |  |   documentation, and that the name of the author not be used in | 
					
						
							|  |  |  |   advertising or publicity pertaining to distribution of the | 
					
						
							|  |  |  |   software without specific, written prior permission. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   The author disclaim all warranties with regard to this | 
					
						
							|  |  |  |   software, including all implied warranties of merchantability | 
					
						
							|  |  |  |   and fitness.  In no event shall the author be liable for any | 
					
						
							|  |  |  |   special, indirect or consequential damages or any damages | 
					
						
							|  |  |  |   whatsoever resulting from loss of use, data or profits, whether | 
					
						
							|  |  |  |   in an action of contract, negligence or other tortious action, | 
					
						
							|  |  |  |   arising out of or in connection with the use or performance of | 
					
						
							|  |  |  |   this software. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Transmission Control Protocol (TCP) packet handling routines. This protocol handles the reliable in-order transmission | 
					
						
							|  |  |  |  *  and reception of packets to and from devices on a network, to "ports" on the device. It is used in situations where data | 
					
						
							|  |  |  |  *  delivery must be reliable and correct, e.g. HTTP, TELNET and most other non-streaming protocols. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define  INCLUDE_FROM_TCP_C
 | 
					
						
							|  |  |  | #include "TCP.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Port state table array. This contains the current status of TCP ports in the device. To save on space, only open ports are
 | 
					
						
							|  |  |  |  *  stored - closed ports may be overwritten at any time, and the system will assume any ports not present in the array are closed. This | 
					
						
							|  |  |  |  *  allows for MAX_OPEN_TCP_PORTS to be less than the number of ports used by the application if desired. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | TCP_PortState_t        PortStateTable[MAX_OPEN_TCP_PORTS]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Connection state table array. This contains the current status of TCP connections in the device. To save on space, only active
 | 
					
						
							|  |  |  |  *  (non-closed) connections are stored - closed connections may be overwritten at any time, and the system will assume any connections | 
					
						
							|  |  |  |  *  not present in the array are closed. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | TCP_ConnectionState_t  ConnectionStateTable[MAX_TCP_CONNECTIONS]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Task to handle the calling of each registered application's callback function, to process and generate TCP packets at the application
 | 
					
						
							|  |  |  |  *  level. If an application produces a response, this task constructs the appropriate Ethernet frame and places it into the Ethernet OUT | 
					
						
							|  |  |  |  *  buffer for later transmission. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void TCP_Task(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Run each application in sequence, to process incoming and generate outgoing packets */ | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find the corresponding port entry in the port table */ | 
					
						
							|  |  |  | 		for (uint8_t PTableEntry = 0; PTableEntry < MAX_OPEN_TCP_PORTS; PTableEntry++) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			/* Run the application handler for the port */ | 
					
						
							|  |  |  | 			if ((PortStateTable[PTableEntry].Port  == ConnectionStateTable[CSTableEntry].Port) && | 
					
						
							|  |  |  | 			    (PortStateTable[PTableEntry].State == TCP_Port_Open)) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				PortStateTable[PTableEntry].ApplicationHandler(&ConnectionStateTable[CSTableEntry], | 
					
						
							|  |  |  | 				                                               &ConnectionStateTable[CSTableEntry].Info.Buffer); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Bail out early if there is already a frame waiting to be sent in the Ethernet OUT buffer */ | 
					
						
							|  |  |  | 	if (FrameOUT.FrameLength) | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Send response packets from each application as the TCP packet buffers are filled by the applications */ | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* For each completely received packet, pass it along to the listening application */ | 
					
						
							|  |  |  | 		if ((ConnectionStateTable[CSTableEntry].Info.Buffer.Direction == TCP_PACKETDIR_OUT) && | 
					
						
							|  |  |  | 		    (ConnectionStateTable[CSTableEntry].Info.Buffer.Ready)) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Ethernet_Frame_Header_t* FrameOUTHeader = (Ethernet_Frame_Header_t*)&FrameOUT.FrameData; | 
					
						
							|  |  |  | 			IP_Header_t*             IPHeaderOUT    = (IP_Header_t*)&FrameOUT.FrameData[sizeof(Ethernet_Frame_Header_t)]; | 
					
						
							|  |  |  | 			TCP_Header_t*            TCPHeaderOUT   = (TCP_Header_t*)&FrameOUT.FrameData[sizeof(Ethernet_Frame_Header_t) + | 
					
						
							|  |  |  | 			                                                                             sizeof(IP_Header_t)]; | 
					
						
							|  |  |  | 			void*                    TCPDataOUT     = &FrameOUT.FrameData[sizeof(Ethernet_Frame_Header_t) + | 
					
						
							|  |  |  | 			                                                              sizeof(IP_Header_t) + | 
					
						
							|  |  |  | 			                                                              sizeof(TCP_Header_t)]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			uint16_t PacketSize = ConnectionStateTable[CSTableEntry].Info.Buffer.Length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Fill out the TCP data */ | 
					
						
							|  |  |  | 			TCPHeaderOUT->SourcePort           = ConnectionStateTable[CSTableEntry].Port; | 
					
						
							|  |  |  | 			TCPHeaderOUT->DestinationPort      = ConnectionStateTable[CSTableEntry].RemotePort; | 
					
						
							|  |  |  | 			TCPHeaderOUT->SequenceNumber       = SwapEndian_32(ConnectionStateTable[CSTableEntry].Info.SequenceNumberOut); | 
					
						
							|  |  |  | 			TCPHeaderOUT->AcknowledgmentNumber = SwapEndian_32(ConnectionStateTable[CSTableEntry].Info.SequenceNumberIn); | 
					
						
							|  |  |  | 			TCPHeaderOUT->DataOffset           = (sizeof(TCP_Header_t) / sizeof(uint32_t)); | 
					
						
							|  |  |  | 			TCPHeaderOUT->WindowSize           = SwapEndian_16(TCP_WINDOW_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			TCPHeaderOUT->Flags                = TCP_FLAG_ACK; | 
					
						
							|  |  |  | 			TCPHeaderOUT->UrgentPointer        = 0; | 
					
						
							|  |  |  | 			TCPHeaderOUT->Checksum             = 0; | 
					
						
							|  |  |  | 			TCPHeaderOUT->Reserved             = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			memcpy(TCPDataOUT, ConnectionStateTable[CSTableEntry].Info.Buffer.Data, PacketSize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].Info.SequenceNumberOut += PacketSize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			TCPHeaderOUT->Checksum             = TCP_Checksum16(TCPHeaderOUT, ServerIPAddress, | 
					
						
							|  |  |  | 			                                                    ConnectionStateTable[CSTableEntry].RemoteAddress, | 
					
						
							|  |  |  | 			                                                    (sizeof(TCP_Header_t) + PacketSize)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			PacketSize += sizeof(TCP_Header_t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Fill out the response IP header */ | 
					
						
							|  |  |  | 			IPHeaderOUT->TotalLength        = SwapEndian_16(sizeof(IP_Header_t) + PacketSize); | 
					
						
							|  |  |  | 			IPHeaderOUT->TypeOfService      = 0; | 
					
						
							|  |  |  | 			IPHeaderOUT->HeaderLength       = (sizeof(IP_Header_t) / sizeof(uint32_t)); | 
					
						
							|  |  |  | 			IPHeaderOUT->Version            = 4; | 
					
						
							|  |  |  | 			IPHeaderOUT->Flags              = 0; | 
					
						
							|  |  |  | 			IPHeaderOUT->FragmentOffset     = 0; | 
					
						
							|  |  |  | 			IPHeaderOUT->Identification     = 0; | 
					
						
							|  |  |  | 			IPHeaderOUT->HeaderChecksum     = 0; | 
					
						
							|  |  |  | 			IPHeaderOUT->Protocol           = PROTOCOL_TCP; | 
					
						
							|  |  |  | 			IPHeaderOUT->TTL                = DEFAULT_TTL; | 
					
						
							|  |  |  | 			IPHeaderOUT->SourceAddress      = ServerIPAddress; | 
					
						
							|  |  |  | 			IPHeaderOUT->DestinationAddress = ConnectionStateTable[CSTableEntry].RemoteAddress; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			IPHeaderOUT->HeaderChecksum     = Ethernet_Checksum16(IPHeaderOUT, sizeof(IP_Header_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			PacketSize += sizeof(IP_Header_t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Fill out the response Ethernet frame header */ | 
					
						
							|  |  |  | 			FrameOUTHeader->Source          = ServerMACAddress; | 
					
						
							|  |  |  | 			FrameOUTHeader->Destination     = (MAC_Address_t){{0x02, 0x00, 0x02, 0x00, 0x02, 0x00}}; | 
					
						
							|  |  |  | 			FrameOUTHeader->EtherType       = SwapEndian_16(ETHERTYPE_IPV4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			PacketSize += sizeof(Ethernet_Frame_Header_t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Set the response length in the buffer and indicate that a response is ready to be sent */ | 
					
						
							|  |  |  | 			FrameOUT.FrameLength            = PacketSize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].Info.Buffer.Ready = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Initializes the TCP protocol handler, clearing the port and connection state tables. This must be called before TCP packets are
 | 
					
						
							|  |  |  |  *  processed. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void TCP_Init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Initialize the port state table with all CLOSED entries */ | 
					
						
							|  |  |  | 	for (uint8_t PTableEntry = 0; PTableEntry < MAX_OPEN_TCP_PORTS; PTableEntry++) | 
					
						
							|  |  |  | 	  PortStateTable[PTableEntry].State = TCP_Port_Closed; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Initialize the connection table with all CLOSED entries */ | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	  ConnectionStateTable[CSTableEntry].State = TCP_Connection_Closed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Sets the state and callback handler of the given port, specified in big endian to the given state.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] Port     Port whose state and callback function to set, specified in big endian | 
					
						
							|  |  |  |  *  \param[in] State    New state of the port, a value from the \ref TCP_PortStates_t enum | 
					
						
							|  |  |  |  *  \param[in] Handler  Application callback handler for the port | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return Boolean true if the port state was set, false otherwise (no more space in the port state table) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool TCP_SetPortState(const uint16_t Port, | 
					
						
							|  |  |  |                       const uint8_t State, | 
					
						
							|  |  |  |                       void (*Handler)(TCP_ConnectionState_t*, TCP_ConnectionBuffer_t*)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Note, Port number should be specified in BIG endian to simplify network code */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Check to see if the port entry is already in the port state table */ | 
					
						
							|  |  |  | 	for (uint8_t PTableEntry = 0; PTableEntry < MAX_OPEN_TCP_PORTS; PTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find existing entry for the port in the table, update it if found */ | 
					
						
							|  |  |  | 		if (PortStateTable[PTableEntry].Port == Port) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			PortStateTable[PTableEntry].State = State; | 
					
						
							|  |  |  | 			PortStateTable[PTableEntry].ApplicationHandler = Handler; | 
					
						
							|  |  |  | 			return true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Check if trying to open the port -- if so we need to find an unused (closed) entry and replace it */ | 
					
						
							|  |  |  | 	if (State == TCP_Port_Open) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		for (uint8_t PTableEntry = 0; PTableEntry < MAX_OPEN_TCP_PORTS; PTableEntry++) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			/* Find a closed port entry in the table, change it to the given port and state */ | 
					
						
							|  |  |  | 			if (PortStateTable[PTableEntry].State == TCP_Port_Closed) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				PortStateTable[PTableEntry].Port  = Port; | 
					
						
							|  |  |  | 				PortStateTable[PTableEntry].State = State; | 
					
						
							|  |  |  | 				PortStateTable[PTableEntry].ApplicationHandler = Handler; | 
					
						
							|  |  |  | 				return true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Port not in table and no room to add it, return failure */ | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Port not in table but trying to close it, so operation successful */ | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Retrieves the current state of a given TCP port, specified in big endian.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] Port  TCP port whose state is to be retrieved, given in big-endian | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return A value from the \ref TCP_PortStates_t enum | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | uint8_t TCP_GetPortState(const uint16_t Port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Note, Port number should be specified in BIG endian to simplify network code */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint8_t PTableEntry = 0; PTableEntry < MAX_OPEN_TCP_PORTS; PTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find existing entry for the port in the table, return the port status if found */ | 
					
						
							|  |  |  | 		if (PortStateTable[PTableEntry].Port == Port) | 
					
						
							|  |  |  | 		  return PortStateTable[PTableEntry].State; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Port not in table, assume closed */ | 
					
						
							|  |  |  | 	return TCP_Port_Closed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Sets the connection state of the given port, remote address and remote port to the given TCP connection state. If the
 | 
					
						
							|  |  |  |  *  connection exists in the connection state table it is updated, otherwise it is created if possible. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] Port           TCP port of the connection on the device, specified in big endian | 
					
						
							|  |  |  |  *  \param[in] RemoteAddress  Remote protocol IP address of the connected device | 
					
						
							|  |  |  |  *  \param[in] RemotePort     TCP port of the remote device in the connection, specified in big endian | 
					
						
							|  |  |  |  *  \param[in] State          TCP connection state, a value from the \ref TCP_ConnectionStates_t enum | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return Boolean true if the connection was updated or created, false otherwise (no more space in the connection state table) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool TCP_SetConnectionState(const uint16_t Port, | 
					
						
							|  |  |  |                             const IP_Address_t RemoteAddress, | 
					
						
							|  |  |  |                             const uint16_t RemotePort, | 
					
						
							|  |  |  |                             const uint8_t State) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Note, Port number should be specified in BIG endian to simplify network code */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find port entry in the table */ | 
					
						
							|  |  |  | 		if ((ConnectionStateTable[CSTableEntry].Port == Port) && | 
					
						
							|  |  |  | 		     IP_COMPARE(&ConnectionStateTable[CSTableEntry].RemoteAddress, &RemoteAddress) && | 
					
						
							|  |  |  | 			 ConnectionStateTable[CSTableEntry].RemotePort == RemotePort) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].State = State; | 
					
						
							|  |  |  | 			return true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find empty entry in the table */ | 
					
						
							|  |  |  | 		if (ConnectionStateTable[CSTableEntry].State == TCP_Connection_Closed) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].Port          = Port; | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].RemoteAddress = RemoteAddress; | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].RemotePort    = RemotePort; | 
					
						
							|  |  |  | 			ConnectionStateTable[CSTableEntry].State         = State; | 
					
						
							|  |  |  | 			return true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Retrieves the current state of a given TCP connection to a host.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] Port           TCP port on the device in the connection, specified in big endian | 
					
						
							|  |  |  |  *  \param[in] RemoteAddress  Remote protocol IP address of the connected host | 
					
						
							|  |  |  |  *  \param[in] RemotePort     Remote TCP port of the connected host, specified in big endian | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return A value from the \ref TCP_ConnectionStates_t enum | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | uint8_t TCP_GetConnectionState(const uint16_t Port, | 
					
						
							|  |  |  |                                const IP_Address_t RemoteAddress, | 
					
						
							|  |  |  |                                const uint16_t RemotePort) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Note, Port number should be specified in BIG endian to simplify network code */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find port entry in the table */ | 
					
						
							|  |  |  | 		if ((ConnectionStateTable[CSTableEntry].Port == Port) && | 
					
						
							|  |  |  | 		     IP_COMPARE(&ConnectionStateTable[CSTableEntry].RemoteAddress, &RemoteAddress) && | 
					
						
							|  |  |  | 			 ConnectionStateTable[CSTableEntry].RemotePort == RemotePort) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return ConnectionStateTable[CSTableEntry].State; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return TCP_Connection_Closed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Retrieves the connection info structure of a given connection to a host.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] Port           TCP port on the device in the connection, specified in big endian | 
					
						
							|  |  |  |  *  \param[in] RemoteAddress  Remote protocol IP address of the connected host | 
					
						
							|  |  |  |  *  \param[in] RemotePort     Remote TCP port of the connected host, specified in big endian | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return ConnectionInfo structure of the connection if found, NULL otherwise | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | TCP_ConnectionInfo_t* TCP_GetConnectionInfo(const uint16_t Port, | 
					
						
							|  |  |  |                                             const IP_Address_t RemoteAddress, | 
					
						
							|  |  |  |                                             const uint16_t RemotePort) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Note, Port number should be specified in BIG endian to simplify network code */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint8_t CSTableEntry = 0; CSTableEntry < MAX_TCP_CONNECTIONS; CSTableEntry++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Find port entry in the table */ | 
					
						
							|  |  |  | 		if ((ConnectionStateTable[CSTableEntry].Port == Port) && | 
					
						
							|  |  |  | 		     IP_COMPARE(&ConnectionStateTable[CSTableEntry].RemoteAddress, &RemoteAddress) && | 
					
						
							|  |  |  | 			 ConnectionStateTable[CSTableEntry].RemotePort == RemotePort) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return &ConnectionStateTable[CSTableEntry].Info; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Processes a TCP packet inside an Ethernet frame, and writes the appropriate response
 | 
					
						
							|  |  |  |  *  to the output Ethernet frame if one is created by a application handler. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] IPHeaderInStart     Pointer to the start of the incoming packet's IP header | 
					
						
							|  |  |  |  *  \param[in] TCPHeaderInStart    Pointer to the start of the incoming packet's TCP header | 
					
						
							|  |  |  |  *  \param[out] TCPHeaderOutStart  Pointer to the start of the outgoing packet's TCP header | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return The number of bytes written to the out Ethernet frame if any, NO_RESPONSE if no | 
					
						
							|  |  |  |  *           response was generated, NO_PROCESS if the packet processing was deferred until the | 
					
						
							|  |  |  |  *           next Ethernet packet handler iteration | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int16_t TCP_ProcessTCPPacket(void* IPHeaderInStart, | 
					
						
							|  |  |  |                              void* TCPHeaderInStart, | 
					
						
							|  |  |  |                              void* TCPHeaderOutStart) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	IP_Header_t*  IPHeaderIN   = (IP_Header_t*)IPHeaderInStart; | 
					
						
							|  |  |  | 	TCP_Header_t* TCPHeaderIN  = (TCP_Header_t*)TCPHeaderInStart; | 
					
						
							|  |  |  | 	TCP_Header_t* TCPHeaderOUT = (TCP_Header_t*)TCPHeaderOutStart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TCP_ConnectionInfo_t* ConnectionInfo; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	DecodeTCPHeader(TCPHeaderInStart); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool PacketResponse = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Check if the destination port is open and allows incoming connections */ | 
					
						
							|  |  |  | 	if (TCP_GetPortState(TCPHeaderIN->DestinationPort) == TCP_Port_Open) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Detect SYN from host to start a connection */ | 
					
						
							|  |  |  | 		if (TCPHeaderIN->Flags & TCP_FLAG_SYN) | 
					
						
							|  |  |  | 		  TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, TCPHeaderIN->SourcePort, TCP_Connection_Listen); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Detect RST from host to abort existing connection */ | 
					
						
							|  |  |  | 		if (TCPHeaderIN->Flags & TCP_FLAG_RST) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 			                           TCPHeaderIN->SourcePort, TCP_Connection_Closed)) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				TCPHeaderOUT->Flags = (TCP_FLAG_RST | TCP_FLAG_ACK); | 
					
						
							|  |  |  | 				PacketResponse = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			/* Process the incoming TCP packet based on the current connection state for the sender and port */ | 
					
						
							|  |  |  | 			switch (TCP_GetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, TCPHeaderIN->SourcePort)) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				case TCP_Connection_Listen: | 
					
						
							|  |  |  | 					if (TCPHeaderIN->Flags == TCP_FLAG_SYN) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						/* SYN connection starts a connection with a peer */ | 
					
						
							|  |  |  | 						if (TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 						                           TCPHeaderIN->SourcePort, TCP_Connection_SYNReceived)) | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							TCPHeaderOUT->Flags = (TCP_FLAG_SYN | TCP_FLAG_ACK); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							ConnectionInfo->SequenceNumberIn  = (SwapEndian_32(TCPHeaderIN->SequenceNumber) + 1); | 
					
						
							|  |  |  | 							ConnectionInfo->SequenceNumberOut = 0; | 
					
						
							|  |  |  | 							ConnectionInfo->Buffer.InUse      = false; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						else | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							TCPHeaderOUT->Flags = TCP_FLAG_RST; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						PacketResponse      = true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				case TCP_Connection_SYNReceived: | 
					
						
							|  |  |  | 					if (TCPHeaderIN->Flags == TCP_FLAG_ACK) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						/* ACK during the connection process completes the connection to a peer */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_Established); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 															   TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberOut++; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				case TCP_Connection_Established: | 
					
						
							|  |  |  | 					if (TCPHeaderIN->Flags == (TCP_FLAG_FIN | TCP_FLAG_ACK)) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						/* FIN ACK when connected to a peer starts the finalization process */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCPHeaderOUT->Flags = (TCP_FLAG_FIN | TCP_FLAG_ACK); | 
					
						
							|  |  |  | 						PacketResponse      = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_CloseWait); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 															   TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberIn++; | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberOut++; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if ((TCPHeaderIN->Flags == TCP_FLAG_ACK) || (TCPHeaderIN->Flags == (TCP_FLAG_ACK | TCP_FLAG_PSH))) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 															   TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						/* Check if the buffer is currently in use either by a buffered data to send, or receive */ | 
					
						
							|  |  |  | 						if ((ConnectionInfo->Buffer.InUse == false) && (ConnectionInfo->Buffer.Ready == false)) | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							ConnectionInfo->Buffer.Direction = TCP_PACKETDIR_IN; | 
					
						
							|  |  |  | 							ConnectionInfo->Buffer.InUse     = true; | 
					
						
							|  |  |  | 							ConnectionInfo->Buffer.Length    = 0; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						/* Check if the buffer has been claimed by us to read in data from the peer */ | 
					
						
							|  |  |  | 						if ((ConnectionInfo->Buffer.Direction == TCP_PACKETDIR_IN) && | 
					
						
							|  |  |  | 							(ConnectionInfo->Buffer.Length != TCP_WINDOW_SIZE)) | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							uint16_t IPOffset   = (IPHeaderIN->HeaderLength * sizeof(uint32_t)); | 
					
						
							|  |  |  | 							uint16_t TCPOffset  = (TCPHeaderIN->DataOffset * sizeof(uint32_t)); | 
					
						
							|  |  |  | 							uint16_t DataLength = (SwapEndian_16(IPHeaderIN->TotalLength) - IPOffset - TCPOffset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							/* Copy the packet data into the buffer */ | 
					
						
							|  |  |  | 							memcpy(&ConnectionInfo->Buffer.Data[ConnectionInfo->Buffer.Length], | 
					
						
							|  |  |  | 								   &((uint8_t*)TCPHeaderInStart)[TCPOffset], | 
					
						
							|  |  |  | 								   DataLength); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							ConnectionInfo->SequenceNumberIn += DataLength; | 
					
						
							|  |  |  | 							ConnectionInfo->Buffer.Length    += DataLength; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							/* Check if the buffer is full or if the PSH flag is set, if so indicate buffer ready */ | 
					
						
							|  |  |  | 							if ((!(TCP_WINDOW_SIZE - ConnectionInfo->Buffer.Length)) || (TCPHeaderIN->Flags & TCP_FLAG_PSH)) | 
					
						
							|  |  |  | 							{ | 
					
						
							|  |  |  | 								ConnectionInfo->Buffer.InUse = false; | 
					
						
							|  |  |  | 								ConnectionInfo->Buffer.Ready = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								TCPHeaderOUT->Flags = TCP_FLAG_ACK; | 
					
						
							|  |  |  | 								PacketResponse      = true; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						else | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							/* Buffer is currently in use by the application, defer processing of the incoming packet */ | 
					
						
							|  |  |  | 							return NO_PROCESS; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				case TCP_Connection_Closing: | 
					
						
							|  |  |  | 						ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 															   TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCPHeaderOUT->Flags = (TCP_FLAG_ACK | TCP_FLAG_FIN); | 
					
						
							|  |  |  | 						PacketResponse      = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo->Buffer.InUse = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_FINWait1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				case TCP_Connection_FINWait1: | 
					
						
							|  |  |  | 					if (TCPHeaderIN->Flags == (TCP_FLAG_FIN | TCP_FLAG_ACK)) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 															   TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCPHeaderOUT->Flags = TCP_FLAG_ACK; | 
					
						
							|  |  |  | 						PacketResponse      = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberIn++; | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberOut++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_Closed); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if (TCPHeaderIN->Flags == TCP_FLAG_ACK) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_FINWait2); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				case TCP_Connection_FINWait2: | 
					
						
							|  |  |  | 					if (TCPHeaderIN->Flags == (TCP_FLAG_FIN | TCP_FLAG_ACK)) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 															   TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCPHeaderOUT->Flags = TCP_FLAG_ACK; | 
					
						
							|  |  |  | 						PacketResponse      = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberIn++; | 
					
						
							|  |  |  | 						ConnectionInfo->SequenceNumberOut++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_Closed); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				case TCP_Connection_CloseWait: | 
					
						
							|  |  |  | 					if (TCPHeaderIN->Flags == TCP_FLAG_ACK) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						TCP_SetConnectionState(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 											   TCPHeaderIN->SourcePort, TCP_Connection_Closed); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Port is not open, indicate via a RST/ACK response to the sender */ | 
					
						
							|  |  |  | 		TCPHeaderOUT->Flags = (TCP_FLAG_RST | TCP_FLAG_ACK); | 
					
						
							|  |  |  | 		PacketResponse      = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Check if we need to respond to the sent packet */ | 
					
						
							|  |  |  | 	if (PacketResponse) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		ConnectionInfo = TCP_GetConnectionInfo(TCPHeaderIN->DestinationPort, IPHeaderIN->SourceAddress, | 
					
						
							|  |  |  | 		                                       TCPHeaderIN->SourcePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		TCPHeaderOUT->SourcePort           = TCPHeaderIN->DestinationPort; | 
					
						
							|  |  |  | 		TCPHeaderOUT->DestinationPort      = TCPHeaderIN->SourcePort; | 
					
						
							|  |  |  | 		TCPHeaderOUT->SequenceNumber       = SwapEndian_32(ConnectionInfo->SequenceNumberOut); | 
					
						
							|  |  |  | 		TCPHeaderOUT->AcknowledgmentNumber = SwapEndian_32(ConnectionInfo->SequenceNumberIn); | 
					
						
							|  |  |  | 		TCPHeaderOUT->DataOffset           = (sizeof(TCP_Header_t) / sizeof(uint32_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!(ConnectionInfo->Buffer.InUse)) | 
					
						
							|  |  |  | 		  TCPHeaderOUT->WindowSize         = SwapEndian_16(TCP_WINDOW_SIZE); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		  TCPHeaderOUT->WindowSize         = SwapEndian_16(TCP_WINDOW_SIZE - ConnectionInfo->Buffer.Length); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		TCPHeaderOUT->UrgentPointer        = 0; | 
					
						
							|  |  |  | 		TCPHeaderOUT->Checksum             = 0; | 
					
						
							|  |  |  | 		TCPHeaderOUT->Reserved             = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		TCPHeaderOUT->Checksum             = TCP_Checksum16(TCPHeaderOUT, IPHeaderIN->DestinationAddress, | 
					
						
							|  |  |  | 		                                                    IPHeaderIN->SourceAddress, sizeof(TCP_Header_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return sizeof(TCP_Header_t); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NO_RESPONSE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Calculates the appropriate TCP checksum, consisting of the addition of the one's compliment of each word,
 | 
					
						
							|  |  |  |  *  complimented. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \param[in] TCPHeaderOutStart   Pointer to the start of the packet's outgoing TCP header | 
					
						
							|  |  |  |  *  \param[in] SourceAddress       Source protocol IP address of the outgoing IP header | 
					
						
							|  |  |  |  *  \param[in] DestinationAddress  Destination protocol IP address of the outgoing IP header | 
					
						
							|  |  |  |  *  \param[in] TCPOutSize          Size in bytes of the TCP data header and payload | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \return A 16-bit TCP checksum value | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static uint16_t TCP_Checksum16(void* TCPHeaderOutStart, | 
					
						
							|  |  |  |                                const IP_Address_t SourceAddress, | 
					
						
							|  |  |  |                                const IP_Address_t DestinationAddress, | 
					
						
							|  |  |  |                                uint16_t TCPOutSize) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	uint32_t Checksum = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* TCP/IP checksums are the addition of the one's compliment of each word including the IP pseudo-header,
 | 
					
						
							|  |  |  | 	   complimented */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Checksum += ((uint16_t*)&SourceAddress)[0]; | 
					
						
							|  |  |  | 	Checksum += ((uint16_t*)&SourceAddress)[1]; | 
					
						
							|  |  |  | 	Checksum += ((uint16_t*)&DestinationAddress)[0]; | 
					
						
							|  |  |  | 	Checksum += ((uint16_t*)&DestinationAddress)[1]; | 
					
						
							|  |  |  | 	Checksum += SwapEndian_16(PROTOCOL_TCP); | 
					
						
							|  |  |  | 	Checksum += SwapEndian_16(TCPOutSize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint16_t CurrWord = 0; CurrWord < (TCPOutSize >> 1); CurrWord++) | 
					
						
							|  |  |  | 	  Checksum += ((uint16_t*)TCPHeaderOutStart)[CurrWord]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (TCPOutSize & 0x01) | 
					
						
							|  |  |  | 	  Checksum += (((uint16_t*)TCPHeaderOutStart)[TCPOutSize >> 1] & 0x00FF); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (Checksum & 0xFFFF0000) | 
					
						
							|  |  |  | 	  Checksum = ((Checksum & 0xFFFF) + (Checksum >> 16)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ~Checksum; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |