当前位置:首页 > 单片机 > embed linux share
[导读]源码剖析tcp协议栈!直接在任意Linux系统下运行调试,直面玩转tcp协议zhan的快感!


前言

阅读本文需要对level-ip的整体架构有所了解,如果读者尚未接触过level-ip,请先阅读下面文章:
分享一款Linux平台下的tcp协议栈!超级透彻!
level-ip之虚拟网卡接口封装
level-ip之以太网数据接口封装
请根据上述文章中的指引获取leve-ip的全部源码,并且尝试在任意Linux发行版本上编译运行。

知识回顾

在前面的文章中,我们已经介绍了以太网卡的封装接口,其中主要是以下几个接口:
  • netdev_init():初始化网卡的ip地址、mac地址和mtu的值
  • netdev_receive():发送以太网帧数据
  • netdev_transmit():发送以太网帧数据
这几个接口是我们封装ARP数据接口的基础,最好还是先搞明白原理。

ARP协议的由来

在上面,我们介绍netdev_receive()函数的时候,已经发现了以太网帧类型主要分两大类型,一种是IP数据帧,另一种是ARP数据帧。也就是说ARP数据帧与IP数据帧同属于网络层的数据帧。如下图:
IP数据帧我们知道,是用来传输用户数据的。哪ARP数据帧有什么用呢?
其实,ARP协议是用来将目标主机的IP地址转换为对应的以太网(MAC)地址的。因为当我们的应用程序要向目标主机发送信息时,它只知道目标主机的IP地址,而IP地址是无法直接用于物理链路上传输数据的,所以需要ARP数据帧来把IP地址转化为对应的MAC地址。
我们可以主动发起ARP查询帧,在本地建立起IP地址和MAC地址的映射关系,也必须要及时回复别人的ARP查询帧!

ARP报文组织结构

ARP数据帧位于以太网数据帧的上一层,我们先来了解一下它的报文结构,如下图:
我们来详解学习一下,里面每个字段所代表的具体含义;
  • 硬件协议:发送方想要知道的硬件接口类型,对于以太网接口来说,该值为1
  • 协议类型:映射的协议地址类型,我们要把MAC地址映射为IP地址,该值为0x0800
  • 硬件地址长度:对于MAC地址来说,该值为6
  • 协议地址长度:对于IP地址来说,该值为4
  • OP:表示ARP数据包的具体类型,1为ARP请求,2为ARP应答
剩余四个字段的具体含义非常简单易懂,就不罗列出来讲解了。
了解ARP报文组织结构之后,下一步,自然就是用c语言结构体来构造这个ARP报文组织,level-ip的ARP报文组织结构体保存在include\ethernet.h文件中,如下图:
这两个结构体的成员变量,与我们刚才介绍的ARP报文的每个字段是一一对应的,这里不再重复解析。

ARP请求发送接口

ARP数据帧的发送接口为arp_request()函数。该函数保存在src/arp.c文件中。当我们在发送IP数据帧时,如果在ARP缓存表中找不到该IP所对应的MAC地址时,就会通过广播的形式,来进行ARP请求数据包的发送。
如下图:
  • 第8行,动态申请一个sk_buff来继续发送数据的存储。
  • 第12行,选择使用哪个网卡来继续数据帧的发送
  • 第13行,在sk_buff中,向前移动arp_ipv4结构体大小的位置,把得到的指针赋值给payload指针
  • 第14行,用网卡(netdev)中记录的源主机mac地址,填充arp-ipv4结构体中的源主机mac地址(smac)
  • 第15行,填充arp-ipv4结构体中的源主机ip地址(sip)
  • 第16行,用广播地址(broadcast_hw),填充arp-ipv4结构体中的目的主机mac地址(dmac)
  • 第17行,填充arp-ipv4结构体中的目的主机ip地址(dip)
  • 第18行,在sk_buff中,向前移动arp_hdr结构体大小的位置,把得到的指针赋值给arp指针
  • 第19~29行,初始化ARP报文的硬件协议、协议类型、报文类型等等,htons()函数为进行数据的大小端切换。到这里ARP报文就初始化好了
  • 第31行,调用netdev_transmit()函数,进一步构建以太网数据帧发送

ARP数据读取接口

ARP数据接收接口为arp_rcv()函数。该函数在以太网数据帧读取接口netdev_receive()函数中调用。我们来了解一下这个函数,如下图:
  • 第8行,从读取到的数据中获取arp数据帧
  • 第10~12行,获取arp数据帧中的硬件类型、协议类型、报文类型
  • 第25~28行,获取源主机和目的主机的ip地址
  • 第30行,继续arp缓存表数据的更新
  • 第32行,判断该arp数据帧,是不是发送给本机的
  • 第37行,如果arp数据帧中的IP地址还没有缓存在本机的ARP缓存表中的话,那么把这个IP地址插入到ARP缓存表中保存
  • 第42行,判断ARP数据帧的报文类型
  • 第43、44行,如果报文类型为ARP请求帧,那么调用arp_reply()函数进行ARP应答帧的发送

ARP应答帧发送接口

在上面我们介绍ARP数据读取接口时,当我们如果接收到了ARP请求帧,那么我们要调用arp_reply()函数进行ARP应答帧的发送,我们来学习一下这个函数。
如下图:
  • 第6行,获取arp报文的数据
  • 第8行,使用skb_reserve()函数来调整sk_buff中数据指针的位置,表示以太网首部和ARP报文的数据都还没有填充
  • 第9行,使用skb_push()函数,参数为ARP_HDR_LEN ARP_DATA_LEN,表示填充了ARP报文
  • 第11~28行,将该ARP请求数据包的源主机信息和目的主机信息交换位置,并把操作字段op置为2
  • 第30行,选择发送网卡
  • 第32行,调用netdev_transmit()函数,进一步构建以太网数据帧发送

总结

通过我们这边文章,我们已经明白了ARP协议的报文结构、ARP数据包的发送、ARP数据包的接收处理等等。知道了ARP协议在TCP协议栈中的重要地位。不过文中对ARP缓存表没有做深入介绍,这是因为该知识点比较基础,主要是对链表的插入、删除等操作。




本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭