mirror of
				https://github.com/FreeRTOS/FreeRTOS-Plus-TCP
				synced 2025-10-25 04:56:15 +08:00 
			
		
		
		
	 0a8ad98ffc
			
		
	
	0a8ad98ffc
	
	
	
		
			
			* Catch allocation failure in prvCreateSectors * Fix unit tests * Fix UTs 2 * Fix formatting
		
			
				
	
	
		
			524 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			524 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * FreeRTOS+TCP <DEVELOPMENT BRANCH>
 | |
|  * Copyright (C) 2022 Amazon.com, Inc. or its affiliates.  All Rights Reserved.
 | |
|  *
 | |
|  * SPDX-License-Identifier: MIT
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining a copy of
 | |
|  * this software and associated documentation files (the "Software"), to deal in
 | |
|  * the Software without restriction, including without limitation the rights to
 | |
|  * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
 | |
|  * the Software, and to permit persons to whom the Software is furnished to do so,
 | |
|  * subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be included in all
 | |
|  * copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
 | |
|  * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
 | |
|  * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
 | |
|  * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 | |
|  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
|  *
 | |
|  * http://aws.amazon.com/freertos
 | |
|  * http://www.FreeRTOS.org
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @file FreeRTOS_TCP_Transmission_IPv4.c
 | |
|  * @brief Module which prepares the packet to be sent through
 | |
|  * a socket for FreeRTOS+TCP.
 | |
|  * It depends on  FreeRTOS_TCP_WIN.c, which handles the TCP windowing
 | |
|  * schemes.
 | |
|  *
 | |
|  * Endianness: in this module all ports and IP addresses are stored in
 | |
|  * host byte-order, except fields in the IP-packets
 | |
|  */
 | |
| 
 | |
| /* Standard includes. */
 | |
| #include <stdint.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| /* FreeRTOS includes. */
 | |
| #include "FreeRTOS.h"
 | |
| #include "task.h"
 | |
| #include "queue.h"
 | |
| #include "semphr.h"
 | |
| 
 | |
| /* FreeRTOS+TCP includes. */
 | |
| #include "FreeRTOS_IP.h"
 | |
| #include "FreeRTOS_Sockets.h"
 | |
| #include "FreeRTOS_IP_Private.h"
 | |
| #include "NetworkInterface.h"
 | |
| #include "NetworkBufferManagement.h"
 | |
| #include "FreeRTOS_ARP.h"
 | |
| #include "FreeRTOSIPConfigDefaults.h"
 | |
| 
 | |
| #include "FreeRTOS_TCP_IP.h"
 | |
| #include "FreeRTOS_TCP_Reception.h"
 | |
| #include "FreeRTOS_TCP_Transmission.h"
 | |
| #include "FreeRTOS_TCP_State_Handling.h"
 | |
| #include "FreeRTOS_TCP_Utils.h"
 | |
| 
 | |
| /* Just make sure the contents doesn't get compiled if TCP is not enabled. */
 | |
| /* *INDENT-OFF* */
 | |
| #if( ipconfigUSE_IPv4 != 0 ) && ( ipconfigUSE_TCP == 1 )
 | |
| /* *INDENT-ON* */
 | |
| 
 | |
| /*------------------------------------------------------------------------*/
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @brief  Return (or send) a packet to the peer. The data is stored in pxBuffer,
 | |
|  *         which may either point to a real network buffer or to a TCP socket field
 | |
|  *         called 'xTCP.xPacket'. A temporary xNetworkBuffer will be used to pass
 | |
|  *         the data to the NIC.
 | |
|  *
 | |
|  * @param[in] pxSocket The socket owning the connection.
 | |
|  * @param[in] pxDescriptor The network buffer descriptor carrying the packet.
 | |
|  * @param[in] ulLen Length of the packet being sent.
 | |
|  * @param[in] xReleaseAfterSend pdTRUE if the ownership of the descriptor is
 | |
|  *                               transferred to the network interface.
 | |
|  */
 | |
| void prvTCPReturnPacket_IPV4( FreeRTOS_Socket_t * pxSocket,
 | |
|                               NetworkBufferDescriptor_t * pxDescriptor,
 | |
|                               uint32_t ulLen,
 | |
|                               BaseType_t xReleaseAfterSend )
 | |
| {
 | |
|     TCPPacket_t * pxTCPPacket = NULL;
 | |
|     ProtocolHeaders_t * pxProtocolHeaders = NULL;
 | |
|     IPHeader_t * pxIPHeader = NULL;
 | |
|     BaseType_t xDoRelease = xReleaseAfterSend;
 | |
|     EthernetHeader_t * pxEthernetHeader = NULL;
 | |
|     NetworkBufferDescriptor_t * pxNetworkBuffer = pxDescriptor;
 | |
|     NetworkBufferDescriptor_t xTempBuffer;
 | |
|     /* memcpy() helper variables for MISRA Rule 21.15 compliance*/
 | |
|     MACAddress_t xMACAddress;
 | |
|     const void * pvCopySource = NULL;
 | |
|     void * pvCopyDest = NULL;
 | |
|     const size_t uxIPHeaderSize = ipSIZE_OF_IPv4_HEADER;
 | |
|     uint32_t ulDestinationIPAddress;
 | |
|     eARPLookupResult_t eResult;
 | |
|     NetworkEndPoint_t * pxEndPoint = NULL;
 | |
| 
 | |
|     do
 | |
|     {
 | |
|         /* For sending, a pseudo network buffer will be used, as explained above. */
 | |
| 
 | |
|         if( pxNetworkBuffer == NULL )
 | |
|         {
 | |
|             pxNetworkBuffer = &xTempBuffer;
 | |
| 
 | |
|             ( void ) memset( &xTempBuffer, 0, sizeof( xTempBuffer ) );
 | |
|             #if ( ipconfigUSE_LINKED_RX_MESSAGES != 0 )
 | |
|             {
 | |
|                 pxNetworkBuffer->pxNextBuffer = NULL;
 | |
|             }
 | |
|             #endif
 | |
|             pxNetworkBuffer->pucEthernetBuffer = pxSocket->u.xTCP.xPacket.u.ucLastPacket;
 | |
|             pxNetworkBuffer->xDataLength = sizeof( pxSocket->u.xTCP.xPacket.u.ucLastPacket );
 | |
|             xDoRelease = pdFALSE;
 | |
|         }
 | |
| 
 | |
|         #if ( ipconfigZERO_COPY_TX_DRIVER != 0 )
 | |
|         {
 | |
|             if( xDoRelease == pdFALSE )
 | |
|             {
 | |
|                 /* A zero-copy network driver wants to pass the packet buffer
 | |
|                  * to DMA, so a new buffer must be created. */
 | |
|                 pxNetworkBuffer = pxDuplicateNetworkBufferWithDescriptor( pxNetworkBuffer, ( size_t ) pxNetworkBuffer->xDataLength );
 | |
| 
 | |
|                 if( pxNetworkBuffer != NULL )
 | |
|                 {
 | |
|                     xDoRelease = pdTRUE;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     FreeRTOS_debug_printf( ( "prvTCPReturnPacket: duplicate failed\n" ) );
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         #endif /* ipconfigZERO_COPY_TX_DRIVER */
 | |
| 
 | |
|         #ifndef __COVERITY__
 | |
|             if( pxNetworkBuffer != NULL ) /* LCOV_EXCL_BR_LINE the 2nd branch will never be reached */
 | |
|         #endif
 | |
|         {
 | |
|             NetworkInterface_t * pxInterface;
 | |
| 
 | |
|             /* MISRA Ref 11.3.1 [Misaligned access] */
 | |
|             /* More details at: https://github.com/FreeRTOS/FreeRTOS-Plus-TCP/blob/main/MISRA.md#rule-113 */
 | |
|             /* coverity[misra_c_2012_rule_11_3_violation] */
 | |
|             pxIPHeader = ( ( IPHeader_t * ) &( pxNetworkBuffer->pucEthernetBuffer[ ipSIZE_OF_ETH_HEADER ] ) );
 | |
| 
 | |
|             /* Map the Ethernet buffer onto a TCPPacket_t struct for easy access to the fields. */
 | |
| 
 | |
|             /* MISRA Ref 11.3.1 [Misaligned access] */
 | |
|             /* More details at: https://github.com/FreeRTOS/FreeRTOS-Plus-TCP/blob/main/MISRA.md#rule-113 */
 | |
|             /* coverity[misra_c_2012_rule_11_3_violation] */
 | |
|             pxTCPPacket = ( TCPPacket_t * ) pxNetworkBuffer->pucEthernetBuffer;
 | |
|             pxEthernetHeader = ( EthernetHeader_t * ) &( pxTCPPacket->xEthernetHeader );
 | |
|             /* MISRA Ref 11.3.1 [Misaligned access] */
 | |
|             /* More details at: https://github.com/FreeRTOS/FreeRTOS-Plus-TCP/blob/main/MISRA.md#rule-113 */
 | |
|             /* coverity[misra_c_2012_rule_11_3_violation] */
 | |
|             pxProtocolHeaders = ( ProtocolHeaders_t * ) &( pxNetworkBuffer->pucEthernetBuffer[ ipSIZE_OF_ETH_HEADER + uxIPHeaderSize ] );
 | |
| 
 | |
|             if( pxNetworkBuffer->pxEndPoint == NULL )
 | |
|             {
 | |
|                 prvTCPReturn_SetEndPoint( pxSocket, pxNetworkBuffer, uxIPHeaderSize );
 | |
| 
 | |
|                 if( pxNetworkBuffer->pxEndPoint == NULL )
 | |
|                 {
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             /* Fill the packet, using hton translations. */
 | |
|             if( pxSocket != NULL )
 | |
|             {
 | |
|                 prvTCPReturn_CheckTCPWindow( pxSocket, pxNetworkBuffer, uxIPHeaderSize );
 | |
|                 prvTCPReturn_SetSequenceNumber( pxSocket, pxNetworkBuffer, uxIPHeaderSize, ulLen );
 | |
|                 pxIPHeader->ulDestinationIPAddress = FreeRTOS_htonl( pxSocket->u.xTCP.xRemoteIP.ulIP_IPv4 );
 | |
|                 pxIPHeader->ulSourceIPAddress = pxNetworkBuffer->pxEndPoint->ipv4_settings.ulIPAddress;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 /* Sending data without a socket, probably replying with a RST flag
 | |
|                  * Just swap the two sequence numbers. */
 | |
|                 vFlip_32( pxProtocolHeaders->xTCPHeader.ulSequenceNumber, pxProtocolHeaders->xTCPHeader.ulAckNr );
 | |
|                 vFlip_32( pxIPHeader->ulDestinationIPAddress, pxIPHeader->ulSourceIPAddress );
 | |
|             }
 | |
| 
 | |
|             pxIPHeader->ucTimeToLive = ( uint8_t ) ipconfigTCP_TIME_TO_LIVE;
 | |
|             pxIPHeader->usLength = FreeRTOS_htons( ulLen );
 | |
| 
 | |
|             /* Just an increasing number. */
 | |
|             pxIPHeader->usIdentification = FreeRTOS_htons( usPacketIdentifier );
 | |
|             usPacketIdentifier++;
 | |
| 
 | |
|             /* The stack doesn't support fragments, so the fragment offset field must always be zero.
 | |
|              * The header was never memset to zero, so set both the fragment offset and fragmentation flags in one go.
 | |
|              */
 | |
|             #if ( ipconfigFORCE_IP_DONT_FRAGMENT != 0 )
 | |
|                 pxIPHeader->usFragmentOffset = ipFRAGMENT_FLAGS_DONT_FRAGMENT;
 | |
|             #else
 | |
|                 pxIPHeader->usFragmentOffset = 0U;
 | |
|             #endif
 | |
| 
 | |
|             #if ( ipconfigDRIVER_INCLUDED_TX_IP_CHECKSUM == 0 )
 | |
|             {
 | |
|                 /* calculate the IP header checksum, in case the driver won't do that. */
 | |
|                 pxIPHeader->usHeaderChecksum = 0x00U;
 | |
|                 pxIPHeader->usHeaderChecksum = usGenerateChecksum( 0U, ( uint8_t * ) &( pxIPHeader->ucVersionHeaderLength ), uxIPHeaderSize );
 | |
|                 pxIPHeader->usHeaderChecksum = ( uint16_t ) ~FreeRTOS_htons( pxIPHeader->usHeaderChecksum );
 | |
| 
 | |
|                 /* calculate the TCP checksum for an outgoing packet. */
 | |
|                 ( void ) usGenerateProtocolChecksum( ( uint8_t * ) pxTCPPacket, pxNetworkBuffer->xDataLength, pdTRUE );
 | |
|             }
 | |
|             #endif /* if ( ipconfigDRIVER_INCLUDED_TX_IP_CHECKSUM == 0 ) */
 | |
| 
 | |
|             vFlip_16( pxProtocolHeaders->xTCPHeader.usSourcePort, pxProtocolHeaders->xTCPHeader.usDestinationPort );
 | |
| 
 | |
|             /* Important: tell NIC driver how many bytes must be sent. */
 | |
|             pxNetworkBuffer->xDataLength = ( size_t ) ulLen;
 | |
|             pxNetworkBuffer->xDataLength += ipSIZE_OF_ETH_HEADER;
 | |
| 
 | |
|             #if ( ipconfigUSE_LINKED_RX_MESSAGES != 0 )
 | |
|             {
 | |
|                 pxNetworkBuffer->pxNextBuffer = NULL;
 | |
|             }
 | |
|             #endif
 | |
| 
 | |
|             pvCopySource = &pxEthernetHeader->xSourceAddress;
 | |
|             ulDestinationIPAddress = pxIPHeader->ulDestinationIPAddress;
 | |
| 
 | |
|             eResult = eARPGetCacheEntry( &ulDestinationIPAddress, &xMACAddress, &pxEndPoint );
 | |
| 
 | |
|             if( eResult == eARPCacheHit )
 | |
|             {
 | |
|                 pvCopySource = &xMACAddress;
 | |
|                 pxNetworkBuffer->pxEndPoint = pxEndPoint;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 pvCopySource = &pxEthernetHeader->xSourceAddress;
 | |
|             }
 | |
| 
 | |
|             if( pxNetworkBuffer->pxEndPoint == NULL )
 | |
|             {
 | |
|                 break;
 | |
|             }
 | |
| 
 | |
|             /* Fill in the destination MAC addresses. */
 | |
|             pvCopyDest = &pxEthernetHeader->xDestinationAddress;
 | |
|             ( void ) memcpy( pvCopyDest, pvCopySource, sizeof( pxEthernetHeader->xDestinationAddress ) );
 | |
| 
 | |
|             /*
 | |
|              * Use helper variables for memcpy() to remain
 | |
|              * compliant with MISRA Rule 21.15.  These should be
 | |
|              * optimized away.
 | |
|              */
 | |
|             pvCopySource = pxNetworkBuffer->pxEndPoint->xMACAddress.ucBytes;
 | |
|             pvCopyDest = &pxEthernetHeader->xSourceAddress;
 | |
|             ( void ) memcpy( pvCopyDest, pvCopySource, ( size_t ) ipMAC_ADDRESS_LENGTH_BYTES );
 | |
| 
 | |
|             #if ( ipconfigETHERNET_MINIMUM_PACKET_BYTES > 0 )
 | |
|             {
 | |
|                 if( pxNetworkBuffer->xDataLength < ( size_t ) ipconfigETHERNET_MINIMUM_PACKET_BYTES )
 | |
|                 {
 | |
|                     BaseType_t xIndex;
 | |
| 
 | |
|                     for( xIndex = ( BaseType_t ) pxNetworkBuffer->xDataLength; xIndex < ( BaseType_t ) ipconfigETHERNET_MINIMUM_PACKET_BYTES; xIndex++ )
 | |
|                     {
 | |
|                         pxNetworkBuffer->pucEthernetBuffer[ xIndex ] = 0U;
 | |
|                     }
 | |
| 
 | |
|                     pxNetworkBuffer->xDataLength = ( size_t ) ipconfigETHERNET_MINIMUM_PACKET_BYTES;
 | |
|                 }
 | |
|             }
 | |
|             #endif /* if( ipconfigETHERNET_MINIMUM_PACKET_BYTES > 0 ) */
 | |
| 
 | |
|             /* Send! */
 | |
|             iptraceNETWORK_INTERFACE_OUTPUT( pxNetworkBuffer->xDataLength, pxNetworkBuffer->pucEthernetBuffer );
 | |
| 
 | |
|             configASSERT( pxNetworkBuffer->pxEndPoint->pxNetworkInterface != NULL );
 | |
|             configASSERT( pxNetworkBuffer->pxEndPoint->pxNetworkInterface->pfOutput != NULL );
 | |
| 
 | |
|             pxInterface = pxNetworkBuffer->pxEndPoint->pxNetworkInterface;
 | |
|             ( void ) pxInterface->pfOutput( pxInterface, pxNetworkBuffer, xDoRelease );
 | |
| 
 | |
|             if( xDoRelease == pdFALSE )
 | |
|             {
 | |
|                 /* Swap-back some fields, as pxBuffer probably points to a socket field
 | |
|                  * containing the packet header. */
 | |
|                 vFlip_16( pxTCPPacket->xTCPHeader.usSourcePort, pxTCPPacket->xTCPHeader.usDestinationPort );
 | |
| 
 | |
|                 pxIPHeader->ulSourceIPAddress = pxIPHeader->ulDestinationIPAddress;
 | |
| 
 | |
|                 ( void ) memcpy( pxEthernetHeader->xSourceAddress.ucBytes, pxEthernetHeader->xDestinationAddress.ucBytes, ( size_t ) ipMAC_ADDRESS_LENGTH_BYTES );
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 xDoRelease = pdFALSE;
 | |
|                 /* The buffer has been passed to DMA and will be released after use */
 | |
|             }
 | |
|         } /* if( pxNetworkBuffer != NULL ) */
 | |
|     } while( ipFALSE_BOOL );
 | |
| 
 | |
|     if( xDoRelease == pdTRUE )
 | |
|     {
 | |
|         vReleaseNetworkBufferAndDescriptor( pxNetworkBuffer );
 | |
|     }
 | |
| }
 | |
| /*-----------------------------------------------------------*/
 | |
| 
 | |
| /**
 | |
|  * @brief Let ARP look-up the MAC-address of the peer and initialise the first SYN
 | |
|  *        packet.
 | |
|  *
 | |
|  * @param[in] pxSocket The socket owning the TCP connection. The first packet shall
 | |
|  *               be created in this socket.
 | |
|  *
 | |
|  * @return pdTRUE: if the packet was successfully created and the first SYN can be sent.
 | |
|  *         Else pdFALSE.
 | |
|  *
 | |
|  * @note Connecting sockets have a special state: eCONNECT_SYN. In this phase,
 | |
|  *       the Ethernet address of the target will be found using ARP. In case the
 | |
|  *       target IP address is not within the netmask, the hardware address of the
 | |
|  *       gateway will be used.
 | |
|  */
 | |
| BaseType_t prvTCPPrepareConnect_IPV4( FreeRTOS_Socket_t * pxSocket )
 | |
| {
 | |
|     TCPPacket_t * pxTCPPacket;
 | |
|     IPHeader_t * pxIPHeader;
 | |
|     eARPLookupResult_t eReturned;
 | |
|     uint32_t ulRemoteIP;
 | |
|     MACAddress_t xEthAddress;
 | |
|     BaseType_t xReturn = pdTRUE;
 | |
|     uint32_t ulInitialSequenceNumber = 0;
 | |
| 
 | |
|     #if ( ipconfigHAS_PRINTF != 0 )
 | |
|     {
 | |
|         /* Only necessary for nicer logging. */
 | |
|         ( void ) memset( xEthAddress.ucBytes, 0, sizeof( xEthAddress.ucBytes ) );
 | |
|     }
 | |
|     #endif /* ipconfigHAS_PRINTF != 0 */
 | |
| 
 | |
|     ulRemoteIP = FreeRTOS_htonl( pxSocket->u.xTCP.xRemoteIP.ulIP_IPv4 );
 | |
|     /* Determine the ARP cache status for the requested IP address. */
 | |
|     eReturned = eARPGetCacheEntry( &( ulRemoteIP ), &( xEthAddress ), &( pxSocket->pxEndPoint ) );
 | |
| 
 | |
|     switch( eReturned )
 | |
|     {
 | |
|         case eARPCacheHit:    /* An ARP table lookup found a valid entry. */
 | |
|             break;            /* We can now prepare the SYN packet. */
 | |
| 
 | |
|         case eARPCacheMiss:   /* An ARP table lookup did not find a valid entry. */
 | |
|         case eCantSendPacket: /* There is no IP address, or an ARP is still in progress. */
 | |
|         default:
 | |
|             /* Count the number of times it could not find the ARP address. */
 | |
|             pxSocket->u.xTCP.ucRepCount++;
 | |
| 
 | |
|             FreeRTOS_debug_printf( ( "ARP for %xip (using %xip): rc=%d %02x-%02x-%02x-%02x-%02x-%02x\n",
 | |
|                                      ( unsigned ) pxSocket->u.xTCP.xRemoteIP.ulIP_IPv4,
 | |
|                                      ( unsigned ) FreeRTOS_htonl( ulRemoteIP ),
 | |
|                                      eReturned,
 | |
|                                      xEthAddress.ucBytes[ 0 ],
 | |
|                                      xEthAddress.ucBytes[ 1 ],
 | |
|                                      xEthAddress.ucBytes[ 2 ],
 | |
|                                      xEthAddress.ucBytes[ 3 ],
 | |
|                                      xEthAddress.ucBytes[ 4 ],
 | |
|                                      xEthAddress.ucBytes[ 5 ] ) );
 | |
| 
 | |
|             /* And issue a (new) ARP request */
 | |
|             FreeRTOS_OutputARPRequest( ulRemoteIP );
 | |
|             xReturn = pdFALSE;
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     if( xReturn != pdFALSE )
 | |
|     {
 | |
|         /* Get a difficult-to-predict initial sequence number for this 4-tuple. */
 | |
|         ulInitialSequenceNumber = ulApplicationGetNextSequenceNumber( pxSocket->xLocalAddress.ulIP_IPv4,
 | |
|                                                                       pxSocket->usLocalPort,
 | |
|                                                                       pxSocket->u.xTCP.xRemoteIP.ulIP_IPv4,
 | |
|                                                                       pxSocket->u.xTCP.usRemotePort );
 | |
| 
 | |
|         /* Check for a random number generation error. */
 | |
|         if( ulInitialSequenceNumber == 0U )
 | |
|         {
 | |
|             xReturn = pdFALSE;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if( xReturn != pdFALSE )
 | |
|     {
 | |
|         uint16_t usLength;
 | |
| 
 | |
|         /* The MAC-address of the peer (or gateway) has been found,
 | |
|          * now prepare the initial TCP packet and some fields in the socket. Map
 | |
|          * the buffer onto the TCPPacket_t struct to easily access it's field. */
 | |
| 
 | |
|         /* MISRA Ref 11.3.1 [Misaligned access] */
 | |
| /* More details at: https://github.com/FreeRTOS/FreeRTOS-Plus-TCP/blob/main/MISRA.md#rule-113 */
 | |
|         /* coverity[misra_c_2012_rule_11_3_violation] */
 | |
|         pxTCPPacket = ( ( TCPPacket_t * ) pxSocket->u.xTCP.xPacket.u.ucLastPacket );
 | |
|         pxIPHeader = &pxTCPPacket->xIPHeader;
 | |
| 
 | |
|         /* reset the retry counter to zero. */
 | |
|         pxSocket->u.xTCP.ucRepCount = 0U;
 | |
| 
 | |
|         /* And remember that the connect/SYN data are prepared. */
 | |
|         pxSocket->u.xTCP.bits.bConnPrepared = pdTRUE_UNSIGNED;
 | |
| 
 | |
|         /* Now that the Ethernet address is known, the initial packet can be
 | |
|          * prepared. */
 | |
|         ( void ) memset( pxSocket->u.xTCP.xPacket.u.ucLastPacket, 0, sizeof( pxSocket->u.xTCP.xPacket.u.ucLastPacket ) );
 | |
| 
 | |
|         /* Write the Ethernet address in Source, because it will be swapped by
 | |
|          * prvTCPReturnPacket(). */
 | |
|         ( void ) memcpy( ( void * ) ( &pxTCPPacket->xEthernetHeader.xSourceAddress ), ( const void * ) ( &xEthAddress ), sizeof( xEthAddress ) );
 | |
| 
 | |
|         /* 'ipIPv4_FRAME_TYPE' is already in network-byte-order. */
 | |
|         pxTCPPacket->xEthernetHeader.usFrameType = ipIPv4_FRAME_TYPE;
 | |
| 
 | |
|         pxIPHeader->ucVersionHeaderLength = 0x45U;
 | |
|         usLength = ( uint16_t ) ( sizeof( TCPPacket_t ) - sizeof( pxTCPPacket->xEthernetHeader ) );
 | |
|         pxIPHeader->usLength = FreeRTOS_htons( usLength );
 | |
|         pxIPHeader->ucTimeToLive = ( uint8_t ) ipconfigTCP_TIME_TO_LIVE;
 | |
| 
 | |
|         pxIPHeader->ucProtocol = ( uint8_t ) ipPROTOCOL_TCP;
 | |
| 
 | |
|         /* Addresses and ports will be stored swapped because prvTCPReturnPacket
 | |
|          * will swap them back while replying. */
 | |
|         pxIPHeader->ulSourceIPAddress = FreeRTOS_htonl( pxSocket->u.xTCP.xRemoteIP.ulIP_IPv4 );
 | |
| 
 | |
|         pxTCPPacket->xTCPHeader.usSourcePort = FreeRTOS_htons( pxSocket->u.xTCP.usRemotePort );
 | |
|         pxTCPPacket->xTCPHeader.usDestinationPort = FreeRTOS_htons( pxSocket->usLocalPort );
 | |
| 
 | |
|         /* We are actively connecting, so the peer's Initial Sequence Number (ISN)
 | |
|          * isn't known yet. */
 | |
|         pxSocket->u.xTCP.xTCPWindow.rx.ulCurrentSequenceNumber = 0U;
 | |
| 
 | |
|         /* Start with ISN (Initial Sequence Number). */
 | |
|         pxSocket->u.xTCP.xTCPWindow.ulOurSequenceNumber = ulInitialSequenceNumber;
 | |
| 
 | |
|         /* The TCP header size is 20 bytes, divided by 4 equals 5, which is put in
 | |
|          * the high nibble of the TCP offset field. */
 | |
|         pxTCPPacket->xTCPHeader.ucTCPOffset = 0x50U;
 | |
| 
 | |
|         /* Only set the SYN flag. */
 | |
|         pxTCPPacket->xTCPHeader.ucTCPFlags = tcpTCP_FLAG_SYN;
 | |
| 
 | |
|         /* Set the value of usMSS for this socket. */
 | |
|         prvSocketSetMSS( pxSocket );
 | |
| 
 | |
|         /* The initial sequence numbers at our side are known.  Later
 | |
|          * vTCPWindowInit() will be called to fill in the peer's sequence numbers, but
 | |
|          * first wait for a SYN+ACK reply. */
 | |
|         if( prvTCPCreateWindow( pxSocket ) != pdTRUE )
 | |
|         {
 | |
|             xReturn = pdFALSE;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return xReturn;
 | |
| }
 | |
| /*-----------------------------------------------------------*/
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @brief Common code for sending a TCP protocol control packet (i.e. no options, no
 | |
|  *        payload, just flags).
 | |
|  *
 | |
|  * @param[in] pxNetworkBuffer The network buffer received from the peer.
 | |
|  * @param[in] ucTCPFlags The flags to determine what kind of packet this is.
 | |
|  *
 | |
|  * @return pdFAIL always indicating that the packet was not consumed.
 | |
|  */
 | |
| BaseType_t prvTCPSendSpecialPktHelper_IPV4( NetworkBufferDescriptor_t * pxNetworkBuffer,
 | |
|                                             uint8_t ucTCPFlags )
 | |
| {
 | |
|     #if ( ipconfigIGNORE_UNKNOWN_PACKETS == 1 )
 | |
|         /* Configured to ignore unknown packets just suppress a compiler warning. */
 | |
|         ( void ) pxNetworkBuffer;
 | |
|         ( void ) ucTCPFlags;
 | |
|     #else
 | |
|     {
 | |
|         /* Map the ethernet buffer onto the TCPPacket_t struct for easy access to the fields. */
 | |
| 
 | |
|         /* MISRA Ref 11.3.1 [Misaligned access] */
 | |
| /* More details at: https://github.com/FreeRTOS/FreeRTOS-Plus-TCP/blob/main/MISRA.md#rule-113 */
 | |
|         /* coverity[misra_c_2012_rule_11_3_violation] */
 | |
|         TCPPacket_t * pxTCPPacket = ( ( TCPPacket_t * ) pxNetworkBuffer->pucEthernetBuffer );
 | |
|         const uint32_t ulSendLength =
 | |
|             ipSIZE_OF_IPv4_HEADER + ipSIZE_OF_TCP_HEADER; /* Plus 0 options. */
 | |
| 
 | |
|         uint8_t ucFlagsReceived = pxTCPPacket->xTCPHeader.ucTCPFlags;
 | |
|         pxTCPPacket->xTCPHeader.ucTCPFlags = ucTCPFlags;
 | |
|         pxTCPPacket->xTCPHeader.ucTCPOffset = ( ipSIZE_OF_TCP_HEADER ) << 2;
 | |
| 
 | |
|         if( ( ucFlagsReceived & tcpTCP_FLAG_SYN ) != 0U )
 | |
|         {
 | |
|             /* A synchronize packet is received. It counts as 1 pseudo byte of data,
 | |
|              * so increase the variable with 1. Before sending a reply, the values of
 | |
|              * 'ulSequenceNumber' and 'ulAckNr' will be swapped. */
 | |
|             uint32_t ulSequenceNumber = FreeRTOS_ntohl( pxTCPPacket->xTCPHeader.ulSequenceNumber );
 | |
|             ulSequenceNumber++;
 | |
|             pxTCPPacket->xTCPHeader.ulSequenceNumber = FreeRTOS_htonl( ulSequenceNumber );
 | |
|         }
 | |
| 
 | |
|         prvTCPReturnPacket( NULL, pxNetworkBuffer, ulSendLength, pdFALSE );
 | |
|     }
 | |
|     #endif /* !ipconfigIGNORE_UNKNOWN_PACKETS */
 | |
| 
 | |
|     /* The packet was not consumed. */
 | |
|     return pdFAIL;
 | |
| }
 | |
| /*-----------------------------------------------------------*/
 | |
| 
 | |
| /* *INDENT-OFF* */
 | |
| #endif /* ( ipconfigUSE_IPv4 != 0 ) && ( ipconfigUSE_TCP == 1 ) */
 | |
| /* *INDENT-ON* */
 |