Linux网络编程:原始套接字的魔力【续】

如何从链路层直接发送数据帧
       本来以为这部分都弄完了,结果有朋友反映说看了半天还是没看到如何从链路层直接发送数据。因为上一篇里面提到的是从链路层“收发”数据,结果只“收”完,忘了“发”,实在抱歉,所以就有这篇续出来了。
       上一节我们主要研究了如何从链路层直接接收数据帧,可以通过 bind函数来将原始套接字绑定到本地一个接口上,然后该套接字就只接收从该接口收上来的对应的数据包。今天我们用原始套接字来手工实现链路层 ARP报文的发送和接收,以便大家对原始套接字有更深刻的掌握和理解。
       ARP全称为地址解析协议,是链路层广泛使用的一种寻址协议,完成 32比特 IP地址到 48比特 MAC地址的映射转换。在以太网中,当一台主机需要向另外一台主机发送消息时,它会首先在自己本地的 ARP缓存表中根据目的主机的 IP地址查找其对应的 MAC地址,如果找到了则直接向其发送消息。如果未找到,它首先会在全网发送一个 ARP广播查询,这个查询的消息会被以太网中所有主机接收到,然后每个主机就根据 ARP查询报文中所指定的 IP地址来检查该报文是不是发给自己的,如果不是则直接丢弃;只有被查询的目的主机才会对这个消息进行响应,然后将自己的 MAC地址通告给发送者。
       也就是说,链路层中是根据 MAC地址来确定唯一一台主机。以太帧格式如下:
       以太帧首部中 2字节的帧类型字段指定了其上层所承载的具体协议,常见的有 0x0800表示是 IP报文、 0x0806表示 RARP协议、 0x0806即为我们将要讨论的 ARP协议。
 硬件类型: 1表示以太网。
 协议类型: 0x0800表示 IP地址。和以太头部中帧类型字段相同。
 硬件地址长度和协议地址长度:对于以太网中的 ARP协议而言,分别为 64
 操作码: 1表示 ARP请求; 2表示 ARP应答; 3表示 RARP请求; 4表示 RARP应答。
       我们这里只讨论硬件地址为以太网地址、协议地址为 IP地址的情形,所以剩下四个字段就分别表示发送方的 MACIP地址、接收方的 MACIP地址了。
       注意:对于一个 ARP请求报文来说,除了接收方硬件地址外,其他字段都要填充。当系统收到一个 ARP请求时,会查询该请求报文中接收方的协议地址是否和自己的 IP地址相等,如果相等,它就把自己的硬件地址和协议地址填充进去,将发送和接收方的地址互换,然后将操作码改为 2,发送回去。

       下面看一个使用原始套接字发送 ARP请求的例子:
点击( 此处)折叠或打开
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <errno.h>
  6. #include <sys/socket.h>
  7. #include <sys/ioctl.h>
  8. #include <sys/types.h>
  9. #include <netinet/in.h>
  10. #include <netinet/ip.h>
  11. #include <netinet/if_ether.h>
  12. #include <net/if_arp.h>
  13. #include <netpacket/packet.h>
  14. #include <net/if.h>
  15. #include <net/ethernet.h>

  16. #define BUFLEN 42

  17. int main(int argc,char** argv){
  18.     int skfd,n;
  19.     char buf[BUFLEN]={0};
  20.     struct ether_header *eth;
  21.     struct ether_arp *arp;
  22.     struct sockaddr_ll toaddr;
  23.     struct in_addr targetIP,srcIP;
  24.     struct ifreq ifr;

  25.     unsigned char src_mac[ETH_ALEN]={0};
  26.     unsigned char dst_mac[ETH_ALEN]={0xff,0xff,0xff,0xff,0xff,0xff}; //全网广播ARP请求
  27.     if(3 != argc){
  28.             printf("Usage: %s netdevName dstIP\n",argv[0]);
  29.             exit(1);
  30.     }

  31.     if(0>(skfd=socket(PF_PACKET,SOCK_RAW,htons(ETH_P_ALL)))){
  32.             perror("Create Error");
  33.             exit(1);
  34.     }

  35.     bzero(&toaddr,sizeof(toaddr));
  36.     bzero(&ifr,sizeof(ifr));
  37.     strcpy(ifr.ifr_name,argv[1]);

  38.     //获取接口索引
  39.     if(-1 == ioctl(skfd,SIOCGIFINDEX,&ifr)){
  40.            perror("get dev index error:");
  41.            exit(1);
  42.     }
  43.     toaddr.sll_ifindex = ifr.ifr_ifindex;
  44.     printf("interface Index:%d\n",ifr.ifr_ifindex);
  45.     //获取接口IP地址
  46.     if(-1 == ioctl(skfd,SIOCGIFADDR,&ifr)){
  47.            perror("get IP addr error:");
  48.            exit(1);
  49.     }
  50.     srcIP.s_addr = ((struct sockaddr_in*)&(ifr.ifr_addr))->sin_addr.s_addr;
  51.     printf("IP addr:%s\n",inet_ntoa(((struct sockaddr_in*)&(ifr.ifr_addr))->sin_addr));

  52.     //获取接口的MAC地址
  53.     if(-1 == ioctl(skfd,SIOCGIFHWADDR,&ifr)){
  54.            perror("get dev MAC addr error:");
  55.            exit(1);
  56.     }

  57.     memcpy(src_mac,ifr.ifr_hwaddr.sa_data,ETH_ALEN);
  58.     printf("MAC :%02X-%02X-%02X-%02X-%02X-%02X\n",src_mac[0],src_mac[1],src_mac[2],src_mac[3],src_mac[4],src_mac[5]);


  59.     //开始填充,构造以太头部
  60.     eth=(struct ether_header*)buf;
  61.     memcpy(eth->ether_dhost,dst_mac,ETH_ALEN);
  62.     memcpy(eth->ether_shost,src_mac,ETH_ALEN);
  63.     eth->ether_type = htons(ETHERTYPE_ARP);

  64.     //手动开始填充用ARP报文首部
  65.     arp=(struct arphdr*)(buf+sizeof(struct ether_header));
  66.     arp->arp_hrd = htons(ARPHRD_ETHER); //硬件类型为以太
  67.     arp->arp_pro = htons(ETHERTYPE_IP); //协议类型为IP

  68.     //硬件地址长度和IPV4地址长度分别是6字节和4字节
  69.     arp->arp_hln = ETH_ALEN;
  70.     arp->arp_pln = 4;

  71.     //操作码,这里我们发送ARP请求
  72.     arp->arp_op = htons(ARPOP_REQUEST);
  73.       
  74.     //填充发送端的MAC和IP地址
  75.     memcpy(arp->arp_sha,src_mac,ETH_ALEN);
  76.     memcpy(arp->arp_spa,&srcIP,4);

  77.     //填充目的端的IP地址,MAC地址不用管
  78.     inet_pton(AF_INET,argv[2],&targetIP);
  79.     memcpy(arp->arp_tpa,&targetIP,4);

  80.     toaddr.sll_family = PF_PACKET;
  81.     n=sendto(skfd,buf,BUFLEN,0,(struct sockaddr*)&toaddr,sizeof(toaddr));

  82.     close(skfd);
  83.     return 0;
  84. }
      结果如下:
       可以看到,我向网关发送一个 ARP查询请求,报文中携带了网关的 IP地址以及我本地主机的 IPMAC地址。网关收到该请求后,对我的这个报文进行了回应,将它的 MAC地址在 ARP应答报文中发给我了。
       在这个示例程序中,我们完全自己手动构造了以太帧头部,并完成了整个 ARP请求报文的填充,最后用 sendto函数,将我们的数据通过 eth0接口发送出去。这个程序的灵活性还在于支持多网卡,使用时只要指定网卡名称 (eth0eth1),程序便会自动去获取指定接口相应的 IPMAC地址,然后用它们去填充 ARP请求报文中对应的各字段。
       在头文件 <net/thernet.h>里,主要对以太帧首部进行了封装:
点击( 此处)折叠或打开
  1. struct ether_header
  2. {
  3.    u_int8_t ether_dhost[ETH_ALEN]; /* destination eth addr */
  4.    u_int8_t ether_shost[ETH_ALEN]; /* source ether addr */
  5.    u_int16_t ether_type; /* packet type ID field */
  6. } __attribute__ ((__packed__));
      在头文件 <net/if_arp.h>中,对 ARP首部进行了封装:
点击( 此处)折叠或打开
  1. struct arphdr
  2. {
  3.     unsigned short ar_hrd; /* format of hardware address */
  4.     unsigned short ar_pro; /* format of protocol address */
  5.     unsigned char ar_hln; /* length of hardware address */
  6.     unsigned char ar_pln; /* length of protocol address */
  7.     unsigned short ar_op; /* ARP opcode (command) */
  8. }
       而头文件 <netinet/if_ether.h>里,又对 ARP整个报文进行了封装:
点击( 此处)折叠或打开
  1. struct ether_arp {
  2.     struct arphdr ea_hdr; /* fixed-size 8 bytes header */
  3.     u_int8_t arp_sha[ETH_ALEN]; /* sender hardware address */
  4.     u_int8_t arp_spa[4]; /* sender protocol address */
  5.     u_int8_t arp_tha[ETH_ALEN]; /* target hardware address */
  6.     u_int8_t arp_tpa[4]; /* target protocol address */
  7. };

  8. #define arp_hrd ea_hdr.ar_hrd
  9. #define arp_pro ea_hdr.ar_pro
  10. #define arp_hln ea_hdr.ar_hln
  11. #define arp_pln ea_hdr.ar_pln
  12. #define arp_op ea_hdr.ar_op
    最后再看一个简单的接收 ARP 报文的小程序: 
点击( 此处)折叠或打开
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <errno.h>
  6. #include <sys/socket.h>
  7. #include <sys/ioctl.h>
  8. #include <sys/types.h>
  9. #include <netinet/in.h>
  10. #include <netinet/ip.h>
  11. #include <netinet/if_ether.h>
  12. #include <net/if_arp.h>
  13. #include <netpacket/packet.h>
  14. #include <net/if.h>
  15. #define BUFLEN 60

  16. int main(int argc,char** argv){
  17.     int i,skfd,n;
  18.     char buf[ETH_FRAME_LEN]={0};
  19.     struct ethhdr *eth;
  20.     struct ether_arp *arp;
  21.     struct sockaddr_ll fromaddr;
  22.     struct ifreq ifr;

  23.     unsigned char src_mac[ETH_ALEN]={0};

  24.     if(2 != argc){
  25.         printf("Usage: %s netdevName\n",argv[0]);
  26.         exit(1);
  27.     }

  28.     //只接收发给本机的ARP报文
  29.     if(0>(skfd=socket(PF_PACKET,SOCK_RAW,htons(ETH_P_ARP)))){
  30.         perror("Create Error");
  31.         exit(1);
  32.     }

  33.     bzero(&fromaddr,sizeof(fromaddr));
  34.     bzero(&ifr,sizeof(ifr));
  35.     strcpy(ifr.ifr_name,argv[1]);

  36.     //获取接口索引
  37.     if(-1 == ioctl(skfd,SIOCGIFINDEX,&ifr)){
  38.         perror("get dev index error:");
  39.         exit(1);
  40.     }
  41.     fromaddr.sll_ifindex = ifr.ifr_ifindex;
  42.     printf("interface Index:%d\n",ifr.ifr_ifindex);

  43.     //获取接口的MAC地址
  44.     if(-1 == ioctl(skfd,SIOCGIFHWADDR,&ifr)){
  45.         perror("get dev MAC addr error:");
  46.         exit(1);
  47.     }

  48.     memcpy(src_mac,ifr.ifr_hwaddr.sa_data,ETH_ALEN);
  49.     printf("MAC :%02X-%02X-%02X-%02X-%02X-%02X\n",src_mac[0],src_mac[1],src_mac[2],src_mac[3],src_mac[4],src_mac[5]);

  50.     fromaddr.sll_family = PF_PACKET;
  51.     fromaddr.sll_protocol=htons(ETH_P_ARP);
  52.     fromaddr.sll_hatype=ARPHRD_ETHER;
  53.     fromaddr.sll_pkttype=PACKET_HOST;
  54.     fromaddr.sll_halen=ETH_ALEN;
  55.     memcpy(fromaddr.sll_addr,src_mac,ETH_ALEN);

  56.     bind(skfd,(struct sockaddr*)&fromaddr,sizeof(struct sockaddr));

  57.     while(1){
  58.         memset(buf,0,ETH_FRAME_LEN);
  59.         n=recvfrom(skfd,buf,ETH_FRAME_LEN,0,NULL,NULL);
  60.         eth=(struct ethhdr*)buf;
  61.         arp=(struct ether_arp*)(buf+14);

  62.         printf("Dest MAC:");
  63.         for(i=0;i<ETH_ALEN;i++){
  64.             printf("%02X-",eth->h_dest[i]);
  65.         }
  66.         printf("Sender MAC:");
  67.         for(i=0;i<ETH_ALEN;i++){
  68.             printf("%02X-",eth->h_source[i]);
  69.         }

  70.         printf("\n");
  71.         printf("Frame type:%0X\n",ntohs(eth->h_proto));

  72.         if(ntohs(arp->arp_op)==2){
  73.             printf("Get an ARP replay!\n");
  74.         }
  75.     }
  76.     close(skfd);
  77.     return 0;
  78. }
  该示例程序中,调用 recvfrom之前我们调用了 bind系统调用,目的是仅从指定的接口接收 ARP报文 (socket函数的第三个参数“ ETH_P_ARP”决定 )。可以对比一下,该程序与博文“ Linux网络编程:原始套接字的魔力【下】 ”里介绍的抓包程序的区别。
 小 结:通过这几个章节的热身,相信大家对网络编程中常见的一系列API函数 socket,bind,listen,connect,sendto,recvfrom,close等的认识应该会有一个较高的突破。当然,你也必须赶 快对它们熟悉起来,因为后面我们不但要“知其然”,还要知其“所以然”。后面,我们会以这些函数调用为主线,看看它们到底在内核中做些哪些事情,而这又对 我们理解协议栈的实现原理有什么帮助做进一步的分析和讨论。

你可能感兴趣的:(Linux网络编程:原始套接字的魔力【续】)