5

这是一项任务,它必须与原始套接字一起使用。我需要编写一个简单的 icmp ping。我以此为基础http://www.pdbuchan.com/rawsock/icmp4.c。第 127 行是 wlan0,我使用的是 eth0。在第 290 行,我编写了以下代码:

struct sockaddr_in rec;
  
unsigned char * pkt = (unsigned char *) malloc (IP_MAXPACKET * sizeof (unsigned char));

if (recvfrom (sd, (void*)pkt, IP4_HDRLEN + ICMP_HDRLEN+datalen , 0, NULL, (socklen_t*)sizeof (struct sockaddr)) < 0)  {
  perror ("recvfrom() failed ");
  exit (EXIT_FAILURE);
}
struct ip *ip = (struct ip *)pkt;
struct icmphdr *icmp = (struct icmphdr *)(pkt + IP4_HDRLEN);

printf("%s %s %d\n",(char*)inet_ntoa(*(struct in_addr*)&ip->ip_dst),
        (char*)inet_ntoa(*(struct in_addr*)&ip->ip_src),
        icmp->type);
free (pkt);

问题是 ip_dst 和 ip_src 显示为我机器的 IP,icmp 类型为 0 而不是 8。Wireshark 显示 icmp 回复和请求。可能我的 recvfrom 是错误的,但我听说 linux 自己的 TCP/IP 可能正在处理数据包。如果这是真的,那么解决方法是什么?

编辑:我检查了这个原始套接字侦听器,但它没有解决我的问题。

4

1 回答 1

7

我认为您在使用 IPPROTO_RAW 时无法得到回复。

你必须使用

socket (AF_INET, SOCK_RAW, IPPROTO_ICMP);

使用 IPPROTO_ICMP 您只能发送一个 ICMP 数据包,而不是整个 IP 数据包。

然而,在接收时,您将获得整个 IP 数据包,并且必须提取 ICMP 回复。请注意,您将获得发送到主机的所有 ICMP 数据包的副本,因此您必须过滤它们。

于 2012-11-25T05:35:37.140 回答