Use the BSD license Fix errors detected by GCC compiler in WebServer/ConfigurationTable.c Add libraries: CpuLib, DxeServicesTableLib and MtrrLib Signed-off-by: lpleahy git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@13115 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			382 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			382 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   Set the socket options
 | |
| 
 | |
|   Copyright (c) 2011-2012, Intel Corporation
 | |
|   All rights reserved. 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 <errno.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <Uefi.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #include <Library/DebugLib.h>
 | |
| #include <Library/UefiLib.h>
 | |
| 
 | |
| #include <sys/socket.h>
 | |
| #include <sys/time.h>
 | |
| 
 | |
| typedef enum _DATA_TYPE {
 | |
|   DATA_TYPE_UNKNOWN = 0,
 | |
|   DATA_TYPE_INT32_DECIMAL,
 | |
|   DATA_TYPE_SOCKET_TYPE,
 | |
|   DATA_TYPE_TIMEVAL
 | |
| } DATA_TYPE;
 | |
| 
 | |
| typedef struct {
 | |
|   char * pOptionName;
 | |
|   int OptionValue;
 | |
|   int OptionLevel;
 | |
|   BOOLEAN bSetAllowed;
 | |
|   DATA_TYPE DataType;
 | |
| } OPTIONS;
 | |
| 
 | |
| CONST OPTIONS mOptions[] = {
 | |
|   { "SO_ACCEPTCONN", SO_ACCEPTCONN, SOL_SOCKET, FALSE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_BROADCAST", SO_BROADCAST, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_DEBUG", SO_DEBUG, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_DONTROUTE", SO_DONTROUTE, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_ERROR", SO_ERROR, SOL_SOCKET, FALSE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_KEEPALIVE", SO_KEEPALIVE, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_OOBINLINE", SO_OOBINLINE, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_OVERFLOWED", SO_OVERFLOWED, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_RCVBUF", SO_RCVBUF, SOL_SOCKET, TRUE, DATA_TYPE_INT32_DECIMAL },
 | |
|   { "SO_RCVLOWAT", SO_RCVLOWAT, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_RCVTIMEO", SO_RCVTIMEO, SOL_SOCKET, TRUE, DATA_TYPE_TIMEVAL },
 | |
|   { "SO_REUSEADDR", SO_REUSEADDR, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_REUSEPORT", SO_REUSEPORT, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_SNDBUF", SO_SNDBUF, SOL_SOCKET, TRUE, DATA_TYPE_INT32_DECIMAL },
 | |
|   { "SO_SNDLOWAT", SO_SNDLOWAT, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_SNDTIMEO", SO_SNDTIMEO, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_TIMESTAMP", SO_TIMESTAMP, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN },
 | |
|   { "SO_TYPE", SO_TYPE, SOL_SOCKET, FALSE, DATA_TYPE_SOCKET_TYPE },
 | |
|   { "SO_USELOOPBACK", SO_USELOOPBACK, SOL_SOCKET, TRUE, DATA_TYPE_UNKNOWN }
 | |
| };
 | |
| 
 | |
| 
 | |
| UINT8 mBuffer[ 65536 ];
 | |
| UINT8 mValue[ 65536 ];
 | |
| char * mSocketType[] = {
 | |
|   "SOCK_STREAM",
 | |
|   "SOCK_DGRAM",
 | |
|   "SOCK_RAW",
 | |
|   "SOCK_RDM",
 | |
|   "SOCK_SEQPACKET"
 | |
| };
 | |
| 
 | |
| void
 | |
| DisplayOption (
 | |
|   CONST OPTIONS * pOption,
 | |
|   socklen_t LengthInBytes,
 | |
|   BOOLEAN bDisplayUpdate,
 | |
|   BOOLEAN bDisplayCrLf
 | |
|   )
 | |
| {
 | |
|   UINT8 * pEnd;
 | |
|   char * pString;
 | |
|   union {
 | |
|     UINT8 * u8;
 | |
|     INT32 * i32;
 | |
|     struct timeval * TimeVal;
 | |
|   } Value;
 | |
| 
 | |
|   //
 | |
|   //  Display the value length
 | |
|   //
 | |
|   if ( !bDisplayUpdate ) {
 | |
|     Print ( L"LengthInBytes: %d\r\n", LengthInBytes );
 | |
|     Print ( L"%a: ", pOption->pOptionName );
 | |
|   }
 | |
|   else {
 | |
|     Print ( L" --> " );
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Display the value
 | |
|   //
 | |
|   Value.u8 = &mBuffer[0];
 | |
|   switch ( pOption->DataType ) {
 | |
|   case DATA_TYPE_UNKNOWN:
 | |
|     Print ( L"%a:", pOption->pOptionName );
 | |
|     pEnd = &Value.u8[ LengthInBytes ];
 | |
|     while ( pEnd > Value.u8 ) {
 | |
|       Print ( L" %02x", *Value.u8 );
 | |
|       Value.u8 += 1;
 | |
|     }
 | |
|     break;
 | |
|     
 | |
|   case DATA_TYPE_INT32_DECIMAL:
 | |
|     if ( 4 == LengthInBytes ) {
 | |
|       Print ( L"%d", *Value.i32 );
 | |
|     }
 | |
|     else {
 | |
|       errno = ( 4 > LengthInBytes ) ? EBUFSIZE : ERANGE;
 | |
|       Print ( L"\r\nERROR - Invalid length, errno: %d\r\n", errno );
 | |
|     }
 | |
|     break;
 | |
| 
 | |
|   case DATA_TYPE_SOCKET_TYPE:
 | |
|     if ( 4 == LengthInBytes ) {
 | |
|       if (( SOCK_STREAM <= *Value.i32 ) && ( SOCK_SEQPACKET >= *Value.i32 )) {
 | |
|         pString = mSocketType[ *Value.i32 - SOCK_STREAM ];
 | |
|         Print ( L"%a", pString );
 | |
|       }
 | |
|       else {
 | |
|         Print ( L"%08x (unknown type)", *Value.i32 );
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       errno = ( 4 > LengthInBytes ) ? EBUFSIZE : ERANGE;
 | |
|       Print ( L"\r\nERROR - Invalid length, errno: %d\r\n", errno );
 | |
|     }
 | |
|     break;
 | |
| 
 | |
|   case DATA_TYPE_TIMEVAL:
 | |
|     if ( sizeof ( *Value.TimeVal ) == LengthInBytes ) {
 | |
|       if (( 0 == Value.TimeVal->tv_sec )
 | |
|         && ( 0 == Value.TimeVal->tv_usec )) {
 | |
|         Print ( L"Infinite" );
 | |
|       }
 | |
|       else {
 | |
|         Print ( L"%d.%06d sec",
 | |
|                 Value.TimeVal->tv_sec,
 | |
|                 Value.TimeVal->tv_usec );
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       errno = ( 4 > LengthInBytes ) ? EBUFSIZE : ERANGE;
 | |
|       Print ( L"\r\nERROR - Invalid length, errno: %d\r\n", errno );
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Terminate the line
 | |
|   //
 | |
|   if ( bDisplayCrLf ) {
 | |
|     Print ( L"\r\n" );
 | |
|   }
 | |
| }
 | |
| 
 | |
| socklen_t
 | |
| GetOptionValue (
 | |
|   CONST OPTIONS * pOption,
 | |
|   char * pValue
 | |
|   )
 | |
| {
 | |
|   socklen_t BytesToWrite;
 | |
|   union {
 | |
|     UINT8 * u8;
 | |
|     INT32 * i32;
 | |
|     struct timeval * TimeVal;
 | |
|   } Value;
 | |
|   int Values;
 | |
| 
 | |
|   //
 | |
|   //  Assume failure
 | |
|   //
 | |
|   errno = EINVAL;
 | |
|   BytesToWrite = 0;
 | |
| 
 | |
|   //
 | |
|   //  Determine the type of parameter
 | |
|   //
 | |
|   if ( pOption->bSetAllowed ) {
 | |
|     Value.u8 = &mValue[0];
 | |
|     switch ( pOption->DataType ) {
 | |
|     default:
 | |
|       break;
 | |
| 
 | |
|     case DATA_TYPE_INT32_DECIMAL:
 | |
|       Values = sscanf ( pValue, "%d", Value.i32 );
 | |
|       if ( 1 == Values ) {
 | |
|         BytesToWrite = sizeof ( *Value.i32);
 | |
|         errno = 0;
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case DATA_TYPE_TIMEVAL:
 | |
|       Values = sscanf ( pValue, "%d.%d",
 | |
|                         &Value.TimeVal->tv_sec,
 | |
|                         &Value.TimeVal->tv_usec );
 | |
|       if (( 2 == Values )
 | |
|         && ( 0 <= Value.TimeVal->tv_sec )
 | |
|         && ( 0 <= Value.TimeVal->tv_usec )
 | |
|         && ( 1000000 > Value.TimeVal->tv_usec )){
 | |
|         BytesToWrite = sizeof ( *Value.TimeVal );
 | |
|         errno = 0;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Display the error
 | |
|   //
 | |
|   if ( 0 == BytesToWrite ) {
 | |
|     Print ( L"ERROR - Invalid value!\r\n" );
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Return the number of bytes to be written
 | |
|   //
 | |
|   return BytesToWrite;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Set the socket options
 | |
| 
 | |
|   @param [in] Argc  The number of arguments
 | |
|   @param [in] Argv  The argument value array
 | |
| 
 | |
|   @retval  0        The application exited normally.
 | |
|   @retval  Other    An error occurred.
 | |
| **/
 | |
| int
 | |
| main (
 | |
|   IN int Argc,
 | |
|   IN char **Argv
 | |
|   )
 | |
| {
 | |
|   socklen_t BytesToWrite;
 | |
|   socklen_t LengthInBytes;
 | |
|   CONST OPTIONS * pEnd;
 | |
|   CONST OPTIONS * pOption;
 | |
|   int s;
 | |
|   int Status;
 | |
| 
 | |
|   DEBUG (( DEBUG_INFO,
 | |
|             "%a starting\r\n",
 | |
|             Argv[0]));
 | |
| 
 | |
|   //
 | |
|   //  Parse the socket option
 | |
|   //
 | |
|   pOption = &mOptions[0];
 | |
|   pEnd = &pOption[sizeof ( mOptions ) / sizeof ( mOptions[0])];
 | |
|   if ( 2 <= Argc ) {
 | |
|     while ( pEnd > pOption ) {
 | |
|       if ( 0 == strcmp ( Argv[1], pOption->pOptionName )) {
 | |
|         break;
 | |
|       }
 | |
|       pOption += 1;
 | |
|     }
 | |
|     if ( pEnd <= pOption ) {
 | |
|       Print ( L"ERROR: Invalid option: %a\r\n", Argv[1]);
 | |
|       Argc = 1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  Display the help if necessary
 | |
|   //
 | |
|   if (( 2 > Argc ) || ( 3 < Argc )) {
 | |
|     Print ( L"%a <option>\r\n", Argv[0]);
 | |
|     Print ( L"\r\n" );
 | |
|     Print ( L"Option one of:\r\n" );
 | |
|     pOption = &mOptions[0];
 | |
|     while ( pEnd > pOption ) {
 | |
|       Print ( L"   %a: %a\r\n",
 | |
|               pOption->pOptionName,
 | |
|               pOption->bSetAllowed ? "get/set" : "get" );
 | |
|       pOption += 1;
 | |
|     }
 | |
|     errno = EINVAL;
 | |
|   }
 | |
|   else {
 | |
|     //
 | |
|     //  Determine if the value is to be set
 | |
|     //
 | |
|     BytesToWrite = 0;
 | |
|     if (( 3 > Argc )
 | |
|       || ( 0 < ( BytesToWrite = GetOptionValue ( pOption, Argv[2])))) {
 | |
|       //
 | |
|       //  Get the socket
 | |
|       //
 | |
|       s = socket ( AF_INET, 0, 0 );
 | |
|       if ( -1 == s ) {
 | |
|         Print ( L"ERROR - Unable to open the socket, errno: %d\r\n", errno );
 | |
|       }
 | |
|       else {
 | |
|         //
 | |
|         //  Display the option value
 | |
|         //
 | |
|         LengthInBytes = sizeof ( mBuffer );
 | |
|         Status = getsockopt ( s,
 | |
|                               pOption->OptionLevel,
 | |
|                               pOption->OptionValue,
 | |
|                               &mBuffer,
 | |
|                               &LengthInBytes );
 | |
|         if ( -1 == Status ) {
 | |
|           Print ( L"ERROR - getsockopt failed, errno: %d\r\n", errno );
 | |
|         }
 | |
|         else {
 | |
|           DisplayOption ( pOption,
 | |
|                           LengthInBytes,
 | |
|                           FALSE,
 | |
|                           (BOOLEAN)( 0 == BytesToWrite ));
 | |
| 
 | |
|           //
 | |
|           //  Determine if the value is to be set
 | |
|           //
 | |
|           if (( 0 < BytesToWrite )
 | |
|               && ( BytesToWrite == LengthInBytes )) {
 | |
|             //
 | |
|             //  Set the option value
 | |
|             //
 | |
|             Status = setsockopt ( s,
 | |
|                                   pOption->OptionLevel,
 | |
|                                   pOption->OptionValue,
 | |
|                                   &mValue,
 | |
|                                   BytesToWrite );
 | |
|             if ( -1 == Status ) {
 | |
|               Print ( L"ERROR - setsockopt failed, errno: %d\r\n", errno );
 | |
|             }
 | |
|             else {
 | |
|               //
 | |
|               //  Display the updated option value
 | |
|               //
 | |
|               Status = getsockopt ( s,
 | |
|                                     pOption->OptionLevel,
 | |
|                                     pOption->OptionValue,
 | |
|                                     &mBuffer,
 | |
|                                     &LengthInBytes );
 | |
|               if ( -1 == Status ) {
 | |
|                 Print ( L"ERROR - getsockopt failed, errno: %d\r\n", errno );
 | |
|               }
 | |
|               else {
 | |
|                 DisplayOption ( pOption,
 | |
|                                 LengthInBytes,
 | |
|                                 TRUE,
 | |
|                                 TRUE );
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         //
 | |
|         //  Done with the socket
 | |
|         //
 | |
|         close ( s );
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   //  All done
 | |
|   //
 | |
|   DEBUG (( DEBUG_INFO,
 | |
|             "%a exiting, errno: %d\r\n",
 | |
|             Argv[0],
 | |
|             errno ));
 | |
|   return errno;
 | |
| }
 |