audk/AppPkg/Applications/Sockets/RawIp4Tx/RawIp4Tx.c

158 lines
4.3 KiB
C

/** @file
Raw IP4 transmit application
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 "RawIp4Tx.h"
UINT8 mBuffer[1024];
/**
Transmit raw IP4 packets to the remote system.
@param [in] ArgC Argument count
@param [in] ArgV Argument value array
@retval 0 Successfully operation
**/
int
RawIp4Tx (
IN int ArgC,
IN char **ArgV
)
{
UINT32 BytesSent;
ssize_t BytesTransmitted;
struct sockaddr_in LocalPort;
UINT32 RemoteAddress[4];
struct sockaddr_in RemotePort;
int RetVal;
UINT32 TotalSent;
SOCKET s;
//
// Create the socket
//
s = socket ( AF_INET, SOCK_RAW, RAW_PROTOCOL );
if ( -1 == s ) {
RetVal = GET_ERRNO;
printf ( "ERROR - socket error, errno: %d\r\n", RetVal );
}
else {
//
// Use for/break; instead of goto
//
for ( ; ; ) {
//
// Validate the arguments
//
if (( 2 > ArgC )
|| ( 4 != sscanf ( ArgV[1],
"%d.%d.%d.%d",
&RemoteAddress[0],
&RemoteAddress[1],
&RemoteAddress[2],
&RemoteAddress[3]))
|| ( 224 < RemoteAddress[0])
|| ( 255 < RemoteAddress[1])
|| ( 255 < RemoteAddress[2])
|| ( 255 < RemoteAddress[3])
|| (( 0 == RemoteAddress[0])
&& ( 0 == RemoteAddress[1])
&& ( 0 == RemoteAddress[2])
&& ( 0 == RemoteAddress[3]))) {
printf ( "%s <remote IP address>\r\n", ArgV[0]);
RetVal = EINVAL;
break;
}
//
// Bind the socket to a local port
//
memset ( &LocalPort, 0, sizeof ( LocalPort ));
SIN_LEN ( LocalPort ) = sizeof ( LocalPort );
SIN_FAMILY ( LocalPort ) = AF_INET;
SIN_ADDR ( LocalPort ) = 0;
SIN_PORT ( LocalPort ) = 0;
RetVal = bind ( s,
(struct sockaddr *)&LocalPort,
sizeof ( LocalPort ));
if ( -1 == RetVal ) {
RetVal = GET_ERRNO;
printf ( "ERROR - bind error, errno: %d\r\n", RetVal );
break;
}
//
// Specify the remote port
//
memset ( &RemotePort, 0, sizeof ( RemotePort ));
SIN_LEN ( RemotePort ) = sizeof ( RemotePort );
SIN_FAMILY ( RemotePort ) = AF_INET;
SIN_ADDR ( RemotePort ) = ( RemoteAddress[3] << 24 )
| ( RemoteAddress[2] << 16 )
| ( RemoteAddress[1] << 8 )
| RemoteAddress[0];
SIN_PORT ( RemotePort ) = 0;
//
// Initialize the messages
//
memset ( &mBuffer[0], 0, sizeof ( mBuffer ));
//
// Send the data before the out-of-band message
//
TotalSent = 0;
BytesSent = 0;
do {
BytesTransmitted = sendto ( s,
&mBuffer[BytesSent],
sizeof ( mBuffer ) - BytesSent,
0,
(struct sockaddr *)&RemotePort,
sizeof ( RemotePort ));
if ( -1 == BytesTransmitted ) {
RetVal = GET_ERRNO;
printf ( "ERROR - send before error, errno: %d\r\n", RetVal );
break;
}
BytesSent += (UINT32)BytesTransmitted;
RetVal = 0;
} while ( sizeof ( mBuffer ) > BytesSent );
if ( 0 != RetVal ) {
break;
}
TotalSent += BytesSent;
//
// Test completed successfully
//
if ( 0 == RetVal ) {
printf ( "Bytes sent: %8d\r\n", TotalSent );
}
break;
}
//
// Close the socket
//
CLOSE_SOCKET ( s );
}
//
// Return the operation status
//
return RetVal;
}