The Linux Kernel Module Programming Guide简译(一)
sighofwraith 发表于 2005-6-12 17:35:19
How do modules get into the kernel?
首先,如果你想知道哪些模块已经载入,可以用一下lsmod,它会从
/proc/modules中把你想看的东西读出来。
模块是怎么的载入的呢~~
当内核发现有一个功能,它需要,但是本来却没有,就会有一个叫kmod(老版本上它叫kerneld)的内核精灵程序运行modprobe去把相应的模块装进来。Modprobe需要个参数来知道到底要作虾米,这个参数可以有两种形式:
·模块名,比方softdog或者ppp之类的东东。
·一个很一般化的描述符,比方char-major-10-30。
如果可怜的modprobe只搞到了一个相貌平平,扔到街上就找不到的虾米描述符,它首先会去/etc/modules.conf找找这东西有没有马甲,比方如果找到下面这么一行:
alias char-major-10-30 softdog
它就知道了,原来char-major-10-30对应的模块应该是softdog.o。
下一步,modprobe会去/lib/modules/version/modules.dep中找找加载这个模块之前是不是还有别的什么要先装进去。比方说要装msdos.o之前就要先把fat.o装进去。(modules.dep这个文件是depmod –a建立的,提供依赖关系用)
最后,modprobe用insmod把这些需要装的modules挨着装进去。
Modprobe会告诉insmod模块们的标准目录——/lib/modules/version/(如果你怕覆盖掉原来的内容,可以在内核编译的时候在Makefile里改一下EXTRAVERSION)。Insmod是个需要制定模块所在位置的家伙,而modprobe则有默认的位置可用。所以,比方说你想装载msdos模块,你就要这么写:
insmod /lib/modules/2.5.1/kernel/fs/fat/fat.o
insmod /lib/modules/2.5.1/kernel/fs/msdos/msdos.o
或者光执行一个“modprobe –a msdos”。
Linux发行版都提供了modprobe、insmod、depmod,它们打成了一个叫modutils或者mod-utils的包。
结束这段内容之前,我们先从/etc/modules.conf弄个片断来看看:
# This file is automatically generated by update-modules
path[misc]=/lib/modules/2.4.?/local
keep
path[net]=~p/mymodules
options mydriver irq=10
alias eth0 eepro
# 开头的行是注释。
path[misc]这行告诉modprobe找misc模块的时候把默认的目录换成/lib/modules/2.4.?/local。
path[net]这行告诉modprobe找net模块的时候在~p/modules目录找,但是前面的“keep”则告诉modprobe是把这个新的目录加进原来的搜索目录,而不是向对misc那样替换掉。
alias这行是说kmod见到eth0这个描述符的时候装载eepro.o这个模块(声明马甲!)。
呵呵,现在大家知道模块是怎么加载的了。不过如果你想写一个依赖于其他的模块的模块还是有点问题的,不过这个比较“高级”的问题容后再说啦~~:)
1.2.1 Before we begin
研究代码之前,先介绍一点小tip。毕竟大家的系统多少都有点差别,而且每个人的习惯也不一样,所以即便是简简单单的一个“hello world”有时候可能也会变成麻烦。所以先看看下面这些在动手还是比较有用的。
1.2.1.1 Modversioning
除非你使能了内核中的CONFIG_MODVERSION,否则针对一个内核编译的模块是不能加载进其他版本系统的。我们现在不去深入到modules versioning。在我们加上modversion内容之前,如果你运行在一个打开了modversioning的系统上,这里的例子可能不能正常工作。不过,大部分的linux发行版都是打开了这个开关的。如果你因为版本错误不能加载模块,那你可以试试编译一个关上modversioning的。
1.2.1.2 Using x
这里强烈推荐你自己输入、编译、加载所有的例子。也强烈建议你在console下做这些事情,而不是在X下面。
在console上,模块会把log信息和警告之类的东西直接显示出来。而在X下的xterm里insmod加载一个模块,这些信息会被记录下来,但不在xterm上显示。这样显得比较麻烦一点。所以推荐使用console。
1.2.1.3 Compiling issues and kernel version
linux发行版往往发布一个使用各种不标准方法补丁过的内核给你,这可能会带来很多麻烦。
一个很常见的问题就是,一些linux发行版的内核和它给你的头文件不完全一样。所以最好自己下载内核code安装上(其实。。。这主意也很麻烦的说。。。。如果只是gcc默认的头文件版本不对,那就自己找对应版本的内核code用-I开关选正确的include来编译吧。)