如何从链路层直接发送数据帧
       本来以为这部分都弄完了,结果有朋友反映说看了半天还是没看到如何从链路层直接发送数据。因为上一篇里面提到的是从链路层“收发”数据,结果只“收”完,忘了“发”,实在抱歉,所以就有这篇续出来了。
       上一节我们主要研究了如何从链路层直接接收数据帧,可以通过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协议而言,分别为6和4;
 操作码:1表示ARP请求;2表示ARP应答;3表示RARP请求;4表示RARP应答。
       我们这里只讨论硬件地址为以太网地址、协议地址为IP地址的情形,所以剩下四个字段就分别表示发送方的MAC和IP地址、接收方的MAC和IP地址了。
       注意:对于一个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. //填充发送端的MAC和IP地址
  74. memcpy(arp->arp_sha,src_mac,ETH_ALEN);
  75. memcpy(arp->arp_spa,&srcIP,4);
  76. //填充目的端的IP地址,MAC地址不用管
  77. inet_pton(AF_INET,argv[2],&targetIP);
  78. memcpy(arp->arp_tpa,&targetIP,4);
  79. toaddr.sll_family = PF_PACKET;
  80. n=sendto(skfd,buf,BUFLEN,0,(struct sockaddr*)&toaddr,sizeof(toaddr));
  81. close(skfd);
  82. return 0;
  83. }

结果如下:

       可以看到,我向网关发送一个ARP查询请求,报文中携带了网关的IP地址以及我本地主机的IP和MAC地址。网关收到该请求后,对我的这个报文进行了回应,将它的MAC地址在ARP应答报文中发给我了。
       在这个示例程序中,我们完全自己手动构造了以太帧头部,并完成了整个ARP请求报文的填充,最后用sendto函数,将我们的数据通过eth0接口发送出去。这个程序的灵活性还在于支持多网卡,使用时只要指定网卡名称(如eth0或eth1),程序便会自动去获取指定接口相应的IP和MAC地址,然后用它们去填充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等的认识应该会有一个较高的突破。当然,你也必须赶 快对它们熟悉起来,因为后面我们不但要“知其然”,还要知其“所以然”。后面,我们会以这些函数调用为主线,看看它们到底在内核中做些哪些事情,而这又对 我们理解协议栈的实现原理有什么帮助做进一步的分析和讨论。

转载于:https://blog.51cto.com/yehubilee/1069078

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

  1. Linux网络编程——原始套接字编程

    Linux网络编程--原始套接字编程 转自:http://blog.csdn.net/tennysonsky/article/details/44676377 原始套接字编程和之前的 UDP 编程差不 ...

  2. Linux网络编程——原始套接字能干什么?

    一.知识回顾: 通常情况下程序员接所接触到的套接字(Socket)为两类: (1)流式套接字(SOCK_STREAM):一种面向连接的 Socket,针对于面向连接的TCP 服务应用: (2)数据报式 ...

  3. Linux 网络编程——原始套接字实例:MAC 地址扫描器

    如果 A (192.168.1.1 )向 B (192.168.1.2 )发送一个数据包,那么需要的条件有 ip.port.使用的协议(TCP/UDP)之外还需要 MAC 地址,因为在以太网数据包中 ...

  4. Linux原始网络编程,Linux操作系统网络编程 原始套接字 (1)

    Linux操作系统网络编程--原始套接字 (1) http://soft.zdnet.com.cn/software_zone/2007/1020/568223.shtml 我们在前面已经学习过了网络 ...

  5. Linux网络编程之套接字基础

    Linux网络编程之套接字基础 1.套接字的基本结构 struct sockaddr 这个结构用来存储套接字地址. 数据定义: struct sockaddr { unsigned short sa_ ...

  6. linux串口编程实例_Linux 网络编程——原始套接字实例:发送 UDP 数据包

    以太网报文格式: IP 报文格式: UDP 报文格式: 校验和函数: /*******************************************************功能:校验和函数参 ...

  7. 【Linux网络编程】套接字简介

    00. 目录 文章目录 00. 目录 01. 概述 02. 套接字属性 03. socket函数 04. 套接字地址结构 05. 附录 01. 概述 Socket套接字由远景研究规划局(Advance ...

  8. 【Linux网络编程】套接字的介绍

    套接字是一种通信机制(通信的两方的一种约定),凭借这种机制,不同主机之间的进程可以进行通信.我们可以用套接字中的相关函数来完成通信过程. 套接字的特性有三个属性确定,它们是:域(domain),类型( ...

  9. Linux网络编程 之 套接字(四)

    目录 1. 套接字的定义 2. 套接字的创建方法 3. 套接字的地址 本地套接字 网络套接字 1. 套接字的定义 套接字是一种通信机制(通信的两方的一种约定),凭借这种机制,不同主机之间的进程可以进行 ...

  10. linux网络编程 华清,Linux网络编程之套接字

    一 :套接字属性 套接字由域(domain),类型(type)和协议(protocol)三个属性确定其特性. 1)套接字的域 域指定套接字通信中使用的网络 介质,常见的套接字域是AF_INET,它指的 ...

最新文章

  1. 采集虚拟机_系列文章:Kubernetes日志采集最佳实践
  2. C语言函数strstr 分析及实现
  3. NeHe教程Qt实现——lesson14
  4. java byte 循环左移 循环右移 rotateLeft rotateRight
  5. 【PC工具】简单好用的截屏gif录制小软件
  6. C++ COM编程之接口背后的虚函数表
  7. java异常处理框架_深入探索 高效的Java异常处理框架(1)
  8. ROS笔记(21) 地图
  9. Seata多微服务互相调用_全局分布式事物使用案例_@GlobalTransactional验证---微服务升级_SpringCloud Alibaba工作笔记0065
  10. php 怎么复制一个文件,php如何复制文件夹?
  11. ssm read time out的原因_为什么得肝病的男人越来越多?爱喝酒不是原因,或跟老婆有关系!...
  12. C#获取程序运行时间
  13. centos 卸载 jdk
  14. 一整个网站的全部数据,我只能给你这么多了。
  15. 人工智能AI系列 - 问答系统
  16. 数据结构中的英文及算法缩写
  17. 企业信息系统战略规划
  18. petalinux - 修改fsbl
  19. python tkinter 表格 怎么设置字体大小_Tkinter动态字体大小更改
  20. 我越脱俗,就会越世俗

热门文章

  1. java 图片组合 分解_切分和组合图片(二)
  2. oracle rac standby,oracle RAC数据库建立STANDBY(二)
  3. vue防抖和节流是什么_JavaScript防抖与节流,你知道多少?
  4. 当电压放大电路的开路增益和输出电阻固定后_晶体管放大电路的性能分析与应用...
  5. 用Java写有关早上的语录,实用的适合早上发的早安问候语语录汇编39句
  6. 使用JAVA爬取博客里面的所有文章
  7. 骑驴找马!在职期间如何优雅的去面试?
  8. 头条 上传图片大小_【标签头条】北京市启用进口冷链食品追溯平台;全球包裹热潮助推标签业发展;数字水印实现大规模垃圾分类;安慕希的麻将酸奶包装好真实...
  9. 作者:石勇(1956-),男,中国科学院大学经济管理学院教授、博士生导师
  10. 作者:赵江华(1989-),女,中国科学院计算机网络信息中心研究实习员