《Linux内核设计与分析》第十七章读书笔记

 设备与模块

关于设备驱动和设备管理,四种内核成分。网络

  • 设备类型:在全部Unix 系统中为了统一普通设备的操做所采用的分类.
  • 模块: Linux 内核中用于按需加载和卸载目标码的机制.
  • 内核对象:内核数据结构中支持面向对象的简单操做,还支持维护对象之间的父子关系。
  • sysfs :表示系统中设备树的一个文件系统。

17 .1 设备类型

在Linux 以及全部Unix 系统中,设备被分为如下三种类型数据结构

  • 块设备
  • 字符设备
  • 网络设备
块设备一般缩写为blkdev,它是可寻址的,寻址以块为单位,块大小随设备不一样而不一样;

块设备一般支持重定位( seeking )操做,也就是对数据的随机访问。

块设备是经过称为“块设备节点”的特殊文件来访问的,井且一般被挂载为文件系统。

  

字符设备一般缩写为cdev,它是不可寻址的,仅提供数据的流式访问,就是一个个字符,或者一个个字节。

字符设备是经过称为“字符设备节点”的特殊文件来访问的。

与块设备不一样,应用程序经过直接访问设备节点与字符设备交互。

  

网络设备最多见的类型:以太网设备(ethemet devices)
它提供了对网络的访问,

经过-个物理适配器和一种特定的协议(如IP 协议)进行的。

网络设备打破了Unix 的“全部东西都是文件”的设计原则,
它不是经过设备节点来访问,
而是经过套接字API 这样的特殊接口来访问。

  

有些设备驱动是虚拟的,仅提供访问内核功能而已.咱们称为“伪设备”(pseudo device )

  

17.2 模块模块化

尽管Linux 是“单块内核”( monolithic )的操做系统
整个系统内核都运行于一个单独的保护域中,
可是Linux 内核是模块化组成的,
它容许内核在运行时动态地向其中插入或从中删除代码。
这些代码(包括相关的子例程、数据、函数人口和函数出口〉被一并组合在一个单独的二进制镜像中,
即所谓的可装载内核模块中,或简称为模块。

支持模块的好处是基本内核镜像能够尽量地小,
由于可选的功能和驱动程序能够利用模块形式再提供。

模块容许咱们方便地删除和从新载入内核代码,也方便了调试工做。

并且当热插拔新设备时,可经过命令载入新的驱动程序。

  

 17.2.1 Hello, World函数

hel lo_init -初始化函散, 当模块装载时被调用,若是成功装载返回零,不然返回非零值工具

hello_exit -退出函数,当模块卸载时披调用 spa

module init(hello init);
module exit(hello exit);

 

hello_init的函数是模块的入口点,
它经过module_ init()例程注册到系统中,在内核装载时被调用。

调用module_initO 实际上不是真正的函数调用,而是一个宏调用,它惟一的参数即是模块的初始化函数。

模块的全部初始化函数必须符合下面的形式:
int my _ init (void) ;

由于init 函数一般不会被外部函数直接调用,因此你没必要导出该函数,故它可标记为static
类型.

  

 

hello_ exit()函数是模块的出口函数,
它由module_exit()例程应册到系统.在模块从内存卸载时,内核便会调用hello_exit。

退出函数可能会在返回前负责清理资源,以保证硬件处于一致状态:或者作其余的一些操做。
 
exit 函数负责对init 函数以及在模块生命周期过程当中所作的一切事情进行撤销工做,
在退出函数返回后, 模块就被卸载了。
退出函数必须符合如下形式:
void my_exit (void);
与init 函数同样,你也能够标记其为static.

  

 

若是上述文件被静态地编译到内核映像中,那么退出函数将不被包含,并且永远都不会被调

MODULE_LICENSE()宏用于指定模块的版权。
 
MODULE_AUTHOR()宏和MODULE_DESCRIPTION()宏指定了代码做者
和模块的简要描述,它们彻底是用做信息记录目的。

  


17.2.2 构建模块

1. 放在内核派代码树中

当你决定了把你的模块放入内核源代码树中,下一步要清楚你的模块应在内核源代码树中
处于何处。操作系统

设备驱动程序存放在内核源码树根目录下/drivers 的子目录下,在其内部,设备驱动
文件被进一步按照类别、类型或特殊驱动程序等更有序地组织起来。设计

2. 放在内核代码外

17.2.3 安装模块调试

编译后的模块将被装入到目录/li~/modules/version/kemel/ 下,对象

在kernel/ 目录下的每个目录都对应着内核源码树中的模块位置。

若是使用的是2 .6.34 内核,并且将你的模块源代码直接
放在drivers/char/下,那么编译后的钓鱼竿驱动程序的存放路径将是:

/lib/modules/2.6.34/kemel/drivers/chai/fisbing.ko.

下面的构建命令用来安装编译的模块到合适的目录下
make modules install
一般须要以root 权限运行。


17.2.4 产生模块依赖性
Linux 模块之间存在依赖性

  • 若想产生内核依赖关系的信息, root 用户可运行

        depmod

  • 为了执行更快的更新操做,那么能够只为新模块生成依赖信息,而不是生成全部的依赖关

       系,这时root 用户可运行命令
       depmod -A

  模块依赖关系信息存放在/lib/modules/version/modules.dep 文件中.


17.2.5 戴入模块

载入模块最简单的方法是经过insmod 命令,请求内核载入指定的模块。

insmod 程序不执行任何依赖性分析或进一步的错误检查。

以root 身份运行命令:
insmod module .ko
这里, module.ko 是要载入的模块名称。

执行命令
insmod fishing.ko

卸载一个模块,你可以使用rmmod 命令,它一样须要以root 身份运行
rmmod module

先进工具modprobe 提供了模块依赖性分析、错误智能检查、错误报告以及许多其余功能和选项。
为了在内核via modprobe 中插入模块,须要以root 身份运行
modprobe module [ module parameters ]
其中,参数module 指定了须要载入的模块各称,后面的参数将在模块加载时传入内核。

  

modprobe 命令不但会加载指定的模块,并且会自动加载任何它所依赖的有关模块.因此说
它是加载模块的最佳机制。
modprobe 命令也可用来从内核中卸载模块,固然这也须要以root 身份运行:
modprobe -r modules
参数modules 指定一个或多个须要卸载的模块.与rmmod 命令不一样, modprobe 也会装载给
定模块所依赖的相关模块,但其前提是这些相关模块没有被使用. 

  



17.2.6 管理配置选项


17.2. 7 模块参数

 

17.2.8 导出符号表

模块被载入后,就会被动态地链接到内核。注意,它与用户空间中的动态连接库相似,只有当被显式导出后的外部函数,才能够被动态库调用。在内核中,导出内核函数须要使用特殊的指令: EXPORT_ SYMBOL()和EXPORT_SYMBOL_GPL()。导出的内核函数能够被模块调用,而来导出的函数模块则无陆被调用。

相关文章
相关标签/搜索