Что означает __init в коде ядра Linux? - PullRequest
78 голосов
/ 12 января 2012

В исходном коде ядра Linux я нашел эту функцию:

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

Здесь я не могу понять, что означает __init.

Ответы [ 6 ]

63 голосов
/ 12 января 2012

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)
44 голосов
/ 12 января 2012

Это только макросы для определения местоположения некоторых частей кода Linux в специальных областях в конечном исполняемом двоичном файле.Например, __init (или лучше __attribute__ ((__section__ (".init.text"))), в который расширяется этот макрос) указывает компилятору специальным образом пометить эту функцию.В конце компоновщик собирает все функции с этой отметкой в ​​конце (или начале) двоичного файла.

Когда ядро ​​запускается, этот код выполняется только один раз (инициализация).После запуска ядро ​​может освободить эту память для повторного использования, и вы увидите сообщение ядра:

Освобождение неиспользуемой памяти ядра: освобождено 108 КБ

Чтобы использовать эту функциювам нужен специальный файл сценария компоновщика, который сообщает компоновщику, где найти все отмеченные функции.

4 голосов
/ 06 декабря 2012

Это демонстрирует особенность ядра 2.2 и выше.Обратите внимание на изменения в определениях функций init и cleanup.Макрос __init вызывает сброс функции init и освобождение ее памяти после завершения работы функции init для встроенных драйверов, но не для загружаемых модулей.Если вы думаете о том, когда вызывается функция init, это имеет смысл.

source

3 голосов
/ 12 января 2012

__ init - это макрос, определенный в ./include/linux/init.h, который расширяется до __attribute__ ((__section__(".init.text"))).

. Он указывает компилятору специальным образом пометить эту функцию.В конце компоновщик собирает все функции с этой отметкой в ​​конце (или начале) двоичного файла.Когда ядро ​​запускается, этот код запускается только один раз (инициализация).После запуска ядро ​​может освободить эту память для повторного использования, и вы увидите ядро ​​

2 голосов
/ 12 января 2012

Прочитать комментарий (и документы одновременно) в linux / init.h .

Вы также должны знать, что у gcc есть некоторые расширения, созданные специально для кода ядра Linux, и похоже, что этот макрос использует одно из них.

0 голосов
/ 09 апреля 2019

Когда вы компилируете и вставляете модуль ядра Linux в ядро, первой выполняемой функцией является __init. Эта функция в основном используется для выполнения инициализации перед выполнением основных операций, таких как регистрация драйвера устройства и т. Д. Есть еще одна функция с противоположный эффект __exit, который вызывается при удалении модуля ядра, который снова используется для удаления какого-либо зарегистрированного устройства или любой подобной функции

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...