解析ip數(shù)據(jù)包課程設(shè)計_第1頁
已閱讀1頁,還剩15頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)

文檔簡介

1、<p><b>  課程設(shè)計任務(wù)書</b></p><p><b>  目錄</b></p><p><b>  1.實驗?zāi)康?lt;/b></p><p><b>  2.實驗要求</b></p><p><b>  3.預(yù)備知識</

2、b></p><p><b>  4.課程設(shè)計分析</b></p><p><b>  5.實現(xiàn)過程</b></p><p><b>  6.程序流程圖</b></p><p><b>  7.相關(guān)擴(kuò)展</b></p><p>

3、;<b>  8.實習(xí)體會</b></p><p><b>  9.參考文獻(xiàn)</b></p><p><b>  一.實驗?zāi)康模?lt;/b></p><p>  設(shè)計一個解析IP數(shù)據(jù)包的程序,并根據(jù)這個程序,說明IP數(shù)據(jù)包的結(jié)構(gòu)及IP協(xié)議的相關(guān)問題,從而IP層的工作原理有更好的理解和認(rèn)識.</p&g

4、t;<p>  二.實驗要求 : </p><p>  本設(shè)計的目標(biāo)是捕獲網(wǎng)絡(luò)中數(shù)據(jù)包,解析數(shù)據(jù)包的內(nèi)容,將、結(jié)果顯示在標(biāo)準(zhǔn)輸出上,并同時寫入日志文件.</p><p>  程序的具體要求如下:</p><p>  以命令行形式運(yùn)行:ipparse logfile,其中ipparse是程序名,而logfile則代表記錄結(jié)果的日志文

5、件.</p><p>  在標(biāo)準(zhǔn)輸出和日志文件中寫入捕獲的IP數(shù)據(jù)包的版本,頭長度,服務(wù)類型,數(shù)據(jù)包總長度,數(shù)據(jù)包標(biāo)識,分段標(biāo)志,分段偏移值,生存時間,上層協(xié)議類型,頭校驗和,源IP地址和目的IP地址等內(nèi)容.</p><p>  當(dāng)程序接收到鍵盤輸入Ctrl+C時退出.</p><p><b>  三.預(yù)備知識 </b></p>

6、<p>  互聯(lián)網(wǎng)絡(luò)層是TCP/IP協(xié)議參考模型中的關(guān)鍵部分.IP協(xié)議把傳輸層送來的消息組裝成IP數(shù)據(jù)包,并把IP數(shù)據(jù)包傳送給數(shù)據(jù)鏈層.IP協(xié)議在TCP/IP協(xié)議族中處于核心地位,IP協(xié)議制定了統(tǒng)一的IP數(shù)據(jù)包格式,以消除個通信子網(wǎng)中的差異,從而為信息發(fā)送方和接收方提供了透明的傳輸通道.編制本程序前,首先要對IP包的格式有一定了解,圖1給出了IP協(xié)議的數(shù)據(jù)包格式.</p><p>  IP數(shù)據(jù)包的第一

7、個字段是版本字段,其度是4位,表示所使用的IP協(xié)議的版本.目前的版本是IPV4,版本字段的值是4,下一代版本是IPV6,版本字段值是6.本程序主要針對版本是IPV4的數(shù)據(jù)包的解析.</p><p>  報頭標(biāo)長字段為4位,它定義了以4B為一個單位的IP包的報文長度.報頭中除了選項字段和填充域字段外,其他各字段是定長的.因此,IP數(shù)據(jù)包的頭長度在20—40B之間,是可變的.</p><p>

8、  0 4 8 16 19 24 31</p><p>  圖1 IP數(shù)據(jù)包的格式</p><p>  服務(wù)類型字段共8位,用于指示路由器如何處理該數(shù)據(jù)包.該字段長度由4位服務(wù)類型(TOS)子域和3位優(yōu)先級子域組成,1位為保留位,該字段結(jié)構(gòu)如圖2所示.</p&

9、gt;<p>  B7 b6 b5 b4 b3 b2 b1 b0</p><p>  圖2 服務(wù)類型字段結(jié)構(gòu)</p><p>  優(yōu)先級共有8種,優(yōu)先級越高表明數(shù)據(jù)包越重要.表1中列出了各種優(yōu)先級所代表的意義.</p><p>  表一 優(yōu)先子域

10、的說明</p><p>  在4位服務(wù)類型子域中b4,b3,b2,b1分別表示D(延遲),T(吞吐量),R(可靠性)與C(成本).表2列出了服務(wù)器類型自域的構(gòu)成.</p><p>  總長度字段為2B,它定義了以字節(jié)為單位的數(shù)據(jù)包的總長度.IP數(shù)據(jù)包的最大長度為65535B.</p><p>  標(biāo)識字段的長度為16位,用于識別IP數(shù)據(jù)包的編號.每批數(shù)據(jù)都要有一個標(biāo)

11、識值,用于讓目的主機(jī)判斷新來的數(shù)據(jù)屬于哪個分組.</p><p>  報頭中的標(biāo)志字段如圖7-3所示.標(biāo)志字段共3位,最高位是0.禁止分片標(biāo)志DF(do not fragment)字段的值若為1,表示不能對數(shù)據(jù)包分片;若DF值為0,則表明可以分片.分片標(biāo)志MF(more fragment)的值為1,表示接收到的不是最后一個分片;若MF值為0,表示接收到的是最后一個分片.</p><p>

12、  片偏移字段共13位,說明分片在整個數(shù)據(jù)包中的相對位置.片偏移值是以8B為單位來記數(shù)的,因此選擇的分片長度應(yīng)該是8B的整數(shù)倍.</p><p>  生存時間(TTL)字段為8位,用來設(shè)置數(shù)據(jù)包在互聯(lián)網(wǎng)絡(luò)的傳輸過程的壽命,通常是用一個數(shù)據(jù)包可以經(jīng)過的最多的路由器跳步數(shù)來限定的.</p><p>  協(xié)議字段為8位,表示使用此IP數(shù)據(jù)包的高層協(xié)議類型,常用的協(xié)議號如表7-3所示.</p

13、><p>  表7-3 典型的協(xié)議號</p><p>  頭校驗和字段為16位,用于存放檢查報頭錯誤的校驗碼。檢驗的范圍是整個IP包的報頭。校驗和按如下方法計算:</p><p>  1)將頭校驗和的字段置為0。</p><p>  2)將報頭部分的所有數(shù)據(jù)以16位為單位進(jìn)行累加,累加方式是求異或。</p><p>  3

14、)將累加的結(jié)果取反碼,就是頭校驗和。</p><p>  當(dāng)收到一個IP包時,要檢查報頭是否出錯,就把報頭中的所有數(shù)據(jù)以16位為單位進(jìn)行累加,若累加的結(jié)果為0,則報文沒有出錯。</p><p>  地址字段包括源地址和目的地址。源地址和目的地址的長度都是32位,分別表示發(fā)送數(shù)據(jù)包的源主機(jī)和目的主機(jī)的IP地址。</p><p>  選項字段的長度范圍為0~40B,主要

15、用于控制和測試。在使用選項字段的過程中,有可能出現(xiàn)報頭部分的長度不是32位的整數(shù)倍的情況。如果出現(xiàn)這種情況,就需要通過填充位來湊齊。</p><p><b>  四.課程設(shè)計分析</b></p><p>  為了獲取網(wǎng)絡(luò)中的IP數(shù)據(jù)包,必須對網(wǎng)卡進(jìn)行編程,在這里我們使用套接字(socket)進(jìn)行編程。但是,在通常情況下,網(wǎng)絡(luò)通信的套接字程序只能響應(yīng)與自己硬件地址相匹配

16、的數(shù)據(jù)包或是以廣播形式出發(fā)的數(shù)據(jù)包。對于其他形式的數(shù)據(jù)包,如已到達(dá)網(wǎng)絡(luò)接口但卻不是發(fā)送到此地址的數(shù)據(jù)包,網(wǎng)絡(luò)接口在驗證投遞地址并非自身地址之后將不引起響應(yīng),也就是說應(yīng)用程序無法收取與自己無關(guān)的數(shù)據(jù)包。我們要想獲取流經(jīng)網(wǎng)絡(luò)設(shè)備的所有數(shù)據(jù)包,就需要將網(wǎng)卡設(shè)置為混雜模式。</p><p>  本程序主要由三部分構(gòu)成:初始化原始套接字,反復(fù)監(jiān)聽捕獲數(shù)據(jù)包和解析數(shù)據(jù)包。下面就結(jié)合核心代碼對程序的具體實現(xiàn)進(jìn)行講解,同時使程序

17、流程更加清晰,去掉了錯誤檢查等保護(hù)性代碼。</p><p><b>  使用原始套接字</b></p><p>  套接字分為三種,即流套接字(Stream Socket)、數(shù)據(jù)報套接字(Datagram Socket)和原始套接字(Raw Socket)。要進(jìn)行IP數(shù)據(jù)包的接受與發(fā)送,應(yīng)使用原始套接字。創(chuàng)建原始套接字的代碼如下:</p><p&g

18、t;  SOCKET sock;</p><p>  Sock=WSASoccet(AF_INET,SOCK_RAW,IPPROTO_IP,NULL,0,WSA_FLAG_OVERRLAPPED);</p><p>  在WSASoccet函數(shù)中,第一個參數(shù)指定通信發(fā)生的區(qū)字段,AF_INET是針對Internet的,允許在遠(yuǎn)程主機(jī)之間通信。第二個參數(shù)是套接字的類型,AF_INET地址族下

19、,有SOCK_STREAM、SOCK_DGRAM、SOCK_RAW三種套接字類型。在這里,我們設(shè)置為SOCK_RAW,表示我們聲明的是一個原始套接字類型。第三個參數(shù)依賴于第二個參數(shù),用于指定套接字所用的特定協(xié)議,這里使用IP協(xié)議。第四個參數(shù)為WSAPROTOCOL_INFO位,該位可以置空,永遠(yuǎn)置0。第六個參數(shù)是標(biāo)志位,WSA_FLAG_OVERRLAPPED表明可以使用發(fā)送接收超時設(shè)置,本課程設(shè)計也可以把這個標(biāo)志位設(shè)置為NULL,因為

20、本設(shè)計不用考慮超時情況。</p><p>  創(chuàng)建原始套接字后,IP頭就會包含在接收的數(shù)據(jù)中。然后,我們可以設(shè)置IP頭操作選項,調(diào)用sotscockpot函數(shù)。其中flag設(shè)置為TRUE,并設(shè)定IP_HDRINCL選項,表明用戶可以親自對IP頭進(jìn)行處理。</p><p>  BOOL flag=true;</p><p>  setsockopt (sock,IPP

21、ROTO_IP,IP_HDRINCL,(char*)&flag,sizeof(flag));</p><p>  之后,使用如下代碼完成對socket的初始化工作</p><p><b>  /*獲取主機(jī)名*/</b></p><p>  char hostname[128];</p><p>  gethost

22、name(hostname, 100);</p><p>  /*獲取IP地址*/</p><p>  hostent *pHostIP;</p><p>  pHostIP=gethostbyname(hostname);</p><p>  /* 填充SOCKADDR_IN的結(jié)構(gòu)內(nèi)容*/</p><p>  soc

23、kaddr_in addr_in;</p><p>  addr_in.sin_addr= *(in_addr*)pHostIP->h_addr_list[0];</p><p>  addr_in.sin_family=AF_TNET;</p><p>  addr-in.sin_port=htons(6000);</p><p> 

24、 /* 綁定socket */</p><p>  bind(sock, (POSCKADDR)&addr_in,sizeof(addr_in));</p><p>  填寫sockaddr_in的內(nèi)容時,其地址值應(yīng)填寫為本機(jī)IP地址可以通過gethostbyname()函數(shù)獲??;端口號可以隨便填寫,但不能與系統(tǒng)沖突;協(xié)議族應(yīng)填寫為AF_INET。注意,sockaddr_in 結(jié)構(gòu)

25、的值必須是以網(wǎng)絡(luò)字節(jié)順序表示的值,而不能直接使用本機(jī)字節(jié)順序的值,使用htoms()函數(shù)可以將無符號短整型的主機(jī)數(shù)據(jù)轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)的順序的數(shù)據(jù)。最后使用bind()函數(shù)將socket綁定到本地網(wǎng)卡上。</p><p>  綁定網(wǎng)卡后,需要WSAIoctl()函數(shù)把網(wǎng)卡設(shè)置為混雜模式,使網(wǎng)卡能夠接收所有網(wǎng)絡(luò)數(shù)據(jù),其關(guān)鍵代碼如下:</p><p>  #define SIO_RCVALL_WS

26、AIOW(IOC_VENDOR,1)</p><p>  DWORD dwBufferLen[10];</p><p>  DWORD dwBufferInLen=1;</p><p>  DWORD dwBytesReturned=0;</p><p>  WSAIoctl(SnifferSocket,IO-RCVALL,&dwBu

27、fferInLen,sizeof(dwBufferInLen),&dwBufferLen,Sizeof(dwBufferLen),&dwByteReturned,NULL,NULL);</p><p>  如果接收的數(shù)據(jù)包中的協(xié)議類型和定義的原始套接字匹配,那么接收到的數(shù)據(jù)就拷貝到套接字中。因此,網(wǎng)卡就可以接收所有經(jīng)過的IP包。</p><p><b>  2.接

28、收數(shù)據(jù)包</b></p><p>  在程序中可使用RECV()函數(shù)接收經(jīng)過的IP包。該函數(shù)有四個參數(shù),第一個參數(shù)接收操作所用的套接字描述符;第二個參數(shù)接收到緩沖區(qū)的地址;第二個參數(shù)接收緩沖區(qū)的地址;第三個參數(shù)接收緩沖區(qū)的大小,也就是所要接收的字節(jié)數(shù);第四個參數(shù)是一個附加標(biāo)志,如果對所發(fā)送的數(shù)據(jù)沒特殊要求,直接設(shè)為0。因為IP數(shù)據(jù)包的最大長度是65536B,因此緩沖區(qū)的大小不能小于65535B。設(shè)置緩

29、沖區(qū)后,可利用循環(huán)來反復(fù)監(jiān)聽接收IP包,用recv()函數(shù)接收功能的代碼如下:</p><p>  #dedine BUFFER_SIZE 65535</p><p>  Char buffer[BUFFER_SIZE]; //設(shè)置緩沖區(qū)</p><p>  While(true)</p><p>  {recv(sock,buffer,

30、BUFFER_SIZE,0); //j接收數(shù)據(jù)包</p><p><b>  ……..</b></p><p>  3.定義IP頭部的數(shù)據(jù)結(jié)構(gòu)</p><p>  程序需要定義一個數(shù)據(jù)結(jié)構(gòu)表示IP頭部。這個數(shù)據(jù)結(jié)構(gòu)應(yīng)該和圖7-1吻合,其代碼如下:</p><p>  typedef struct _IP_HEADE

31、R //定義IP頭</p><p><b>  {</b></p><p><b>  union</b></p><p><b>  {</b></p><p>  BYTE Version; //版本前4位</p><p>  BYTE HdrLe

32、n; //報頭標(biāo)長(后四位),IP頭長度</p><p><b>  };</b></p><p>  BYTE ServiceType;//服務(wù)類型</p><p>  WORD TotalLen; //總長度</p><p>  WORD ID; //標(biāo)識</p><p><b>

33、;  union </b></p><p><b>  {</b></p><p>  WORD Flags; //標(biāo)志</p><p>  Word FragOff; //分段偏移</p><p><b>  };</b></p><p>  BYTE Ti

34、meToLive; //生命期</p><p>  BYTE Protiocol; //協(xié)議</p><p>  WORD HdrChksum; //頭校驗和</p><p>  DWORD SrcAddr; //源地址</p><p>  DWORD DstAddr: //目的地址</p><p>  BYTE

35、 Options; //選項</p><p>  }IP_HEADER;</p><p>  這是我們只考慮IP頭部結(jié)構(gòu),不考慮數(shù)據(jù)部分。在捕獲IP數(shù)據(jù)包后,可以通過指針把緩沖區(qū)的內(nèi)容強(qiáng)制轉(zhuǎn)化為IP_HEADER數(shù)據(jù)結(jié)構(gòu)。</p><p>  IP_HEADER ip = *( IP_HEADER *)buffer;</p><p><

36、;b>  4.IP包的解析</b></p><p>  解析IP包的字段有兩種策略。針對長度為8位、16位和32位的字段 (或子字段)時,可以利用IP_HEADER的成員指教獲取。要解析長度不是9位倍數(shù)的字段(或子字段)時,可以利用C語言中的位移以及與、或操作完成。下面給出了通過IP_HEADER解析IP頭各個字段的代碼。</p><p>  /*獲取版本字段*/<

37、/p><p>  ip.Version>>4;</p><p>  /*獲取頭部長度字段*/</p><p>  ip.HdrLen & 0x0f;</p><p>  /*獲取服務(wù)類型字段中的優(yōu)先級子域*/</p><p>  ip.ServiceType>>5;</p>&

38、lt;p>  /*獲取服務(wù)類型字段中的TOS子域*/</p><p>  (IP.sERVICEtYPE>>1)&0X0F;</p><p>  /*獲取總長度字段*/</p><p>  ip.TotalLEN;</p><p>  /*獲取標(biāo)識字段*/</p><p><b> 

39、 ip.ID;</b></p><p>  /*解析標(biāo)識字段*/</p><p>  DF=(ip.Flags>>14) &0x01;</p><p>  MF=(ip.Flags>>13) &0X01;</p><p>  /*獲取分段偏移字段*/</p><p>

40、  ip.FragOff &0x1fff;</p><p>  /*獲取生存時間字段*/</p><p>  ip.TimeToLive;</p><p>  /*獲取協(xié)議字段*/</p><p>  ip.Protocol;</p><p>  /*獲取頭校驗和字段*/</p><p>

41、;  ip.HdrChksum;</p><p>  /*解析源IP地址字段*/</p><p>  inet_ntoa(*(in_addr*)&ip.SrcAddr;</p><p>  /*解析目的的IP地址字段*/</p><p>  inet_ntoa(*(in_addr*)&ip.DstAddr);</p>

42、;<p><b>  5.參考程序流程圖</b></p><p>  圖7-4給出一個供參考的程序流程圖。 </p><p><b>  五.實現(xiàn)過程</b></p><p>  #include "stdafx.h"</p><p>

43、  #include "winsock2.h"</p><p>  #include "ws2tcpip.h"</p><p>  #include "stdio.h"</p><p>  typedef struct _IP_HEADER</p><p><b>  {&

44、lt;/b></p><p><b>  union</b></p><p><b>  {</b></p><p>  BYTE Version;</p><p>  BYTE HdrLen;</p><p><b>  };</b></p

45、><p>  BYTE ServiceType;</p><p>  WORD TotalLen;</p><p><b>  WORD ID;</b></p><p><b>  union</b></p><p><b>  {</b></p>

46、;<p>  WORD Flags;</p><p>  WORD Fragoff;</p><p><b>  };</b></p><p>  BYTE TimeToLive;</p><p>  BYTE Protocol;</p><p>  WORD HdrChksum;&

47、lt;/p><p>  DWORD SrcAddr;</p><p>  DWORD DstAddr;</p><p>  BYTE Options;</p><p>  }IP_HEADER;</p><p>  void getVersion(BYTE b,BYTE & version)</p>

48、<p><b>  {</b></p><p>  version =b>>4;</p><p><b>  }</b></p><p>  void getIHL(BYTE b,BYTE & result)</p><p><b>  {</b>

49、</p><p>  result = (b & 0x0f) *4;</p><p><b>  }</b></p><p>  char * parseServiceType_getProcedence(BYTE b)</p><p><b>  {</b></p><

50、p>  switch(b>>5)</p><p><b>  {</b></p><p><b>  case 7:</b></p><p>  return "Network Control";</p><p><b>  break;</b&

51、gt;</p><p><b>  case 6:</b></p><p>  return "Internet work Control";</p><p><b>  break;</b></p><p><b>  case 5:</b></p&

52、gt;<p>  return "CRITIC/ECP";</p><p><b>  break;</b></p><p><b>  case 4:</b></p><p>  return "Flash Override";</p><p>

53、;<b>  break;</b></p><p><b>  case 3:</b></p><p>  return "Falsh";</p><p><b>  break;</b></p><p><b>  case 2:</b&g

54、t;</p><p>  return "Immediate";</p><p><b>  break;</b></p><p><b>  case 1:</b></p><p>  return "Priority";</p><p&

55、gt;<b>  break;</b></p><p><b>  case 0:</b></p><p>  return "Routine";</p><p><b>  break;</b></p><p><b>  default :&l

56、t;/b></p><p>  return "Unknown"</p><p><b>  }</b></p><p><b>  }</b></p><p>  char * parseServiceType_getTOS(BYTE b)</p><

57、p><b>  {</b></p><p>  b=(b>>1)&0x0f;</p><p><b>  switch(b)</b></p><p><b>  {</b></p><p><b>  case 0:</b><

58、;/p><p>  return "Normal service";</p><p><b>  break;</b></p><p><b>  case 1:</b></p><p>  return "Minimize monetary cost";<

59、/p><p><b>  break;</b></p><p><b>  case 2:</b></p><p>  return "Maximize reliability";</p><p><b>  break;</b></p><

60、p><b>  case 4:</b></p><p>  return "Maximize throughput";</p><p><b>  break;</b></p><p><b>  case 8:</b></p><p>  retur

61、n "Minimize delay";</p><p><b>  break;</b></p><p><b>  case 15:</b></p><p>  return "Maximize security";</p><p><b>  b

62、reak;</b></p><p><b>  default:</b></p><p>  return "Unknown";</p><p><b>  }</b></p><p><b>  }</b></p><p&g

63、t;  void getFlags(WORD w,BYTE & DF, BYTE & MF)</p><p><b>  {</b></p><p>  DF=(w>>14)&0x01;</p><p>  MF=(w>>13)&0x01;</p><p><

64、b>  }</b></p><p>  void getFragoff(WORD w,WORD & fragoff)</p><p><b>  {</b></p><p>  fragoff=w&0x1ffff;</p><p><b>  }</b></p

65、><p>  char * getProtocol(BYTE Protocol)</p><p><b>  {</b></p><p>  switch (Protocol)</p><p><b>  {</b></p><p><b>  case 1:</

66、b></p><p>  return "ICMP";</p><p><b>  case 2:</b></p><p>  return "IGMP";</p><p><b>  case 4:</b></p><p>  

67、return "IP in IP ";</p><p><b>  case 6:</b></p><p>  return "TCP";</p><p><b>  case 8:</b></p><p>  return "EGP";&

68、lt;/p><p><b>  case 17:</b></p><p>  return "UPD";</p><p><b>  case 41:</b></p><p>  return "IPv6";</p><p><b&g

69、t;  case 46:</b></p><p>  return "OSPF";</p><p><b>  default:</b></p><p>  return "UNKNOWN";</p><p><b>  }</b></p&g

70、t;<p><b>  }</b></p><p>  void ipparse(FILE * file,char *buffer)</p><p><b>  {</b></p><p>  IP_HEADER ip=*(IP_HEADER *)buffer;</p><p>  f

71、seek(file,0,SEEK_END);</p><p>  BYTE version;</p><p>  getVersion(ip.Version,version);</p><p>  fprintf(file,"版本=%d\r\n",version);</p><p>  BYTE headerLen;<

72、;/p><p>  getIHL(ip.HdrLen,headerLen);</p><p>  fprintf(file,"頭長度=%d(BYTE)\r\n",headerLen);</p><p>  fprintf(file,"服務(wù)類型=%s,%s\r\n");</p><p>  parseServ

73、iceType_getProcedence(ip.ServiceType);</p><p>  parseServiceType_getTOS(ip.ServiceType);</p><p>  fprintf(file,"數(shù)據(jù)報長度=%d(BYTE)\r\n",ip.TotalLen);</p><p>  fprintf(file,&qu

74、ot;數(shù)據(jù)報ID=%d\r\n",ip.ID);</p><p>  BYTE DF,MF;</p><p>  getFlags(ip.Flags,DF,MF);</p><p>  fprintf(file,"分段標(biāo)志 DF=%d,MF=%d\r\n",DF,MF);</p><p>  WORD fragO

75、ff;</p><p>  getFragOff(ip.FragOff,fragOff);</p><p>  fprintf(file,"分段偏移值=%d\r\n",fragOff);</p><p>  fprintf(file," 生存期=%d(hops)\r\n",ip.TimeToLive);</p>

76、<p>  fprintf(file,"協(xié)議=%s\r\n",getProtocol(ip.Protocol));</p><p>  fprintf(file,"頭校驗和=0x%0x\r\n",ip.HdrChksum);</p><p>  fprintf(file,"源IP地址=%s\r\n",inet-ntoa(

77、*(in-addr*)&ip.SrcAddr));</p><p>  fprintf(file,"目的IP地址=%s\r\n",inet_ntoa(*(in-addr*)&ip.DstAddr));</p><p>  fprintf(file,"__________________________________\r\n");&l

78、t;/p><p><b>  }</b></p><p>  int main(int argc,char *argv[])</p><p><b>  {</b></p><p>  if(argc!=2)</p><p><b>  {</b></

79、p><p>  printf("usage error!\n");</p><p>  return -1;</p><p><b>  }</b></p><p>  FILE * file;</p><p>  if((file=foopen(argv[1],"wb+

80、"))==NULL)</p><p><b>  {</b></p><p>  printf("fail to open file %s",argv{1});</p><p>  return -1;</p><p><b>  }</b></p>&l

81、t;p>  WSADATA wsData;</p><p>  if(WSAStartup(MAKEWORD(2,2),$WSdATA)!=0)</p><p><b>  {</b></p><p>  PRINTF("WSAStartup FAILED!\n");</p><p>  ret

82、urn -1;</p><p><b>  }</b></p><p>  SOCKET sock;</p><p>  if((sock=socket(AF_INET,SOCK_RAW,ippROTO_IP))==INVALID_SOCKET)</p><p><b>  {</b></p&

83、gt;<p>  PRINTF("CREATE socket failed!\n");</p><p>  return -1;</p><p><b>  }</b></p><p>  BOOL flag=TRUE;</p><p>  IF(setsockopt(sock,IPPR

84、OTO_IP,IP_HDRINCL,(CHAR*)&FLAG,sizeof(flag))==SOCKET_ERROR)</p><p><b>  {</b></p><p>  printf("setsockopt failed!\n");</p><p>  return -1;</p><p

85、><b>  }</b></p><p>  char hostName[128];</p><p>  if(gethostname(hostName,100)==SOCKET_ERROR)</p><p><b>  {</b></p><p>  printf("gethost

86、name failed!\n");</p><p>  return -1;</p><p><b>  }</b></p><p>  hostent * pHostIP;</p><p>  if(pHostIP=gethostbyname(hostName))==NULL)</p><

87、p><b>  { </b></p><p>  printf("gethostbyname failed!\n");</p><p>  return -1;</p><p><b>  }</b></p><p>  sockaddr_in addr_in;</p

88、><p>  addr_in.sin_addr=*(in_addr*)pHostIP->h_addr_list[0];</p><p>  addr_in.sin_family=AF_INET;</p><p>  addr_in.sin_port=htone(6000);</p><p>  if(bind(sock,(PSOCKADDR

89、)&addr_in,sizeof(addr_in))==SOCKET_ERROR)</p><p><b>  {</b></p><p>  printf("bind failed");</p><p>  return -1;</p><p><b>  }</b>&

90、lt;/p><p>  DWORD dwValue=1;</p><p>  #define IO_RCVALL_WSAIOW(IOC_VENDOR,1)</p><p>  DWORD dwBufferLen[10];</p><p>  DWORD dwBufferInLen=1;</p><p>  DWORD dw

91、BytesReturned=0;</p><p>  if(WSAIoctl(sock,IO_RCVALL,&dwBufferInLen,sizeof(dwBufferInLen),</p><p>  &dwBufferLen,sizeof(dwBufferLen),&dwBytesReturned,NULL,NULL)==</p><p>

92、;  NULL,NULL)==SOCKET_ERROR)</p><p><b>  {</b></p><p>  printf("icotlsocket failed\n");</p><p>  return -1;</p><p><b>  }</b></p>

93、;<p>  #define BUFFER_SIZE 65535</p><p>  char buffer[BUFFER_SIZE];</p><p>  printf("開始解析\n\n");</p><p>  while(true)</p><p><b>  {</b><

94、/p><p>  int size=recv(sock,buffer,BUFFER_SIZE,0);</p><p>  if(size>0)</p><p><b>  { </b></p><p>  ipparse(stdout,buffer);</p><p>  ipparse(fil

95、e,buffer); </p><p><b>  }</b></p><p><b>  }</b></p><p>  fclose(file);</p><p><b>  return 0:</b></p><p><b>  }&l

96、t;/b></p><p><b>  六.程序流程圖:</b></p><p><b>  N</b></p><p><b>  Y</b></p><p><b>  七.相關(guān)擴(kuò)展</b></p><p>  本程序設(shè)計

97、也可以利用Winpcap完成,部分核心代碼的簡略提示如下。</p><p>  獲取所有存在的網(wǎng)絡(luò)設(shè)備的鏈表。</p><p>  Pcap_if_t *alldevs; //網(wǎng)絡(luò)設(shè)備結(jié)構(gòu)鏈表</p><p>  Char errbuf[PCAP_ERRBUF_SIZE]; //錯誤信息</p><p>  /*所有網(wǎng)絡(luò)設(shè)備

98、的信息以鏈表形式存在alldevs中*/</p><p>  pcap_findalldevs(&alldevs,errbuf);</p><p>  2) 從鏈表中選擇物理Ethernet卡后,用混雜模式打開,調(diào)用的函數(shù)為pcap_open_live(const char*device,int snaplen,int promisc,int to_ms,char*errbuf).

99、在這個函數(shù)中,第一個參數(shù)為要打開的設(shè)備名稱,這里是Ethernet卡,可以從設(shè)備鏈表alldevs中選出。第二個參數(shù)應(yīng)為捕獲的數(shù)據(jù)包長度,填入65535以保證在鏈路層的整個數(shù)據(jù)包都被捕獲。第三個參數(shù)為打開模式,填入1表明用混雜模式打開網(wǎng)卡。最后兩個參數(shù)本別為讀入超時的時間和保存錯誤信息。</p><p><b>  編譯、設(shè)置過濾器。</b></p><p>  ch

100、ar packet_filter[]=”ip”;</p><p>  pcap_compile(adhandle,&fcode,packet_filter,1,netmask);//編譯過濾器</p><p>  pcap_setfilter(adhandle,&fcode); //設(shè)置過濾器</p><p>  adhandle參數(shù)為

101、網(wǎng)卡描述符,fcode參數(shù)是一個BPF偽匯編程序,packet_filter參數(shù)用于設(shè)置的過濾規(guī)則,在這里我們只需要捕獲IP包。</p><p>  4) 可利用pcap_loop函數(shù)捕獲數(shù)據(jù)包。對于捕獲的數(shù)據(jù)包,去掉數(shù)據(jù)鏈路層的14B的頭部后才是真正的IP包信息。利用winpcap編程和利用socket編程在處理IP包上并沒有太大區(qū)別,我們依然可以使用7。4節(jié)中采用的數(shù)據(jù)結(jié)構(gòu)IP——HEADER來保存、解析IP

102、頭部信息。</p><p><b>  八.實習(xí)體會</b></p><p>  通過這次實驗,了解到關(guān)于計算機(jī)網(wǎng)絡(luò)數(shù)據(jù)傳送及處理過程中,軟件起到了巨大的作用。熟悉了VC++在計算機(jī)網(wǎng)絡(luò)方面的應(yīng)用,是一次難得的機(jī)會。</p><p>  同學(xué)們的默鍥配合和合作精神是實驗成功的必要條件,而謹(jǐn)慎對待事物的態(tài)度是成功的關(guān)鍵。</p>&

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論