В исходном коде ядра Linux я нашел эту функцию:
static int __init clk_disable_unused(void)
{
// some code
}
Здесь я не могу понять, что означает __init
.
В исходном коде ядра Linux я нашел эту функцию:
static int __init clk_disable_unused(void)
{
// some code
}
Здесь я не могу понять, что означает __init
.
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
Чтобы использовать эту функцию, вам нужен специальный файл-компоновщик script, который сообщает компоновщик, где можно найти все отмеченные функции.
Это демонстрирует особенность ядра 2.2 и более поздних версий. Обратите внимание на изменение в определениях функций init
и cleanup
. Макрос __init
вызывает отбрасывание функции init
и освобождение ее памяти после завершения функции init
для встроенных драйверов, но не для загружаемых модулей. Если вы думаете о том, когда вызывается функция init
, это имеет смысл.
__ init - это макрос, определенный в. /include/linux/init.h, который расширяется до __attribute__ ((__section__(".init.text")))
.
Он инструктирует компилятор отмечать эту функцию специальным образом. В конце компоновщик собирает все функции с этой меткой в конце (или начале) двоичного файла. Когда ядро запускается, этот код запускается только один раз (инициализация). После запуска ядро может освободить эту память для повторного использования, и вы увидите ядро
Прочитайте комментарий (и документы одновременно) в linux/init.h.
Вы также должны знать, что у gcc есть некоторые расширения, созданные специально для кода ядра Linux, и похоже, что этот макрос использует одно из них.
Когда вы компилируете и вставляете модуль ядра Linux в ядро, первой выполняемой функцией является __init. Эта функция в основном используется для выполнения инициализации перед выполнением основных операций, таких как регистрация драйвера устройства и т.д. Есть другая функция с противоположным эффектом. __exit, который вызывается при удалении модуля ядра, который снова используется для удаления какого-либо зарегистрированного устройства или любой подобной функции