linuxlinux-kernel

How to build a Linux kernel module so that it is compatible with all kernel releases?


I want to build a kernel module that will be compatible with all the kernel’s releases. For example, if I build a kernel module on kernel 3.2.0-29 and try to load it on 3.2.0-86, I get the following error:

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 ]

How can I build a kernel module on 3.2.0-29 that will be good for all 3.2.0 releases.


Solution

  • In short: you can hardly write a useful kernel module which can be loaded to kernels of relatively wide version ranges.

    When you build the module against a kernel compiled with CONFIG_MODVERSIONS (like in your case), for every symbol exported from the kernel, the CRC (Cyclic Redundancy Check) for this symbol is stored in module's file. A CRC is a sort of control sum which takes into account, among other things, the layout of function parameter types. For example, if the layout of a hypothetical struct A differs in two kernels, the CRC for function f(struct A *a) in these kernels differs too.

    When a module is loaded into a running kernel, the CRC for all functions in the module are compared with those of the kernel. If they differ, the kernel refuses to load the module. To read more about this mechanism see the kernel's documentation (Documentation/kbuild/modules.txt).

    So, to make a module loadable into two different kernels, you are restricted to functions whose parameters have the same layout in both kernels. In particular, if the layout of type struct module differs, no single module can be loaded for both kernels.


    There are several approaches to deliver a driver compatible with multiple kernels. The simplest way is to deliver the driver's sources and add them to dkms. In this way, if the running kernel doesn't have the driver built, the driver will be compiled automatically using its sources.