1. 程式人生 > >Linux核心2.4和2.6編譯模組的方法

Linux核心2.4和2.6編譯模組的方法

2.6下的模組,副檔名為.ko,而不是2.4下的.o。很多初學者寫完模組之後,會使用2.4的方法來編譯模組。

/*filename: test.c*/
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>

static int dummy_init(void)
{
    printk("hello,world.\n");
    return 0;
}
static void dummy_exit(void)
{
    return;
}

module_init(dummy_init);
module_exit(dummy_exit);

MODULE_LICENSE("GPL")


執行如下命令:

$ gcc -c -O2 -DMODULE -D__KERNEL__ -I/usr/src/linux test.c


$ insmod test.o

No module found in object
insmod: error inserting 'test.o': -1 Invalid module format


正確的做法是寫一個Makefile,由核心的Kbuild來幫你編譯。
#filename: Makefile
obj-:= test.o
KDIR := /lib/modules/$(shell uname -r)/build
PWD := $(shell pwd)
default:
    $(MAKE) -$(KDIR) SUBDIRS=$(PWD) modules

執行如下命令:

$make


make -/lib/modules/2.6.5-1.358/build SUBDIRS=/test modules
make[1]: Entering directory `/lib/modules/2.6.5-1.358/build'
  CC [M] /test/modinject/test.o
  Building modules, stage 2.
  MODPOST
  CC /test/modinject/test.mod.o
  LD [M] /test/modinject/test.ko
make[1]: Leaving directory `/lib/modules/2.6.5-1.358/build'

$ls -l

-rw-r--r-- 1 root root 268 Jan 7 08:31 test.c
-rw-r--r-- 1 root root 2483 Jan 8 09:19 test.ko
-rw-r--r-- 1 root root 691 Jan 8 09:19 test.mod.c
-rw-r--r-- 1 root root 1964 Jan 8 09:19 test.mod.o
-rw-r--r-- 1 root root 1064 Jan 8 09:19 test.o


其實上邊的test.o就是用gcc生成的test.o,而test.ko是使用下列命令來生成的。

$ld -m elf_i386 -r -o test.ko test.o test.mod.o

再來看看test.mod.c,它是由/usr/src/linux/scripts/modpost.c來生成的。

$ cat test.mod.c
#include <linux/module.h>
#include <linux/vermagic.h>
#include <linux/compiler.h>

MODULE_INFO(vermagic, VERMAGIC_STRING);
#undef unix

struct module __this_module
__attribute__((section(".gnu.linkonce.this_module"))) = {
.name = __stringify(KBUILD_MODNAME),
.init = init_module,
#ifdef CONFIG_MODULE_UNLOAD

.exit = cleanup_module,
#endif

};
static const struct modversion_info ____versions[]
__attribute_used__
__attribute__((section("__versions"))) = {
        { 0, "cleanup_module" },
        { 0, "init_module" },
        { 0, "struct_module" },
        { 0, "printk" },
};
static const char __module_depends[]
__attribute_used__
__attribute__((section(".modinfo"))) =
"depends=";


可見,test.mod.o只是產生了幾個ELF的節,分別是modinfo, .gun.linkonce.this_module(用於重定位,引進了rel.gnu.linkonce.this_module), __versions。而test.ko是test.o和test.mod.o合併的結果。