mirror of
https://github.com/FreeRTOS/FreeRTOS-Plus-TCP
synced 2025-10-24 12:02:37 +08:00

* 556 Initial Cmake Module definition. #557 * renaming variables that have conflicting names with MSC and *nix headers * fix build issue for posix port * Fix warning: -Waddress-of-packed-member when calculating checksum directly from network packets * fix warnings with prvInitialiseTCPFields declaration * removing macros that hides the structure fields * Updating build check enable all config to enable all config macros * CMake: Fix GIT_REPOSITORY and GIT_TAG (#742) * moving ipTRUE_BOOL and ipFALSE_BOOL out of #ifndef pdTRUE_SIGNED check as they are not defined in kernel * minor fix to the cmake files and main file * Uncrustify: triggered by comment. * adding doxygen comments to new functions * Uncrustify: triggered by comment * Add more warnings check and fix warnings * Uncrustify: triggered by comment * fix review feedback and more debug printf warnings fix * more warnings fix * fix misra issues * Uncrustify: triggered by comment * replace sin_addr with sin_address.ulIP_IPv4 in +TCP demos * replace sin_addr6 with sin_address.xIP_IPv6 in +TCP demos * replace freertos_sockaddr6 with freertos_sockaddr in +TCP demos * review feedback changes * removing duplicate def for prvStreamBufferAdd from winpcap * fix more warnings from MSVC * Uncrustify: triggered by comment * review feedback changes * Uncrustify: triggered by comment --------- Co-authored-by: phelter <paulheltera@gmail.com> Co-authored-by: Nikhil Kamath <110539926+amazonKamath@users.noreply.github.com> Co-authored-by: GitHub Action <action@github.com>
727 lines
31 KiB
C
727 lines
31 KiB
C
/*
|
|
* FreeRTOS+TCP V2.3.1
|
|
* Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
*
|
|
* 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_RA.c
|
|
* @brief A client implementation of Router advertisement protocol.
|
|
*/
|
|
|
|
/* Standard includes. */
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
|
|
|
|
/* FreeRTOS includes. */
|
|
#include "FreeRTOS.h"
|
|
#include "task.h"
|
|
|
|
/* FreeRTOS+TCP includes. */
|
|
#include "FreeRTOS_IP.h"
|
|
#include "FreeRTOS_Sockets.h"
|
|
#include "FreeRTOS_IP_Private.h"
|
|
#include "FreeRTOS_IP_Timers.h"
|
|
#include "FreeRTOS_ARP.h"
|
|
#include "FreeRTOS_UDP_IP.h"
|
|
#include "FreeRTOS_Routing.h"
|
|
#include "FreeRTOS_ND.h"
|
|
#if ( ipconfigUSE_LLMNR == 1 )
|
|
#include "FreeRTOS_DNS.h"
|
|
#endif /* ipconfigUSE_LLMNR */
|
|
#include "NetworkBufferManagement.h"
|
|
|
|
/* This define may exclude the entire source file. */
|
|
#if ( ipconfigUSE_IPv6 != 0 ) && ( ipconfigUSE_RA != 0 )
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/** A block time of 0 simply means "don't block". */
|
|
#define raDONT_BLOCK ( ( TickType_t ) 0 )
|
|
|
|
/** The default value for the IPv6-field 'ucVersionTrafficClass'. */
|
|
#define raDEFAULT_VERSION_TRAFFIC_CLASS 0x60U
|
|
|
|
/** The default value for the IPv6-field 'ucHopLimit'. */
|
|
#define raDEFAULT_HOP_LIMIT 255U
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/* Initialise the Router Advertisement process for a given end-point. */
|
|
static void vRAProcessInit( NetworkEndPoint_t * pxEndPoint );
|
|
|
|
/* Find a link-local address that is bound to a given interface. */
|
|
static BaseType_t xGetLinkLocalAddress( const NetworkInterface_t * pxInterface,
|
|
IPv6_Address_t * pxAddress );
|
|
|
|
/* Read the reply received from the RA server. */
|
|
static ICMPPrefixOption_IPv6_t * vReceiveRA_ReadReply( const NetworkBufferDescriptor_t * pxNetworkBuffer );
|
|
|
|
/* Handle the states that are limited by a timer. See if any of the timers has expired. */
|
|
static TickType_t xRAProcess_HandleWaitStates( NetworkEndPoint_t * pxEndPoint,
|
|
TickType_t uxReloadTime );
|
|
|
|
/* Handle the other states. */
|
|
static TickType_t xRAProcess_HandleOtherStates( NetworkEndPoint_t * pxEndPoint,
|
|
TickType_t uxReloadTime );
|
|
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Find a link-local address that is bound to a given interface.
|
|
*
|
|
* @param[in] pxInterface: The interface for which a link-local address is looked up.
|
|
* @param[out] pxAddress: The IP address will be copied to this parameter.
|
|
*
|
|
* @return pdPASS in case a link-local address was found, otherwise pdFAIL.
|
|
*/
|
|
static BaseType_t xGetLinkLocalAddress( const NetworkInterface_t * pxInterface,
|
|
IPv6_Address_t * pxAddress )
|
|
{
|
|
BaseType_t xResult = pdFAIL;
|
|
NetworkEndPoint_t * pxEndPoint;
|
|
|
|
for( pxEndPoint = FreeRTOS_FirstEndPoint( pxInterface );
|
|
pxEndPoint != NULL;
|
|
pxEndPoint = FreeRTOS_NextEndPoint( pxInterface, pxEndPoint ) )
|
|
{
|
|
/* Check if it has the link-local prefix FE80::/10 */
|
|
if( ( pxEndPoint->ipv6_settings.xIPAddress.ucBytes[ 0 ] == 0xfeU ) &&
|
|
( ( pxEndPoint->ipv6_settings.xIPAddress.ucBytes[ 1 ] & 0xc0U ) == 0x80U ) )
|
|
{
|
|
( void ) memcpy( pxAddress->ucBytes, pxEndPoint->ipv6_settings.xIPAddress.ucBytes, ipSIZE_OF_IPv6_ADDRESS );
|
|
xResult = pdPASS;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return xResult;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Send an ICMPv6 message of the type: Router Solicitation.
|
|
*
|
|
* @param[in] pxNetworkBuffer: The network buffer which can be used for this.
|
|
* @param[in] pxIPAddress: The target address, normally ff02::2
|
|
*
|
|
*/
|
|
void vNDSendRouterSolicitation( NetworkBufferDescriptor_t * pxNetworkBuffer,
|
|
IPv6_Address_t * pxIPAddress )
|
|
{
|
|
ICMPPacket_IPv6_t * pxICMPPacket;
|
|
ICMPRouterSolicitation_IPv6_t * xRASolicitationRequest;
|
|
const NetworkEndPoint_t * pxEndPoint = pxNetworkBuffer->pxEndPoint;
|
|
const size_t uxNeededSize = ipSIZE_OF_ETH_HEADER + ipSIZE_OF_IPv6_HEADER + sizeof( ICMPRouterSolicitation_IPv6_t );
|
|
MACAddress_t xMultiCastMacAddress;
|
|
NetworkBufferDescriptor_t * pxDescriptor = pxNetworkBuffer;
|
|
IPv6_Address_t xSourceAddress;
|
|
BaseType_t xHasLocal;
|
|
|
|
configASSERT( pxEndPoint != NULL );
|
|
|
|
xHasLocal = xGetLinkLocalAddress( pxEndPoint->pxNetworkInterface, &( xSourceAddress ) );
|
|
|
|
if( xHasLocal == pdFAIL )
|
|
{
|
|
FreeRTOS_printf( ( "RA: can not find a Link-local address\n" ) );
|
|
( void ) memset( xSourceAddress.ucBytes, 0, ipSIZE_OF_IPv6_ADDRESS );
|
|
}
|
|
else
|
|
{
|
|
FreeRTOS_printf( ( "RA: source %pip\n", xSourceAddress.ucBytes ) );
|
|
}
|
|
|
|
if( pxDescriptor->xDataLength < uxNeededSize )
|
|
{
|
|
pxDescriptor = pxDuplicateNetworkBufferWithDescriptor( pxDescriptor, uxNeededSize );
|
|
}
|
|
|
|
if( pxDescriptor != NULL )
|
|
{
|
|
/* 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] */
|
|
pxICMPPacket = ( ( ICMPPacket_IPv6_t * ) pxDescriptor->pucEthernetBuffer );
|
|
xRASolicitationRequest = ( ( ICMPRouterSolicitation_IPv6_t * ) &( pxICMPPacket->xICMPHeaderIPv6 ) );
|
|
|
|
pxDescriptor->xDataLength = uxNeededSize;
|
|
|
|
( void ) eNDGetCacheEntry( pxIPAddress, &( xMultiCastMacAddress ), NULL );
|
|
|
|
/* Set Ethernet header. Will be swapped. */
|
|
( void ) memcpy( pxICMPPacket->xEthernetHeader.xSourceAddress.ucBytes, xMultiCastMacAddress.ucBytes, ipMAC_ADDRESS_LENGTH_BYTES );
|
|
( void ) memcpy( pxICMPPacket->xEthernetHeader.xDestinationAddress.ucBytes, pxEndPoint->xMACAddress.ucBytes, ipMAC_ADDRESS_LENGTH_BYTES );
|
|
pxICMPPacket->xEthernetHeader.usFrameType = ipIPv6_FRAME_TYPE;
|
|
|
|
/* Set IP-header. */
|
|
pxICMPPacket->xIPHeader.ucVersionTrafficClass = raDEFAULT_VERSION_TRAFFIC_CLASS;
|
|
pxICMPPacket->xIPHeader.ucTrafficClassFlow = 0U;
|
|
pxICMPPacket->xIPHeader.usFlowLabel = 0U;
|
|
pxICMPPacket->xIPHeader.usPayloadLength = FreeRTOS_htons( sizeof( ICMPRouterSolicitation_IPv6_t ) );
|
|
pxICMPPacket->xIPHeader.ucNextHeader = ipPROTOCOL_ICMP_IPv6;
|
|
pxICMPPacket->xIPHeader.ucHopLimit = raDEFAULT_HOP_LIMIT;
|
|
|
|
configASSERT( pxEndPoint != NULL );
|
|
configASSERT( pxEndPoint->bits.bIPv6 != pdFALSE_UNSIGNED );
|
|
|
|
/* Normally, the source address is set as 'ipv6_settings.xIPAddress'.
|
|
* But is some routers will not accept a public IP-address, the original
|
|
* default address will be used. It must be a link-local address. */
|
|
( void ) memcpy( pxICMPPacket->xIPHeader.xSourceAddress.ucBytes, xSourceAddress.ucBytes, ipSIZE_OF_IPv6_ADDRESS );
|
|
|
|
( void ) memcpy( pxICMPPacket->xIPHeader.xDestinationAddress.ucBytes, pxIPAddress->ucBytes, ipSIZE_OF_IPv6_ADDRESS );
|
|
|
|
/* Set ICMP header. */
|
|
( void ) memset( xRASolicitationRequest, 0, sizeof( *xRASolicitationRequest ) );
|
|
xRASolicitationRequest->ucTypeOfMessage = ipICMP_ROUTER_SOLICITATION_IPv6;
|
|
|
|
/* __XX__ revisit on why commented out
|
|
* xRASolicitationRequest->ucOptionType = ndICMP_SOURCE_LINK_LAYER_ADDRESS;
|
|
* xRASolicitationRequest->ucOptionLength = 1;
|
|
* ( void ) memcpy( xRASolicitationRequest->ucOptionBytes, pxEndPoint->xMACAddress.ucBytes, ipMAC_ADDRESS_LENGTH_BYTES );
|
|
*/
|
|
/* Checksums. */
|
|
xRASolicitationRequest->usChecksum = 0U;
|
|
/* calculate the UDP checksum for outgoing package */
|
|
( void ) usGenerateProtocolChecksum( pxDescriptor->pucEthernetBuffer, pxDescriptor->xDataLength, pdTRUE );
|
|
|
|
/* This function will fill in the eth addresses and send the packet */
|
|
vReturnEthernetFrame( pxDescriptor, pdTRUE );
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Receive a NA ( Neighbour Advertisement ) message to see if a chosen IP-address is already in use.
|
|
*
|
|
* @param[in] pxNetworkBuffer: The buffer that contains the message.
|
|
*/
|
|
void vReceiveNA( const NetworkBufferDescriptor_t * pxNetworkBuffer )
|
|
{
|
|
const NetworkInterface_t * pxInterface = pxNetworkBuffer->pxInterface;
|
|
NetworkEndPoint_t * pxPoint;
|
|
|
|
/* 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] */
|
|
const ICMPPacket_IPv6_t * pxICMPPacket = ( ( const ICMPPacket_IPv6_t * ) pxNetworkBuffer->pucEthernetBuffer );
|
|
const ICMPHeader_IPv6_t * pxICMPHeader_IPv6 = ( ( const ICMPHeader_IPv6_t * ) &( pxICMPPacket->xICMPHeaderIPv6 ) );
|
|
|
|
for( pxPoint = FreeRTOS_FirstEndPoint( pxInterface );
|
|
pxPoint != NULL;
|
|
pxPoint = FreeRTOS_NextEndPoint( pxInterface, pxPoint ) )
|
|
{
|
|
if( ( pxPoint->bits.bWantRA != pdFALSE_UNSIGNED ) && ( pxPoint->xRAData.eRAState == eRAStateIPWait ) )
|
|
{
|
|
if( memcmp( pxPoint->ipv6_settings.xIPAddress.ucBytes, pxICMPHeader_IPv6->xIPv6Address.ucBytes, ipSIZE_OF_IPv6_ADDRESS ) == 0 )
|
|
{
|
|
pxPoint->xRAData.bits.bIPAddressInUse = pdTRUE_UNSIGNED;
|
|
vDHCP_RATimerReload( pxPoint, 100U );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Read a received RA reply and return the prefix option from the packet.
|
|
*
|
|
* @param[in] pxNetworkBuffer: The buffer that contains the message.
|
|
*
|
|
* @returns Returns the ICMP prefix option pointer, pointing to its location in the
|
|
* input RA reply message buffer.
|
|
*/
|
|
static ICMPPrefixOption_IPv6_t * vReceiveRA_ReadReply( const NetworkBufferDescriptor_t * pxNetworkBuffer )
|
|
{
|
|
size_t uxIndex = 0U;
|
|
const size_t uxICMPSize = sizeof( ICMPRouterAdvertisement_IPv6_t );
|
|
const size_t uxNeededSize = ipSIZE_OF_ETH_HEADER + ipSIZE_OF_IPv6_HEADER + uxICMPSize;
|
|
/* uxLast points to the first byte after the buffer. */
|
|
const size_t uxLast = pxNetworkBuffer->xDataLength - uxNeededSize;
|
|
uint8_t * pucBytes = &( pxNetworkBuffer->pucEthernetBuffer[ uxNeededSize ] );
|
|
ICMPPrefixOption_IPv6_t * pxPrefixOption = NULL;
|
|
|
|
while( ( uxIndex + 1U ) < uxLast )
|
|
{
|
|
uint8_t ucType = pucBytes[ uxIndex ];
|
|
size_t uxLength = ( size_t ) pucBytes[ uxIndex + 1U ] * 8U;
|
|
|
|
if( uxLast < ( uxIndex + uxLength ) )
|
|
{
|
|
FreeRTOS_printf( ( "RA: Not enough bytes ( %u > %u )\n", ( unsigned ) ( uxIndex + uxLength ), ( unsigned ) uxLast ) );
|
|
break;
|
|
}
|
|
|
|
switch( ucType )
|
|
{
|
|
case ndICMP_SOURCE_LINK_LAYER_ADDRESS: /* 1 */
|
|
FreeRTOS_printf( ( "RA: Source = %02x-%02x-%02x-%02x-%02x-%02x\n",
|
|
pucBytes[ uxIndex + 2U ],
|
|
pucBytes[ uxIndex + 3U ],
|
|
pucBytes[ uxIndex + 4U ],
|
|
pucBytes[ uxIndex + 5U ],
|
|
pucBytes[ uxIndex + 6U ],
|
|
pucBytes[ uxIndex + 7U ] ) );
|
|
break;
|
|
|
|
case ndICMP_TARGET_LINK_LAYER_ADDRESS: /* 2 */
|
|
break;
|
|
|
|
case ndICMP_PREFIX_INFORMATION: /* 3 */
|
|
/* 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] */
|
|
pxPrefixOption = ( ( ICMPPrefixOption_IPv6_t * ) &( pucBytes[ uxIndex ] ) );
|
|
|
|
FreeRTOS_printf( ( "RA: Prefix len %d Life %u, %u (%pip)\n",
|
|
pxPrefixOption->ucPrefixLength,
|
|
FreeRTOS_ntohl( pxPrefixOption->ulValidLifeTime ),
|
|
FreeRTOS_ntohl( pxPrefixOption->ulPreferredLifeTime ),
|
|
pxPrefixOption->ucPrefix ) );
|
|
break;
|
|
|
|
case ndICMP_REDIRECTED_HEADER: /* 4 */
|
|
break;
|
|
|
|
case ndICMP_MTU_OPTION: /* 5 */
|
|
{
|
|
uint32_t ulMTU;
|
|
|
|
/* ulChar2u32 returns host-endian numbers. */
|
|
ulMTU = ulChar2u32( &( pucBytes[ uxIndex + 4U ] ) );
|
|
FreeRTOS_printf( ( "RA: MTU = %u\n", ( unsigned int ) ulMTU ) );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
FreeRTOS_printf( ( "RA: Type 0x%02x not implemented\n", ucType ) );
|
|
break;
|
|
}
|
|
|
|
uxIndex = uxIndex + uxLength;
|
|
} /* while( ( uxIndex + 1 ) < uxLast ) */
|
|
|
|
return pxPrefixOption;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Receive and analyse a RA ( Router Advertisement ) message.
|
|
* If the reply is satisfactory, the end-point will do SLAAC: choose an IP-address using the
|
|
* prefix offered, and completed with random bits. It will start testing if another device
|
|
* already exists that uses the same IP-address.
|
|
*
|
|
* @param[in] pxNetworkBuffer: The buffer that contains the message.
|
|
*/
|
|
void vReceiveRA( const NetworkBufferDescriptor_t * pxNetworkBuffer )
|
|
{
|
|
/* 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] */
|
|
const ICMPPacket_IPv6_t * pxICMPPacket = ( ( const ICMPPacket_IPv6_t * ) pxNetworkBuffer->pucEthernetBuffer );
|
|
const ICMPPrefixOption_IPv6_t * pxPrefixOption = NULL;
|
|
const size_t uxICMPSize = sizeof( ICMPRouterAdvertisement_IPv6_t );
|
|
const size_t uxNeededSize = ipSIZE_OF_ETH_HEADER + ipSIZE_OF_IPv6_HEADER + uxICMPSize;
|
|
|
|
/* A Router Advertisement was received, handle it here. */
|
|
if( uxNeededSize > pxNetworkBuffer->xDataLength )
|
|
{
|
|
FreeRTOS_printf( ( "vReceiveRA: The buffer provided is too small\n" ) );
|
|
}
|
|
else
|
|
{
|
|
/* 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] */
|
|
const ICMPRouterAdvertisement_IPv6_t * pxAdvertisement = ( ( const ICMPRouterAdvertisement_IPv6_t * ) &( pxICMPPacket->xICMPHeaderIPv6 ) );
|
|
FreeRTOS_printf( ( "RA: Type %02x Srv %02x Checksum %04x Hops %d Flags %02x Life %d\n",
|
|
pxAdvertisement->ucTypeOfMessage,
|
|
pxAdvertisement->ucTypeOfService,
|
|
FreeRTOS_ntohs( pxAdvertisement->usChecksum ),
|
|
pxAdvertisement->ucHopLimit,
|
|
pxAdvertisement->ucFlags,
|
|
FreeRTOS_ntohs( pxAdvertisement->usLifetime ) ) );
|
|
|
|
if( pxAdvertisement->usLifetime != 0U )
|
|
{
|
|
pxPrefixOption = vReceiveRA_ReadReply( pxNetworkBuffer );
|
|
|
|
configASSERT( pxNetworkBuffer->pxInterface != NULL );
|
|
|
|
if( pxPrefixOption != NULL )
|
|
{
|
|
NetworkEndPoint_t * pxEndPoint;
|
|
|
|
for( pxEndPoint = FreeRTOS_FirstEndPoint( pxNetworkBuffer->pxInterface );
|
|
pxEndPoint != NULL;
|
|
pxEndPoint = FreeRTOS_NextEndPoint( pxNetworkBuffer->pxInterface, pxEndPoint ) )
|
|
{
|
|
if( ( pxEndPoint->bits.bWantRA != pdFALSE_UNSIGNED ) && ( pxEndPoint->xRAData.eRAState == eRAStateWait ) )
|
|
{
|
|
pxEndPoint->ipv6_settings.uxPrefixLength = pxPrefixOption->ucPrefixLength;
|
|
( void ) memcpy( pxEndPoint->ipv6_settings.xPrefix.ucBytes, pxPrefixOption->ucPrefix, ipSIZE_OF_IPv6_ADDRESS );
|
|
( void ) memcpy( pxEndPoint->ipv6_settings.xGatewayAddress.ucBytes, pxICMPPacket->xIPHeader.xSourceAddress.ucBytes, ipSIZE_OF_IPv6_ADDRESS );
|
|
|
|
pxEndPoint->xRAData.bits.bRouterReplied = pdTRUE_UNSIGNED;
|
|
pxEndPoint->xRAData.uxRetryCount = 0U;
|
|
pxEndPoint->xRAData.ulPreferredLifeTime = FreeRTOS_ntohl( pxPrefixOption->ulPreferredLifeTime );
|
|
/* Force taking a new random IP-address. */
|
|
pxEndPoint->xRAData.bits.bIPAddressInUse = pdTRUE_UNSIGNED;
|
|
pxEndPoint->xRAData.eRAState = eRAStateIPTest;
|
|
vRAProcess( pdFALSE, pxEndPoint );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* The life-time field contains zero. */
|
|
}
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief This is an option to test SLAAC. This device will take the IP-address of a
|
|
* known device in the LAN, just to simulate a IP-address clash.
|
|
*
|
|
* @param[in] xIndex: the index to be used in the list of IP-addresses.
|
|
* @param[out] pxIPAddress: Here the IP-address will be written.
|
|
*
|
|
* @return pdPASS if an existing IP-address has been found and written to pxIPAddress.
|
|
*/
|
|
static BaseType_t prvGetTestAddress( BaseType_t xIndex,
|
|
const IPv6_Address_t * pxIPAddress )
|
|
{
|
|
( void ) xIndex;
|
|
( void ) pxIPAddress;
|
|
return 0;
|
|
|
|
#if 0
|
|
BaseType_t xResult;
|
|
|
|
/* For testing only: return an IPv6 address that is already taken in the LAN. */
|
|
const char * ip_address[] =
|
|
{
|
|
"fe80::6816:5e9b:80a0:9edb", /* laptop _HT_ */
|
|
"fe80::9355:69c7:585a:afe7", /* raspberry */
|
|
};
|
|
|
|
if( xIndex < ARRAY_SIZE_X( ip_address ) )
|
|
{
|
|
( void ) FreeRTOS_inet_pton6( ip_address[ xIndex ], pxIPAddress->ucBytes );
|
|
xResult = pdPASS;
|
|
}
|
|
else
|
|
{
|
|
xResult = pdFAIL;
|
|
}
|
|
return xResult;
|
|
#endif /* 0 */
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Handles the RA wait state and calculates the new timer reload value
|
|
* based on the wait state. Also checks if any timer has expired. If its found that
|
|
* there is no other device using the same IP-address vIPNetworkUpCalls() is called
|
|
* to send the network up event.
|
|
*
|
|
* @param[in] pxEndPoint: The end point for which RA assignment is required.
|
|
* @param[out] uxReloadTime: Timer reload value in ticks.
|
|
*
|
|
* @return New timer reload value.
|
|
*/
|
|
static TickType_t xRAProcess_HandleWaitStates( NetworkEndPoint_t * pxEndPoint,
|
|
TickType_t uxReloadTime )
|
|
{
|
|
TickType_t uxNewReloadTime = uxReloadTime;
|
|
|
|
if( pxEndPoint->xRAData.eRAState == eRAStateWait )
|
|
{
|
|
/* A Router Solicitation has been sent, waited for a reply, but no came.
|
|
* All replies will be handled in the function vReceiveRA(). */
|
|
pxEndPoint->xRAData.uxRetryCount++;
|
|
|
|
if( pxEndPoint->xRAData.uxRetryCount < ( UBaseType_t ) ipconfigRA_SEARCH_COUNT )
|
|
{
|
|
pxEndPoint->xRAData.eRAState = eRAStateApply;
|
|
}
|
|
else
|
|
{
|
|
FreeRTOS_printf( ( "RA: Giving up waiting for a Router.\n" ) );
|
|
( void ) memcpy( &( pxEndPoint->ipv6_settings ), &( pxEndPoint->ipv6_defaults ), sizeof( pxEndPoint->ipv6_settings ) );
|
|
|
|
pxEndPoint->xRAData.bits.bRouterReplied = pdFALSE_UNSIGNED;
|
|
pxEndPoint->xRAData.uxRetryCount = 0U;
|
|
/* Force taking a new random IP-address. */
|
|
pxEndPoint->xRAData.bits.bIPAddressInUse = pdTRUE_UNSIGNED;
|
|
pxEndPoint->xRAData.eRAState = eRAStateIPTest;
|
|
}
|
|
}
|
|
else if( pxEndPoint->xRAData.eRAState == eRAStateIPWait )
|
|
{
|
|
/* A Neighbour Solicitation has been sent, waited for a reply.
|
|
* Repeat this 'ipconfigRA_IP_TEST_COUNT' times to be sure. */
|
|
if( pxEndPoint->xRAData.bits.bIPAddressInUse != pdFALSE_UNSIGNED )
|
|
{
|
|
/* Another device has responded with the same IPv4 address. */
|
|
pxEndPoint->xRAData.uxRetryCount = 0U;
|
|
pxEndPoint->xRAData.eRAState = eRAStateIPTest;
|
|
uxNewReloadTime = pdMS_TO_TICKS( ipconfigRA_IP_TEST_TIME_OUT_MSEC );
|
|
}
|
|
else if( pxEndPoint->xRAData.uxRetryCount < ( UBaseType_t ) ipconfigRA_IP_TEST_COUNT )
|
|
{
|
|
/* Try again. */
|
|
pxEndPoint->xRAData.uxRetryCount++;
|
|
pxEndPoint->xRAData.eRAState = eRAStateIPTest;
|
|
uxNewReloadTime = pdMS_TO_TICKS( ipconfigRA_IP_TEST_TIME_OUT_MSEC );
|
|
}
|
|
else
|
|
{
|
|
/* Now it is assumed that there is no other device using the same IP-address. */
|
|
if( pxEndPoint->xRAData.bits.bRouterReplied != pdFALSE_UNSIGNED )
|
|
{
|
|
/* Obtained configuration from a router. */
|
|
uxNewReloadTime = pdMS_TO_TICKS( 1000U * pxEndPoint->xRAData.ulPreferredLifeTime );
|
|
pxEndPoint->xRAData.eRAState = eRAStatePreLease;
|
|
iptraceRA_SUCCEDEED( &( pxEndPoint->ipv6_settings.xIPAddress ) );
|
|
FreeRTOS_printf( ( "RA: succeeded, using IP address %pip Reload after %u seconds\n",
|
|
pxEndPoint->ipv6_settings.xIPAddress.ucBytes,
|
|
( unsigned ) pxEndPoint->xRAData.ulPreferredLifeTime ) );
|
|
}
|
|
else
|
|
{
|
|
/* Using the default network parameters. */
|
|
pxEndPoint->xRAData.eRAState = eRAStateFailed;
|
|
|
|
iptraceRA_REQUESTS_FAILED_USING_DEFAULT_IP_ADDRESS( &( pxEndPoint->ipv6_settings.xIPAddress ) );
|
|
|
|
FreeRTOS_printf( ( "RA: failed, using default parameters and IP address %pip\n", pxEndPoint->ipv6_settings.xIPAddress.ucBytes ) );
|
|
/* Disable the timer. */
|
|
uxNewReloadTime = 0U;
|
|
}
|
|
|
|
/* Now call vIPNetworkUpCalls() to send the network-up event and
|
|
* start the ARP timer. */
|
|
vIPNetworkUpCalls( pxEndPoint );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Do nothing */
|
|
}
|
|
|
|
return uxNewReloadTime;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Handles the RA states other than the wait states.
|
|
*
|
|
* @param[in] pxEndPoint: The end point for which RA assignment is required.
|
|
* @param[out] uxReloadTime: Timer reload value in ticks.
|
|
*
|
|
* @return New timer reload value.
|
|
*/
|
|
static TickType_t xRAProcess_HandleOtherStates( NetworkEndPoint_t * pxEndPoint,
|
|
TickType_t uxReloadTime )
|
|
{
|
|
TickType_t uxNewReloadTime = uxReloadTime;
|
|
|
|
switch( pxEndPoint->xRAData.eRAState )
|
|
{
|
|
case eRAStateApply:
|
|
{
|
|
IPv6_Address_t xIPAddress;
|
|
size_t uxNeededSize;
|
|
NetworkBufferDescriptor_t * pxNetworkBuffer;
|
|
|
|
/* Send a Router Solicitation to ff02::2 */
|
|
( void ) memset( xIPAddress.ucBytes, 0, sizeof( xIPAddress.ucBytes ) );
|
|
xIPAddress.ucBytes[ 0 ] = 0xffU;
|
|
xIPAddress.ucBytes[ 1 ] = 0x02U;
|
|
xIPAddress.ucBytes[ 15 ] = 0x02U;
|
|
uxNeededSize = ipSIZE_OF_ETH_HEADER + ipSIZE_OF_IPv6_HEADER + sizeof( ICMPRouterSolicitation_IPv6_t );
|
|
pxNetworkBuffer = pxGetNetworkBufferWithDescriptor( uxNeededSize, raDONT_BLOCK );
|
|
|
|
if( pxNetworkBuffer != NULL )
|
|
{
|
|
pxNetworkBuffer->pxEndPoint = pxEndPoint;
|
|
vNDSendRouterSolicitation( pxNetworkBuffer, &( xIPAddress ) );
|
|
}
|
|
|
|
FreeRTOS_printf( ( "vRAProcess: Router Solicitation, attempt %lu/%u\n",
|
|
pxEndPoint->xRAData.uxRetryCount + 1U,
|
|
ipconfigRA_SEARCH_COUNT ) );
|
|
/* Wait a configurable time for a router advertisement. */
|
|
uxNewReloadTime = pdMS_TO_TICKS( ipconfigRA_SEARCH_TIME_OUT_MSEC );
|
|
pxEndPoint->xRAData.eRAState = eRAStateWait;
|
|
}
|
|
break;
|
|
|
|
case eRAStateWait:
|
|
/* Waiting for a router advertisement. */
|
|
/* Handled here above. */
|
|
break;
|
|
|
|
case eRAStateIPTest: /* Assuming an IP address, test if another device is using it already. */
|
|
{
|
|
size_t uxNeededSize;
|
|
NetworkBufferDescriptor_t * pxNetworkBuffer;
|
|
|
|
/* Get an IP-address, using the network prefix and a random host address. */
|
|
if( pxEndPoint->xRAData.bits.bIPAddressInUse != 0U )
|
|
{
|
|
static BaseType_t xUseIndex = 0;
|
|
|
|
pxEndPoint->xRAData.bits.bIPAddressInUse = pdFALSE_UNSIGNED;
|
|
|
|
if( prvGetTestAddress( xUseIndex, &( pxEndPoint->ipv6_settings.xIPAddress ) ) == pdPASS )
|
|
{
|
|
/* TESTING ONLY */
|
|
xUseIndex++;
|
|
}
|
|
else
|
|
{
|
|
( void ) FreeRTOS_CreateIPv6Address( &pxEndPoint->ipv6_settings.xIPAddress, &pxEndPoint->ipv6_settings.xPrefix, pxEndPoint->ipv6_settings.uxPrefixLength, pdTRUE );
|
|
}
|
|
|
|
FreeRTOS_printf( ( "RA: Creating a random IP-address\n" ) );
|
|
}
|
|
|
|
FreeRTOS_printf( ( "RA: Neighbour solicitation for %pip\n", pxEndPoint->ipv6_settings.xIPAddress.ucBytes ) );
|
|
|
|
uxNeededSize = ipSIZE_OF_ETH_HEADER + ipSIZE_OF_IPv6_HEADER + sizeof( ICMPHeader_IPv6_t );
|
|
pxNetworkBuffer = pxGetNetworkBufferWithDescriptor( uxNeededSize, raDONT_BLOCK );
|
|
|
|
if( pxNetworkBuffer != NULL )
|
|
{
|
|
pxNetworkBuffer->pxEndPoint = pxEndPoint;
|
|
vNDSendNeighbourSolicitation( pxNetworkBuffer, &( pxEndPoint->ipv6_settings.xIPAddress ) );
|
|
}
|
|
|
|
uxNewReloadTime = pdMS_TO_TICKS( 1000U );
|
|
pxEndPoint->xRAData.eRAState = eRAStateIPWait;
|
|
}
|
|
break;
|
|
|
|
case eRAStateIPWait:
|
|
/* Assuming an IP address, test if another device is using it already. */
|
|
/* Handled here above. */
|
|
break;
|
|
|
|
case eRAStatePreLease:
|
|
pxEndPoint->xRAData.eRAState = eRAStateLease;
|
|
break;
|
|
|
|
case eRAStateLease:
|
|
|
|
vRAProcessInit( pxEndPoint );
|
|
uxNewReloadTime = pdMS_TO_TICKS( 1000U );
|
|
|
|
break;
|
|
|
|
case eRAStateFailed:
|
|
break;
|
|
|
|
default:
|
|
/* All states were handled. */
|
|
break;
|
|
}
|
|
|
|
return uxNewReloadTime;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Initialise the RA state machine.
|
|
*
|
|
* @param[in] pxEndPoint: The end-point for which Router Advertisement is required.
|
|
*/
|
|
static void vRAProcessInit( NetworkEndPoint_t * pxEndPoint )
|
|
{
|
|
pxEndPoint->xRAData.uxRetryCount = 0U;
|
|
pxEndPoint->xRAData.eRAState = eRAStateApply;
|
|
}
|
|
|
|
/**
|
|
* @brief Do a single cycle of the RA state machine.
|
|
*
|
|
* @param[in] xDoReset: pdTRUE if the state machine must be reset.
|
|
* @param[in] pxEndPoint: The end-point for which a RA assignment is required.
|
|
*/
|
|
void vRAProcess( BaseType_t xDoReset,
|
|
NetworkEndPoint_t * pxEndPoint )
|
|
{
|
|
TickType_t uxReloadTime = pdMS_TO_TICKS( 5000U );
|
|
|
|
#if ( ipconfigHAS_PRINTF == 1 )
|
|
/* Remember the initial state, just for logging. */
|
|
eRAState_t eRAState = pxEndPoint->xRAData.eRAState;
|
|
#endif
|
|
|
|
configASSERT( pxEndPoint != NULL );
|
|
|
|
if( xDoReset != pdFALSE )
|
|
{
|
|
vRAProcessInit( pxEndPoint );
|
|
}
|
|
|
|
/* First handle the states that are limited by a timer. See if some
|
|
* timer has expired. */
|
|
uxReloadTime = xRAProcess_HandleWaitStates( pxEndPoint, uxReloadTime );
|
|
|
|
/* Now handle the other states. */
|
|
uxReloadTime = xRAProcess_HandleOtherStates( pxEndPoint, uxReloadTime );
|
|
|
|
#if ( ipconfigHAS_PRINTF == 1 )
|
|
{
|
|
FreeRTOS_printf( ( "vRAProcess( %ld, %pip) bRouterReplied=%d bIPAddressInUse=%d state %d -> %d\n",
|
|
xDoReset,
|
|
pxEndPoint->ipv6_defaults.xIPAddress.ucBytes,
|
|
pxEndPoint->xRAData.bits.bRouterReplied,
|
|
pxEndPoint->xRAData.bits.bIPAddressInUse,
|
|
eRAState,
|
|
pxEndPoint->xRAData.eRAState ) );
|
|
}
|
|
#endif /* ( ipconfigHAS_PRINTF == 1 ) */
|
|
|
|
if( uxReloadTime != 0U )
|
|
{
|
|
FreeRTOS_printf( ( "RA: Reload %u seconds\n", ( unsigned ) ( uxReloadTime / 1000U ) ) );
|
|
vDHCP_RATimerReload( pxEndPoint, uxReloadTime );
|
|
}
|
|
else
|
|
{
|
|
/* Disable the timer, this function vRAProcess() won't be called anymore for this end-point. */
|
|
FreeRTOS_printf( ( "RA: Disabled timer.\n" ) );
|
|
vIPSetDHCP_RATimerEnableState( pxEndPoint, pdFALSE );
|
|
}
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
#endif /* ( ipconfigUSE_IPv6 != 0 ) && ( ipconfigUSE_RA != 0 ) */
|