在Linux内核源代码中,我找到了以下功能:
static int __init clk_disable_unused(void)
{
// some code
}
在这里我不明白这__init
意味着什么。
在Linux内核源代码中,我找到了以下功能:
static int __init clk_disable_unused(void)
{
// some code
}
在这里我不明白这__init
意味着什么。
Answers:
include/linux/init.h
/* These macros are used to mark some functions or
* initialized data (doesn't apply to uninitialized data)
* as `initialization' functions. The kernel can take this
* as hint that the function is used only during the initialization
* phase and free up used memory resources after
*
* Usage:
* For functions:
*
* You should add __init immediately before the function name, like:
*
* static void __init initme(int x, int y)
* {
* extern int z; z = x * y;
* }
*
* If the function has a prototype somewhere, you can also add
* __init between closing brace of the prototype and semicolon:
*
* extern int initialize_foobar_device(int, int, int) __init;
*
* For initialized data:
* You should insert __initdata between the variable name and equal
* sign followed by value, e.g.:
*
* static int init_variable __initdata = 0;
* static const char linux_logo[] __initconst = { 0x32, 0x36, ... };
*
* Don't forget to initialize data not at file scope, i.e. within a function,
* as gcc otherwise puts the data into the bss section and not into the init
* section.
*
* Also note, that this data cannot be "const".
*/
/* These are for everybody (although not all archs will actually
discard it in modules) */
#define __init __section(.init.text) __cold notrace
#define __initdata __section(.init.data)
#define __initconst __section(.init.rodata)
#define __exitdata __section(.exit.data)
#define __exit_call __used __section(.exitcall.exit)
这些只是将linux代码的某些部分放置在最终执行的二进制文件中的特殊区域中的宏。
__init
,例如(或更好地__attribute__ ((__section__
(".init.text")))
将此宏扩展为),指示编译器以特殊方式标记此函数。最后,链接器会在二进制文件的末尾(或开头)收集带有此标记的所有功能。
当内核启动时,此代码仅运行一次(初始化)。运行之后,内核可以释放该内存以重新使用它,您将看到内核消息:
释放未使用的内核内存:已释放108k
要使用此功能,您需要一个特殊的链接程序脚本文件,该文件告诉链接程序在哪里可以找到所有标记的功能。
阅读linux / init.h中的注释(和文档同时)。
您还应该知道gcc有一些专门针对linux内核代码进行的扩展,并且看起来该宏使用了其中之一。