《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 -退出函数,当模块卸载时披调用 

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()。
导出的内核函数可以被模块调用,而来导出的函数模块则无陆被调用。

原文地址:https://www.cnblogs.com/zhengwei0712/p/5456305.html