亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

用C語言實現Ping程序功能

2019-11-17 05:19:18
字體:
來源:轉載
供稿:網友
大部分人用ping命令只是作為查看另一個系統的網絡連接是否正常的一種簡單方法。在這篇文章中,作者將介紹如何用C語言編寫一個模擬ping命令功能的程序。

ping命令是用來查看網絡上另一個主機系統的網絡連接是否正常的一個工具。ping命令的工作原理是:向網絡上的另一個主機系統發送ICMP報文,假如指定系統得到了報文,它將把報文一模一樣地傳回給發送者,這有點象潛水艇聲納系統中使用的發聲裝置。

例如,在linux終端上執行ping localhost命令將會看到以下結果:
PING localhost.localdomain (127.0.0.1) from 127.0.0.1 : 56(84) bytes of data.
64 bytes from localhost.localdomain (127.0.0.1): icmp_seq=0 ttl=255 time=112 usec
64 bytes from localhost.localdomain (127.0.0.1): icmp_seq=1 ttl=255 time=79 usec
64 bytes from localhost.localdomain (127.0.0.1): icmp_seq=2 ttl=255 time=78 usec
64 bytes from localhost.localdomain (127.0.0.1): icmp_seq=3 ttl=255 time=82 usec

--- localhost.localdomain ping statistics ---
4 packets transmitted, 4 packets received, 0% packet loss
round-trip min/avg/max/mdev = 0.078/0.087/0.112/0.018 ms


由上面的執行結果可以看到,ping命令執行后顯示出被測試系統主機名和相應IP地址、返回給當前主機的ICMP報文順序號、ttl生存時間和往返時間rtt(單位是毫秒,即千分之一秒)。要寫一個模擬ping命令,這些信息有啟示作用。

要真正了解ping命令實現原理,就要了解ping命令所使用到的TCP/IP協議。

ICMP(Internet Control Message,網際控制報文協議)是為網關和目標主機而提供的一種差錯控制機制,使它們在碰到差錯時能把錯誤報告給報文源發方。ICMP協議是IP層的一個協議,但是由于差錯報告在發送給報文源發方時可能也要經過若干子網,因此牽涉到路由選擇等問題,所以ICMP報文需通過IP協議來發送。ICMP數據報的數據發送前需要兩級封裝:首先添加ICMP報頭形成ICMP報文,再添加IP報頭形成IP數據報。如下圖所示

IP報頭
ICMP報頭
ICMP數據報


IP報頭格式
由于IP層協議是一種點對點的協議,而非端對端的協議,它提供無連接的數據報服務,沒有端口的概念,因此很少使用bind()和connect()函數,若有使用也只是用于設置IP地址。發送數據使用sendto()函數,接收數據使用recvfrom()函數。IP報頭格式如下圖:
用C語言實現Ping程序功能

在Linux中,IP報頭格式數據結構()定義如下:
strUCt ip
{
#if __BYTE_ORDER == __LITTLE_ENDIAN
unsigned int ip_hl:4; /* header length */
unsigned int ip_v:4; /* version */
#endif
#if __BYTE_ORDER == __BIG_ENDIAN
unsigned int ip_v:4; /* version */
unsigned int ip_hl:4; /* header length */
#endif
u_int8_t ip_tos; /* type of service */
u_short ip_len; /* total length */
u_short ip_id; /* identification */
u_short ip_off; /* fragment offset field */
#define IP_RF 0x8000 /* reserved fragment flag */
#define IP_DF 0x4000 /* dont fragment flag */
#define IP_MF 0x2000 /* more fragments flag */
#define IP_OFFMASK 0x1fff /* mask for fragmenting bits */
u_int8_t ip_ttl; /* time to live */
u_int8_t ip_p; /* PRotocol */
u_short ip_sum; /* checksum */
struct in_addr ip_src, ip_dst; /* source and dest address */
};



其中ping程序只使用以下數據:

IP報頭長度IHL(Internet Header Length)――以4字節為一個單位來記錄IP報頭的長度,是上述IP數據結構的ip_hl變量。
生存時間TTL(Time To Live)――以秒為單位,指出IP數據報能在網絡上停留的最長時間,其值由發送方設定,并在經過路由的每一個節點時減一,當該值為0時,數據報將被丟棄,是上述IP數據結構的ip_ttl變量
 

ICMP報頭格式
ICMP報文分為兩種,一是錯誤報告報文,二是查詢報文。每個ICMP報頭均包含類型、編碼和校驗和這三項內容,長度為8位,8位和16位,其余選項則隨ICMP的功能不同而不同。

Ping命令只使用眾多ICMP報文中的兩種:"請求回送'(ICMP_ECHO)和"請求回應'(ICMP_ECHOREPLY)。在Linux中定義如下:
#define ICMP_ECHO 0
#define ICMP_ECHOREPLY 8



這兩種ICMP類型報頭格式如下: 



在Linux中ICMP數據結構()定義如下:
struct icmp
{
u_int8_t icmp_type; /* type of message, see below */
u_int8_t icmp_code; /* type sub code */
u_int16_t icmp_cksum; /* ones complement checksum of struct */
union
{
u_char ih_PPTr; /* ICMP_PARAMPROB */
struct in_addr ih_gwaddr; /* gateway address */
struct ih_idseq /* echo datagram */
{
u_int16_t icd_id;
u_int16_t icd_seq;
} ih_idseq;
u_int32_t ih_void;

/* ICMP_UNREACH_NEEDFRAG -- Path MTU Discovery (RFC1191) */
struct ih_pmtu
{
u_int16_t ipm_void;
u_int16_t ipm_nextmtu;
} ih_pmtu;

struct ih_rtradv
{
u_int8_t irt_num_addrs;
u_int8_t irt_wpa;
u_int16_t irt_lifetime;
} ih_rtradv;
} icmp_hun;
#define icmp_pptr icmp_hun.ih_pptr
#define icmp_gwaddr icmp_hun.ih_gwaddr
#define icmp_id icmp_hun.ih_idseq.icd_id
#define icmp_seq icmp_hun.ih_idseq.icd_seq
#define icmp_void icmp_hun.ih_void
#define icmp_pmvoid icmp_hun.ih_pmtu.ipm_void
#define icmp_nextmtu icmp_hun.ih_pmtu.ipm_nextmtu
#define icmp_num_addrs icmp_hun.ih_rtradv.irt_num_addrs
#define icmp_wpa icmp_hun.ih_rtradv.irt_wpa
#define icmp_lifetime icmp_hun.ih_rtradv.irt_lifetime
union
{
struct
{
u_int32_t its_otime;
u_int32_t its_rtime;
u_int32_t its_ttime;
} id_ts;
struct
{
struct ip idi_ip;
/* options and then 64 bits of data */
} id_ip;
struct icmp_ra_addr id_radv;
u_int32_t id_mask;
u_int8_t id_data[1];
} icmp_dun;
#define icmp_otime icmp_dun.id_ts.its_otime
#define icmp_rtime icmp_dun.id_ts.its_rtime
#define icmp_ttime icmp_dun.id_ts.its_ttime
#define icmp_ip icmp_dun.id_ip.idi_ip
#define icmp_radv icmp_dun.id_radv
#define icmp_mask icmp_dun.id_mask
#define icmp_data icmp_dun.id_data
};




使用宏定義令表達更簡潔,其中ICMP報頭為8字節,數據報長度最大為64K字節。

校驗和算法――這一算法稱為網際校驗和算法,把被校驗的數據16位進行累加,然后取反碼,若數據字節長度為奇數,則數據尾部補一個字節的0以湊成偶數。此算法適用于IPv4、ICMPv4、IGMPV4、ICMPv6、UDP和TCP校驗和,更具體的信息請參考RFC1071,校驗和字段為上述ICMP數據結構的icmp_cksum變量。
標識符――用于唯一標識ICMP報文, 為上述ICMP數據結構的icmp_id宏所指的變量。
順序號――ping命令的icmp_seq便由這里讀出,代表ICMP報文的發送順序,為上述ICMP數據結構的icmp_seq宏所指的變量。

ICMP數據報
Ping命令中需要顯示的信息,包括icmp_seq和ttl都已有實現的辦法,但還缺rtt往返時間。為了實現這一功能,可利用ICMP數據報攜帶一個時間戳。使用以下函數生成時間戳:
#include
int gettimeofday(struct timeval *tp,void *tzp)
其中timeval結構如下:
struct timeval{
long tv_sec;
long tv_usec;
}



其中tv_sec為秒數,tv_usec微秒數。在發送和接收報文時由gettimeofday分別生成兩個timeval結構,兩者之差即為往返時間,即ICMP報文發送與接收的時間差,而timeval結構由ICMP數據報攜帶,tzp指針表示時區,一般都不使用,賦NULL值。


數據統計
系統自帶的ping命令當它接送完所有ICMP報文后,會對所有發送和所有接收的ICMP報文進行統計,從而計算ICMP報文丟失的比率。為達此目的,定義兩個全局變量:接收計數器和發送計數器,用于記錄ICMP報文接受和發送數目。丟失數目=發送總數-接收總數,丟失比率=丟失數目/發送總數。

現給出模擬Ping程序功能的代碼如下:
/***********************************************************
* 作者:梁俊輝 *
* 時間:2001年10月 *
* 名稱:myping.c *
* 說明:本程序用于演示ping命令的實現原理 *
***********************************************************/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include

#define PACKET_SIZE 4096
#define MAX_WAIT_TIME 5
#define MAX_NO_PACKETS 3

char sendpacket[PACKET_SIZE];
char recvpacket[PACKET_SIZE];
int sockfd,datalen=56;
int nsend=0,nreceived=0;
struct sockaddr_in dest_addr;
pid_t pid;
struct sockaddr_in from;
struct timeval tvrecv;

void statistics(int signo);
unsigned short cal_chksum(unsigned short *addr,int len);
int pack(int pack_no);
void send_packet(void);
void recv_packet(void);
int unpack(char *buf,int len);
void tv_sub(struct timeval *out,struct timeval *in);

void statistics(int signo)
{ printf("/n--------------------PING statistics-------------------/n");
printf("%d packets transmitted, %d received , %%%d lost/n",nsend,nreceived,
(nsend-nreceived)/nsend*100);
close(sockfd);
exit(1);
}
/*校驗和算法*/
unsigned short cal_chksum(unsigned short *addr,int len)
{ int nleft=len;
int sum=0;
unsigned short *w=addr;
unsigned short answer=0;

/*把ICMP報頭二進制數據以2字節為單位累加起來*/
while(nleft>1)
{ sum+=*w++;
nleft-=2;
}
/*若ICMP報頭為奇數個字節,會剩下最后一字節。把最后一個字節視為一個2字節數據的高字節,這個2字節數據的低字節為0,繼續累加*/
if( nleft==1)
{ *(unsigned char *)(&answer)=*(unsigned char *)w;
sum+=answer;
}
sum=(sum>>16)+(sum&0xffff);
sum+=(sum>>16);
answer=~sum;
return answer;
}
/*設置ICMP報頭*/
int pack(int pack_no)
{ int i,packsize;
struct icmp *icmp;
struct timeval *tval;

icmp=(struct icmp*)sendpacket;
icmp->icmp_type=ICMP_ECHO;
icmp->icmp_code=0;
icmp->icmp_cksum=0;
icmp->icmp_seq=pack_no;
icmp->icmp_id=pid;
packsize=8+datalen;
tval= (struct timeval *)icmp->icmp_data;
gettimeofday(tval,NULL); /*記錄發送時間*/
icmp->icmp_cksum=cal_chksum( (unsigned short *)icmp,packsize); /*校驗算法*/
return packsize;
}

/*發送三個ICMP報文*/
void send_packet()
{ int packetsize;
while( nsend { nsend++;
packetsize=pack(nsend); /*設置ICMP報頭*/
if( sendto(sockfd,sendpacket,packetsize,0,
(struct sockaddr *)&dest_addr,sizeof(dest_addr) )<0 )
{ perror("sendto error");
continue;
}
sleep(1); /*每隔一秒發送一個ICMP報文*/
}
}

/*接收所有ICMP報文*/
void recv_packet()
{ int n,fromlen;
extern int errno;

signal(SIGALRM,statistics);
fromlen=sizeof(from);
while( nreceived { alarm(MAX_WAIT_TIME);
if( (n=recvfrom(sockfd,recvpacket,sizeof(recvpacket),0,
(struct sockaddr *)&from,&fromlen)) <0)
{ if(errno==EINTR)continue;
perror("recvfrom error");
continue;
}
gettimeofday(&tvrecv,NULL); /*記錄接收時間*/
if(unpack(recvpacket,n)==-1)continue;
nreceived++;
}

}
/*剝去ICMP報頭*/
int unpack(char *buf,int len)
{ int i,iphdrlen;
struct ip *ip;
struct icmp *icmp;
struct timeval *tvsend;
double rtt;

ip=(struct ip *)buf;
iphdrlen=ip->ip_hl<<2; /*求ip報頭長度,即ip報頭的長度標志乘4*/
icmp=(struct icmp *)(buf+iphdrlen); /*越過ip報頭,指向ICMP報頭*/
len-=iphdrlen; /*ICMP報頭及ICMP數據報的總長度*/
if( len<8) /*小于ICMP報頭長度則不合理*/
{ printf("ICMP packets/'s length is less than 8/n");
return -1;
}
/*確保所接收的是我所發的的ICMP的回應*/
if( (icmp->icmp_type==ICMP_ECHOREPLY) && (icmp->icmp_id==pid) )
{ tvsend=(struct timeval *)icmp->icmp_data;
tv_sub(&tvrecv,tvsend); /*接收和發送的時間差*/
rtt=tvrecv.tv_sec*1000+tvrecv.tv_usec/1000; /*以毫秒為單位計算rtt*/
/*顯示相關信息*/
printf("%d byte from %s: icmp_seq=%u ttl=%d rtt=%.3f ms/n",
len,
inet_ntoa(from.sin_addr),
icmp->icmp_seq,
ip->ip_ttl,
rtt);
}
else return -1;
}

main(int argc,char *argv[])
{ struct hostent *host;
struct protoent *protocol;
unsigned long inaddr=0l;
int waittime=MAX_WAIT_TIME;
int size=50*1024;

if(argc<2)
{ printf("usage:%s hostname/IP address/n",argv[0]);
exit(1);
}

if( (protocol=getprotobyname("icmp") )==NULL)
{ perror("getprotobyname");
exit(1);
}
/*生成使用ICMP的原始套接字,這種套接字只有root才能生成*/
if( (sockfd=socket(AF_INET,SOCK_RAW,protocol->p_proto) )<0)
{ perror("socket error");
exit(1);
}
/* 回收root權限,設置當前用戶權限*/
setuid(getuid());
/*擴大套接字接收緩沖區到50K這樣做主要為了減小接收緩沖區溢出的
的可能性,若無意中ping一個廣播地址或多播地址,將會引來大量應答*/
setsockopt(sockfd,SOL_SOCKET,SO_RCVBUF,&size,sizeof(size) );
bzero(&dest_addr,sizeof(dest_addr));
dest_addr.sin_family=AF_INET;

/*判定是主機名還是ip地址*/
if( inaddr=inet_addr(argv[1])==INADDR_NONE)
{ if((host=gethostbyname(argv[1]) )==NULL) /*是主機名*/
{ perror("gethostbyname error");
exit(1);
}
memcpy( (char *)&dest_addr.sin_addr,host->h_addr,host->h_length);
}
else /*是ip地址*/
memcpy( (char *)&dest_addr,(char *)&inaddr,host->h_length);
/*獲取main的進程id,用于設置ICMP的標志符*/
pid=getpid();
printf("PING %s(%s): %d bytes data in ICMP packets./n",argv[1],
inet_ntoa(dest_addr.sin_addr),datalen);
send_packet(); /*發送所有ICMP報文*/
recv_packet(); /*接收所有ICMP報文*/
statistics(SIGALRM); /*進行統計*/

return 0;

}
/*兩個timeval結構相減*/
void tv_sub(struct timeval *out,struct timeval *in)
{ if( (out->tv_usec-=in->tv_usec)<0)
{ --out->tv_sec;
out->tv_usec+=1000000;
}
out->tv_sec-=in->tv_sec;
}
/*------------- The End -----------*/




非凡注重
只有root用戶才能利用socket()函數生成原始套接字,要讓Linux的一般用戶能執行以上程序,需進行如下的非凡操作:

用root登陸,編譯以上程序:gcc -o myping myping.c,其目的有二:一是編譯,二是讓myping屬于root用戶。

再執行chmod u+s myping,目的是把myping程序設成SUID的屬性。

退出root,用一般用戶登陸,執行./myping www.cn.ibm.com,有以下執行結果:
PING www.cn.ibm.com(202.95.2.148): 56 bytes data in ICMP packets.
64 byte from 202.95.2.148: icmp_seq=1 ttl=242 rtt=3029.000 ms
64 byte from 202.95.2.148: icmp_seq=2 ttl=242 rtt=2020.000 ms
64 byte from 202.95.2.148: icmp_seq=3 ttl=242 rtt=1010.000 ms

--------------------PING statistics-------------------
3 packets transmitted, 3 received , %0 lost




由于myping.c是發送完所有的ICMP報文才去接收,因此第一、第二和第三個ICMP報文的往返時間依此是3秒,2秒,1秒,上述結果中rtt信息正反映這一事實。

作者簡介
梁俊輝,對Linux的網絡應用和程序設計有濃厚愛好,并且專注于這一方面研究,在IBM developerWorks――Linux專區上發表過《NEWT程序設計指南》一文。



發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国模gogo一区二区大胆私拍| 欧美极品少妇与黑人| 亚洲视频电影图片偷拍一区| 国内成人精品一区| 国产精品国产福利国产秒拍| 日韩资源在线观看| 久久噜噜噜精品国产亚洲综合| 性欧美长视频免费观看不卡| 久久久久久久亚洲精品| 九色91av视频| 日韩精品中文在线观看| 丝袜亚洲另类欧美重口| 欧美影院在线播放| 成人观看高清在线观看免费| 2018中文字幕一区二区三区| 亚洲深夜福利网站| 中文.日本.精品| 国产精品成人国产乱一区| 久久久久久国产三级电影| 奇米四色中文综合久久| 国产精品一久久香蕉国产线看观看| 狠狠躁天天躁日日躁欧美| 亚洲xxxxx性| 欧美日韩综合视频| 日韩成人激情视频| 欧美日韩美女在线观看| 国内免费精品永久在线视频| 久热爱精品视频线路一| 精品视频在线观看日韩| 九九精品在线观看| 欧美美最猛性xxxxxx| 91免费国产网站| 视频直播国产精品| 精品性高朝久久久久久久| 国产欧美日韩丝袜精品一区| 国产成人精品综合久久久| 懂色av影视一区二区三区| 国产成人一区二区在线| 亚洲视频在线观看免费| 伊人一区二区三区久久精品| 色偷偷噜噜噜亚洲男人的天堂| 亚洲精品欧美极品| 911国产网站尤物在线观看| 激情av一区二区| 狠狠躁夜夜躁人人爽超碰91| 日韩av免费在线| 欧美精品久久久久| 欧美激情奇米色| 久久免费少妇高潮久久精品99| 国产高清视频一区三区| 久久青草精品视频免费观看| 欧美中文字幕在线视频| 亚洲欧洲一区二区三区在线观看| 日韩久久免费视频| 久久精彩免费视频| 中文在线资源观看视频网站免费不卡| 红桃av永久久久| 97久久久久久| 亚洲qvod图片区电影| 日韩精品福利在线| 亚洲色图欧美制服丝袜另类第一页| 亚洲免费一级电影| 91免费人成网站在线观看18| 欧美日韩加勒比精品一区| 日韩电影免费在线观看中文字幕| 亚洲图片欧美午夜| 91精品国产九九九久久久亚洲| 日本精品视频在线播放| 亚洲v日韩v综合v精品v| 国产日韩欧美成人| 国产精品欧美风情| 亚洲大尺度美女在线| 欧美第一页在线| 亚洲成人黄色在线| 日韩高清欧美高清| 亚洲精品欧美日韩| 久久国产精品免费视频| 亚洲欧美国产日韩天堂区| 色综合伊人色综合网站| 97成人在线视频| 日韩不卡在线观看| 中文字幕亚洲欧美日韩高清| 久久久久久久爱| 国产一区二区三区在线观看视频| 中文字幕日韩有码| 欧美亚洲午夜视频在线观看| 欧美激情精品久久久久久免费印度| 成人淫片在线看| 午夜精品久久久久久久99黑人| 亚洲午夜未满十八勿入免费观看全集| 国产精品极品在线| 亚洲国产精品推荐| 久久精品国产96久久久香蕉| 欧美精品18videosex性欧美| 亚洲欧美日韩精品久久| 一区二区福利视频| 国产精品一区二区久久久| 欧美国产极速在线| 欧美性色视频在线| 成人有码视频在线播放| 国产视频精品va久久久久久| 亚洲国产美女久久久久| 中文字幕亚洲欧美一区二区三区| 狠狠操狠狠色综合网| 国产精品视频在线播放| 欧美重口另类videos人妖| 91精品国产91| 亚洲欧美日韩天堂一区二区| 中文字幕亚洲一区二区三区| 日韩电影在线观看永久视频免费网站| 91久久精品国产91久久性色| 中文字幕一区二区三区电影| 日韩亚洲欧美中文在线| 欧美中在线观看| 成人夜晚看av| 欧美专区国产专区| 国产精品久久久久av免费| 欧美日韩在线一区| 91精品国产综合久久男男| 欧美日韩在线另类| 成人免费午夜电影| 欧美寡妇偷汉性猛交| 亚洲高清在线观看| 国产精品福利在线观看| 欧美猛少妇色xxxxx| 欧洲精品在线视频| 亚洲精品自在久久| 国产va免费精品高清在线观看| 丝袜亚洲欧美日韩综合| 日韩av影院在线观看| 136fldh精品导航福利| 亚洲激情小视频| 欧美成人免费va影院高清| 久久免费成人精品视频| 欧洲美女7788成人免费视频| 亚洲国产成人在线视频| 在线亚洲国产精品网| 日本亚洲欧洲色| 国产免费一区二区三区在线能观看| 日韩中文在线不卡| 欧美高清无遮挡| yw.139尤物在线精品视频| 国产婷婷97碰碰久久人人蜜臀| 欧美精品videossex性护士| 亚洲欧美福利视频| xvideos亚洲| 欧美亚洲国产成人精品| 国产精品综合不卡av| 日韩av网址在线观看| 亚洲天堂免费在线| 一本色道久久88综合亚洲精品ⅰ| 日韩风俗一区 二区| 中文在线不卡视频| 久久久久国产精品一区| 这里只有视频精品| 精品成人乱色一区二区| 上原亚衣av一区二区三区| 91综合免费在线| 欧美在线欧美在线| 色综合天天狠天天透天天伊人| 91欧美精品午夜性色福利在线| 国产精品海角社区在线观看| 国产精品久久91| 亚洲偷熟乱区亚洲香蕉av|