ping程序使用原始套接字,套接口使用
socket(AF_INET,SOCK_RAW,IPPROTO_ICMP)来开启。
报文中的LLC和IP报头由kernel自动填写,其中L2 layer有14bytes分别为:
source mac (6bytes)
destination mac (6bytes)
type (2bytes)
ip报头为20个字节。
ICMP部分由用户自己填写:
pkt->icmp_type = ICMP_ECHO; //1byte
pkt->icmp_code= 0; //1byte
pkt->icmp_cksum= 0; //2byte
pkt->icmp_seq= 0; //2byte
pkt->icmp_id=getpid(); //2byte
pkt->icmp_cksum = 0;checkSum((unsigned short *) pkt, sizeof(packet));
剩下的为ping带出去的数据,可以任意填。
接收回来的数据中跳过ip头,然后可以判断ping的结果。
pkt = (struct icmp *) (packet + sizeof(struct iphdr));
if (pkt->icmp_type == ICMP_ECHOREPLY && pkt->icmp_id == getpid()){
pingResult = 0;
close(pingsock);
}
在发送和接收之间使用select函数进行超时的处理。
FD_ZERO(&fds);
FD_SET(pingsock, &fds);
ret = select(pingsock+1,&fds,NULL,NULL,&tmout);
if (ret <= 0){
DDBG("timeout\n");
close(pingsock);
return -1;
}
阅读(2276) | 评论(0) | 转发(0) |