www.pudn.com > 11_code.rar > ping.c


(1)主体代码 
ping代码的主体部分可以四部分,首先是一些头函数及宏定义:  
#include  
#include  
#include  
#include  
#include  
 
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#define	F_FLOOD	0x001 
#define	F_INTERVAL	0x002 
#define	F_NUMERIC	0x004 
#define	F_PINGFILLED	0x008 
#define	F_QUIET	0x010 
#define	F_RROUTE	0x020 
#define	F_SO_DEBUG	0x040 
#define	F_SO_DONTROUTE	0x080 
#define	F_VERBOSE	0x100 
 
/* 多播选项 */ 
int moptions; 
#define MULTICAST_NOLOOP	0x001 
#define MULTICAST_TTL		0x002 
#define MULTICAST_IF		0x004 
… 
接下来的第二部分是建立socket并处理选项: 
Int main(int argc, char *argv[]) 
{ 
	struct timeval timeout; 
	struct hostent *hp; 
	struct sockaddr_in *to; 
	struct protoent *proto; 
	struct in_addr ifaddr; 
	int i; 
	int ch, fdmask, hold, packlen, preload; 
	u_char *datap, *packet; 
	char *target, hnamebuf[MAXHOSTNAMELEN]; 
	u_char ttl, loop; 
	int am_i_root; 
… 
	static char *null = NULL; 
 
	/*__environ = &null;*/ 
	am_i_root = (getuid()==0); 
 
	/* 
	 *建立socket连接,并且测试是否是root用户 
	 */ 
	if ((s = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP)) < 0) { 
		if (errno==EPERM) { 
			fprintf(stderr, "ping: ping must run as root\n"); 
		} 
		else perror("ping: socket"); 
		exit(2); 
	} 
… 
	preload = 0; 
	datap = &outpack[8 + sizeof(struct timeval)]; 
	while ((ch = getopt(argc, argv, "I:LRc:dfh:i:l:np:qrs:t:v")) != EOF) 
		switch(ch) { 
		case 'c': 
			npackets = atoi(optarg); 
			if (npackets <= 0) { 
				(void)fprintf(stderr, 
				    "ping: bad number of packets to transmit.\n"); 
				exit(2); 
			} 
			break; 
/*调用选项*/ 
		case 'd': 
			options |= F_SO_DEBUG; 
			break; 
/*flood选项*/ 
		case 'f': 
			if (!am_i_root) { 
				(void)fprintf(stderr, 
				    "ping: %s\n", strerror(EPERM)); 
				exit(2); 
			} 
			options |= F_FLOOD; 
			setbuf(stdout, NULL); 
			break; 
/*等待选项*/ 
		case 'i':		/* wait between sending packets */ 
			interval = atoi(optarg); 
			if (interval <= 0) { 
				(void)fprintf(stderr, 
				    "ping: bad timing interval.\n"); 
				exit(2); 
			} 
			options |= F_INTERVAL; 
			break; 
		case 'l': 
			if (!am_i_root) { 
				(void)fprintf(stderr, 
				    "ping: %s\n", strerror(EPERM)); 
				exit(2); 
			} 
			preload = atoi(optarg); 
			if (preload < 0) { 
				(void)fprintf(stderr, 
				    "ping: bad preload value.\n"); 
				exit(2); 
			} 
			break; 
	… 
		default: 
			usage(); 
		} 
	argc -= optind; 
	argv += optind; 
	 
	if (argc != 1) 
		usage(); 
	target = *argv; 
 
接下来的第三部分是用于获取地址,这里主要使用了inet_aton函数,将点分十进制地址转化为二进制地址。当然,作为完整的ping程序有较完善的出错处理: 
	memset(&whereto, 0, sizeof(struct sockaddr)); 
	to = (struct sockaddr_in *)&whereto; 
	to->sin_family = AF_INET; 
/*地址转换函数*/ 
	if (inet_aton(target, &to->sin_addr)) { 
		hostname = target; 
	} 
	else { 
#if 0 
		char * addr = resolve_name(target, 0); 
		if (!addr) { 
			(void)fprintf(stderr, 
			    "ping: unknown host %s\n", target); 
			exit(2); 
		} 
		to->sin_addr.s_addr = inet_addr(addr); 
		hostname = target; 
#else 
/*调用gethostbyname识别主机名*/ 
		hp = gethostbyname(target); 
		if (!hp) { 
			(void)fprintf(stderr, 
			    "ping: unknown host %s\n", target); 
			exit(2); 
		} 
		to->sin_family = hp->h_addrtype; 
		if (hp->h_length > (int)sizeof(to->sin_addr)) { 
			hp->h_length = sizeof(to->sin_addr); 
		} 
		memcpy(&to->sin_addr, hp->h_addr, hp->h_length); 
		(void)strncpy(hnamebuf, hp->h_name, sizeof(hnamebuf) - 1); 
		hostname = hnamebuf; 
#endif 
	} 
接下来的一部分主要是对各个选项(如路由、多播)选项的处理,这里就不做介绍。再接下来是ping函数的最主要部分,就是接收无限循环接收回应信息,这里主要用到了函数recvfrom。另外,对用户中断信息也有相应的处理,如下所示: 
	if (to->sin_family == AF_INET) 
		(void)printf("PING %s (%s): %d data bytes\n", hostname, 
		    inet_ntoa(*(struct in_addr *)&to->sin_addr.s_addr), 
		    datalen); 
	else 
		(void)printf("PING %s: %d data bytes\n", hostname, datalen); 
/*若程序接收到SIGINT或SIGALRM信号,调用相关的函数*/ 
	(void)signal(SIGINT, finish); 
	(void)signal(SIGALRM, catcher); 
… 
/*循环等待客户端的回应信息*/ 
	for (;;) { 
		struct sockaddr_in from; 
		register int cc; 
		int fromlen; 
 
		if (options & F_FLOOD) { 
/*形成ICMP回应数据包,在后面会有讲解*/ 
			pinger(); 
/*设定等待实践*/ 
			timeout.tv_sec = 0; 
			timeout.tv_usec = 10000; 
			fdmask = 1 << s; 
/*调用select函数*/ 
			if (select(s + 1, (fd_set *)&fdmask, (fd_set *)NULL, 
			    (fd_set *)NULL, &timeout) < 1) 
				continue; 
		} 
		fromlen = sizeof(from); 
/*接收客户端信息*/ 
		if ((cc = recvfrom(s, (char *)packet, packlen, 0, 
		    (struct sockaddr *)&from, &fromlen)) < 0) { 
			if (errno == EINTR) 
				continue; 
			perror("ping: recvfrom"); 
			continue; 
		} 
		pr_pack((char *)packet, cc, &from); 
		if (npackets && nreceived >= npackets) 
			break; 
	} 
	finish(0); 
	/* NOTREACHED */ 
	return 0; 
} 
(2)其他函数 
下面的函数也是ping程序中用到的重要函数。首先catcher函数是用户在发送SIGINT时调用的函数,在该函数中又调用了SIGALARM信号的处理来结束程序。 
static void 
catcher(int ignore) 
{ 
	int waittime; 
 
	(void)ignore; 
	pinger(); 
/*调用catcher函数*/ 
	(void)signal(SIGALRM, catcher); 
	if (!npackets || ntransmitted < npackets) 
		alarm((u_int)interval); 
	else { 
		if (nreceived) { 
			waittime = 2 * tmax / 1000; 
			if (!waittime) 
				waittime = 1; 
			if (waittime > MAXWAIT) 
				waittime = MAXWAIT; 
		} else 
			waittime = MAXWAIT; 
/*调用finish函数,并设定一定的等待实践*/ 
		(void)signal(SIGALRM, finish); 
		(void)alarm((u_int)waittime); 
	} 
} 
Pinger函数也是一个非常重要的函数,用于形成ICMP回应数据包,其中ID是该进程的ID,数据段中的前8字节用于存放时间间隔,从而可以计算ping程序从对端返回的往返时延差,这里的数据校验用到了后面定义的in_cksum函数。其代码如下所示: 
static void 
pinger(void) 
{ 
	register struct icmphdr *icp; 
	register int cc; 
	int i; 
 
/*形成icmp信息包,填写icmphdr结构体中的各项数据*/ 
	icp = (struct icmphdr *)outpack; 
	icp->icmp_type = ICMP_ECHO; 
	icp->icmp_code = 0; 
	icp->icmp_cksum = 0; 
	icp->icmp_seq = ntransmitted++; 
	icp->icmp_id = ident;			/* ID */ 
 
	CLR(icp->icmp_seq % mx_dup_ck); 
 
/*设定等待实践*/ 
	if (timing) 
		(void)gettimeofday((struct timeval *)&outpack[8], 
		    (struct timezone *)NULL); 
 
	cc = datalen + 8;			/* skips ICMP portion */ 
 
	/* compute ICMP checksum here */ 
	icp->icmp_cksum = in_cksum((u_short *)icp, cc); 
 
	i = sendto(s, (char *)outpack, cc, 0, &whereto, 
	    sizeof(struct sockaddr)); 
 
	if (i < 0 || i != cc)  { 
		if (i < 0) 
			perror("ping: sendto"); 
		(void)printf("ping: wrote %s %d chars, ret=%d\n", 
		    hostname, cc, i); 
	} 
	if (!(options & F_QUIET) && options & F_FLOOD) 
		(void)write(STDOUT_FILENO, &DOT, 1); 
} 
pr_pack是数据包显示函数,分别打印出IP数据包部分和ICMP回应信息。在规范的程序中通常将数据的显示部分独立出来,这样可以很好地加强程序的逻辑性和结构性。 
void 
pr_pack(char *buf, int cc, struct sockaddr_in *from) 
{ 
	register struct icmphdr *icp; 
	register int i; 
	register u_char *cp,*dp; 
/*#if 0*/ 
	register u_long l; 
	register int j; 
	static int old_rrlen; 
	static char old_rr[MAX_IPOPTLEN]; 
/*#endif*/ 
	struct iphdr *ip; 
	struct timeval tv, *tp; 
	long triptime = 0; 
	int hlen, dupflag; 
 
	(void)gettimeofday(&tv, (struct timezone *)NULL); 
 
	/* 检查IP数据包头信息 */ 
	ip = (struct iphdr *)buf; 
	hlen = ip->ip_hl << 2; 
	if (cc < datalen + ICMP_MINLEN) { 
		if (options & F_VERBOSE) 
			(void)fprintf(stderr, 
			  "ping: packet too short (%d bytes) from %s\n", cc, 
			  inet_ntoa(*(struct in_addr *)&from->sin_addr.s_addr)); 
		return; 
	} 
 
	/* ICMP部分显示 */ 
	cc -= hlen; 
	icp = (struct icmphdr *)(buf + hlen); 
	if (icp->icmp_type == ICMP_ECHOREPLY) { 
		if (icp->icmp_id != ident) 
			return;			/* 'Twas not our ECHO */ 
		++nreceived; 
		if (timing) { 
#ifndef icmp_data 
			tp = (struct timeval *)(icp + 1); 
#else 
			tp = (struct timeval *)icp->icmp_data; 
#endif 
			tvsub(&tv, tp); 
			triptime = tv.tv_sec * 10000 + (tv.tv_usec / 100); 
			tsum += triptime; 
			if (triptime < tmin) 
				tmin = triptime; 
			if (triptime > tmax) 
				tmax = triptime; 
		} 
 
		if (TST(icp->icmp_seq % mx_dup_ck)) { 
			++nrepeats; 
			--nreceived; 
			dupflag = 1; 
		} else { 
			SET(icp->icmp_seq % mx_dup_ck); 
			dupflag = 0; 
		} 
 
		if (options & F_QUIET) 
			return; 
 
		if (options & F_FLOOD) 
			(void)write(STDOUT_FILENO, &BSPACE, 1); 
		else { 
			(void)printf("%d bytes from %s: icmp_seq=%u", cc, 
			   inet_ntoa(*(struct in_addr *)&from->sin_addr.s_addr), 
			   icp->icmp_seq); 
			(void)printf(" ttl=%d", ip->ip_ttl); 
			if (timing) 
				(void)printf(" time=%ld.%ld ms", triptime/10, 
						triptime%10); 
			if (dupflag) 
				(void)printf(" (DUP!)"); 
			/* check the data */ 
#ifndef icmp_data 
			cp = ((u_char*)(icp + 1) + 8); 
#else 
			cp = (u_char*)icp->icmp_data + 8; 
#endif 
			dp = &outpack[8 + sizeof(struct timeval)]; 
			for (i = 8; i < datalen; ++i, ++cp, ++dp) { 
				if (*cp != *dp) { 
	(void)printf("\nwrong data byte #%d should be 0x%x but was 0x%x", 
	    i, *dp, *cp); 
					cp = (u_char*)(icp + 1); 
					for (i = 8; i < datalen; ++i, ++cp) { 
						if ((i % 32) == 8) 
							(void)printf("\n\t"); 
						(void)printf("%x ", *cp); 
					} 
					break; 
				} 
			} 
		} 
	} else { 
		/* We've got something other than an ECHOREPLY */ 
		if (!(options & F_VERBOSE)) 
			return; 
		(void)printf("%d bytes from %s: ", cc, 
		    pr_addr(from->sin_addr.s_addr)); 
		pr_icmph(icp); 
	} 
 
/*#if 0*/ 
	/*显示其他IP选项 */ 
	cp = (u_char *)buf + sizeof(struct iphdr); 
 
	for (; hlen > (int)sizeof(struct iphdr); --hlen, ++cp) 
		switch (*cp) { 
		case IPOPT_EOL: 
			hlen = 0; 
			break; 
		case IPOPT_LSRR: 
			(void)printf("\nLSRR: "); 
			hlen -= 2; 
			j = *++cp; 
			++cp; 
			if (j > IPOPT_MINOFF) 
				for (;;) { 
					l = *++cp; 
					l = (l<<8) + *++cp; 
					l = (l<<8) + *++cp; 
					l = (l<<8) + *++cp; 
					if (l == 0) 
						(void)printf("\t0.0.0.0"); 
				else 
					(void)printf("\t%s", pr_addr(ntohl(l))); 
				hlen -= 4; 
				j -= 4; 
				if (j <= IPOPT_MINOFF) 
					break; 
				(void)putchar('\n'); 
			} 
			break; 
		case IPOPT_RR: 
			j = *++cp;		/* get length */ 
			i = *++cp;		/* and pointer */ 
			hlen -= 2; 
			if (i > j) 
				i = j; 
			i -= IPOPT_MINOFF; 
			if (i <= 0) 
				continue; 
			if (i == old_rrlen 
			    && cp == (u_char *)buf + sizeof(struct iphdr) + 2 
			    && !memcmp((char *)cp, old_rr, i) 
			    && !(options & F_FLOOD)) { 
				(void)printf("\t(same route)"); 
				i = ((i + 3) / 4) * 4; 
				hlen -= i; 
				cp += i; 
				break; 
			} 
			old_rrlen = i; 
			memcpy(old_rr, cp, i); 
			(void)printf("\nRR: "); 
			for (;;) { 
				l = *++cp; 
				l = (l<<8) + *++cp; 
				l = (l<<8) + *++cp; 
				l = (l<<8) + *++cp; 
				if (l == 0) 
					(void)printf("\t0.0.0.0"); 
				else 
					(void)printf("\t%s", pr_addr(ntohl(l))); 
				hlen -= 4; 
				i -= 4; 
				if (i <= 0) 
					break; 
				(void)putchar('\n'); 
			} 
			break; 
		case IPOPT_NOP: 
			(void)printf("\nNOP"); 
			break; 
		default: 
			(void)printf("\nunknown option %x", *cp); 
			break; 
		} 
/*#endif*/ 
	if (!(options & F_FLOOD)) { 
		(void)putchar('\n'); 
		(void)fflush(stdout); 
	} 
} 
in_cksum是数据校验程序,如下所示: 
static int 
in_cksum(u_short *addr, int len) 
{ 
	register int nleft = len; 
	register u_short *w = addr; 
	register int sum = 0; 
	u_short answer = 0; 
 
	/*这里的算法很简单,就采用32bit的加法*/ 
	while (nleft > 1)  { 
		sum += *w++; 
		nleft -= 2; 
	} 
 
	if (nleft == 1) { 
		*(u_char *)(&answer) = *(u_char *)w ; 
		sum += answer; 
	} 
 
	/*把高16bit加到低16bit上去*/ 
	sum = (sum >> 16) + (sum & 0xffff); 
	sum += (sum >> 16); 
	answer = ~sum; 
	return(answer); 
} 
Finish程序是ping程序的结束程序,主要是打印出来一些统计信息,如下所示: 
static void 
finish(int ignore) 
{ 
	(void)ignore; 
 
	(void)signal(SIGINT, SIG_IGN); 
	(void)putchar('\n'); 
	(void)fflush(stdout); 
	(void)printf("--- %s ping statistics ---\n", hostname); 
	(void)printf("%ld packets transmitted, ", ntransmitted); 
	(void)printf("%ld packets received, ", nreceived); 
	if (nrepeats) 
		(void)printf("+%ld duplicates, ", nrepeats); 
	if (ntransmitted) 
		if (nreceived > ntransmitted) 
			(void)printf("-- somebody's printing up packets!"); 
		else 
			(void)printf("%d%% packet loss", 
			    (int) (((ntransmitted - nreceived) * 100) / 
			    ntransmitted)); 
	(void)putchar('\n'); 
	if (nreceived && timing) 
		(void)printf("round-trip min/avg/max = %ld.%ld/%lu.%ld/%ld.%ld ms\n", 
			tmin/10, tmin%10, 
			(tsum / (nreceived + nrepeats))/10, 
			(tsum / (nreceived + nrepeats))%10, 
			tmax/10, tmax%10); 
 
	if (nreceived==0) exit(1); 
	exit(0); 
} 
 
#ifdef notdef 
static char *ttab[] = { 
	"Echo Reply",		/* ip + seq + udata */ 
	"Dest Unreachable",	/* net, host, proto, port, frag, sr + IP */ 
	"Source Quench",	/* IP */ 
	"Redirect",		/* redirect 类型, gateway, + IP  */ 
	"Echo", 
	"Time Exceeded",	/*传输超时*/ 
	"Parameter Problem",	/* IP参数问题 */ 
	"Timestamp",		/* id + seq + three timestamps */ 
	"Timestamp Reply",	/* " */ 
	"Info Request",		/* id + sq */ 
	"Info Reply"		/* " */ 
}; 
#endif 
pr_icmph函数是用于打印出ICMP回应信息,如下所示: 
static void 
pr_icmph(struct icmphdr *icp) 
{ 
	switch(icp->icmp_type) { 
/*ICMP回应*/ 
	case ICMP_ECHOREPLY: 
		(void)printf("Echo Reply\n"); 
		/* XXX ID + Seq + Data */ 
		break; 
/*ICMP终点不可达*/ 
	case ICMP_DEST_UNREACH: 
		switch(icp->icmp_code) { 
		case ICMP_NET_UNREACH: 
			(void)printf("Destination Net Unreachable\n"); 
			break; 
		case ICMP_HOST_UNREACH: 
			(void)printf("Destination Host Unreachable\n"); 
			break; 
		case ICMP_PROT_UNREACH: 
			(void)printf("Destination Protocol Unreachable\n"); 
			break; 
		… 
		default: 
			(void)printf("Dest Unreachable, Unknown Code: %d\n", 
			    icp->icmp_code); 
			break; 
		} 
		/* Print returned IP header information */ 
#ifndef icmp_data 
		pr_retip((struct iphdr *)(icp + 1)); 
#else 
		pr_retip((struct iphdr *)icp->icmp_data); 
#endif 
		break; 
		… 
		default: 
			(void)printf("Redirect, Bad Code: %d", icp->icmp_code); 
			break; 
		} 
		(void)printf("(New addr: %s)\n",  
			     inet_ntoa(icp->icmp_gwaddr)); 
#ifndef icmp_data 
		pr_retip((struct iphdr *)(icp + 1)); 
#else 
		pr_retip((struct iphdr *)icp->icmp_data); 
#endif 
		break; 
	case ICMP_ECHO: 
		(void)printf("Echo Request\n"); 
		/* XXX ID + Seq + Data */ 
		break; 
	case ICMP_TIME_EXCEEDED: 
		switch(icp->icmp_code) { 
		case ICMP_EXC_TTL: 
			(void)printf("Time to live exceeded\n"); 
			break; 
		case ICMP_EXC_FRAGTIME: 
			(void)printf("Frag reassembly time exceeded\n"); 
			break; 
		default: 
			(void)printf("Time exceeded, Bad Code: %d\n", 
			    icp->icmp_code); 
			break; 
		} 
	… 
	default: 
		(void)printf("Bad ICMP type: %d\n", icp->icmp_type); 
	} 
} 
pr_iph函数是用于打印IP数据包头选项,如下所示: 
static void 
pr_iph(struct iphdr *ip) 
{ 
	int hlen; 
	u_char *cp; 
 
	hlen = ip->ip_hl << 2; 
	cp = (u_char *)ip + 20;		/* point to options */ 
	(void)printf("Vr HL TOS  Len   ID Flg  off TTL Pro  cks      Src      Dst Data\n"); 
	(void)printf(" %1x  %1x  %02x %04x %04x", 
	    ip->ip_v, ip->ip_hl, ip->ip_tos, ip->ip_len, ip->ip_id); 
	(void)printf("   %1x %04x", ((ip->ip_off) & 0xe000) >> 13, 
	    (ip->ip_off) & 0x1fff); 
	(void)printf("  %02x  %02x %04x", ip->ip_ttl, ip->ip_p, ip->ip_sum); 
	(void)printf(" %s ", inet_ntoa(*((struct in_addr *) &ip->ip_src))); 
	(void)printf(" %s ", inet_ntoa(*((struct in_addr *) &ip->ip_dst))); 
	/* dump and option bytes */ 
	while (hlen-- > 20) { 
		(void)printf("%02x", *cp++); 
	} 
	(void)putchar('\n'); 
} 
pr_addr是用于将ascii主机地址转换为十进制点分形式并打印出来,这里使用的函数是inet_ntoa,如下所示: 
static char * 
pr_addr(u_long l) 
{ 
	struct hostent *hp; 
	static char buf[256]; 
 
	if ((options & F_NUMERIC) || 
	    !(hp = gethostbyaddr((char *)&l, 4, AF_INET))) 
		(void)sprintf(buf, /*sizeof(buf),*/ "%s",inet_ntoa(*(struct in_addr *)&l)); 
	else 
		(void)sprintf(buf, /*sizeof(buf),*/ "%s (%s)", hp->h_name,inet_ntoa(*(struct in_addr *)&l)); 
	return(buf); 
} 
Usage函数是用于显示帮助信息,如下所示: 
static void 
usage(void) 
{ 
	(void)fprintf(stderr, 
	    "usage: ping [-LRdfnqrv] [-c count] [-i wait] [-l preload]\n\t[-p pattern] [-s packetsize] [-t ttl] [-I interface address] host\n"); 
	exit(2); 
}