Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

358 lines
8.6 KiB

/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
ping.c
Abstract:
Packet INternet Groper utility for TCP/IP.
Author:
Numerous TCP/IP folks.
Revision History:
Who When What
-------- -------- ----------------------------------------------
WeihaiC 5-Dec-98. Moved from \nt\private\net\sockets\tcpcmd\ping
MohsinA, 21-Oct-96. INADDR_NONE check to avoid broadcast.
MohsinA, 13-Nov-96. Max packet size < 64K.
Notes:
--*/
//:ts=4
typedef unsigned long ULONG;
typedef unsigned short ushort;
typedef unsigned int UINT;
typedef unsigned char UCHAR;
#include "precomp.h"
#include "icmpapi.h"
#define NOGDI
#define NOMINMAX
#define MAX_BUFFER_SIZE (sizeof(ICMP_ECHO_REPLY) + 0xfff7 + MAX_OPT_SIZE)
#define DEFAULT_BUFFER_SIZE (0x2000 - 8)
#define DEFAULT_SEND_SIZE 32
#define DEFAULT_COUNT 4
#define DEFAULT_TTL 32
#define DEFAULT_TOS 0
#define DEFAULT_TIMEOUT 5000L
#define MIN_INTERVAL 1000L
#define TRUE 1
#define FALSE 0
#define STDOUT 1
#define net_long(x) (((((ULONG)(x))&0xffL)<<24) | \
((((ULONG)(x))&0xff00L)<<8) | \
((((ULONG)(x))&0xff0000L)>>8) | \
((((ULONG)(x))&0xff000000L)>>24))
#ifdef VXD
#define FAR _far
#endif // VXD
WSADATA WsaData;
// ========================================================================
// Caveat: return 0 for invalid, else internet address.
// I would prefer -1 for error. - MohsinA, 21-Nov-96.
unsigned long
get_pingee(char *ahstr, char **hstr, int *was_inaddr, int dnsreq)
{
struct hostent *hostp = NULL;
long inaddr;
if( strcmp( ahstr, "255.255.255.255" ) == 0 ){
return 0L;
}
if ((inaddr = inet_addr(ahstr)) == -1L) {
hostp = gethostbyname(ahstr);
if (hostp) {
/*
* If we find a host entry, set up the internet address
*/
inaddr = *(long *)hostp->h_addr;
*was_inaddr = 0;
} else {
// Neither dotted, not name.
return(0L);
}
} else {
// Is dotted.
*was_inaddr = 1;
if (dnsreq == 1) {
hostp = gethostbyaddr((char *)&inaddr,sizeof(inaddr),AF_INET);
}
}
*hstr = hostp ? hostp->h_name : (char *)NULL;
return(inaddr);
}
// ========================================================================
BOOL Ping (LPTSTR pszServerName)
{
UINT i;
UINT j;
int found_addr = 0;
int dnsreq = 0;
char *hostname = NULL;
int was_inaddr;
DWORD numberOfReplies;
UINT Count = DEFAULT_COUNT;
UCHAR TTL = DEFAULT_TTL;
UCHAR FAR *Opt = (UCHAR FAR *)0; // Pointer to send options
UINT OptLength = 0;
int OptIndex = 0; // Current index into SendOptions
int SRIndex = -1; // Where to put address, if source routing
UCHAR TOS = DEFAULT_TOS;
UCHAR Flags = 0;
ULONG Timeout = DEFAULT_TIMEOUT;
IP_OPTION_INFORMATION SendOpts;
int EndOffset;
ULONG TempAddr;
UCHAR TempCount;
DWORD errorCode;
HANDLE IcmpHandle = NULL;
int err;
struct in_addr addr;
BOOL result;
PICMP_ECHO_REPLY reply;
BOOL sourceRouting = FALSE;
char *SendBuffer = NULL;
char *RcvBuffer = NULL;
UINT RcvSize;
UINT SendSize = DEFAULT_SEND_SIZE;
BOOL bRet = FALSE;
IPAddr address = 0; // was local to main earlier.
char *arg;
// ====================================================================
#ifdef UNICODE
LPSTR pszAnsiServerName = NULL;
DWORD uSize;
if (pszServerName && (uSize = WideCharToMultiByte(CP_ACP,
0,
pszServerName,
-1,
NULL,
0,
NULL,
NULL))) {
if (pszAnsiServerName = (LPSTR) LocalAlloc(LPTR, uSize)) {
if (!WideCharToMultiByte (CP_ACP,
0,
pszServerName,
-1,
pszAnsiServerName,
uSize,
NULL,
NULL))
goto CleanUp;
}
else
goto CleanUp;
}
arg = pszAnsiServerName;
#else
arg = pszServerName;
#endif
if (WSAStartup( 0x0101, &WsaData))
goto CleanUp;
// Added check for INADDR_NONE, MohsinA, 21-Oct-96.
address = get_pingee(arg, &hostname, &was_inaddr, dnsreq);
if(!address || (address == INADDR_NONE) ){
SetLastError (DNS_ERROR_INVALID_IP_ADDRESS);
goto CleanUp;
}
IcmpHandle = IcmpCreateFile();
if (IcmpHandle == INVALID_HANDLE_VALUE) {
goto CleanUp;
}
if (! (SendBuffer = (char *) LocalAlloc(LMEM_FIXED, SendSize))) {
goto CleanUp;
}
//
// Calculate receive buffer size and try to allocate it.
//
if (SendSize <= DEFAULT_SEND_SIZE) {
RcvSize = DEFAULT_BUFFER_SIZE;
}
else {
RcvSize = MAX_BUFFER_SIZE;
}
if (! (RcvBuffer = (char *)LocalAlloc(LMEM_FIXED, RcvSize))) {
goto CleanUp;
}
//
// Initialize the send buffer pattern.
//
for (i = 0; i < SendSize; i++) {
SendBuffer[i] = 'a' + (i % 23);
}
//
// Initialize the send options
//
SendOpts.OptionsData = Opt;
SendOpts.OptionsSize = (UCHAR)OptLength;
SendOpts.Ttl = TTL;
SendOpts.Tos = TOS;
SendOpts.Flags = Flags;
addr.s_addr = address;
#if 0
if (hostname) {
NlsPutMsg(
STDOUT,
PING_HEADER1,
hostname,
inet_ntoa(addr),
SendSize
);
// printf("Pinging Host %s [%s]\n", hostname, inet_ntoa(addr));
} else {
NlsPutMsg(
STDOUT,
PING_HEADER2,
inet_ntoa(addr),
SendSize
);
// printf("Pinging Host [%s]\n", inet_ntoa(addr));
}
#endif
// for (i = 0; i < Count; i++) {
numberOfReplies = IcmpSendEcho(
IcmpHandle,
address,
SendBuffer,
(unsigned short) SendSize,
&SendOpts,
RcvBuffer,
RcvSize,
Timeout
);
if (numberOfReplies == 0) {
errorCode = GetLastError();
goto CleanUp;
// Need to try again? - weihaic
}
else {
reply = (PICMP_ECHO_REPLY) RcvBuffer;
while (numberOfReplies--) {
struct in_addr addr;
addr.S_un.S_addr = reply->Address;
// printf(
// "Reply from %s:",
// inet_ntoa(addr),
// );
if (reply->Status == IP_SUCCESS) {
// printf(
// "Echo size=%d ",
// reply->DataSize
// );
bRet = TRUE;
break;
// printf("\n time rt=%dms min %d, max %d, total %d\n",
// reply->RoundTripTime,
// time_min, time_max, time_total );
// printf("TTL=%u\n", (UINT)reply->Options.Ttl);
//
// Ignore Option fields
// if (reply->Options.OptionsSize) {
// ProcessOptions(reply, (BOOLEAN) dnsreq);
//}
}
/* else {
for (j=0; ErrorTable[j].Error != IP_GENERAL_FAILURE; j++) {
if (ErrorTable[j].Error == reply->Status) {
break;
}
}
} */
reply++;
}
}
// }
CleanUp:
#ifdef UNICODE
if (pszAnsiServerName) {
LocalFree(pszAnsiServerName);
}
#endif
if (IcmpHandle) {
IcmpCloseHandle(IcmpHandle);
}
if (SendBuffer) {
LocalFree(SendBuffer);
}
if (RcvBuffer) {
LocalFree(RcvBuffer);
}
return bRet;
}