我想构建一个与所有内核版本兼容的内核模块.
例如,如果我在内核3.2.0-29上构建内核模块并尝试在3.2.0-86上加载它,我会收到以下错误:
modprobe my_driver
FATAL: Error inserting my_driver (/lib/modules/3.2.0-86-generic/kernel/fs/my_drv/my_drv.ko): Invalid module format
[ In the log messeges: my_drv: disagrees about version of symbol module_layout ]
如何在3.2.0-29上构建一个适用于所有3.2.0版本的内核模块.
最佳答案
简而言之:您几乎无法编写有用的内核模块,可以将其加载到相对广泛版本的内核中.
当您针对使用CONFIG_MODVERSIONS编译的内核构建模块时(就像您的情况一样),对于从内核导出的每个符号,此符号的CRC都存储在模块的文件中. CRC是某种控制和,除其他外,它考虑了用于函数参数的类型布局.例如,如果假设结构A的布局在两个内核中不同,则这些内核中函数f(struct A * a)的CRC也不同.
将模块加载到运行内核时,将模块中所有函数的CRC与内核的CRC进行比较.如果它们不同,则内核拒绝加载模块.要阅读有关此机制的更多信息,请参阅内核的文档(Documentation/kbuild/modules.txt).
因此,为了使一个模块可以加载到两个不同的内核中,仅限于两个内核中参数具有相同布局的函数.特别是,如果struct模块的布局不同,则不能为两个内核加载单个模块.
有几种方法可以提供适合多个内核的驱动程序.最简单的方法是提供驱动程序的源并将它们添加到dkms.这样一来,如果运行内核没有内置驱动程序,驱动程序将使用其源自动编译.