Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: jiaxinwu <jiaxin.wu@intel.com> Reviewed-by: "Leahy, Leroy P" <leroy.p.leahy@intel.com> git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17875 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			1381 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1381 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   Implement the IP4 driver support for the socket layer.
 | |
| 
 | |
|   Copyright (c) 2011 - 2015, Intel Corporation. All rights reserved.<BR>
 | |
|   This program and the accompanying materials are licensed and made available
 | |
|   under the terms and conditions of the BSD License which accompanies this
 | |
|   distribution.  The full text of the license may be found at
 | |
|   http://opensource.org/licenses/bsd-license.php.
 | |
| 
 | |
|   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
 | |
|   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
 | |
| **/
 | |
| #include "Socket.h"
 | |
| 
 | |
| 
 | |
| /** Get the local socket address.
 | |
| 
 | |
|   This routine returns the IPv4 address associated with the local
 | |
|   socket.
 | |
| 
 | |
|   This routine is called by ::EslSocketGetLocalAddress to determine the
 | |
|   network address for the SOCK_RAW socket.
 | |
| 
 | |
|   @param [in] pPort       Address of an ::ESL_PORT structure.
 | |
|   @param [out] pAddress   Network address to receive the local system address
 | |
| **/
 | |
| VOID
 | |
| EslIp4LocalAddressGet (
 | |
|   IN ESL_PORT * pPort,
 | |
|   OUT struct sockaddr * pAddress
 | |
|   )
 | |
| {
 | |
|   struct sockaddr_in * pLocalAddress;
 | |
|   ESL_IP4_CONTEXT * pIp4;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //  Return the local address
 | |
|   pIp4 = &pPort->Context.Ip4;
 | |
|   pLocalAddress = (struct sockaddr_in *)pAddress;
 | |
|   pLocalAddress->sin_family = AF_INET;
 | |
|   CopyMem ( &pLocalAddress->sin_addr,
 | |
|             &pIp4->ModeData.ConfigData.StationAddress.Addr[0],
 | |
|             sizeof ( pLocalAddress->sin_addr ));
 | |
| 
 | |
|   DBG_EXIT ( );
 | |
| }
 | |
| 
 | |
| 
 | |
| /** Set the local port address.
 | |
| 
 | |
|   This routine sets the local port address.
 | |
| 
 | |
|   This support routine is called by ::EslSocketPortAllocate.
 | |
| 
 | |
|   @param [in] pPort       Address of an ESL_PORT structure
 | |
|   @param [in] pSockAddr   Address of a sockaddr structure that contains the
 | |
|                           connection point on the local machine.  An IPv4 address
 | |
|                           of INADDR_ANY specifies that the connection is made to
 | |
|                           all of the network stacks on the platform.  Specifying a
 | |
|                           specific IPv4 address restricts the connection to the
 | |
|                           network stack supporting that address.  Specifying zero
 | |
|                           for the port causes the network layer to assign a port
 | |
|                           number from the dynamic range.  Specifying a specific
 | |
|                           port number causes the network layer to use that port.
 | |
| 
 | |
|   @param [in] bBindTest   TRUE = run bind testing
 | |
| 
 | |
|   @retval EFI_SUCCESS     The operation was successful
 | |
|  **/
 | |
| EFI_STATUS
 | |
| EslIp4LocalAddressSet (
 | |
|   IN ESL_PORT * pPort,
 | |
|   IN CONST struct sockaddr * pSockAddr,
 | |
|   IN BOOLEAN bBindTest
 | |
|   )
 | |
| {
 | |
|   EFI_IP4_CONFIG_DATA * pConfig;
 | |
|   CONST struct sockaddr_in * pIpAddress;
 | |
|   CONST UINT8 * pIpv4Address;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //  Validate the address
 | |
|   pIpAddress = (struct sockaddr_in *)pSockAddr;
 | |
|   if ( INADDR_BROADCAST == pIpAddress->sin_addr.s_addr ) {
 | |
|     //  The local address must not be the broadcast address
 | |
|     Status = EFI_INVALID_PARAMETER;
 | |
|     pPort->pSocket->errno = EADDRNOTAVAIL;
 | |
|   }
 | |
|   else {
 | |
|     Status = EFI_SUCCESS;
 | |
| 
 | |
|     //  Set the local address
 | |
|     pIpAddress = (struct sockaddr_in *)pSockAddr;
 | |
|     pIpv4Address = (UINT8 *)&pIpAddress->sin_addr.s_addr;
 | |
|     pConfig = &pPort->Context.Ip4.ModeData.ConfigData;
 | |
|     pConfig->StationAddress.Addr[0] = pIpv4Address[0];
 | |
|     pConfig->StationAddress.Addr[1] = pIpv4Address[1];
 | |
|     pConfig->StationAddress.Addr[2] = pIpv4Address[2];
 | |
|     pConfig->StationAddress.Addr[3] = pIpv4Address[3];
 | |
| 
 | |
|     //  Determine if the default address is used
 | |
|     pConfig->UseDefaultAddress = (BOOLEAN)( 0 == pIpAddress->sin_addr.s_addr );
 | |
| 
 | |
|     //  Display the local address
 | |
|     DEBUG (( DEBUG_BIND,
 | |
|               "0x%08x: Port, Local IP4 Address: %d.%d.%d.%d\r\n",
 | |
|               pPort,
 | |
|               pConfig->StationAddress.Addr[0],
 | |
|               pConfig->StationAddress.Addr[1],
 | |
|               pConfig->StationAddress.Addr[2],
 | |
|               pConfig->StationAddress.Addr[3]));
 | |
| 
 | |
|     //  Set the subnet mask
 | |
|     if ( pConfig->UseDefaultAddress ) {
 | |
|       pConfig->SubnetMask.Addr[0] = 0;
 | |
|       pConfig->SubnetMask.Addr[1] = 0;
 | |
|       pConfig->SubnetMask.Addr[2] = 0;
 | |
|       pConfig->SubnetMask.Addr[3] = 0;
 | |
|     }
 | |
|     else {
 | |
|       pConfig->SubnetMask.Addr[0] = 0xff;
 | |
|       pConfig->SubnetMask.Addr[1] = ( 128 <= pConfig->StationAddress.Addr[0]) ? 0xff : 0;
 | |
|       pConfig->SubnetMask.Addr[2] = ( 192 <= pConfig->StationAddress.Addr[0]) ? 0xff : 0;
 | |
|       pConfig->SubnetMask.Addr[3] = ( 224 <= pConfig->StationAddress.Addr[0]) ? 0xff : 0;
 | |
|     }
 | |
|   }
 | |
|   //  Return the operation status
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /** Get the option value.
 | |
| 
 | |
|   This routine handles the IPv4 level options.
 | |
| 
 | |
|   The ::EslSocketOptionGet routine calls this routine to retrieve
 | |
|   the IPv4 options one at a time by name.
 | |
| 
 | |
|   @param [in] pSocket           Address of an ::ESL_SOCKET structure
 | |
|   @param [in] OptionName        Name of the option
 | |
|   @param [out] ppOptionData     Buffer to receive address of option value
 | |
|   @param [out] pOptionLength    Buffer to receive the option length
 | |
| 
 | |
|   @retval EFI_SUCCESS - Socket data successfully received
 | |
|  **/
 | |
| EFI_STATUS
 | |
| EslIp4OptionGet (
 | |
|   IN ESL_SOCKET * pSocket,
 | |
|   IN int OptionName,
 | |
|   OUT CONST void ** __restrict ppOptionData,
 | |
|   OUT socklen_t * __restrict pOptionLength
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //  Assume success
 | |
|   pSocket->errno = 0;
 | |
|   Status = EFI_SUCCESS;
 | |
| 
 | |
|   //  Attempt to get the option
 | |
|   switch ( OptionName ) {
 | |
|   default:
 | |
|     //  Option not supported
 | |
|     pSocket->errno = ENOPROTOOPT;
 | |
|     Status = EFI_INVALID_PARAMETER;
 | |
|     break;
 | |
| 
 | |
|   case IP_HDRINCL:
 | |
|     *ppOptionData = (void *)&pSocket->bIncludeHeader;
 | |
|     *pOptionLength = sizeof ( pSocket->bIncludeHeader );
 | |
|     break;
 | |
|   }
 | |
|   //  Return the operation status
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /** Set the option value.
 | |
| 
 | |
|   This routine handles the IPv4 level options.
 | |
| 
 | |
|   The ::EslSocketOptionSet routine calls this routine to adjust
 | |
|   the IPv4 options one at a time by name.
 | |
| 
 | |
|   @param [in] pSocket         Address of an ::ESL_SOCKET structure
 | |
|   @param [in] OptionName      Name of the option
 | |
|   @param [in] pOptionValue    Buffer containing the option value
 | |
|   @param [in] OptionLength    Length of the buffer in bytes
 | |
| 
 | |
|   @retval EFI_SUCCESS - Option successfully set
 | |
|  **/
 | |
| EFI_STATUS
 | |
| EslIp4OptionSet (
 | |
|   IN ESL_SOCKET * pSocket,
 | |
|   IN int OptionName,
 | |
|   IN CONST void * pOptionValue,
 | |
|   IN socklen_t OptionLength
 | |
|   )
 | |
| {
 | |
|   BOOLEAN bTrueFalse;
 | |
|   //socklen_t LengthInBytes;
 | |
|   //UINT8 * pOptionData;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //  Assume success
 | |
|   pSocket->errno = 0;
 | |
|   Status = EFI_SUCCESS;
 | |
| 
 | |
|   //  Determine if the option protocol matches
 | |
|   //LengthInBytes = 0;
 | |
|   //pOptionData = NULL;
 | |
|   switch ( OptionName ) {
 | |
|   default:
 | |
|     //  Protocol level not supported
 | |
|     DEBUG (( DEBUG_INFO | DEBUG_OPTION, "ERROR - Invalid protocol option\r\n" ));
 | |
|     pSocket->errno = ENOTSUP;
 | |
|     Status = EFI_UNSUPPORTED;
 | |
|     break;
 | |
| 
 | |
|   case IP_HDRINCL:
 | |
| 
 | |
|     //  Validate the option length
 | |
|     if ( sizeof ( UINT32 ) == OptionLength ) {
 | |
|       //  Restrict the input to TRUE or FALSE
 | |
|       bTrueFalse = TRUE;
 | |
|       if ( 0 == *(UINT32 *)pOptionValue ) {
 | |
|         bTrueFalse = FALSE;
 | |
|       }
 | |
|       pOptionValue = &bTrueFalse;
 | |
| 
 | |
|       //  Set the option value
 | |
|       //pOptionData = (UINT8 *)&pSocket->bIncludeHeader;
 | |
|       //LengthInBytes = sizeof ( pSocket->bIncludeHeader );
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   //  Return the operation status
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Free a receive packet
 | |
| 
 | |
|   This routine performs the network specific operations necessary
 | |
|   to free a receive packet.
 | |
| 
 | |
|   This routine is called by ::EslSocketPortCloseTxDone to free a
 | |
|   receive packet.
 | |
| 
 | |
|   @param [in] pPacket         Address of an ::ESL_PACKET structure.
 | |
|   @param [in, out] pRxBytes   Address of the count of RX bytes
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| EslIp4PacketFree (
 | |
|   IN ESL_PACKET * pPacket,
 | |
|   IN OUT size_t * pRxBytes
 | |
|   )
 | |
| {
 | |
|   EFI_IP4_RECEIVE_DATA * pRxData;
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Account for the receive bytes
 | |
|   //
 | |
|   pRxData = pPacket->Op.Ip4Rx.pRxData;
 | |
|   *pRxBytes -= pRxData->HeaderLength + pRxData->DataLength;
 | |
| 
 | |
|   //
 | |
|   //  Disconnect the buffer from the packet
 | |
|   //
 | |
|   pPacket->Op.Ip4Rx.pRxData = NULL;
 | |
| 
 | |
|   //
 | |
|   //  Return the buffer to the IP4 driver
 | |
|   //
 | |
|   gBS->SignalEvent ( pRxData->RecycleSignal );
 | |
|   DBG_EXIT ( );
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Initialize the network specific portions of an ::ESL_PORT structure.
 | |
| 
 | |
|   This routine initializes the network specific portions of an
 | |
|   ::ESL_PORT structure for use by the socket.
 | |
| 
 | |
|   This support routine is called by ::EslSocketPortAllocate
 | |
|   to connect the socket with the underlying network adapter
 | |
|   running the IPv4 protocol.
 | |
| 
 | |
|   @param [in] pPort       Address of an ESL_PORT structure
 | |
|   @param [in] DebugFlags  Flags for debug messages
 | |
| 
 | |
|   @retval EFI_SUCCESS - Socket successfully created
 | |
| 
 | |
|  **/
 | |
| EFI_STATUS
 | |
| EslIp4PortAllocate (
 | |
|   IN ESL_PORT * pPort,
 | |
|   IN UINTN DebugFlags
 | |
|   )
 | |
| {
 | |
|   EFI_IP4_CONFIG_DATA * pConfig;
 | |
|   ESL_SOCKET * pSocket;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Initialize the port
 | |
|   //
 | |
|   pSocket = pPort->pSocket;
 | |
|   pSocket->TxPacketOffset = OFFSET_OF ( ESL_PACKET, Op.Ip4Tx.TxData );
 | |
|   pSocket->TxTokenEventOffset = OFFSET_OF ( ESL_IO_MGMT, Token.Ip4Tx.Event );
 | |
|   pSocket->TxTokenOffset = OFFSET_OF ( EFI_IP4_COMPLETION_TOKEN, Packet.TxData );
 | |
| 
 | |
|   //
 | |
|   //  Save the cancel, receive and transmit addresses
 | |
|   //
 | |
|   pPort->pfnConfigure = (PFN_NET_CONFIGURE)pPort->pProtocol.IPv4->Configure;
 | |
|   pPort->pfnRxCancel = (PFN_NET_IO_START)pPort->pProtocol.IPv4->Cancel;
 | |
|   pPort->pfnRxPoll = (PFN_NET_POLL)pPort->pProtocol.IPv4->Poll;
 | |
|   pPort->pfnRxStart = (PFN_NET_IO_START)pPort->pProtocol.IPv4->Receive;
 | |
|   pPort->pfnTxStart = (PFN_NET_IO_START)pPort->pProtocol.IPv4->Transmit;
 | |
| 
 | |
|   //
 | |
|   //  Set the configuration flags
 | |
|   //
 | |
|   pConfig = &pPort->Context.Ip4.ModeData.ConfigData;
 | |
|   pConfig->AcceptIcmpErrors = FALSE;
 | |
|   pConfig->AcceptBroadcast = FALSE;
 | |
|   pConfig->AcceptPromiscuous = FALSE;
 | |
|   pConfig->TypeOfService = 0;
 | |
|   pConfig->TimeToLive = 255;
 | |
|   pConfig->DoNotFragment = FALSE;
 | |
|   pConfig->RawData = FALSE;
 | |
|   pConfig->ReceiveTimeout = 0;
 | |
|   pConfig->TransmitTimeout = 0;
 | |
| 
 | |
|   //
 | |
|   //  Set the default protocol
 | |
|   //
 | |
|   pConfig->DefaultProtocol = (UINT8)pSocket->Protocol;
 | |
|   pConfig->AcceptAnyProtocol = (BOOLEAN)( 0 == pConfig->DefaultProtocol );
 | |
|   Status = EFI_SUCCESS;
 | |
| 
 | |
|   //
 | |
|   //  Return the operation status
 | |
|   //
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Receive data from a network connection.
 | |
| 
 | |
|   This routine attempts to return buffered data to the caller.  The
 | |
|   data is removed from the urgent queue if the message flag MSG_OOB
 | |
|   is specified, otherwise data is removed from the normal queue.
 | |
|   See the \ref ReceiveEngine section.
 | |
| 
 | |
|   This routine is called by ::EslSocketReceive to handle the network
 | |
|   specific receive operation to support SOCK_RAW sockets.
 | |
| 
 | |
|   @param [in] pPort           Address of an ::ESL_PORT structure.
 | |
| 
 | |
|   @param [in] pPacket         Address of an ::ESL_PACKET structure.
 | |
| 
 | |
|   @param [in] pbConsumePacket Address of a BOOLEAN indicating if the packet is to be consumed
 | |
| 
 | |
|   @param [in] BufferLength    Length of the the buffer
 | |
| 
 | |
|   @param [in] pBuffer         Address of a buffer to receive the data.
 | |
| 
 | |
|   @param [in] pDataLength     Number of received data bytes in the buffer.
 | |
| 
 | |
|   @param [out] pAddress       Network address to receive the remote system address
 | |
| 
 | |
|   @param [out] pSkipBytes     Address to receive the number of bytes skipped
 | |
| 
 | |
|   @return   Returns the address of the next free byte in the buffer.
 | |
| 
 | |
|  **/
 | |
| UINT8 *
 | |
| EslIp4Receive (
 | |
|   IN ESL_PORT * pPort,
 | |
|   IN ESL_PACKET * pPacket,
 | |
|   IN BOOLEAN * pbConsumePacket,
 | |
|   IN size_t BufferLength,
 | |
|   IN UINT8 * pBuffer,
 | |
|   OUT size_t * pDataLength,
 | |
|   OUT struct sockaddr * pAddress,
 | |
|   OUT size_t * pSkipBytes
 | |
|   )
 | |
| {
 | |
|   size_t DataBytes;
 | |
|   size_t HeaderBytes;
 | |
|   size_t LengthInBytes;
 | |
|   struct sockaddr_in * pRemoteAddress;
 | |
|   EFI_IP4_RECEIVE_DATA * pRxData;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Return the remote system address if requested
 | |
|   //
 | |
|   pRxData = pPacket->Op.Ip4Rx.pRxData;
 | |
|   if ( NULL != pAddress ) {
 | |
|     //
 | |
|     //  Build the remote address
 | |
|     //
 | |
|     DEBUG (( DEBUG_RX,
 | |
|               "Getting packet remote address: %d.%d.%d.%d\r\n",
 | |
|               pRxData->Header->SourceAddress.Addr[0],
 | |
|               pRxData->Header->SourceAddress.Addr[1],
 | |
|               pRxData->Header->SourceAddress.Addr[2],
 | |
|               pRxData->Header->SourceAddress.Addr[3]));
 | |
|     pRemoteAddress = (struct sockaddr_in *)pAddress;
 | |
|     CopyMem ( &pRemoteAddress->sin_addr,
 | |
|               &pRxData->Header->SourceAddress.Addr[0],
 | |
|               sizeof ( pRemoteAddress->sin_addr ));
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Copy the IP header
 | |
|   //
 | |
|   HeaderBytes = pRxData->HeaderLength;
 | |
|   if ( HeaderBytes > BufferLength ) {
 | |
|     HeaderBytes = BufferLength;
 | |
|   }
 | |
|   DEBUG (( DEBUG_RX,
 | |
|             "0x%08x --> 0x%08x: Copy header 0x%08x bytes\r\n",
 | |
|             pRxData->Header,
 | |
|             pBuffer,
 | |
|             HeaderBytes ));
 | |
|   CopyMem ( pBuffer, pRxData->Header, HeaderBytes );
 | |
|   pBuffer += HeaderBytes;
 | |
|   LengthInBytes = HeaderBytes;
 | |
| 
 | |
|   //
 | |
|   //  Copy the received data
 | |
|   //
 | |
|   if ( 0 < ( BufferLength - LengthInBytes )) {
 | |
|     pBuffer = EslSocketCopyFragmentedBuffer ( pRxData->FragmentCount,
 | |
|                                               &pRxData->FragmentTable[0],
 | |
|                                               BufferLength - LengthInBytes,
 | |
|                                               pBuffer,
 | |
|                                               &DataBytes );
 | |
|     LengthInBytes += DataBytes;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Determine if the data is being read
 | |
|   //
 | |
|   if ( *pbConsumePacket ) {
 | |
|     //
 | |
|     //  Display for the bytes consumed
 | |
|     //
 | |
|     DEBUG (( DEBUG_RX,
 | |
|               "0x%08x: Port account for 0x%08x bytes\r\n",
 | |
|               pPort,
 | |
|               LengthInBytes ));
 | |
| 
 | |
|     //
 | |
|     //  Account for any discarded data
 | |
|     //
 | |
|     *pSkipBytes = pRxData->HeaderLength + pRxData->DataLength - LengthInBytes;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Return the data length and the buffer address
 | |
|   //
 | |
|   *pDataLength = LengthInBytes;
 | |
|   DBG_EXIT_HEX ( pBuffer );
 | |
|   return pBuffer;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Get the remote socket address
 | |
| 
 | |
|   This routine returns the address of the remote connection point
 | |
|   associated with the SOCK_RAW socket.
 | |
| 
 | |
|   This routine is called by ::EslSocketGetPeerAddress to detemine
 | |
|   the IPv4 address associated with the network adapter.
 | |
| 
 | |
|   @param [in] pPort       Address of an ::ESL_PORT structure.
 | |
| 
 | |
|   @param [out] pAddress   Network address to receive the remote system address
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| EslIp4RemoteAddressGet (
 | |
|   IN ESL_PORT * pPort,
 | |
|   OUT struct sockaddr * pAddress
 | |
|   )
 | |
| {
 | |
|   struct sockaddr_in * pRemoteAddress;
 | |
|   ESL_IP4_CONTEXT * pIp4;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Return the remote address
 | |
|   //
 | |
|   pIp4 = &pPort->Context.Ip4;
 | |
|   pRemoteAddress = (struct sockaddr_in *)pAddress;
 | |
|   pRemoteAddress->sin_family = AF_INET;
 | |
|   CopyMem ( &pRemoteAddress->sin_addr,
 | |
|             &pIp4->DestinationAddress.Addr[0],
 | |
|             sizeof ( pRemoteAddress->sin_addr ));
 | |
| 
 | |
|   DBG_EXIT ( );
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Set the remote address
 | |
| 
 | |
|   This routine sets the remote address in the port.
 | |
| 
 | |
|   This routine is called by ::EslSocketConnect to specify the
 | |
|   remote network address.
 | |
| 
 | |
|   @param [in] pPort           Address of an ::ESL_PORT structure.
 | |
| 
 | |
|   @param [in] pSockAddr       Network address of the remote system.
 | |
| 
 | |
|   @param [in] SockAddrLength  Length in bytes of the network address.
 | |
| 
 | |
|   @retval EFI_SUCCESS     The operation was successful
 | |
| 
 | |
|  **/
 | |
| EFI_STATUS
 | |
| EslIp4RemoteAddressSet (
 | |
|   IN ESL_PORT * pPort,
 | |
|   IN CONST struct sockaddr * pSockAddr,
 | |
|   IN socklen_t SockAddrLength
 | |
|   )
 | |
| {
 | |
|   ESL_IP4_CONTEXT * pIp4;
 | |
|   CONST struct sockaddr_in * pRemoteAddress;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Set the remote address
 | |
|   //
 | |
|   pIp4 = &pPort->Context.Ip4;
 | |
|   pRemoteAddress = (struct sockaddr_in *)pSockAddr;
 | |
|   pIp4->DestinationAddress.Addr[0] = (UINT8)( pRemoteAddress->sin_addr.s_addr );
 | |
|   pIp4->DestinationAddress.Addr[1] = (UINT8)( pRemoteAddress->sin_addr.s_addr >> 8 );
 | |
|   pIp4->DestinationAddress.Addr[2] = (UINT8)( pRemoteAddress->sin_addr.s_addr >> 16 );
 | |
|   pIp4->DestinationAddress.Addr[3] = (UINT8)( pRemoteAddress->sin_addr.s_addr >> 24 );
 | |
|   pPort->pSocket->bAddressSet = TRUE;
 | |
|   Status = EFI_SUCCESS;
 | |
| 
 | |
|   //
 | |
|   //  Return the operation status
 | |
|   //
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Process the receive completion
 | |
| 
 | |
|   This routine keeps the IPv4 driver's buffer and queues it in
 | |
|   in FIFO order to the data queue.  The IP4 driver's buffer will
 | |
|   be returned by either ::EslIp4Receive or ::EslSocketPortCloseTxDone.
 | |
|   See the \ref ReceiveEngine section.
 | |
| 
 | |
|   This routine is called by the IPv4 driver when data is
 | |
|   received.
 | |
| 
 | |
|   @param [in] Event     The receive completion event
 | |
| 
 | |
|   @param [in] pIo       The address of an ::ESL_IO_MGMT structure
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| EslIp4RxComplete (
 | |
|   IN EFI_EVENT Event,
 | |
|   IN ESL_IO_MGMT * pIo
 | |
|   )
 | |
| {
 | |
|   size_t LengthInBytes;
 | |
|   ESL_PACKET * pPacket;
 | |
|   EFI_IP4_RECEIVE_DATA * pRxData;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Get the operation status.
 | |
|   //
 | |
|   Status = pIo->Token.Ip4Rx.Status;
 | |
| 
 | |
|   //
 | |
|   //  Get the packet length
 | |
|   //
 | |
|   pRxData = pIo->Token.Ip4Rx.Packet.RxData;
 | |
|   LengthInBytes = pRxData->HeaderLength + pRxData->DataLength;
 | |
| 
 | |
|   //{{
 | |
|   //      +--------------------+   +----------------------+
 | |
|   //      | ESL_IO_MGMT        |   |      Data Buffer     |
 | |
|   //      |                    |   |     (Driver owned)   |
 | |
|   //      |    +---------------+   +----------------------+
 | |
|   //      |    | Token         |               ^
 | |
|   //      |    |      Rx Event |               |
 | |
|   //      |    |               |   +----------------------+
 | |
|   //      |    |        RxData --> | EFI_IP4_RECEIVE_DATA |
 | |
|   //      +----+---------------+   |    (Driver owned)    |
 | |
|   //                               +----------------------+
 | |
|   //      +--------------------+               ^
 | |
|   //      | ESL_PACKET         |               .
 | |
|   //      |                    |               .
 | |
|   //      |    +---------------+               .
 | |
|   //      |    |       pRxData --> NULL  .......
 | |
|   //      +----+---------------+
 | |
|   //
 | |
|   //
 | |
|   //  Save the data in the packet
 | |
|   //}}
 | |
|   pPacket = pIo->pPacket;
 | |
|   pPacket->Op.Ip4Rx.pRxData = pRxData;
 | |
| 
 | |
|   //
 | |
|   //  Complete this request
 | |
|   //
 | |
|   EslSocketRxComplete ( pIo, Status, LengthInBytes, FALSE );
 | |
|   DBG_EXIT ( );
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Determine if the socket is configured.
 | |
| 
 | |
|   This routine uses the flag ESL_SOCKET::bConfigured to determine
 | |
|   if the network layer's configuration routine has been called.
 | |
|   This routine calls the ::EslSocketBind and configuration routines
 | |
|   if they were not already called.  After the port is configured,
 | |
|   the \ref ReceiveEngine is started.
 | |
| 
 | |
|   This routine is called by EslSocketIsConfigured to verify
 | |
|   that the socket is configured.
 | |
| 
 | |
|   @param [in] pSocket         Address of an ::ESL_SOCKET structure
 | |
| 
 | |
|   @retval EFI_SUCCESS - The port is connected
 | |
|   @retval EFI_NOT_STARTED - The port is not connected
 | |
| 
 | |
|  **/
 | |
|  EFI_STATUS
 | |
|  EslIp4SocketIsConfigured (
 | |
|   IN ESL_SOCKET * pSocket
 | |
|   )
 | |
| {
 | |
|   UINTN Index;
 | |
|   ESL_PORT * pPort;
 | |
|   ESL_PORT * pNextPort;
 | |
|   ESL_IP4_CONTEXT * pIp4;
 | |
|   EFI_IP4_PROTOCOL * pIp4Protocol;
 | |
|   EFI_STATUS Status;
 | |
|   struct sockaddr_in LocalAddress;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Assume success
 | |
|   //
 | |
|   Status = EFI_SUCCESS;
 | |
| 
 | |
|   //
 | |
|   //  Configure the port if necessary
 | |
|   //
 | |
|   if ( !pSocket->bConfigured ) {
 | |
|     //
 | |
|     //  Fill in the port list if necessary
 | |
|     //
 | |
|     pSocket->errno = ENETDOWN;
 | |
|     if ( NULL == pSocket->pPortList ) {
 | |
|       LocalAddress.sin_len = sizeof ( LocalAddress );
 | |
|       LocalAddress.sin_family = AF_INET;
 | |
|       LocalAddress.sin_addr.s_addr = 0;
 | |
|       LocalAddress.sin_port = 0;
 | |
|       Status = EslSocketBind ( &pSocket->SocketProtocol,
 | |
|                                (struct sockaddr *)&LocalAddress,
 | |
|                                LocalAddress.sin_len,
 | |
|                                &pSocket->errno );
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  Walk the port list
 | |
|     //
 | |
|     pPort = pSocket->pPortList;
 | |
|     while ( NULL != pPort ) {
 | |
|       //
 | |
|       //  Update the raw setting
 | |
|       //
 | |
|       pIp4 = &pPort->Context.Ip4;
 | |
|       if ( pSocket->bIncludeHeader ) {
 | |
|         //
 | |
|         //  IP header will be included with the data on transmit
 | |
|         //
 | |
|         pIp4->ModeData.ConfigData.RawData = TRUE;
 | |
|       }
 | |
| 
 | |
|       //
 | |
|       //  Attempt to configure the port
 | |
|       //
 | |
|       pNextPort = pPort->pLinkSocket;
 | |
|       pIp4Protocol = pPort->pProtocol.IPv4;
 | |
|       DEBUG (( DEBUG_TX,
 | |
|                 "0x%08x: pPort Configuring for %d.%d.%d.%d --> %d.%d.%d.%d\r\n",
 | |
|                           pPort,
 | |
|                           pIp4->ModeData.ConfigData.StationAddress.Addr[0],
 | |
|                           pIp4->ModeData.ConfigData.StationAddress.Addr[1],
 | |
|                           pIp4->ModeData.ConfigData.StationAddress.Addr[2],
 | |
|                           pIp4->ModeData.ConfigData.StationAddress.Addr[3],
 | |
|                           pIp4->DestinationAddress.Addr[0],
 | |
|                           pIp4->DestinationAddress.Addr[1],
 | |
|                           pIp4->DestinationAddress.Addr[2],
 | |
|                           pIp4->DestinationAddress.Addr[3]));
 | |
|       Status = pIp4Protocol->Configure ( pIp4Protocol,
 | |
|                                           &pIp4->ModeData.ConfigData );
 | |
|       if ( !EFI_ERROR ( Status )) {
 | |
|         //
 | |
|         //  Update the configuration data
 | |
|         //
 | |
|         Status = pIp4Protocol->GetModeData ( pIp4Protocol,
 | |
|                                              &pIp4->ModeData,
 | |
|                                              NULL,
 | |
|                                              NULL );
 | |
|       }
 | |
|       if ( EFI_ERROR ( Status )) {
 | |
|         if ( !pSocket->bConfigured ) {
 | |
|           DEBUG (( DEBUG_LISTEN,
 | |
|                     "ERROR - Failed to configure the Ip4 port, Status: %r\r\n",
 | |
|                     Status ));
 | |
|           switch ( Status ) {
 | |
|           case EFI_ACCESS_DENIED:
 | |
|             pSocket->errno = EACCES;
 | |
|             break;
 | |
| 
 | |
|           default:
 | |
|           case EFI_DEVICE_ERROR:
 | |
|             pSocket->errno = EIO;
 | |
|             break;
 | |
| 
 | |
|           case EFI_INVALID_PARAMETER:
 | |
|             pSocket->errno = EADDRNOTAVAIL;
 | |
|             break;
 | |
| 
 | |
|           case EFI_NO_MAPPING:
 | |
|             pSocket->errno = EAFNOSUPPORT;
 | |
|             break;
 | |
| 
 | |
|           case EFI_OUT_OF_RESOURCES:
 | |
|             pSocket->errno = ENOBUFS;
 | |
|             break;
 | |
| 
 | |
|           case EFI_UNSUPPORTED:
 | |
|             pSocket->errno = EOPNOTSUPP;
 | |
|             break;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       else {
 | |
|         DEBUG (( DEBUG_TX,
 | |
|                   "0x%08x: pPort Configured for %d.%d.%d.%d --> %d.%d.%d.%d\r\n",
 | |
|                   pPort,
 | |
|                   pIp4->ModeData.ConfigData.StationAddress.Addr[0],
 | |
|                   pIp4->ModeData.ConfigData.StationAddress.Addr[1],
 | |
|                   pIp4->ModeData.ConfigData.StationAddress.Addr[2],
 | |
|                   pIp4->ModeData.ConfigData.StationAddress.Addr[3],
 | |
|                   pIp4->DestinationAddress.Addr[0],
 | |
|                   pIp4->DestinationAddress.Addr[1],
 | |
|                   pIp4->DestinationAddress.Addr[2],
 | |
|                   pIp4->DestinationAddress.Addr[3]));
 | |
|         DEBUG (( DEBUG_TX,
 | |
|                   "Subnet Mask: %d.%d.%d.%d\r\n",
 | |
|                   pIp4->ModeData.ConfigData.SubnetMask.Addr[0],
 | |
|                   pIp4->ModeData.ConfigData.SubnetMask.Addr[1],
 | |
|                   pIp4->ModeData.ConfigData.SubnetMask.Addr[2],
 | |
|                   pIp4->ModeData.ConfigData.SubnetMask.Addr[3]));
 | |
|         DEBUG (( DEBUG_TX,
 | |
|                   "Route Count: %d\r\n",
 | |
|                   pIp4->ModeData.RouteCount ));
 | |
|         for ( Index = 0; pIp4->ModeData.RouteCount > Index; Index++ ) {
 | |
|           if ( 0 == Index ) {
 | |
|             DEBUG (( DEBUG_TX, "Route Table:\r\n" ));
 | |
|           }
 | |
|           DEBUG (( DEBUG_TX,
 | |
|                     "%5d: %d.%d.%d.%d, %d.%d.%d.%d ==> %d.%d.%d.%d\r\n",
 | |
|                     Index,
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetAddress.Addr[0],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetAddress.Addr[1],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetAddress.Addr[2],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetAddress.Addr[3],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetMask.Addr[0],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetMask.Addr[1],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetMask.Addr[2],
 | |
|                     pIp4->ModeData.RouteTable[Index].SubnetMask.Addr[3],
 | |
|                     pIp4->ModeData.RouteTable[Index].GatewayAddress.Addr[0],
 | |
|                     pIp4->ModeData.RouteTable[Index].GatewayAddress.Addr[1],
 | |
|                     pIp4->ModeData.RouteTable[Index].GatewayAddress.Addr[2],
 | |
|                     pIp4->ModeData.RouteTable[Index].GatewayAddress.Addr[3]));
 | |
|         }
 | |
|         pPort->bConfigured = TRUE;
 | |
|         pSocket->bConfigured = TRUE;
 | |
| 
 | |
|         //
 | |
|         //  Start the first read on the port
 | |
|         //
 | |
|         EslSocketRxStart ( pPort );
 | |
| 
 | |
|         //
 | |
|         //  The socket is connected
 | |
|         //
 | |
|         pSocket->State = SOCKET_STATE_CONNECTED;
 | |
|         pSocket->errno = 0;
 | |
|       }
 | |
| 
 | |
|       //
 | |
|       //  Set the next port
 | |
|       //
 | |
|       pPort = pNextPort;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Determine the socket configuration status
 | |
|   //
 | |
|   Status = pSocket->bConfigured ? EFI_SUCCESS : EFI_NOT_STARTED;
 | |
| 
 | |
|   //
 | |
|   //  Return the port connected state.
 | |
|   //
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Buffer data for transmission over a network connection.
 | |
| 
 | |
|   This routine buffers data for the transmit engine in the normal
 | |
|   data queue.  When the \ref TransmitEngine has resources, this
 | |
|   routine will start the transmission of the next buffer on the
 | |
|   network connection.
 | |
| 
 | |
|   This routine is called by ::EslSocketTransmit to buffer
 | |
|   data for transmission.  The data is copied into a local buffer
 | |
|   freeing the application buffer for reuse upon return.  When
 | |
|   necessary, this routine starts the transmit engine that
 | |
|   performs the data transmission on the network connection.  The
 | |
|   transmit engine transmits the data a packet at a time over the
 | |
|   network connection.
 | |
| 
 | |
|   Transmission errors are returned during the next transmission or
 | |
|   during the close operation.  Only buffering errors are returned
 | |
|   during the current transmission attempt.
 | |
| 
 | |
|   @param [in] pSocket         Address of an ::ESL_SOCKET structure
 | |
| 
 | |
|   @param [in] Flags           Message control flags
 | |
| 
 | |
|   @param [in] BufferLength    Length of the the buffer
 | |
| 
 | |
|   @param [in] pBuffer         Address of a buffer to receive the data.
 | |
| 
 | |
|   @param [in] pDataLength     Number of received data bytes in the buffer.
 | |
| 
 | |
|   @param [in] pAddress        Network address of the remote system address
 | |
| 
 | |
|   @param [in] AddressLength   Length of the remote network address structure
 | |
| 
 | |
|   @retval EFI_SUCCESS - Socket data successfully buffered
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EslIp4TxBuffer (
 | |
|   IN ESL_SOCKET * pSocket,
 | |
|   IN int Flags,
 | |
|   IN size_t BufferLength,
 | |
|   IN CONST UINT8 * pBuffer,
 | |
|   OUT size_t * pDataLength,
 | |
|   IN const struct sockaddr * pAddress,
 | |
|   IN socklen_t AddressLength
 | |
|   )
 | |
| {
 | |
|   ESL_PACKET * pPacket;
 | |
|   ESL_PACKET * pPreviousPacket;
 | |
|   ESL_PORT * pPort;
 | |
|   const struct sockaddr_in * pRemoteAddress;
 | |
|   ESL_IP4_CONTEXT * pIp4;
 | |
|   size_t * pTxBytes;
 | |
|   ESL_IP4_TX_DATA * pTxData;
 | |
|   EFI_STATUS Status;
 | |
|   EFI_TPL TplPrevious;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Assume failure
 | |
|   //
 | |
|   Status = EFI_UNSUPPORTED;
 | |
|   pSocket->errno = ENOTCONN;
 | |
|   *pDataLength = 0;
 | |
| 
 | |
|   //
 | |
|   //  Verify that the socket is connected
 | |
|   //
 | |
|   if ( SOCKET_STATE_CONNECTED == pSocket->State ) {
 | |
|     //
 | |
|     //  Verify that there is enough room to buffer another
 | |
|     //  transmit operation
 | |
|     //
 | |
|     pTxBytes = &pSocket->TxBytes;
 | |
|     if ( pSocket->MaxTxBuf > *pTxBytes ) {
 | |
|       //
 | |
|       //  Locate the port
 | |
|       //
 | |
|       pPort = pSocket->pPortList;
 | |
|       while ( NULL != pPort ) {
 | |
|         //
 | |
|         //  Determine the queue head
 | |
|         //
 | |
|         pIp4 = &pPort->Context.Ip4;
 | |
| 
 | |
|         //
 | |
|         //  Attempt to allocate the packet
 | |
|         //
 | |
|         Status = EslSocketPacketAllocate ( &pPacket,
 | |
|                                            sizeof ( pPacket->Op.Ip4Tx )
 | |
|                                            - sizeof ( pPacket->Op.Ip4Tx.Buffer )
 | |
|                                            + BufferLength,
 | |
|                                            0,
 | |
|                                            DEBUG_TX );
 | |
|         if ( !EFI_ERROR ( Status )) {
 | |
|           //
 | |
|           //  Initialize the transmit operation
 | |
|           //
 | |
|           pTxData = &pPacket->Op.Ip4Tx;
 | |
|           pTxData->TxData.DestinationAddress.Addr[0] = pIp4->DestinationAddress.Addr[0];
 | |
|           pTxData->TxData.DestinationAddress.Addr[1] = pIp4->DestinationAddress.Addr[1];
 | |
|           pTxData->TxData.DestinationAddress.Addr[2] = pIp4->DestinationAddress.Addr[2];
 | |
|           pTxData->TxData.DestinationAddress.Addr[3] = pIp4->DestinationAddress.Addr[3];
 | |
|           pTxData->TxData.OverrideData = NULL;
 | |
|           pTxData->TxData.OptionsLength = 0;
 | |
|           pTxData->TxData.OptionsBuffer = NULL;
 | |
|           pTxData->TxData.TotalDataLength = (UINT32) BufferLength;
 | |
|           pTxData->TxData.FragmentCount = 1;
 | |
|           pTxData->TxData.FragmentTable[0].FragmentLength = (UINT32) BufferLength;
 | |
|           pTxData->TxData.FragmentTable[0].FragmentBuffer = &pPacket->Op.Ip4Tx.Buffer[0];
 | |
| 
 | |
|           //
 | |
|           //  Set the remote system address if necessary
 | |
|           //
 | |
|           if ( NULL != pAddress ) {
 | |
|             pRemoteAddress = (const struct sockaddr_in *)pAddress;
 | |
|             pTxData->Override.SourceAddress.Addr[0] = pIp4->ModeData.ConfigData.StationAddress.Addr[0];
 | |
|             pTxData->Override.SourceAddress.Addr[1] = pIp4->ModeData.ConfigData.StationAddress.Addr[1];
 | |
|             pTxData->Override.SourceAddress.Addr[2] = pIp4->ModeData.ConfigData.StationAddress.Addr[2];
 | |
|             pTxData->Override.SourceAddress.Addr[3] = pIp4->ModeData.ConfigData.StationAddress.Addr[3];
 | |
|             pTxData->TxData.DestinationAddress.Addr[0] = (UINT8)pRemoteAddress->sin_addr.s_addr;
 | |
|             pTxData->TxData.DestinationAddress.Addr[1] = (UINT8)( pRemoteAddress->sin_addr.s_addr >> 8 );
 | |
|             pTxData->TxData.DestinationAddress.Addr[2] = (UINT8)( pRemoteAddress->sin_addr.s_addr >> 16 );
 | |
|             pTxData->TxData.DestinationAddress.Addr[3] = (UINT8)( pRemoteAddress->sin_addr.s_addr >> 24 );
 | |
|             pTxData->Override.GatewayAddress.Addr[0] = 0;
 | |
|             pTxData->Override.GatewayAddress.Addr[1] = 0;
 | |
|             pTxData->Override.GatewayAddress.Addr[2] = 0;
 | |
|             pTxData->Override.GatewayAddress.Addr[3] = 0;
 | |
|             pTxData->Override.Protocol = (UINT8)pSocket->Protocol;
 | |
|             pTxData->Override.TypeOfService = 0;
 | |
|             pTxData->Override.TimeToLive = 255;
 | |
|             pTxData->Override.DoNotFragment = FALSE;
 | |
| 
 | |
|             //
 | |
|             //  Use the remote system address when sending this packet
 | |
|             //
 | |
|             pTxData->TxData.OverrideData = &pTxData->Override;
 | |
|           }
 | |
| 
 | |
|           //
 | |
|           //  Copy the data into the buffer
 | |
|           //
 | |
|           CopyMem ( &pPacket->Op.Ip4Tx.Buffer[0],
 | |
|                     pBuffer,
 | |
|                     BufferLength );
 | |
| 
 | |
|           //
 | |
|           //  Synchronize with the socket layer
 | |
|           //
 | |
|           RAISE_TPL ( TplPrevious, TPL_SOCKETS );
 | |
| 
 | |
|           //
 | |
|           //  Display the request
 | |
|           //
 | |
|           DEBUG (( DEBUG_TX,
 | |
|                     "Send %d bytes from 0x%08x, %d.%d.%d.%d --> %d.%d.%d.%d\r\n",
 | |
|                     BufferLength,
 | |
|                     pBuffer,
 | |
|                     pIp4->ModeData.ConfigData.StationAddress.Addr[0],
 | |
|                     pIp4->ModeData.ConfigData.StationAddress.Addr[1],
 | |
|                     pIp4->ModeData.ConfigData.StationAddress.Addr[2],
 | |
|                     pIp4->ModeData.ConfigData.StationAddress.Addr[3],
 | |
|                     pTxData->TxData.DestinationAddress.Addr[0],
 | |
|                     pTxData->TxData.DestinationAddress.Addr[1],
 | |
|                     pTxData->TxData.DestinationAddress.Addr[2],
 | |
|                     pTxData->TxData.DestinationAddress.Addr[3]));
 | |
| 
 | |
|           //
 | |
|           //  Queue the data for transmission
 | |
|           //
 | |
|           pPacket->pNext = NULL;
 | |
|           pPreviousPacket = pSocket->pTxPacketListTail;
 | |
|           if ( NULL == pPreviousPacket ) {
 | |
|             pSocket->pTxPacketListHead = pPacket;
 | |
|           }
 | |
|           else {
 | |
|             pPreviousPacket->pNext = pPacket;
 | |
|           }
 | |
|           pSocket->pTxPacketListTail = pPacket;
 | |
|           DEBUG (( DEBUG_TX,
 | |
|                     "0x%08x: Packet on transmit list\r\n",
 | |
|                     pPacket ));
 | |
| 
 | |
|           //
 | |
|           //  Account for the buffered data
 | |
|           //
 | |
|           *pTxBytes += BufferLength;
 | |
|           *pDataLength = BufferLength;
 | |
| 
 | |
|           //
 | |
|           //  Start the transmit engine if it is idle
 | |
|           //
 | |
|           if ( NULL != pPort->pTxFree ) {
 | |
|             EslSocketTxStart ( pPort,
 | |
|                                &pSocket->pTxPacketListHead,
 | |
|                                &pSocket->pTxPacketListTail,
 | |
|                                &pPort->pTxActive,
 | |
|                                &pPort->pTxFree );
 | |
| 
 | |
|             //
 | |
|             //  Ignore any transmit error
 | |
|             //
 | |
|             if ( EFI_ERROR ( pSocket->TxError )) {
 | |
|               DEBUG (( DEBUG_TX,
 | |
|                        "0x%08x: Transmit error, Packet: 0x%08x, Status: %r\r\n",
 | |
|                        pPort,
 | |
|                        pPacket,
 | |
|                        pSocket->TxError ));
 | |
|             }
 | |
|             pSocket->TxError = EFI_SUCCESS;
 | |
|           }
 | |
| 
 | |
|           //
 | |
|           //  Release the socket layer synchronization
 | |
|           //
 | |
|           RESTORE_TPL ( TplPrevious );
 | |
|         }
 | |
|         else {
 | |
|           //
 | |
|           //  Packet allocation failed
 | |
|           //
 | |
|           pSocket->errno = ENOMEM;
 | |
|           break;
 | |
|         }
 | |
| 
 | |
|         //
 | |
|         //  Set the next port
 | |
|         //
 | |
|         pPort = pPort->pLinkSocket;
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       //
 | |
|       //  Not enough buffer space available
 | |
|       //
 | |
|       pSocket->errno = EAGAIN;
 | |
|       Status = EFI_NOT_READY;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Return the operation status
 | |
|   //
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Process the transmit completion
 | |
| 
 | |
|   This routine use ::EslSocketTxComplete to perform the transmit
 | |
|   completion processing for data packets.
 | |
| 
 | |
|   This routine is called by the IPv4 network layer when a data
 | |
|   transmit request completes.
 | |
| 
 | |
|   @param [in] Event     The normal transmit completion event
 | |
| 
 | |
|   @param [in] pIo       The address of an ::ESL_IO_MGMT structure
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| EslIp4TxComplete (
 | |
|   IN EFI_EVENT Event,
 | |
|   IN ESL_IO_MGMT * pIo
 | |
|   )
 | |
| {
 | |
|   UINT32 LengthInBytes;
 | |
|   ESL_PORT * pPort;
 | |
|   ESL_PACKET * pPacket;
 | |
|   ESL_SOCKET * pSocket;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Locate the active transmit packet
 | |
|   //
 | |
|   pPacket = pIo->pPacket;
 | |
|   pPort = pIo->pPort;
 | |
|   pSocket = pPort->pSocket;
 | |
| 
 | |
|   //
 | |
|   //  Get the transmit length and status
 | |
|   //
 | |
|   LengthInBytes = pPacket->Op.Ip4Tx.TxData.TotalDataLength;
 | |
|   pSocket->TxBytes -= LengthInBytes;
 | |
|   Status = pIo->Token.Ip4Tx.Status;
 | |
| 
 | |
|   //
 | |
|   //  Ignore the transmit error
 | |
|   //
 | |
|   if ( EFI_ERROR ( Status )) {
 | |
|     DEBUG (( DEBUG_TX,
 | |
|              "0x%08x: Transmit completion error, Packet: 0x%08x, Status: %r\r\n",
 | |
|              pPort,
 | |
|              pPacket,
 | |
|              Status ));
 | |
|     Status = EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Complete the transmit operation
 | |
|   //
 | |
|   EslSocketTxComplete ( pIo,
 | |
|                         LengthInBytes,
 | |
|                         Status,
 | |
|                         "Raw ",
 | |
|                         &pSocket->pTxPacketListHead,
 | |
|                         &pSocket->pTxPacketListTail,
 | |
|                         &pPort->pTxActive,
 | |
|                         &pPort->pTxFree );
 | |
|   DBG_EXIT ( );
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Verify the adapter's IP address
 | |
| 
 | |
|   This support routine is called by EslSocketBindTest.
 | |
| 
 | |
|   @param [in] pPort       Address of an ::ESL_PORT structure.
 | |
|   @param [in] pConfigData Address of the configuration data
 | |
| 
 | |
|   @retval EFI_SUCCESS - The IP address is valid
 | |
|   @retval EFI_NOT_STARTED - The IP address is invalid
 | |
| 
 | |
|  **/
 | |
| EFI_STATUS
 | |
| EslIp4VerifyLocalIpAddress (
 | |
|   IN ESL_PORT * pPort,
 | |
|   IN EFI_IP4_CONFIG_DATA * pConfigData
 | |
|   )
 | |
| {
 | |
|   UINTN DataSize;
 | |
|   EFI_IP4_CONFIG2_INTERFACE_INFO * pIfInfo;
 | |
|   EFI_IP4_CONFIG2_PROTOCOL * pIpConfig2Protocol;
 | |
|   ESL_SERVICE * pService;
 | |
|   EFI_STATUS Status;
 | |
| 
 | |
|   DBG_ENTER ( );
 | |
| 
 | |
|   //
 | |
|   //  Use break instead of goto
 | |
|   //
 | |
|   pIfInfo = NULL;
 | |
|   for ( ; ; ) {
 | |
|     //
 | |
|     //  Determine if the IP address is specified
 | |
|     //
 | |
|     DEBUG (( DEBUG_BIND,
 | |
|               "UseDefaultAddress: %s\r\n",
 | |
|               pConfigData->UseDefaultAddress ? L"TRUE" : L"FALSE" ));
 | |
|     DEBUG (( DEBUG_BIND,
 | |
|               "Requested IP address: %d.%d.%d.%d\r\n",
 | |
|               pConfigData->StationAddress.Addr [ 0 ],
 | |
|               pConfigData->StationAddress.Addr [ 1 ],
 | |
|               pConfigData->StationAddress.Addr [ 2 ],
 | |
|               pConfigData->StationAddress.Addr [ 3 ]));
 | |
|     if ( pConfigData->UseDefaultAddress
 | |
|       || (( 0 == pConfigData->StationAddress.Addr [ 0 ])
 | |
|       && ( 0 == pConfigData->StationAddress.Addr [ 1 ])
 | |
|       && ( 0 == pConfigData->StationAddress.Addr [ 2 ])
 | |
|       && ( 0 == pConfigData->StationAddress.Addr [ 3 ])))
 | |
|     {
 | |
|       Status = EFI_SUCCESS;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  Open the configuration protocol
 | |
|     //
 | |
|     pService = pPort->pService;
 | |
|     Status = gBS->OpenProtocol ( 
 | |
|                     pService->Controller,
 | |
|                     &gEfiIp4Config2ProtocolGuid,
 | |
|                     (VOID **)&pIpConfig2Protocol,
 | |
|                     NULL,
 | |
|                     NULL,
 | |
|                     EFI_OPEN_PROTOCOL_GET_PROTOCOL 
 | |
|                     );
 | |
|     if ( EFI_ERROR ( Status )) {
 | |
|       DEBUG (( DEBUG_ERROR,
 | |
|                 "ERROR - IP Configuration Protocol not available, Status: %r\r\n",
 | |
|                 Status ));
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Get the interface information size.
 | |
|     //
 | |
|     DataSize = 0;
 | |
|     Status = pIpConfig2Protocol->GetData ( 
 | |
|                                    pIpConfig2Protocol,
 | |
|                                    Ip4Config2DataTypeInterfaceInfo,
 | |
|                                    &DataSize,
 | |
|                                    NULL
 | |
|                                    );
 | |
|     if ( EFI_BUFFER_TOO_SMALL != Status ) {
 | |
|       DEBUG (( DEBUG_ERROR,
 | |
|                 "ERROR - Failed to get the interface information size, Status: %r\r\n",
 | |
|                 Status ));
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  Allocate the interface information buffer
 | |
|     //
 | |
|     pIfInfo = AllocatePool ( DataSize );
 | |
|     if ( NULL == pIfInfo ) {
 | |
|       DEBUG (( DEBUG_ERROR,
 | |
|                 "ERROR - Not enough memory to allocate the interface information buffer!\r\n" ));
 | |
|       Status = EFI_OUT_OF_RESOURCES;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Get the interface info.
 | |
|     //
 | |
|     Status = pIpConfig2Protocol->GetData ( 
 | |
|                                   pIpConfig2Protocol,
 | |
|                                   Ip4Config2DataTypeInterfaceInfo,
 | |
|                                   &DataSize,
 | |
|                                   pIfInfo
 | |
|                                   );
 | |
|     if ( EFI_ERROR ( Status )) {
 | |
|       DEBUG (( DEBUG_ERROR,
 | |
|                 "ERROR - Failed to return the interface info, Status: %r\r\n",
 | |
|                 Status ));
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  Display the current configuration
 | |
|     //
 | |
|     DEBUG (( DEBUG_BIND,
 | |
|               "Actual adapter IP address: %d.%d.%d.%d\r\n",
 | |
|               pIfInfo->StationAddress.Addr [ 0 ],
 | |
|               pIfInfo->StationAddress.Addr [ 1 ],
 | |
|               pIfInfo->StationAddress.Addr [ 2 ],
 | |
|               pIfInfo->StationAddress.Addr [ 3 ]));
 | |
| 
 | |
|     //
 | |
|     //  Assume the port is not configured
 | |
|     //
 | |
|     Status = EFI_SUCCESS;
 | |
|     if (( pConfigData->StationAddress.Addr [ 0 ] == pIfInfo->StationAddress.Addr [ 0 ])
 | |
|       && ( pConfigData->StationAddress.Addr [ 1 ] == pIfInfo->StationAddress.Addr [ 1 ])
 | |
|       && ( pConfigData->StationAddress.Addr [ 2 ] == pIfInfo->StationAddress.Addr [ 2 ])
 | |
|       && ( pConfigData->StationAddress.Addr [ 3 ] == pIfInfo->StationAddress.Addr [ 3 ])) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     //  The IP address did not match
 | |
|     //
 | |
|     Status = EFI_NOT_STARTED;
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Free the buffer if necessary
 | |
|   //
 | |
|   if ( NULL != pIfInfo ) {
 | |
|     FreePool ( pIfInfo );
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Return the IP address status
 | |
|   //
 | |
|   DBG_EXIT_STATUS ( Status );
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Interface between the socket layer and the network specific
 | |
|   code that supports SOCK_RAW sockets over IPv4.
 | |
| **/
 | |
| CONST ESL_PROTOCOL_API cEslIp4Api = {
 | |
|   "IPv4",
 | |
|     IPPROTO_IP,
 | |
|   OFFSET_OF ( ESL_PORT, Context.Ip4.ModeData.ConfigData ),
 | |
|   OFFSET_OF ( ESL_LAYER, pIp4List ),
 | |
|   OFFSET_OF ( struct sockaddr_in, sin_zero ),
 | |
|   sizeof ( struct sockaddr_in ),
 | |
|   AF_INET,
 | |
|   sizeof (((ESL_PACKET *)0 )->Op.Ip4Rx ),
 | |
|   sizeof (((ESL_PACKET *)0 )->Op.Ip4Rx ),
 | |
|   OFFSET_OF ( ESL_IO_MGMT, Token.Ip4Rx.Packet.RxData ),
 | |
|   FALSE,
 | |
|   EADDRNOTAVAIL,
 | |
|   NULL,   //  Accept
 | |
|   NULL,   //  ConnectPoll
 | |
|   NULL,   //  ConnectStart
 | |
|   EslIp4SocketIsConfigured,
 | |
|   EslIp4LocalAddressGet,
 | |
|   EslIp4LocalAddressSet,
 | |
|   NULL,   //  Listen
 | |
|   EslIp4OptionGet,
 | |
|   EslIp4OptionSet,
 | |
|   EslIp4PacketFree,
 | |
|   EslIp4PortAllocate,
 | |
|   NULL,   //  PortClose
 | |
|   NULL,   //  PortCloseOp
 | |
|   TRUE,
 | |
|   EslIp4Receive,
 | |
|   EslIp4RemoteAddressGet,
 | |
|   EslIp4RemoteAddressSet,
 | |
|   EslIp4RxComplete,
 | |
|   NULL,   //  RxStart
 | |
|   EslIp4TxBuffer,
 | |
|   EslIp4TxComplete,
 | |
|   NULL,   //  TxOobComplete
 | |
|   (PFN_API_VERIFY_LOCAL_IP_ADDRESS)EslIp4VerifyLocalIpAddress
 | |
| };
 |