版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)
文檔簡介
1、<p> 計算機網(wǎng)絡(luò)工程課程設(shè)計報告</p><p> 題 目: Ping程序的實現(xiàn) </p><p> 學(xué)生姓名: </p><p> 學(xué) 號: </p><p> 專業(yè)班級: 計科專業(yè)10102班 </p&
2、gt;<p> 同組姓名: </p><p> 指導(dǎo)教師: </p><p> 設(shè)計時間: 2013年下學(xué)期第16周 </p><p><b> 目錄</b></p><p> 一、課程設(shè)計的目的和意義2</p&g
3、t;<p> 二、課程設(shè)計的內(nèi)容和要求2</p><p><b> 1.內(nèi)容:2</b></p><p><b> 2.要求:2</b></p><p> 三、課程設(shè)計的相關(guān)技術(shù)2</p><p> 四、課程設(shè)計過程2</p><p> 1
4、.Ping主模塊2</p><p> 2.功能控制模塊4</p><p> 3.數(shù)據(jù)報解析模塊5</p><p> 五、課程設(shè)計小結(jié)6</p><p><b> 六、參考文獻7</b></p><p> 七、附 錄(程序清單)7</p><p>&
5、lt;b> 八、心得體會11</b></p><p> 一、課程設(shè)計的目的和意義</p><p> 利用ICMP數(shù)據(jù)包、C語言實現(xiàn)Ping命令程序,能實現(xiàn)基本的Ping操作,發(fā)送ICMP回顯請求報文,用于測試—個主機到只一個主機之間的連通情況。通過本程序的訓(xùn)練,使學(xué)生熟悉ICMP報文結(jié)構(gòu),使學(xué)生對ICMP有更深的理解,掌握Ping程序的設(shè)計方法,掌握網(wǎng)絡(luò)編程的方法
6、和技巧,從而編寫出功能更強大的程序。</p><p> 二、課程設(shè)計的內(nèi)容和要求</p><p><b> 1.內(nèi)容:</b></p><p> 用C語言實現(xiàn)Ping命令程序,能實現(xiàn)基本的Ping操作,發(fā)送ICMP回顯請求報文,用于測試—個主機到只一個主機之間的連通情況。</p><p><b> 2.
7、要求: </b></p><p> 獨立完成程序的設(shè)計、編碼和調(diào)試。</p><p> 系統(tǒng)利用C語言實現(xiàn),程序調(diào)試環(huán)境為Turbo C或VC;</p><p> 按照課程設(shè)計規(guī)范書寫課程設(shè)計報告。</p><p> 采用VC環(huán)境進行調(diào)試運行。</p><p> 三、課程設(shè)計的相關(guān)技術(shù)</p
8、><p> 由于Ping程序是面向用戶的應(yīng)用程序,該程序使用ICMP的封裝機制,通過IP協(xié)議來工作。為了實現(xiàn)直接對IP和ICMP包進行操作,實驗中使用RAW模式的socket編程。首先定義IP數(shù)據(jù)報首部,在IP數(shù)據(jù)報的基礎(chǔ)上定義ICMP數(shù)據(jù)報首部,并初始化一些全局變量。接著自定義填充ICMP數(shù)據(jù)報字段函數(shù)FillICMPData()、校驗和函數(shù)checksum()、解讀ICMP報首部函數(shù)DecodeICMPHead
9、er()、釋放資源函Cleanup()。最后主函數(shù)通過調(diào)用這些函數(shù)來實現(xiàn)Ping命令功能。IP頭與ICMP頭的設(shè)置分別參照RFC791及RFC792的標(biāo)準(zhǔn),包含所有必要信息。主程序設(shè)置main()函數(shù),主函數(shù)用庫函數(shù)實現(xiàn)套接字編程用于數(shù)據(jù)包發(fā)送及接收,其中,數(shù)據(jù)包發(fā)送調(diào)用sendto(),數(shù)據(jù)包接收調(diào)用recvfrom( ),由于發(fā)送數(shù)據(jù)包時可能會遇到阻塞或者目標(biāo)主機不通,造成超時,因此需要在發(fā)送數(shù)據(jù)包后調(diào)用一個函數(shù)判斷是否超時,此處調(diào)
10、用庫函數(shù)setsockopt()來實現(xiàn)超時判斷;其次,校驗和函數(shù)采用移位方法進行計算。</p><p> 套接字所需要的文件有頭文件Winsocket2.h、庫文件WS2_32.LIB、動態(tài)庫W32_32.DLL。創(chuàng)建套接字的時候參數(shù)的以及在創(chuàng)建套接字之前必須首先使用WSAStartup函數(shù)。</p><p><b> 四、課程設(shè)計過程</b></p>
11、<p><b> 1.Ping主模塊</b></p><p> Ping()函數(shù)是本程序的核心部分,它基本是調(diào)用其他模塊的函數(shù)來實現(xiàn)最終功能,其主要布驟包括:定義及初始化各個全局變量、打開socket動態(tài)庫、設(shè)置接收和發(fā)送超時值、域名地址解析、分配內(nèi)存、創(chuàng)建及初始化ICMP報文、發(fā)送ICMP請求報文、接收ICMP 應(yīng)答報文以及解讀應(yīng)答報文和輸出Ping結(jié)果,最后釋放占用的資
12、源其流程如下頁圖2.1所示。</p><p><b> 注釋:</b></p><p> 該模塊并非只有處理還包括判斷及輸出判斷結(jié)果的含義;</p><p> 程序沒運行一次就只能輸出四行結(jié)果(前提是輸入的地址有效),欲再次PING其他地址必須要重新啟動程序。</p><p> 輸入時不能輸入目標(biāo)主機名,不然pi
13、ng結(jié)果為TIMEOUT;</p><p><b> 2.功能控制模塊</b></p><p> 功能控制模塊主要是為其他模塊提供可調(diào)用的函數(shù),該模塊主要包括參數(shù)獲取功能、計算ICMP數(shù)據(jù)報文檢驗和、清除SOCKET,ICMP包數(shù)據(jù)以及接受緩沖區(qū)、占用資源釋放功能和顯示用尸幫助功能。該模塊一共包含三個函數(shù)來實現(xiàn)。,流程如圖2.2所示。</p><
14、;p> 圖2.2 功能控制模塊</p><p><b> 注釋:</b></p><p> a.illICMPData是由一系列的初始化的語句在流程圖中不再畫出;</p><p> b.Cleanup()函數(shù)中的WSACleanup(),HeapFree(),closesocket()都是一些庫函數(shù)。checksum()校驗和函
15、數(shù)是冗余校驗的一種形式。 它是通過錯誤檢測方法,對經(jīng)過空間(如通信)或者時間(如計算機存儲)傳送的數(shù)據(jù)的完整性進行檢查的一種簡單方法。</p><p><b> 3.數(shù)據(jù)報解析模塊</b></p><p> 數(shù)據(jù)報解析模塊提供了解讀IP選項和解讀IcMP報文的功能。從本機收到目的主機返回的1cMP回顯應(yīng)答報文,就開始逐個地解讀IcMP報文,如果需要記錄路由的情況下
16、,IcMP解析函數(shù)將調(diào)用IP選項解讀函數(shù)來實現(xiàn)IP路由的輸出(但本程序沒有此功能。該模塊主要由DecodeICMPHeader一個函數(shù)來實現(xiàn),而中間也會調(diào)用其它模塊的相應(yīng)函數(shù)。其流程圖如圖2.3:</p><p><b> 注釋:</b></p><p> a.判斷是否為我們所要的數(shù)據(jù)報回應(yīng)之前,還有一些判斷回應(yīng)多少內(nèi)容的語句未呈現(xiàn)出;</p>&l
17、t;p> b.函數(shù)GetTickCount()是用來記錄此時我機所處的現(xiàn)在時間(毫秒級);</p><p> 圖2.3 數(shù)據(jù)報解析模塊</p><p><b> 五、課程設(shè)計小結(jié)</b></p><p> 1.運行操作結(jié)果:在vc里運行之后界面:</p><p> 2.輸入本機ip地址:</p&g
18、t;<p> 3.輸入網(wǎng)上ip看結(jié)果:</p><p> 4. 但是當(dāng)網(wǎng)絡(luò)連不通時,就會出現(xiàn)下圖結(jié)果</p><p><b> 六、參考文獻</b></p><p> 【1】Visual C++網(wǎng)絡(luò)通信編程實用案例精選(第二版)曹衍龍 劉海英 編著;</p><p> 【2】Windows網(wǎng)絡(luò)
19、編程技術(shù) (美);</p><p> 七、附 錄(程序清單)</p><p> #pragma comment(lib,"ws2_32.lib")</p><p> #include <winsock2.h>//創(chuàng)建套接字頭文件</p><p> #include <ws2tcpip.h>
20、;</p><p> #include <stdio.h>//標(biāo)準(zhǔn)輸入輸出函數(shù)</p><p> #include <stdlib.h>//實用程序庫函數(shù)</p><p> #include <string.h></p><p> typedef struct iphdr </p>&
21、lt;p><b> {</b></p><p> unsigned int h_len:4; // 頭長度</p><p> unsigned int version:4; // IP版本 </p><p> unsigned char service;
22、 // 服務(wù)類型 </p><p> unsigned short total_len; // 包的總長度 </p><p> unsigned short ident; // 包標(biāo)示身份 </p><p> unsig
23、ned short frag_and_flags; // 標(biāo)志 </p><p> unsigned char ttl; // 包生命周期 </p><p> unsigned char proto; // 協(xié)議類型 </p><p> unsigne
24、d short checksum; // IP 校驗 </p><p> unsigned int sourceIP; //源IP</p><p> unsigned int destIP; //目標(biāo)IP</p><p> } IpHeader;</p><p> #defin
25、e ICMP_ECHO 8 //ICMP報文類型,回顯請求 </p><p> #define ICMP_ECHOREPLY 0 //ICMP報文類型,回顯應(yīng)答</p><p> #define ICMP_MIN 8 //最小的ICMP數(shù)據(jù)報大小 </p
26、><p> typedef struct icmphdr </p><p><b> {</b></p><p> BYTE i_type; //ICMP報文類型</p><p> BYTE i_code; //該類型中的代碼號</p>
27、;<p> USHORT i_cksum; //校驗和</p><p> USHORT i_id; //惟一的標(biāo)識符</p><p> USHORT i_seq; //序列號</p><p> ULONG timestamp;
28、 //時間戳 </p><p> } IcmpHeader;</p><p> #define DEF_PACKET_SIZE 32 //默認數(shù)據(jù)報大小</p><p> #define MAX_PACKET 1024 // 最大的ICMP數(shù)據(jù)報大小 </p><p>
29、#define MAX_IP_HDR_SIZE 60 // 最大IP頭長度 </p><p> //初始化全局變量 </p><p> int datasize=DEF_PACKET_SIZE; </p><p> char *icmp_data=NULL;</p><p> c
30、har *recvbuf=NULL; </p><p> SOCKET m_hSocket= INVALID_SOCKET; </p><p> char *lpdest=NULL;</p><p> //填充ICMP數(shù)據(jù)報字段函數(shù)</p><p> void FillICMPData(char *icmp_data, int dat
31、asize)</p><p><b> {</b></p><p> IcmpHeader *icmp_hdr = NULL; </p><p> char*datapart = NULL;</p><p> icmp_hdr = (IcmpHeader*)icmp_data; </p><p
32、> icmp_hdr->i_type = ICMP_ECHO; </p><p> icmp_hdr->i_code = 0; </p><p> icmp_hdr->i_id = (USHORT)GetCurrentProcessId();</p><p> icmp_hdr->i_cksum = 0; <
33、/p><p> icmp_hdr->i_seq = 0;</p><p> datapart = icmp_data + sizeof(IcmpHeader);</p><p><b> }</b></p><p><b> //校驗和函數(shù)</b></p><p>
34、 USHORT checksum(USHORT *buffer, int size)</p><p> { unsigned long cksum=0;</p><p> while (size > 1) </p><p> { cksum += *buffer++; size -= sizeof(USHORT); }</p>
35、;<p> if (size) </p><p> { cksum += *(UCHAR*)buffer; } </p><p> cksum = (cksum >> 16) + (cksum & 0xffff); </p><p> cksum += (cksum >>16);return (USHORT
36、)(~cksum);</p><p><b> }</b></p><p> //解讀ICMP報首部函數(shù)</p><p> void DecodeICMPHeader(char *buf, int bytes, SOCKADDR_IN *from)</p><p><b> {</b><
37、;/p><p> IpHeader*iphdr = NULL; </p><p> IcmpHeader*icmphdr = NULL;</p><p> unsigned short iphdrlen;</p><p> DWORD tick; </p><p> static int icmpcoun
38、t = 0;</p><p> iphdr = (IpHeader *)buf; </p><p> iphdrlen = iphdr->h_len * 4; </p><p> tick = GetTickCount();</p><p> if (bytes < iphdrlen + ICMP_MIN) </
39、p><p> { printf("Too few bytes from %s \r\n",inet_ntoa(from->sin_addr)); }</p><p> icmphdr = (IcmpHeader*)(buf + iphdrlen);</p><p> if (icmphdr->i_type != ICMP_E
40、CHOREPLY) </p><p> { printf("nonecho type %d received \r\n", icmphdr->i_type); }</p><p> if (icmphdr->i_id != (USHORT)GetCurrentProcessId()) </p><p> { printf(
41、"其他程序的回應(yīng)報文! \t錯誤代碼 %d\n", WSAGetLastError()); }</p><p> DWORD tick0[4]; </p><p> tick0[icmpcount]=tick - icmphdr->timestamp;</p><p> if(tick0[icmpcount]<1)</
42、p><p> printf("Replyfrom %s: bytes=%d time<1ms </p><p> icmp_seq=%d\n",inet_ntoa(from->sin_addr), bytes, icmphdr->i_seq );</p><p><b> else </b></
43、p><p> printf("Reply from %s: bytes=%d time=%dms icmp_seq = %d\n",inet_ntoa(from->sin_addr), bytes,tick0[icmpcount], icmphdr->i_seq); </p><p> icmpcount++;</p><p><
44、;b> }</b></p><p><b> //釋放資源函數(shù)</b></p><p> void Cleanup()</p><p><b> {</b></p><p> if (m_hSocket != INVALID_SOCKET) </p><
45、;p> closesocket(m_hSocket);</p><p> HeapFree(GetProcessHeap(), 0, recvbuf);</p><p> HeapFree(GetProcessHeap(), 0, icmp_data); </p><p> WSACleanup();</p><p><b
46、> }</b></p><p><b> //主函數(shù)</b></p><p> void main()</p><p><b> {</b></p><p> WSADATA wsaData; char a[100]; printf("ping &qu
47、ot;);</p><p> scanf("%s",a); lpdest=a; SOCKADDR_IN m_addrDest;//結(jié)構(gòu)體 </p><p> SOCKADDR_IN m_addrFrom; int timeout=1000; USHORT seq_no=0;</p><p> if (WSAStartu
48、p(MAKEWORD(2, 2), &wsaData) != 0)</p><p> { printf("Sorry, you cannot load socket dll!"); }</p><p> m_hSocket = WSASocket (AF_INET, SOCK_RAW, IPPROTO_ICMP, NULL, 0,WSA_FLAG_OVE
49、RLAPPED);//創(chuàng)建原始套接字,該套接字用于ICMP協(xié)議</p><p> if (m_hSocket == INVALID_SOCKET) //如果套接字創(chuàng)建不成功</p><p> { printf("socket 創(chuàng)建失敗!"); }</p><p> int bread = setsockopt(m_hSocket, SO
50、L_SOCKET, SO_RCVTIMEO, (char*)&timeout, sizeof(timeout));//設(shè)置接收的超時值</p><p> if(bread == SOCKET_ERROR) </p><p> { printf("設(shè)置socket接收超時選項錯誤!"); }</p><p> timeout =
51、1000;</p><p> bread = setsockopt(m_hSocket, SOL_SOCKET, SO_SNDTIMEO, (char*)&timeout, sizeof(timeout));//設(shè)置發(fā)送的超時值</p><p> if (bread == SOCKET_ERROR) </p><p> { printf("
52、設(shè)置socket發(fā)送超時選項錯誤!"); }</p><p> memset(&m_addrDest, 0, sizeof(m_addrDest));// 用0初始化目的地地址</p><p> m_addrDest.sin_family = AF_INET;//設(shè)置地址族,這里表示使用IP地址族</p><p> if ((m_addrD
53、est.sin_addr.s_addr = inet_addr(lpdest)) == INADDR_NONE)//地址轉(zhuǎn)化</p><p><b> { </b></p><p> struct hostent *hp = NULL;</p><p> if ((hp = gethostbyname(lpdest)) != NULL
54、) //名字解析,根據(jù)主機名獲取IP地址</p><p><b> {</b></p><p> memcpy(&(m_addrDest.sin_addr), hp->h_addr, hp->h_length);//將獲取到的IP值賦給目的地地址中的相應(yīng)字段</p><p> m_addrDest.sin_fam
55、ily = hp->h_addrtype; //將獲取到的地址族值賦給目的地地址中的相應(yīng)字段</p><p><b> }</b></p><p><b> else</b></p><p><b> {</b></p><p> printf("
56、不能找到名為 %s 的主機\t錯誤代碼 %d\n",lpdest, exit(0);</p><p> } } </p><p> printf("Pinging %s with 64 bytes of data: \n\n", inet_ntoa(m_addrDest.sin_addr)); </p><p
57、> datasize += sizeof(IcmpHeader); //數(shù)據(jù)報文大小需要包含ICMP報頭</p><p> //根據(jù)默認堆句柄,從堆中分配MAX_PACKET內(nèi)存塊,新分配內(nèi)存的內(nèi)容將被初始化為0</p><p> icmp_data=(char*)HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,MAX_PACKET);
58、</p><p> recvbuf =(char*) HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,MAX_PACKET);</p><p> if (!icmp_data) //如果分配內(nèi)存不成功</p><p> { printf("堆分配錯誤!"); }</p><p
59、> memset(icmp_data,0,MAX_PACKET);//創(chuàng)建ICMP報文</p><p> FillICMPData(icmp_data,datasize);</p><p> // 開始發(fā)送或接受ICMP包</p><p> int nCount=0;</p><p><b> while(1) &l
60、t;/b></p><p> { int bwrote; </p><p> if(nCount++ == 4) </p><p> break;//超過指定的記錄條數(shù)則退出</p><p> ((IcmpHeader*)icmp_data)->i_cksum = 0;//計算校驗和前要把校驗和字段設(shè)置為
61、0</p><p> ((IcmpHeader*)icmp_data)->timestamp = GetTickCount();//獲取操作系統(tǒng)啟動到現(xiàn)在所經(jīng)過的毫秒數(shù),設(shè)置時間戳</p><p> ((IcmpHeader*)icmp_data)->i_seq = seq_no++;//設(shè)置序列號</p><p> ((IcmpHeader*)i
62、cmp_data)->i_cksum = checksum((USHORT*)icmp_data, datasize);//計算校驗和</p><p> bwrote = sendto(m_hSocket, icmp_data, datasize, 0, (struct sockaddr*)&m_addrDest, sizeof(m_addrDest));//開始發(fā)送ICMP請求</p>
63、;<p> if (bwrote == SOCKET_ERROR)//如果發(fā)送不成功</p><p><b> {</b></p><p> if (WSAGetLastError() == WSAETIMEDOUT) //如果是由于超時不成功</p><p> { printf("Requrest timed
64、 out ! \r\n"); continue; }</p><p> printf("目標(biāo)不可達!\t錯誤代碼 %d\n", WSAGetLastError());//其他發(fā)送不成功原因</p><p> continue; }</p><p> if (bwrote < datasize) </p>
65、<p> {printf("Wrote %d bytes \r\n", bwrote); }</p><p> int fromlen = sizeof(m_addrFrom);//開始接收ICMP應(yīng)答</p><p> bread=recvfrom(m_hSocket,recvbuf,MAX_PACKET,0,(struct sockaddr*
66、)&m_addrFrom, &fromlen);</p><p> if (bread == SOCKET_ERROR)//如果接收不成功</p><p><b> {</b></p><p> if (WSAGetLastError() == WSAETIMEDOUT) //如果是由于超時不成功</p>&
67、lt;p> { printf("Requrest timed out !\r\n"); continue; } printf("接收數(shù)據(jù)函數(shù)調(diào)用錯誤!\t錯誤代碼 %d\n", WSAGetLastError());//其他接收不成功原因</p><p><b> exit(0);</b></p><
68、;p><b> }</b></p><p> DecodeICMPHeader(recvbuf, bread, &m_addrFrom);//解讀接收到的ICMP數(shù)據(jù)報</p><p> } Cleanup();</p><p><b> }</b></p><p><b
溫馨提示
- 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)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 《計算機網(wǎng)絡(luò)基礎(chǔ)課程設(shè)計》udp
- 計算機網(wǎng)絡(luò)課程設(shè)計--編程實現(xiàn)基于udp的ping
- 計算機網(wǎng)絡(luò)基礎(chǔ)課程設(shè)計---中學(xué)網(wǎng)絡(luò)規(guī)劃
- 計算機網(wǎng)絡(luò)課程設(shè)計-----基于icmp的ping設(shè)計
- 計算機網(wǎng)絡(luò)基礎(chǔ)課件
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 《計算機網(wǎng)絡(luò)課程設(shè)計》
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計---網(wǎng)絡(luò)設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計報告
- 計算機網(wǎng)絡(luò)課程設(shè)計報告
- 計算機網(wǎng)絡(luò)課程設(shè)計報告
- 計算機網(wǎng)絡(luò)組建課程設(shè)計
- 計算機網(wǎng)絡(luò)課程設(shè)計報告
- 《計算機網(wǎng)絡(luò)》課程設(shè)計報告
評論
0/150
提交評論