透析ICMP協議(三): 牛刀初試之一 應用篇ping(ICMP.dll) (轉)

worldblog發表於2007-12-14
透析ICMP協議(三): 牛刀初試之一 應用篇ping(ICMP.dll) (轉)[@more@]

透析ICMP(三): 牛刀初試之一
 應用篇(ICMP.dll)
===============================
這篇文章出自:
翻譯: free/CSDN, 對原始程式碼加了些註釋
平臺: VC6 XP

原理簡介:
--------
這個例子演示了應用的ICMP.DLL怎樣"ping"另一臺機器. 這個DLL是沒有文件話的傳送ICMP回送包介面, 也稱為"pings," 就像潛水員對聲納訊號的術語一樣. 這段程式碼出自一個被一個名叫MarkG的傢伙的GUI, 他的網頁已經消失了.

ICMP.DLL API 現在在Windows平臺上與微軟的Winsocks工作的很好, 但是微軟說更好的產品一出來他們將替換它. 微軟說這個自從Windows 95時代就在用, 這些功能在在上仍然存在.

For more information on the ICMP.DLL API, check out sockets.com's ICMP API page.
更詳細的ICMP.DLL API的資訊到sockets.com的ICMP API網頁獲取.


具體實現:
--------
// Borland C++ 5.0: bcc32.cpp ping.cpp
// Visual C++ 5.0:  cl ping.cpp wsock32.lib
//
// This sample program is hereby placed in the public ain.

#include
#include
#include
#include "icmpdefs.h"

==================ping的實現部分==================
int doit(int argc, char* argv[])
{//[bugfree] 建議將這個argc和argv的處理拿到main中
  // 檢查命令列引數
  if (argc < 2) {
  cerr << "usage: ping " << endl;
  return 1;
  }
 
  // 裝載ICMP.DLL連線庫
  HINSTANCE hIcmp = LoadLibrary("ICMP.DLL");
  if (hIcmp == 0) {
  cerr << "Unable to locate ICMP.DLL!" << endl;
  return 2;
  }

  // 查詢給定機器的資訊
  struct hostent* phe;
  if ((phe = gethostbyname(argv[1])) == 0) {
  cerr << "Could not find address for " << argv[1] << endl;
  return 3;
  }

  // 定義函式三個指標型別
  typedef HANDLE (WINAPI* pfnHV)(VOID);
  typedef BOOL (WINAPI* pfnBH)(HANDLE);
  typedef D (WINAPI* pfnDHDPWPipPDD)(HANDLE, DWORD, LPVOID, WORD,
  PIP_OPTION_INFORMATION, LPVOID, DWORD, DWORD); // evil, no?
  //定義三個指標函式
  pfnHV pIcmpCreateFile;
  pfnBH pIcmpCloseHandle;
  pfnDHDPWPipPDD pIcmpSendEcho;
 
  //從ICMP.DLL中得到函式入口地址
  pIcmpCreateFile = (pfnHV)GetProcAddress(hIcmp,  "IcmpCreateFile");
  pIcmpCloseHandle = (pfnBH)GetProcAddress(hIcmp, "IcmpCloseHandle");
  pIcmpSendEcho = (pfnDHDPWPipPDD)GetProcAddress(hIcmp, "IcmpSendEcho");
  if ((pIcmpCreateFile == 0) || (pIcmpCloseHandle == 0) ||
  (pIcmpSendEcho == 0)) {
  cerr << "Failed to get proc addr for function." << endl;
  return 4;
  }

  // 開啟ping服務
  HANDLE hIP = pIcmpCreateFile();
  if (hIP == INVALID_HANDLE_VALUE) {
  cerr << "Unable to open ping service." << endl;
  return 5;
  }
 
  // 構造ping資料包
  char acPingBuffer[64];
  memset(acPingBuffer, 'xAA', sizeof(acPingBuffer));
  PIP_ECHO_REPLY pIpe = (PIP_ECHO_REPLY)GlobalAlloc( GMEM_FIXED | GMEM_ZEROINIT,
  sizeof(IP_ECHO_REPLY) + sizeof(acPingBuffer));
  if (pIpe == 0) {
  cerr << "Failed to allocate global ping packet buffer." << endl;
  return 6;
  }
  pIpe->Data = acPingBuffer;
  pIpe->DataSize = sizeof(acPingBuffer); 

  // 傳送ping資料包
  DWORD dwStatus = pIcmpSendEcho(hIP, *((DWORD*)phe->h_addr_list[0]),
  acPingBuffer, sizeof(acPingBuffer), NULL, pIpe,
  sizeof(IP_ECHO_REPLY) + sizeof(acPingBuffer), 5000);
  if (dwStatus != 0) {
  cout << "Addr: " <<
  int(LOBYTE(LOWORD(pIpe->Address))) << "." <<
  int(HIBYTE(LOWORD(pIpe->Address))) << "." <<
  int(LOBYTE(HIWORD(pIpe->Address))) << "." <<
  int(HIBYTE(HIWORD(pIpe->Address))) << ", " <<
  "RTT: " << int(pIpe->RoundTripTime) << "ms, " <<
  "TTL: " << int(pIpe->Options.Ttl) << endl;
  }
  else {
  cerr << "Error obtaining info from ping packet." << endl;
  }

  // 關閉,回收資源
  GlobalFree(pIpe);
  FreeLibrary(hIcmp);
  return 0;
}
==================主函式==================
int main(int argc, char* argv[])
{
  WSAData wsaData;
  if (Wtartup(MAKEWORD(1, 1), &wsaData) != 0) {
  return 255;
  }

  int retval = doit(argc, argv);

  WSACleanup();
  return retval;
}

==================頭==================
icmpdefs.h
//ICMP.DLL 函式中需要的結構


typedef struct {
  unsigned char Ttl;  // Time To Live
  unsigned char Tos;  // Type Of Service
  unsigned char Flags;  // IP header flags
  unsigned char OptionsSize;  // Size in bytes of options data
  unsigned char *OptionsData;  // Pointer to options data
} IP_OPTION_INFORMATION, * PIP_OPTION_INFORMATION;

typedef struct {
  DWORD Address;  // Replying address
  unsigned long  Status;  // Reply status
  unsigned long  RoundTripTime;  // RTT in milliseconds
  unsigned short DataSize;  // Echo data size
  unsigned short Reserved;  // Reserved for system use
  void *Data;  // Pointer to the echo data
  IP_OPTION_INFORMATION Options;  // Reply options
} IP_ECHO_REPLY, * PIP_ECHO_REPLY;
 

連結:
-------
我的其它文章,<>, 和其它文章參見:
http://www.csdn.net/develop/author/netauthor/bugfree/

聯絡方式:
-------
 to:zhangliangsd@.com">zhangliangsd@hotmail.com 


來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/10752043/viewspace-993156/,如需轉載,請註明出處,否則將追究法律責任。

相關文章