|
道德三皇五帝,功名夏后商周.英雄五霸鬧春秋,頃刻興亡過手.青史幾行名姓,北邙無數(shù)荒丘.前人田地后人收,說甚龍爭虎斗。
ICMP報文是一種面向無連接的協(xié)議,用于傳輸出錯報告控制信息。它是一個非常重要的協(xié)議,ICMP提供一致易懂的出錯報告信息。發(fā)送的出錯報文返回到發(fā)送原數(shù)據(jù)的設(shè)備,因為只有發(fā)送設(shè)備才是出錯報文的邏輯接受者。發(fā)送設(shè)備隨后可根據(jù)ICMP報文確定發(fā)生錯誤的類型,并確定如何才能更好地重發(fā)失敗的數(shù)據(jù)包。但是ICMP唯一的功能是報告問題而不是糾正錯誤,糾正錯誤的任務(wù)由發(fā)送方完成 。
ICMP報文對于我來說最大的意義是偵測主機是否存在。這一點要命。
windows大名鼎鼎的PING程序就是ICMP協(xié)議,因此就ICMP的SOCKET實現(xiàn)記述一二,
第一部分:實現(xiàn)
ICMP的實現(xiàn)依賴于SOCKET,那么SOCKET是這樣做的流程:
PING 發(fā)送程序
1、socket(AF_INET, SOCK_RAW, IP_PROTO_ICMP)
建立SOCKET,ICMP協(xié)議
2、setsockopt(s, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout))
配置SOCKET的接受超時時間,我配置的為10S
3、 iecho = rt_malloc(ping_size);
構(gòu)建PING數(shù)據(jù),數(shù)據(jù)長度為32字節(jié)+icmp_echo_hdr的頭
4、 /* fill the additional data buffer with some data */
for(i = 0; i < data_len; i++)
{
((char*)iecho)[sizeof(struct icmp_echo_hdr) + i] = (char)i;
}
iecho->chksum = inet_chksum(iecho, len);
填充數(shù)據(jù)
5、sendto(s, iecho, ping_size, 0, (struct sockaddr*)&to, sizeof(to));
發(fā)送數(shù)據(jù)
6、 free(iecho);釋放內(nèi)存
到此PING程序的發(fā)送完成。
PING 接收程序
1、recvfrom(s, buf, sizeof(buf), 0,(struct sockaddr*)&from, (socklen_t*)&fromlen)
阻塞于ICMP的socket上,等待回應(yīng)
2、 addr = (struct _ip_addr *)&(from.sin_addr);
如果收到回應(yīng)那么輸出回應(yīng)的IP地址
3、printf("ping: timeout\n");
如果沒有收到回應(yīng),10S后會返回系統(tǒng)一個 超時
當(dāng)然數(shù)據(jù)已經(jīng)被存入BUFF中,想用隨時取出即可。我目前是丟掉的。
第二部分:測試
程序完成后的測試結(jié)果如下:

可見ICMP報文的包確實是正常發(fā)出來了。
IP地址是192.168.0.105是板子
IP地址是192.168.0.100是電腦。
debug時也可以正常收到電腦的回復(fù)數(shù)據(jù)。
這證明此流程程序可用!
最后socket的收發(fā)程序目前采用的都是堵塞方式實現(xiàn),也就是說底層和上層接口之間用郵箱來通信。有數(shù)據(jù)立馬推送,所以這個也是應(yīng)用程序編寫的關(guān)鍵之所在。如果不能及時的收取那么很可能box溢出,那就不是溢出那么簡單的事情了。所以只能是單線程處理,絕對禁止多線程。前人的經(jīng)驗血的教訓(xùn)!
于日照比特電子
老偉
|
|