这里展示了内核2.2以后引入的一个新特性。注意在负责“初始化”和“清理收尾”的函数定义处的变化。宏 __init的使用会在初始化完成后丢弃该函数并收回所占内存,如果该模块被编译进内核,而不是动态加载。
宏 __initdata 同 __init 类似,只不过对变量有效。
宏 __exit 将忽略“清理收尾”的函数如果该模块被编译进内核。同宏__init一样,对动态加载模块是无效的。这很容易理解。编译进内核的模块是没有清理收尾工作的, 而动态加载的却需要自己完成这些工作。
这些宏在头文件 linux/init.h 定义,用来释放内核占用的内存。当你在启动时看到这样的 Freeing unused kernel memory: 236k freed内核输出,上面的那些正是内核所释放的。
Example 2-5. hello-3.c
/* hello-3.c - Illustrating the __init, __initdata and __exit macros. */ #include <linux/module.h> /* Needed by all modules */ #include <linux/kernel.h> /* Needed for KERN_ALERT */ #include <linux/init.h> /* Needed for the macros */ static int hello3_data __initdata = 3; static int __init hello_3_init(void) { printk(KERN_ALERT "Hello, world %d\n", hello3_data); return 0; } static void __exit hello_3_exit(void) { printk(KERN_ALERT "Goodbye, world 3\n"); } module_init(hello_3_init); module_exit(hello_3_exit); |
附带说一句,你会发现宏__initfunction()在为2.2版本内核编写的模块中出现。
__initfunction(int init_module(void)) { printk(KERN_ALERT "Hi there.\n"); return 0; } |
该宏起到和宏 __init类似的作用,但因为推荐使用宏__init目前已不推荐使用。 我提到它只因为你可能在较新的内核中遇到。像在内核2.4.18中, 总共有38处 __initfunction()的引用,在2.4.20中,有37处。但是,不要在你的代码中使用。