成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

對ICMP路由跟蹤的研究

網(wǎng)絡(luò) 網(wǎng)絡(luò)管理
下面我們來對ICMP的路由跟蹤進行一下講解。首先我們需要對ICMP協(xié)議的基礎(chǔ)知識進行一下分析,然后對于其代碼再來進一步研究。

在ICMP協(xié)議的使用中,我們來介紹一下路由跟蹤的應(yīng)用。那么,在掌握了ICMP協(xié)議的一些基本介紹后,我們來對具體的路由跟蹤的實現(xiàn)進行一下分析和講解,希望對大家有所幫助。

一、概述

計算機在Internet中傳遞信息時,必須要經(jīng)過路由器進行網(wǎng)絡(luò)路由才能找到目的主機,把信息送到目的主機.路由器中都有一張路由表,表中保存了從本路由器到某一主機的路由信息,路由器就是通過該路由表進行網(wǎng)絡(luò)尋徑的.兩臺主機之間并沒有一條固定的路徑(即路由表并不固定),該路徑隨著網(wǎng)絡(luò)的變動而作相應(yīng)的變動,因而我們并不能直接從某一主機上得到去往另一主機的路徑,要得到本機與網(wǎng)絡(luò)上某臺主機的網(wǎng)絡(luò)路徑就必須要進行路由跟蹤.本文將介紹一種實現(xiàn)路由跟蹤的方法.

二、ICMP簡介

ICMP即Internet控制報文協(xié)議是一種用于特殊用途的報文機制,可以使互聯(lián)網(wǎng)中的路由器或主機報告差錯或提供有關(guān)意外情況的信息.

ICMP報文為兩級封裝,ICMP報文放在IP數(shù)據(jù)報的數(shù)據(jù)部分,IP數(shù)據(jù)報則放在幀的數(shù)據(jù)中進行網(wǎng)絡(luò)傳輸(如下圖1所示).ICMP報文與其他普通報文一樣,具有相同的路由選擇,并沒有特殊的優(yōu)先權(quán)和增加可靠性.

(圖1)ICMP報文的封裝

(圖1)ICMP報文的封裝

在ICMP包頭中包含了三個字段:1字節(jié)類型域、1字節(jié)代碼域、2字節(jié)校驗和.類型域表示了該報文的類型,如:回應(yīng)請求報文,數(shù)據(jù)報超時報文等,代碼域表示了該類型的幾種不同情況,如:當類型為11(超時報文)時,代碼為0表示TTL超時,為1表示片重組超時.在實現(xiàn)本文中所述的功能時要發(fā)送回應(yīng)請求報文(類型為8),過程如下:源主機向目的主機發(fā)送一個類型為8的回應(yīng)請求報文,若目的站點收到回應(yīng)請求報文則把報文IP包頭部中的目的IP與源IP地址交換,將類型8改為回應(yīng)類型0,計算出新的校驗和再發(fā)往源主機.若源主機收到了該回應(yīng)報文,則不但說明了目的主機可達,而且說明目的主機與源主機之間的路由器工作正常,源主機和目的主機的IP、ICMP軟件運行正常.但若在傳輸過程中了出現(xiàn)了某些問題,如網(wǎng)絡(luò)不通等,導(dǎo)致數(shù)據(jù)被定向到一個無效的目的地,這時相關(guān)路由器或目的主機將發(fā)回目的不可達報文(類型為3),并在代碼中說明該報文的具體情況:是網(wǎng)絡(luò)不可達還是主機不可達等.若請求報文在傳輸過程中超時,即TTL被減為0(報文每經(jīng)過一個路由器TTL都要減1),則該路由器返回一個TTL超時報文(類型為11),報文IP頭中源IP地址即為本路由器的IP地址.

三、路由跟蹤的實現(xiàn)方法

路由跟蹤的實現(xiàn)就是巧妙地利用了ICMP報文的TTL超時報文.其實現(xiàn)過程如下:源主機先向目的主機發(fā)送一個回應(yīng)請求報文(類型8),TTL值設(shè)為1,第一個路由器收到后將TTL減1,這樣TTL變?yōu)?,分組被廢除,同時路由器向源主機發(fā)送一個TTL超時報文(類型為11),報文的IP包頭中的源IP地址就是第一個路由器的地址,源主機就可以通過對該報文進行分析,得到第一個路由器的地址.接著發(fā)送TTL等于2的報文得到第二個路由器地址,再發(fā)TTL等于3的報文,如此下去直到收到目的主機的回應(yīng)應(yīng)答報文(類型為0)或目的不可達報文(類型為3),或者到了最大跳數(shù)(要檢測路由器個數(shù)的最大值).可以看到,對TTL的設(shè)置是實現(xiàn)跟蹤的關(guān)鍵,使用函數(shù)setsockopt(m_Sock, IPPROTO_IP, IP_TTL,(LPSTR)&TTL,sizeof(int)) 可以對其進行設(shè)置,m_Sock是所創(chuàng)建的套接字,IP_TTL說明是進行TTL設(shè)置,TTL即是要設(shè)置的TTL值,為一個整形數(shù)值 .其實現(xiàn)流程可用下圖2表示:

(圖2) 流程圖

(圖2) 流程圖

四、路由跟蹤程序?qū)崿F(xiàn)

本文所介紹的程序是使用了Visual C++6.0編寫,其過程如下:

1、創(chuàng)建一個新的基于對話框的AppWizard工程,并命名為RouteTrace.

2、在stdafx.h中加入#include "winsock2.h".

3、打開選擇菜單Project->Setting (ALT+F7),進入Project Setting 對話框,在Link下的 Object/library modules 輸入ws2_32.lib,然后點OK.

4、自定義一個ICMP類.點擊菜單中的Insert->New Class,進入New Class對話框,在Class type中選擇Generic Class,在Name中輸入類名CICMP,然后點OK,這樣就新建了一個CICMP類.

5、將對話框設(shè)置成如圖3所示的樣子:

(圖3 程序界面)

(圖3 程序界面)

啟動Class Wizard 為各控件添加響應(yīng)函數(shù)和關(guān)聯(lián)變量,控件對應(yīng)的ID及響應(yīng)函數(shù)或變量為:

控件 ID 響應(yīng)函數(shù) 變量
地址組合框 IDC_COMBO   CComboBox m_comb
最大跳數(shù)編輯框 IDC_MAXHOT   int m_maxhot
跟蹤按鈕 IDC_TRACE OnTrace()  
停止按鈕 IDC_STOP OnStop()  
列表框 IDC_LIST   CListCtrl m_list

#p#

五、路由跟蹤代碼

在完成了對各控件的設(shè)置和類的添加以后就是對代碼的編寫了,這里給出了新建類CICMP和RouteTraceDlg.cpp的代碼,詳細代碼請參看源程序。

ICMP.cpp文件代碼:

// ICMP.cpp: implementation of the CICMP class.
//
//////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "RouteTrace.h"
#include "ICMP.h"
#include "ws2tcpip.h"  //實現(xiàn) IP_TTL 設(shè)置的關(guān)鍵
#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[]=__FILE__;
#define new DEBUG_NEW
#endif
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CICMP::CICMP()
{
  winsock = 0;
  m_pIp = NULL;
  m_pIcmp = NULL;
  m_pIp = (IP_HEAD *)new BYTE[MAX_PACKET];
  m_pIcmp = (ICMP_HEAD *)new BYTE[MAX_PACKET];
}
CICMP::~CICMP()
{
  delete [] m_pIp;
  delete [] m_pIcmp;
}
BOOL CICMP::Initialize()
{
  WSADATA wsadata;
  if( WSAStartup(MAKEWORD(2, 1),&wsadata) ) 
  {
    AfxMessageBox("WSAStartup初始化失敗!");
    return FALSE;
  }
  
  winsock= WSASocket (AF_INET,   //建立socket
          SOCK_RAW,
          IPPROTO_ICMP,
          NULL, 0,0);
  if(!winsock)  {
    AfxMessageBox( "Socket創(chuàng)建失敗!");
    return FALSE;
  }
  int timeout =5000;
  setsockopt(winsock,SOL_SOCKET,SO_RCVTIMEO,(char *)&timeout,  // 設(shè)置接收超時
    sizeof(timeout));
  timeout = 5000;
  setsockopt(winsock,SOL_SOCKET,SO_SNDTIMEO,(char *)&timeout,  //設(shè)置發(fā)送超時
    sizeof(timeout));
  return TRUE;
}
void CICMP::Uninitialize()        //釋放Socket
{
  if(winsock)
    closesocket(winsock);
  WSACleanup();
}
USHORT CICMP::CheckSum(USHORT *buffer, int size) //計算校驗和
{
 unsigned long cksum = 0;
 while(size > 1) {
  cksum+=*buffer++;
  size -=sizeof(USHORT);
 }
 
 if(size ) {
  cksum += *(UCHAR*)buffer;
 }
 cksum = (cksum >> 16) + (cksum & 0xffff);
 cksum += (cksum >>16);
 return (USHORT)(~cksum);
}
//--------------------發(fā)送ICMP回應(yīng)請求報文-------------------
BOOL CICMP::SendICMPPack(char *pAddr)
{
  sockaddr_in sockAddr;
  memset((void *)&sockAddr,0,sizeof(sockAddr));
  sockAddr.sin_family = AF_INET;
  sockAddr.sin_port = 0;
  sockAddr.sin_addr.S_un.S_addr=inet_addr(pAddr);
  return SendICMPPack(&sockAddr);
}
BOOL CICMP::SendICMPPack(sockaddr_in *pAddr)
{
  //填充ICMP數(shù)據(jù)各項
  int state;
  char *p_data;
  m_pIcmp->type = ICMP_ECHO;
  m_pIcmp->code = 0;
  m_pIcmp->ID = (USHORT)GetCurrentProcessId();
  m_pIcmp->number = 0;
  m_pIcmp->time = GetTickCount();
  m_pIcmp->cksum = 0;
  //填充數(shù)據(jù)
  p_data = ((char *)m_pIcmp + sizeof(ICMP_HEAD));
  memset((char *)p_data,''0'',DEF_PACKET);
  //校驗和
  m_pIcmp->cksum = CheckSum((USHORT *)m_pIcmp,
    DEF_PACKET+sizeof(ICMP_HEAD));
  
  //發(fā)送數(shù)據(jù)
  state = sendto(winsock,(char *)m_pIcmp,
    DEF_PACKET+sizeof(ICMP_HEAD),
    NULL,(struct sockaddr *)pAddr,sizeof(sockaddr));
  if(state == SOCKET_ERROR) {
    if(GetLastError()==WSAETIMEDOUT)
      m_strInfo = "連接超時!(發(fā)送)";
    else
      m_strInfo.Format("出現(xiàn)未知發(fā)送錯誤!");
    return FALSE;
  }
  if(state <DEF_PACKET) {
    m_strInfo = "發(fā)送數(shù)據(jù)錯誤!";
    return FALSE;
  }
  
  memcpy((void *)&m_sockAddr,(void *)pAddr,
    sizeof(sockaddr_in));
  return TRUE;
}
//----------------------接收數(shù)據(jù)----------------------------
BOOL CICMP::RecvICMPPack()
{
  int state;
  int len = sizeof(sockaddr_in);
  char * addr;
  struct hostent *lpHostent = NULL;
  addr = inet_ntoa(m_sockAddr.sin_addr);
  state = recvfrom(winsock,(char *)m_pIp,MAX_PACKET,0,
    (struct sockaddr*)&m_sockAddr,&len);
  if (state == SOCKET_ERROR) {
    if (WSAGetLastError() == WSAETIMEDOUT)
    {  m_strInfo="接收超時,路由跟蹤失敗!";
    routestate=0;
    RouteState="路由跟蹤失敗!";
    }
    else
      m_strInfo = "未知接收錯誤!";
    return FALSE;
  }
  //分析數(shù)據(jù)
  int ipheadlen;
  ipheadlen = m_pIp->HeadLen * 4 ;
  if (state < (ipheadlen+MIN_PACKET))  {
    m_strInfo = "目的地址的響應(yīng)數(shù)據(jù)不正確";
    return FALSE;
  }
  ICMP_HEAD * p_icmprev;
  p_icmprev = (ICMP_HEAD*)((char *)m_pIp + ipheadlen);
    switch (p_icmprev->type)
    {
    case ICMP_ECHOREPLY: //收到正常回顯
    {
    m_strInfo.Format("接收到%s %d字節(jié)響應(yīng)數(shù)據(jù),響應(yīng)時間:%dms.",
    inet_ntoa(m_sockAddr.sin_addr),len,GetTickCount()-p_icmprev->time);
    routeaddr=addr;
    routestate=0;  
    RouteState="到達目的主機!";
    return TRUE;
      break;  
    }
    case ICMP_TTLOUT:  // TTL超時
    {  routeaddr=inet_ntoa(m_sockAddr.sin_addr);
      routestate=1;
      RouteState="測試到路由器!";
      return TRUE;
      break;
    }
    
    case ICMP_DESUNREACH: //目的不可達
    {  m_strInfo = "目的不可達!";
        routestate=0;
      RouteState="目的不可達!";
      return TRUE;
      break;
    }
        
    default :{ routestate=0;
          m_strInfo="未知錯誤!";
          RouteState="不明狀態(tài)!";
         }
    }
    return TRUE;
}
//----------------設(shè)置TTL--------------------
int CICMP::SetTTL(int TTL)
{
  int nRet=setsockopt(winsock, IPPROTO_IP, IP_TTL,(LPSTR)&TTL,sizeof(int));
  
  if(nRet==SOCKET_ERROR)
  {  CString ttlerr;
    ttlerr.Format("設(shè)置 TTL 錯誤!");
    AfxMessageBox(ttlerr);
    return 0;
  }
  return 1;
}

  RouteTraceDlg.cpp文件代碼:

// RouteTraceDlg.cpp : implementation file
//
#include "stdafx.h"
#include "RouteTrace.h"
#include "RouteTraceDlg.h"
#include "afxmt.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/////////////////////////////////////////////////////////////////////////////
// CAboutDlg dialog used for App About
struct SubThreadInfo
{
  CDialog* pDialog;
  CListCtrl*  list;
  CStatic* state;
  CString IPStr;
  int Maxhot;
  
} Info;
CEvent eventStopRoute;
//-----------------路由跟蹤線程---------------
UINT ThreadRoute(LPVOID pParam)     
{
  SubThreadInfo* pInfo = (SubThreadInfo*)pParam;
  CRouteTraceDlg* pThreadDlg = (CRouteTraceDlg*)pInfo->pDialog;
  CICMP m_icmp;
  CString IPStr=pInfo->IPStr;
  CString sTTL;
  int nTtl;
  m_icmp.Initialize();
  for(nTtl=1;nTtl<=pInfo->Maxhot;nTtl++)
  {
  if(m_icmp.SetTTL(nTtl)==0)
    return 0;  
  sTTL.Format("%d",nTtl);  
  if(m_icmp.SendICMPPack((char *)(LPCSTR)IPStr)) 
  m_icmp.RecvICMPPack();
  
  
  {  
  int i=pInfo->list->InsertItem(0,sTTL);
   pInfo->list->SetItemText(i,1,m_icmp.routeaddr);
   pInfo->list->SetItemText(i,2,m_icmp.RouteState);
    pInfo->state->SetWindowText(m_icmp.m_strInfo);
    Sleep(100);
  }
if(m_icmp.routestate==0)  //收到非TTL超時報文則跳出循環(huán)
     break;
if(WaitForSingleObject(eventStopRoute.m_hObject, 0) == WAIT_OBJECT_0)
     break;     //收到停止信號則跳出循環(huán)
  }
pThreadDlg->Routeflag=TRUE;
 return 0;
}
……  //系統(tǒng)代碼
BOOL CRouteTraceDlg::OnInitDialog()
{
  CDialog::OnInitDialog();
…… //系統(tǒng)代碼
  
  // TODO: Add extra initialization here
  m_list.InsertColumn(0,"標號",LVCFMT_CENTER,60,0);
  m_list.InsertColumn(1,"路由器地址",HDF_CENTER,200,0);
  m_list.InsertColumn(2,"狀態(tài)",HDF_CENTER,100,0);
  ListView_SetExtendedListViewStyleEx(m_list.m_hWnd, LVS_EX_FULLROWSELECT, 0xFFFFFFFF);
  return TRUE; // return TRUE unless you set the focus to a control
}
…… //系統(tǒng)代碼
void CRouteTraceDlg::OnTrace()
{
  CString str;
  UpdateData(TRUE);
  CWnd * pWnd;
  pWnd = GetDlgItem(IDC_COMBO);
  pWnd->GetWindowText(str);
  if(str.IsEmpty()) {
    MessageBox("請輸入地址!");
    pWnd->SetFocus();
    return;
  }
  if (m_comb.FindStringExact(-1, str) == CB_ERR)
  m_comb.AddString(str);
   m_list.DeleteAllItems();
   if(Routeflag)
   {
    Routeflag=FALSE;
    Info.IPStr=str;
    Info.pDialog=this;
    Info.Maxhot=m_maxhot;
    Info.list=(&m_list);
    Info.state=(&m_statectl);
    AfxBeginThread(ThreadRoute, &Info); //在線程中實現(xiàn)路由跟蹤
   }
}
void CRouteTraceDlg::OnStop()
{
   if(!Routeflag)
   {  
    eventStopRoute.SetEvent();  // 發(fā)出停止信號
   }
}
void CRouteTraceDlg::OnDestroy()
{
  CDialog::OnDestroy();
  
  // TODO: Add your message handler code here
  m_icmp.Uninitialize();
}

代碼運行效果圖如下:

代碼運行效果圖

六、結(jié)束語

本文所述程序在 Windows 98 下 Visual C++6.0 中調(diào)試通過,并在寬帶網(wǎng)中測試成功。讀者可根據(jù)實際需要修改程序中的參數(shù),以實現(xiàn)更強大的功能,比如可以把超時值設(shè)置成更合適的值,或設(shè)置成可動態(tài)輸入的形式。
 

責任編輯:佟健 來源: 賽迪網(wǎng)
相關(guān)推薦

2010-09-16 11:05:43

2010-09-29 12:52:14

2010-12-14 15:07:15

ICMP路由重定向

2010-07-28 17:36:17

2010-08-05 08:49:19

2010-12-22 11:27:23

2015-05-11 09:43:07

路由traceroute

2011-01-18 13:50:20

路由跟蹤tcptracerou

2020-12-18 09:49:48

iOS ChromiWKWebViewJavaScript

2011-04-18 15:24:40

Tracert命令ICMP

2010-08-02 16:59:54

2010-07-13 10:10:07

2009-12-22 15:25:23

2010-05-31 17:45:50

MySQL行鎖

2021-01-18 09:30:25

惡意軟件網(wǎng)絡(luò)攻擊網(wǎng)絡(luò)安全

2010-04-12 16:24:15

Oracle表查詢

2022-11-07 07:54:05

微服務(wù)數(shù)據(jù)庫網(wǎng)關(guān)

2010-06-21 13:07:14

2010-08-03 09:23:25

2011-04-15 13:30:07

Tracert路由
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 99精品欧美一区二区三区综合在线 | 久久久久亚洲 | 亚洲成人av | 国产一级视频在线观看 | 国产一区二区三区久久 | 亚洲国产精品一区二区第一页 | 亚洲高清一区二区三区 | 久草福利 | 国产精品精品视频一区二区三区 | 久草久草久草 | 久久精品综合 | 九九九视频在线观看 | 狠狠色狠狠色综合系列 | 一区不卡在线观看 | 9porny九色视频自拍 | 午夜免费网站 | 免费视频99 | 狠狠爱免费视频 | 国产亚洲精品久久久久久牛牛 | 天天操天天操 | 91av在线不卡| 亚洲欧美精品国产一级在线 | 久久精品色欧美aⅴ一区二区 | 99精品久久 | 玖玖国产 | 国产精品日韩欧美一区二区 | 久久久久久亚洲欧洲 | 亚洲欧美中文日韩在线v日本 | 成人免费在线播放视频 | 四虎网站在线观看 | 天天干夜夜操 | 亚洲综合色自拍一区 | 久久久爽爽爽美女图片 | 精品欧美色视频网站在线观看 | 一区二区三区av | 夏同学福利网 | 久久黄色网 | 日韩免费激情视频 | 黄色网址在线播放 | 国产在线a视频 | 911影院|