【摘要】前文我们分析了一个虚拟硬件的网络驱动例子,从中我们看到了网络设备的一些接口,其实网络设备驱动和块设备驱动的功能比较类似,都是发送和接收数据包(数据请求)。当然它们实际是有很多不同的。

1、引言

首先块设备在/dev目录下有设备节点,而网络设备没有这样的设备入口。read,write等常规的文件接口在网络设备下也没有意义。 
最大的区别在于:块设备只响应内核的数据请求;而网络设备驱动要异步地接收来自外部的数据包。简单地说,块设备驱动是被要求传输数据而网络设备是主动请求传输数据。网络设备驱动还需要支持设置地址,修改传输参数等等这样的操作,所以网络设备驱动的api需要提供这些接口。

本文是对上文虚拟硬件的网络驱动例子进行一个简单的梳理。

(1)网络设备注册

头文件:<linux/netdevice.h>
struct net_device 网络设备结构体
struct net_device *alloc_netdev (int size_priv, const char *name, void (*setup)(struct net_device *));
int register_netdev(struct net_device *device); 注册网络设备
void unregitster_netdev(struct net_device *device); 注销网络设备

(2)打开和关闭 
驱动在加载入内核后,内核会调用probe函数来探测它。在网络接口可以传送数据包时,内核必须首先打开它并给它设置地址。内核打开和关闭网络接口是由ifconfig命令触发的。

int (*open)(struct net_device*); 打开网络设备
int (*stop)(struct net_device*); 关闭网络设备
void netif_start_queue(struct net_device*); 启动网络传输队列
void netif_stop_queue(struct net_device*); 关闭网络传输队列

(3)网络数据的发送 
网络接口最重要的作用是发送和接收网络数据。

头文件:<linux/skbuff.h> 定义了网络驱动中传输的基本单元,struct sk_buff
struct netdeviceops 网络设备驱动需要实现的接口函数
netdev_tx_t (*ndo_start_xmit) (struct sk_buff *skb, struct net_device *dev); 传输网络数据包的函数
void (*ndo_tx_timeout) (struct net_device *dev); 传输超时函数

(4)网络数据的接收 
接收网络数据相对于发送数据要复杂一些,因为你需要在原子上下文中把分配一个 sk_buff 并把它移交给上层处理。

数据包接收有两种实现方式:中断驱动和轮询。大多数驱动都是中断驱动的,有一些高吞吐量的驱动会使用轮询的方式。

struct sk_buff *dev_alloc_skb(unsigned int length); 原子上下文中分配skb
printk_ratelimit() 限制printk的输出频率,在中断相应函数中减少输出
  • 1
  • 2
  • 1
  • 2

实现高吞吐量的网络驱动,要减少网络阻塞最好的方法是使用napi,后面会介绍

(5)中断处理 
硬件可以中断cpu发送两种事件:新的数据包到来和发送的数据包已经发送完成。

判断并处理数据包事件 
如果在其他地方暂时停止了发送队列,应该在中断函数中重新启动它

(6)NAPI  
高吞吐量的网络接口如果每个数据包都用中断来处理的话会给系统带来很大的负担,这个时候应该使用基于轮询的 NAPI。这样可以减轻系统的负担,减少阻塞的时间。

只有极少数的设备实现了NAPI,因为实现起来比中断要复杂,而且有其他的一些条件。

在中断处理函数中,首先禁止进一步的中断处理,然后调度轮询函数,进入轮询函数后连续处理多个数据发送请求。

int (*poll)(struct net_device *dev, int *budget); 网络驱动轮询函数
int netif_rx_schedule(struct net_device *dev); 准备调用轮询函数
int (*poll)(struct net_device *dev, int *budget); 轮询函数

http://blog.csdn.net/xy010902100449/article/details/47132917

Linux 网卡驱动学习(二)(网络驱动接口小结)的更多相关文章

  1. Linux Platform驱动模型(二) _驱动方法

    在Linux设备树语法详解和Linux Platform驱动模型(一) _设备信息中我们讨论了设备信息的写法,本文主要讨论平台总线中另外一部分-驱动方法,将试图回答下面几个问题: 如何填充platfo ...

  2. Linux Platform驱动模型(二) _驱动方法【转】

    转自:http://www.cnblogs.com/xiaojiang1025/archive/2017/02/06/6367910.html 在Linux设备树语法详解和Linux Platform ...

  3. [Binder深入学习二]Binder驱动——基础数据结构二

    Userspace和KernelSpace进行交互时,大部分命令是通过 ioctl 实现的,在这个过程中,最重要的一个便是 BINDER_WRITE_READ 命令了. #define BINDER_ ...

  4. Linux——shell简单学习(二)

    流控制语句: for…done语句 格式:for  变量   in   名字表 do  命令列表 done 例子: #!/bin/sh for DAY in Sunday Monday Tuesday ...

  5. Linux运维学习笔记-网络技术知识体系总结

    传统网络知识体系 未来网络SND模型架构

  6. Linux运维学习笔记-网络安全等级保护

    网络安全等级保护简介与作用: 验证信息系统是否满足相应安全保护等级的一个过程. 对不同信息系统分等级进行保护.

  7. Linux网络驱动--snull

    snull是<Linux Device Drivers>中的一个网络驱动的例子.这里引用这个例子学习Linux网络驱动. 因为snull的源码,网上已经更新到适合最新内核,而我自己用的还是 ...

  8. 嵌入式Linux驱动学习之路(二十五)虚拟网卡驱动程序

    一.协议栈层次对比 设备无关层到驱动层的体系结构 1).网络协议接口层向网络层协议提供提供统一的数据包收发接口,不论上层协议为ARP还是IP,都通过dev_queue_xmit()函数发送数据,并通过 ...

  9. Linux 网卡驱动学习(一)(分析一个虚拟硬件的网络驱动样例)

    在Linux,网络分为两个层,各自是网络堆栈协议支持层,以及接收和发送网络协议的设备驱动程序层. 网络堆栈是硬件中独立出来的部分.主要用来支持TCP/IP等多种协议,网络设备驱动层是连接网络堆栈协议层 ...

随机推荐

  1. HDU 3966 RE 树链剖分 线段树 Aragorn's Story

    题意: 给出一棵树,每个顶点上有一个权值. 操作:选择一条路径,并将路径上所有的点的权值同时加或减某个数. 查询:某个点的当前权值 分析: 树链剖分完毕后,就是简单的线段树区间更新. 提交的时候注意要 ...

  2. 初试webpack打包

    第一次接触webpack,学习了如何用webpack打包,记录一下过程. 1.在项目根目录安装webpack $ npm install webpack --save-dev 2.新建一个webpac ...

  3. mysql primary partition分区

    尝试把数据库一个表分区 ALTER TABLE user PARTITION BY RANGE(TO_DAYS(`date`)) ( PARTITION p1004 VALUES LESS THAN  ...

  4. TOJ4277: Sequence 组合数学

    4277: Sequence Time Limit(Common/Java):2000MS/6000MS     Memory Limit:65536KByte Total Submit: 39   ...

  5. php 审批流程管理

    1.流程管理的用法是什么样的? 2.怎么发起想要的流程? 3.审批的人要是怎么审批通过? 4.流程审核是不是要挨个走过? 一.要有数据库的内容的 肯定会有表的,首先就是用户表了,然后就是流程表,用户编 ...

  6. 九度oj 题目1460:Oil Deposit

    题目描述: The GeoSurvComp geologic survey company is responsible for detecting underground oil deposits. ...

  7. [luoguP2765] 魔术球问题(最大流—最小不相交路径覆盖)

    传送门 枚举球的个数 num 如果 i < j && (i + j) 是完全平方数,那么 i -> j' 连一条边 再加一个超级源点 s,s -> i 再加一个超级汇 ...

  8. 论蛋疼的调戏matrix67的首页

    唔,初学js,然后拿matrix67的首页玩玩23333 console 里输入以下这句话 for(var i = 1; i <= 400; i++){var td = document.get ...

  9. TypeToken获取运行时泛型类型

    最近正好使用到了Guava的TypeToken来获取泛型的类型信息 比如,泛型父类需要获取其子类定义的泛型类型时: public abstract class GenericClazz<V> ...

  10. TSP 旅行商问题(状态压缩dp)

    题意:有n个城市,有p条单向路径,连通n个城市,旅行商从0城市开始旅行,那么旅行完所有城市再次回到城市0至少需要旅行多长的路程. 思路:n较小的情况下可以使用状态压缩dp,设集合S代表还未经过的城市的 ...