Linux 的网络系统主要是基于 BSD UNIX
的套接字机制。

在系统与驱动程序之间定义了数据结构 sk_buff
进行传输数据。系统支持对发送数据和接收数据缓存,提供流控机制并提供对多协议的支持。

1、 linux 网络驱动程序的体系结构

linux 网络驱动程序的体系结构从上到下分为4层。各层作用例如以下:

(1)      网络协议接口层向网络层协议提供统一的数据包收发接口。不论上层是IP还是ARP。都通过
dev_queue_xmit 函数发送数据。并通过 net_rx
函数接收数据。这一层的存在使得上层协议独立于详细的设备。

(2)      网络设备接口层向协议接口层提供统一的用于描写叙述据详细网络设备属性和操作的结构体 net_dev。net_dev是设备驱动功能层中各函数的容器。

网络协议接口层从宏观上规划了详细操作硬件的设备驱动功能层的结构。

(3)      设备驱动功能层各函数是网络设备接口层 net_dev
数据结构的详细成员。是驱使网络设备硬件完毕对应动作的程序。通过 hard_start_xmit
函数启动发送操作,并通过网络设备上的中断触发接收操作。

因为网络数据包的接收由中断引发,设备驱动功能层中还有一个主体部分是中断处理函数,中断处理函数负责读取硬件上接收的数据包并传递给上层协议,可能包括了xxx_interrupt
和 xxx_rx
函数。

xxx_interrupt
函数完毕中断类型推断等基本工作, xxx_rx
函数完毕数据包的生成和递交上层等复杂工作。

(4)      网络设备与媒介层是完毕数据包发送和接收的物理实体,包含了网路适配器饿坏详细的传输媒介,网络设备被设备驱动功能层中的函数物理上驱动。在linux中。网络设备和媒介都能够是虚拟的。

在linux 中全部网络系统都抽象为一个接口。由结构体 struct net_device
来表示网络设备在内核中的执行情况,即网络设备接口。网络设备接口既包含了纯软件网络设备接口。也包含了硬件网络设备接口。全部的网络设备都是通过以 dev_base
为头指针的设备链表来管理的。该设备链表中的每个元素代表一个网络接口。结构体 net_device
中包括了非常多供系统訪问和协议层调用的设备方法,包括了 init
函数(设备初始化和系统注冊)、open
函数(打开网络设备)、stop
函数(关闭网络设备)、hard_start_xmit
函数(处理数据包发送)和中断处理函数等。

2、   网络设备的初始化

网络设备的初始化主要是 struct net_device
中的 init
函数指针所指向的初始化函数来完毕。

当内核启动或者载入网络驱动模块的时候,就会调用 init
函数。init
函数通过检測物理设备的硬件特征推断网络物理设备是不是存在。然后才决定是不是启用这个驱动程序。

接着对设备进行资源配置,配置好之后就能够向系统申请资源,如中断和 i/o
空间。最后对 net_device
对应的成员变量初始化。这样一个网络设备才干被系统使用。

对于一个新加入的设备。则须要构造设备的 net_device
数据结构,然后向linux
内核注冊设备并申请内存空间。

3、   数据包的发送和接收

当网络设备被激活的时候,调用 net_device
中的 open
函数指针。打开网络设备。并使用 net_device
中的 hard_header
函数指针来建立硬件帧头信息。

然后通过 dev_queue_xmit
协议接口层函数来调用 net_device
中的 hard_start_xmit
函数指针完毕数据包的发送。hard_start_xmit
函数负责把存放在套接字缓冲区(sk_buff)中的数据发送到物理设备。

普通情况下,设备受到数据就会产生一个中断,在中断处理时驱动程序申请一块 sk_buff
。从硬件读出数据并放到申请好的缓冲区中,然后填充sk_buff
的一些信息。最后调用 netif_rx
函数把接收到的数据包传到网络协议的上层进行处理。

当网络设备被激活的时候。调用 net_device
中的 open
函数指针,打开网络设备,并使用 net_device
中的 hard_header
函数指针来建立硬件帧头信息。然后通过 dev_queue_xmit
协议接口层函数来调用 net_device
中的 hard_start_xmit
函数指针完毕数据包的发送。hard_start_xmit
函数负责把存放在套接字缓冲区(sk_buff)中的数据发送到物理设备。

普通情况下,设备受到数据就会产生一个中断,在中断处理时驱动程序申请一块 sk_buff
,从硬件读出数据并放到申请好的缓冲区中,然后填充sk_buff
的一些信息。最后调用 netif_rx
函数把接收到的数据包传到网络协议的上层进行处理。

sk_buff 结构体(即套接字缓冲区)用于在linux 网络子系统中的各层之间传递数据。当发送数据包的时候,linux内核的网络处理模块必须建立一个包括要传输的数据包的 sk_buff
,然后将 sk_buff
递交给下层,各层在 sk_buff
中加入不同的协议头直至交给网络设备发送。当网络设备从网络媒介上接收到数据包的时候,必须将接收到的数据装换为 sk_buff
数据结构并传递给上层,各层去掉对应的协议头直至给交用户。

4、   网络设备的模块载入

网络设备驱动程序载入方式有内核载入和模块载入两种方式。

系统内核有专门载入网络设备的过程,也能够通过 module_init
宏为系统加入新网络设备。此处主要讨论网络设备的模块载入。其基本流程为:

(1)      通过 module_init
宏修饰的函数会在模块载入(或系统启动)时被调用;

(2)      网络设备被检測到后,通过调用
register_netdev 函数在 linux
系统中把设备加入到系统的网络设备链表 dev_base
的末尾;

(3)      注冊成功,将调用net_device
中的 init
函数,初始化设备。

通过 register_netdev 函数注冊网络设备是一个动态载入的过程。在 linux 2.6
内核中,全部网络设备都是用该函数载入的。模块载入不存在为网络设备预留空间,从而节省了系统内存资源。

5、 net_device
结构细节

netdevice.h文件保存在 include/linux
文件夹中

(1)      全局信息

char                   name[IFNAMSIZ];

设备名称。

分配的编号从零開始。

unsigned long          state;

设备状态。包括若干标识。

struct net_device      *next;

指向全局链表下一个设备的指针。

(2)      硬件信息

unsignedlong           mem_end;

unsigned long          mem_start;

设备内存信息。保存了设备使用的共享内存值起始和终止地址

unsigned long          base_addr;

网络接口的的I/O基地址。

unsigned int           irq;

被赋予的中断号。

unsigned char          if_port;

指定在多port设备上使用哪个port。

unsigned char          dma;

为设备分配的DMA通道。用于显示信息(ifconfig命令)。

(3)      设备方法

下面仅列举使用接口必需的方法。

int                     (*open)(struct net_device*dev);

打开接口。在ifconfig激活接口时,接口将被打开。

open函数应该注冊全部的系统资源。打开硬件并对设备运行其它所需的设置。

int                     (*stop)(struct net_device*dev);

停止接口。该函数运行的操作与open函数相反。

int                     (*hard_header) (struct sk_buff *skb,

                         structnet_device *dev,

                         unsigned short type,

                         void *daddr,

                         void *saddr,

                         unsigned len);

该函数依据先前检索到的源和目标硬件地址建立硬件头。该函数任务是将作为參数传递进入的信息,组织成设备特有的适当硬件头。

eth_header
是以太网类型接口的默认函数。ether_setup
将该成员赋值成eth_header。

 int                     (*rebuild_header)(structsk_buff *skb);

该函数用来在数据传输包之前,完毕ARP解析之后,又一次建立硬件头。

  void                    (*tx_timeout) (structnet_device *dev);

如果数据包的传输在合理的时间段内失败。则如果丢失了中断或接口被锁住。这时将调用该方法。tx_timeout
负责解决这个问题并又一次開始数据包的传输。

int                     (*set_config)(structnet_device *dev,

                                      struct ifmap *map);

改变接口配置。该函数是配置驱动程序的入口点。利用set_config
能够在执行中改变设备的I/O地址和中断号。

struct net_device_stats* (*get_stats)(struct net_device*dev);

当应用程序须要获得接口上的统计信息的时候。将会调用该函数。

最新文章

  1. ansible使用文档
  2. 测试一下Word发布刚刚发现通过Word 可以直接发Blog 感觉很新奇,先看看,如果可以呢,将发通知的改了。
  3. angular 指令作用域 scope
  4. osc的git服务器简单使用。
  5. C++11 图说VS2013下的引用叠加规则和模板参数类型推导规则
  6. wpf 旋转效果
  7. Block 的基本用法
  8. 【MYSQL】在脚本中使用变量-执行脚本时传参
  9. C# WinForm控件、自定义控件整理(大全)
  10. 网页上PNG透明图片的ie6bug
  11. Java内存模型JMM与可见性
  12. Java基础知识强化58:经典排序之二叉树排序(BinaryTreeSort)
  13. C#学习日志 day 6 ------ 常用正则表达式例举
  14. 解决MVC模型验证在IE 6 7下不起作用或者报错
  15. Ceph: A Scalable, High-Performance Distributed File System译文
  16. sql 选取分组中的第一条,显示聚合以外的列,having字句的使用
  17. [开源] C# 封装 银海医保的接口
  18. VS中自定义代码段
  19. ROS学习笔记三(理解ROS节点)
  20. 使用VMWareWorkstation10搭建学习环境笔记

热门文章

  1. SimpleMappingExceptionResolver异常映射
  2. 导入Excel时去除多余的空白行
  3. 关于react-router-dom入门配置
  4. 如何拿到阿里算法校招offer
  5. 【Oracle】DBA_AUDIT_TRAIL表SES_ACTIONS字段对应的操作列表
  6. 【转载】Java实现word转pdf
  7. C++调用Matlab 注意事项
  8. 易观OLAP算法大赛结果揭晓,开源组黑马放大招!
  9. JS中常见问题
  10. Pycharm 设置