当前位置:首页 > 单片机 > embed linux share
[导读]讲解一个小巧精致、可以直接在Linux平台下运行的tcp协议栈,带你感受源码剖析tcp的快感!

前言

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

知识回顾

在前面的文章中,我们已经介绍了虚拟网卡的封装接口,其中主要是以下几个接口:
  • tun_calloc():打开并设置虚拟网卡
  • tun_read():读取虚拟网卡数据
  • tun_write():发送虚拟网卡数据
这几个接口是我们封装以太网数据接口的基础,最好还是先搞明白原理。

以太网头部结构

在tcp的多层协议中,以太网数据帧位于最底层的数据链路层,如下图:
我们重点关注图片中的MAC头部(以太网头部)结构,因为它是我们最终往发送数据中填充的信息内容。我们先来看一下MAC头部的结构,如下图;
了解了以太网头部结构之后,下一步,自然就是用c语言结构体来构造这个以太网头部结构,level-ip的以太网头部结构保存在include\ethernet.h文件中,如下图:
  • 第3行:以太网目的地址,由6个8位16进制的数字构成。表示目标主机的物理地址
  • 第4行:以太网源地址,由6个8位16进制的数字构成。表示源主机的物理地址
  • 第5行:帧类型,用来区分ip数据包还是arp数据包(后面会详细解析两者区别)
  • 第6行:变长数组,gcc编译支持变长数组的用法
结构体用__attributr__关键字来设置结构体按照实际占用字节数进行对齐,而变长数组又不占据结构体内存,因此结构体实际大小为14个字节。

以太网卡管理结构体

在上面对以太网头部结构的介绍中,我们可以看到它保存着以太网源地址,这个以太网源地址实际上就对应着我们的电脑上网卡,我们知道,电脑上面实际可能会有多个网卡,因此在level-ip中必定需要一个结构体来对这些网卡进行管理,这个结构体就是struct netdev,它定义在level-ip的include\netdev.h文件上,如下图;
  • 第2行,记录本地网卡的静态ip地址,该地址是以网络传输的数值格式保存的
  • 第3行,记录地址长度
  • 第4行,记录本地网卡的以太网地址,由6个8位16进制的数字构成
  • 第5行,记录网卡的最大传输单元
以后,我们将把网卡相关信息记录在这个struct netdev结构体中,然后在发送数据时,从中获取以太网源地址。

网卡结构体初始化

上面介绍了struct netdev结构体,接下来我们看一下,level-ip是如何初始化它的。如下图:
netdev_init()函数负责初始化网卡的ip地址、mac地址和mtu的值,这个函数的调用为:main()->init_stack()->netdev_init()。在该函数里面,使用到了netdev_alloc函数,函数负责具体的初始化工作,如下图:
  • 第3行,动态申请内存给netdev结构体使用
  • 第5行,调用ip_parse函数,将十进制的ip地址转化为用于网络传输的数值格式
  • 第7行,调用sscanf函数,填充以太网地址到netdev结构体
  • 第14行,设置地址长度
  • 第15行,设置最大单元发送长度

以太网发送接口

终于来到重头戏!level-ip的以太网发送接口为netdev_transmit()函数,该函数保存在src\netdev.c文件中,如下图:
  • 第3行,定义了一个netdev结构体指针,前面介绍过了,这个结构体保存了本地网卡的ip、以太网地址等信息。
  • 第4行,定义了一个eth_hdr结构体指针,我们将通过这个结构体来给数据帧填充以太网头部
  • 第7行,从sk_buff结构体中获取netdev结构体。在用户连接网络,准备发送数据的时候,会用sk_buff来记录了待发送的数据帧和选中的网卡型号,因此能通过该结构体来获取netdev结构体,这个地方以后会进一步深入讲解
  • 第9行,skb_push函数会把sk_buff中的数据帧指针,往前移动eth_hdr结构体大小,这样我们就能通过eth_hdr结构体来给数据帧添加以太网头部了。
  • 第13~17行,从netdev结构体提取网卡信息,填充到数据帧头部,得到待发送的完整数据帧。
  • 第19行,调用虚拟网卡发送函数,来进行以太网帧的数据发送。

以太网接收接口

level-ip的以太网发送接口为netdev_receive()函数,该函数保存在src\netdev.c文件中,如下图:
以太网的数据接收过程如下:
main函数中调用run_threads函数,来创建netdev_rx_loop线程,netdev_rx_loop线程负责调用虚拟网卡读取接口tun_read来读取数据,把读取到的数据缓存到sk_buff结构体后,调用netdev_receive函数来进行以太网头部解析。下面我们看netdev_receive函数分析过程:
第3行,调用eth_hdr函数,从sk_buff结构体中,获取以太网头部数据
第7行,判断该帧数据是属于ARP数据帧、IP数据帧、还是IPV6数据帧。这些概念我们下一篇文章再分析,这里只要了解以太网头部数据是怎么获取的就可以了。
第8~19行,对不同的类型的帧作进一步处理。

总结

通过我们这边文章,我们已经明白了以太网帧头部的三要素:
  • 以太网目的地址
  • 以太网源地址
  • 以太网帧类型




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

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 隧道灯 驱动电源
关闭