51 === 2. How to Build External Modules
52
53 To build external modules, you must have a prebuilt kernel available
54 that contains the configuration and header files used in the build.
55 Also, the kernel must have been built with modules enabled. If you are
56 using a distribution kernel, there will be a package for the kernel you
57 are running provided by your distribution.
58
59 An alternative is to use the "make" target "modules_prepare." This will
60 make sure the kernel contains the information required. The target
61 exists solely as a simple way to prepare a kernel source tree for
62 building external modules.
63
64 NOTE: "modules_prepare" will not build Module.symvers even if
65 CONFIG_MODVERSIONS is set; therefore, a full kernel build needs to be
66 executed to make module versioning work.
具体来说,问题是当您构建模块时,内核源代码树可能缺少 Modules.symvers 文件。当您构建模块时,kbuild 系统实际上会警告您这一点。如果 Modules.symvers 缺失,您将看到:
如果您的内核已
CONFIG_MODVERSIONS
启用,那么在构建驱动程序的 modpost 阶段,它将使用 -m 选项运行scripts/mod/modpost 。如果您勇敢地查看scripts/mod/modpost.c源代码,您会看到 -m 选项添加了 vmlinux 中的 _module_layout_ 符号,但是如果您的内核中没有 Modules.symvers,你不会得到这个符号的 CRC 值,你最终会得到这个错误消息。所以有两种方法可以解决这个问题。
1) 运行您正在运行的内核的完整版本以生成 Modules.symvers,然后重建您的模块。[http://www.mjmwired.net/kernel/Documentation/kbuild/modules.txt][1]
2)另一个选项是告诉愚蠢的 modprobe 忽略所有这些废话,并且无论如何都加载您的模块:
我倾向于选择选项2 :)
安装与您的内核对应的
linux-headers
和linux-source
软件包。例如对于内核3.2.0-27-generic-pae
,您需要:linux-headers-3.2.0-27-generic-pae
和linux-source-3.2.0-27-generic-pae
.如果上述软件包的版本与您正在运行的内核版本不匹配,那么您需要
$(uname -r)
使用上面安装的内核软件包中的版本字符串进行替换。对于上面的例子,包版本是
3.2.0-27-generic-pae
. 当您运行uname -r
并且其输出有所不同时,3.2.0-27-generic-pae
您需要替换$(uname -r)
下面的每个以匹配已安装软件包中的版本字符串。cd /usr/src/linux-source-$Version
并解压 .tar.bz2 存档并 cd 进入提取的目录 - 我猜你已经这样做了cp /boot/config-$(uname -r) .config
进入内核源目录cp /usr/src/linux-headers-$(uname -r)/Module.symvers .
进入内核源目录完成后,在内核源目录中,执行以下操作:
make prepare
make scripts
make M=drivers/usb/serial
M=
-根据您的需要更改路径不幸的是,我不知道如何在保持
Module.symvers
不变的情况下构建特定模块。make drivers/usb/serial/option.ko
例如,这样做会杀死Module.symvers
文件,最终会遇到最初的问题。使用M=
参数并不会杀死它,但是您必须在指定路径中构建所有模块 - 我还没有找到解决它的方法。在运行之前,您必须使用完全相同的内核配置
make prepare
。此外,如果您在树外构建它,则需要针对与当前运行的内核(或者如果您在编译时没有运行它,则为目标)匹配的完全相同的内核头文件来构建它。