mirror of
				https://github.com/FreeRTOS/FreeRTOS-Plus-TCP
				synced 2025-10-25 04:56:15 +08:00 
			
		
		
		
	 dbd2b001b1
			
		
	
	dbd2b001b1
	
	
	
		
			
			* Rename FreeRTOS_IPStart to FreeRTOS_IPInit_Multi * Update backward compatability for portable layer
		
			
				
	
	
		
			246 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			246 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Copyright 2018 Espressif Systems (Shanghai) PTE LTD */
 | |
| /* */
 | |
| /* Licensed under the Apache License, Version 2.0 (the "License"); */
 | |
| /* you may not use this file except in compliance with the License. */
 | |
| /* You may obtain a copy of the License at */
 | |
| /* */
 | |
| /*     http://www.apache.org/licenses/LICENSE-2.0 */
 | |
| /* */
 | |
| /* Unless required by applicable law or agreed to in writing, software */
 | |
| /* distributed under the License is distributed on an "AS IS" BASIS, */
 | |
| /* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */
 | |
| /* See the License for the specific language governing permissions and */
 | |
| /* limitations under the License. */
 | |
| 
 | |
| /* Standard includes. */
 | |
| #include <stdint.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.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 "FreeRTOS_DNS.h"
 | |
| #include "NetworkBufferManagement.h"
 | |
| #include "NetworkInterface.h"
 | |
| 
 | |
| #include "esp_log.h"
 | |
| #include "esp_wifi.h"
 | |
| #include "esp_wifi_internal.h"
 | |
| #include "tcpip_adapter.h"
 | |
| 
 | |
| enum if_state_t
 | |
| {
 | |
|     INTERFACE_DOWN = 0,
 | |
|     INTERFACE_UP,
 | |
| };
 | |
| 
 | |
| static const char * TAG = "NetInterface";
 | |
| volatile static uint32_t xInterfaceState = INTERFACE_DOWN;
 | |
| 
 | |
| static NetworkInterface_t * pxMyInterface;
 | |
| 
 | |
| static BaseType_t xESP32_Eth_NetworkInterfaceInitialise( NetworkInterface_t * pxInterface );
 | |
| 
 | |
| static BaseType_t xESP32_Eth_NetworkInterfaceOutput( NetworkInterface_t * pxInterface,
 | |
|                                                      NetworkBufferDescriptor_t * const pxDescriptor,
 | |
|                                                      BaseType_t xReleaseAfterSend );
 | |
| 
 | |
| static BaseType_t xESP32_Eth_GetPhyLinkStatus( NetworkInterface_t * pxInterface );
 | |
| 
 | |
| NetworkInterface_t * pxESP32_Eth_FillInterfaceDescriptor( BaseType_t xEMACIndex,
 | |
|                                                           NetworkInterface_t * pxInterface );
 | |
| 
 | |
| /*-----------------------------------------------------------*/
 | |
| 
 | |
| #if defined( ipconfigIPv4_BACKWARD_COMPATIBLE ) && ( ipconfigIPv4_BACKWARD_COMPATIBLE == 1 )
 | |
| 
 | |
| /* Do not call the following function directly. It is there for downward compatibility.
 | |
|  * The function FreeRTOS_IPInit() will call it to initialice the interface and end-point
 | |
|  * objects.  See the description in FreeRTOS_Routing.h. */
 | |
|     NetworkInterface_t * pxFillInterfaceDescriptor( BaseType_t xEMACIndex,
 | |
|                                                     NetworkInterface_t * pxInterface )
 | |
|     {
 | |
|         pxESP32_Eth_FillInterfaceDescriptor( xEMACIndex, pxInterface );
 | |
|     }
 | |
| 
 | |
| #endif
 | |
| /*-----------------------------------------------------------*/
 | |
| 
 | |
| 
 | |
| NetworkInterface_t * pxESP32_Eth_FillInterfaceDescriptor( BaseType_t xEMACIndex,
 | |
|                                                           NetworkInterface_t * pxInterface )
 | |
| {
 | |
|     static char pcName[ 8 ];
 | |
| 
 | |
| /* This function pxESP32_Eth_FillInterfaceDescriptor() adds a network-interface.
 | |
|  * Make sure that the object pointed to by 'pxInterface'
 | |
|  * is declared static or global, and that it will remain to exist. */
 | |
| 
 | |
|     snprintf( pcName, sizeof( pcName ), "eth%ld", xEMACIndex );
 | |
| 
 | |
|     memset( pxInterface, '\0', sizeof( *pxInterface ) );
 | |
|     pxInterface->pcName = pcName;                    /* Just for logging, debugging. */
 | |
|     pxInterface->pvArgument = ( void * ) xEMACIndex; /* Has only meaning for the driver functions. */
 | |
|     pxInterface->pfInitialise = xESP32_Eth_NetworkInterfaceInitialise;
 | |
|     pxInterface->pfOutput = xESP32_Eth_NetworkInterfaceOutput;
 | |
|     pxInterface->pfGetPhyLinkStatus = xESP32_Eth_GetPhyLinkStatus;
 | |
| 
 | |
|     FreeRTOS_AddNetworkInterface( pxInterface );
 | |
|     pxMyInterface = pxInterface;
 | |
| 
 | |
|     return pxInterface;
 | |
| }
 | |
| /*-----------------------------------------------------------*/
 | |
| 
 | |
| static BaseType_t xESP32_Eth_NetworkInterfaceInitialise( NetworkInterface_t * pxInterface )
 | |
| {
 | |
|     static BaseType_t xMACAdrInitialized = pdFALSE;
 | |
|     uint8_t ucMACAddress[ ipMAC_ADDRESS_LENGTH_BYTES ];
 | |
| 
 | |
|     if( xInterfaceState == INTERFACE_UP )
 | |
|     {
 | |
|         if( xMACAdrInitialized == pdFALSE )
 | |
|         {
 | |
|             esp_wifi_get_mac( ESP_IF_WIFI_STA, ucMACAddress );
 | |
|             FreeRTOS_UpdateMACAddress( ucMACAddress );
 | |
|             xMACAdrInitialized = pdTRUE;
 | |
|         }
 | |
| 
 | |
|         return pdTRUE;
 | |
|     }
 | |
| 
 | |
|     return pdFALSE;
 | |
| }
 | |
| 
 | |
| static BaseType_t xESP32_Eth_GetPhyLinkStatus( NetworkInterface_t * pxInterface )
 | |
| {
 | |
|     BaseType_t xResult = pdFALSE;
 | |
| 
 | |
|     if( xInterfaceState == INTERFACE_UP )
 | |
|     {
 | |
|         xResult = pdTRUE;
 | |
|     }
 | |
| 
 | |
|     return xResult;
 | |
| }
 | |
| 
 | |
| static BaseType_t xESP32_Eth_NetworkInterfaceOutput( NetworkInterface_t * pxInterface,
 | |
|                                                      NetworkBufferDescriptor_t * const pxDescriptor,
 | |
|                                                      BaseType_t xReleaseAfterSend )
 | |
| {
 | |
|     if( ( pxNetworkBuffer == NULL ) || ( pxNetworkBuffer->pucEthernetBuffer == NULL ) || ( pxNetworkBuffer->xDataLength == 0 ) )
 | |
|     {
 | |
|         ESP_LOGE( TAG, "Invalid params" );
 | |
|         return pdFALSE;
 | |
|     }
 | |
| 
 | |
|     esp_err_t ret;
 | |
| 
 | |
|     if( xInterfaceState == INTERFACE_DOWN )
 | |
|     {
 | |
|         ESP_LOGD( TAG, "Interface down" );
 | |
|         ret = ESP_FAIL;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ret = esp_wifi_internal_tx( ESP_IF_WIFI_STA, pxNetworkBuffer->pucEthernetBuffer, pxNetworkBuffer->xDataLength );
 | |
| 
 | |
|         if( ret != ESP_OK )
 | |
|         {
 | |
|             ESP_LOGE( TAG, "Failed to tx buffer %p, len %d, err %d", pxNetworkBuffer->pucEthernetBuffer, pxNetworkBuffer->xDataLength, ret );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #if ( ipconfigHAS_PRINTF != 0 )
 | |
|         {
 | |
|             /* Call a function that monitors resources: the amount of free network
 | |
|              * buffers and the amount of free space on the heap.  See FreeRTOS_IP.c
 | |
|              * for more detailed comments. */
 | |
|             vPrintResourceStats();
 | |
|         }
 | |
|     #endif /* ( ipconfigHAS_PRINTF != 0 ) */
 | |
| 
 | |
|     if( xReleaseAfterSend == pdTRUE )
 | |
|     {
 | |
|         vReleaseNetworkBufferAndDescriptor( pxNetworkBuffer );
 | |
|     }
 | |
| 
 | |
|     return ret == ESP_OK ? pdTRUE : pdFALSE;
 | |
| }
 | |
| 
 | |
| void vNetworkNotifyIFDown()
 | |
| {
 | |
|     IPStackEvent_t xRxEvent = { eNetworkDownEvent, NULL };
 | |
| 
 | |
|     if( xInterfaceState != INTERFACE_DOWN )
 | |
|     {
 | |
|         xInterfaceState = INTERFACE_DOWN;
 | |
|         xSendEventStructToIPTask( &xRxEvent, 0 );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void vNetworkNotifyIFUp()
 | |
| {
 | |
|     xInterfaceState = INTERFACE_UP;
 | |
| }
 | |
| 
 | |
| esp_err_t wlanif_input( void * netif,
 | |
|                         void * buffer,
 | |
|                         uint16_t len,
 | |
|                         void * eb )
 | |
| {
 | |
|     NetworkBufferDescriptor_t * pxNetworkBuffer;
 | |
|     IPStackEvent_t xRxEvent = { eNetworkRxEvent, NULL };
 | |
|     const TickType_t xDescriptorWaitTime = pdMS_TO_TICKS( 250 );
 | |
| 
 | |
|     #if ( ipconfigHAS_PRINTF != 0 )
 | |
|         {
 | |
|             vPrintResourceStats();
 | |
|         }
 | |
|     #endif /* ( ipconfigHAS_PRINTF != 0 ) */
 | |
| 
 | |
|     if( eConsiderFrameForProcessing( buffer ) != eProcessBuffer )
 | |
|     {
 | |
|         ESP_LOGD( TAG, "Dropping packet" );
 | |
|         esp_wifi_internal_free_rx_buffer( eb );
 | |
|         return ESP_OK;
 | |
|     }
 | |
| 
 | |
|     pxNetworkBuffer = pxGetNetworkBufferWithDescriptor( len, xDescriptorWaitTime );
 | |
| 
 | |
|     if( pxNetworkBuffer != NULL )
 | |
|     {
 | |
|         /* Set the packet size, in case a larger buffer was returned. */
 | |
|         pxNetworkBuffer->xDataLength = len;
 | |
|         pxNetworkBuffer->pxInterface = pxMyInterface;
 | |
|         pxNetworkBuffer->pxEndPoint = FreeRTOS_MatchingEndpoint( pxMyInterface, pcBuffer );
 | |
| 
 | |
|         /* Copy the packet data. */
 | |
|         memcpy( pxNetworkBuffer->pucEthernetBuffer, buffer, len );
 | |
|         xRxEvent.pvData = ( void * ) pxNetworkBuffer;
 | |
| 
 | |
|         if( xSendEventStructToIPTask( &xRxEvent, xDescriptorWaitTime ) == pdFAIL )
 | |
|         {
 | |
|             ESP_LOGE( TAG, "Failed to enqueue packet to network stack %p, len %d", buffer, len );
 | |
|             vReleaseNetworkBufferAndDescriptor( pxNetworkBuffer );
 | |
|             return ESP_FAIL;
 | |
|         }
 | |
| 
 | |
|         esp_wifi_internal_free_rx_buffer( eb );
 | |
|         return ESP_OK;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ESP_LOGE( TAG, "Failed to get buffer descriptor" );
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
| }
 |