__init在Linux内核代码中是什么意思?


91

在Linux内核源代码中,我找到了以下功能:

static int __init clk_disable_unused(void) 
{
   // some code
}

在这里我不明白这__init意味着什么。

Answers:


76

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)

55

这些只是将linux代码的某些部分放置在最终执行的二进制文件中的特殊区域中的宏。 __init,例如(或更好地__attribute__ ((__section__ (".init.text")))将此宏扩展为),指示编译器以特殊方式标记此函数。最后,链接器会在二进制文件的末尾(或开头)收集带有此标记的所有功能。

当内核启动时,此代码仅运行一次(初始化)。运行之后,内核可以释放该内存以重新使用它,您将看到内核消息:

释放未使用的内核内存:已释放108k

要使用此功能,您需要一个特殊的链接程序脚本文件,该文件告诉链接程序在哪里可以找到所有标记的功能。


11
聪明!这就是“释放未使用的内核内存:释放108k”的含义。:-)这些年来,我有些纳闷。我认为这是某种缓冲区或某种东西,而不是代码。
法尔肯教授2012年

6

这演示了内核2.2及更高版本的功能。注意initcleanup函数定义的变化。该__init宏导致的init被丢弃的功能和它的内存一旦释放init功能完成对内置的驱动程序,但没有可加载模块。如果考虑何时init调用该函数,这是很合理的。

资源


5

__init是./include/linux/init.h中定义的宏,它扩展为__attribute__ ((__section__(".init.text")))

它指示编译器以特殊方式标记此函数。最后,链接器在二进制文件的末尾(或开头)收集带有此标记的所有功能。当内核启动时,此代码仅运行一次(初始化)。运行之后,内核可以释放该内存以重新使用它,您将看到内核


3

阅读linux / init.h中的注释(和文档同时)。

您还应该知道gcc有一些专门针对linux内核代码进行的扩展,并且看起来该宏使用了其中之一。


1

当您将Linux内核模块编译并插入内核时,要执行的第一个函数是__init。此函数基本上用于执行初始化,然后再执行主要操作(如注册设备驱动程序等),还有另一个作用相反__exit,当您删除内核模块时将调用它,该模块再次用于删除某些已注册的设备或任何类似的功能

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.