xxx_initcall 的调用

Posted gyforever1004

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了xxx_initcall 的调用相关的知识,希望对你有一定的参考价值。

内核版本:linux-4.19

上一篇文章提到了这段代码:

arch_initcall_sync(of_platform_default_populate_init);

它的功能是完成 device_node 到 platform_device 的转换。这篇文章就来大概的分析一下,它是怎样被调用的。

arch_initcall_sync 定义如下:

#define ___define_initcall(fn, id, __sec)     static initcall_t __initcall_##fn##id __used         __attribute__((__section__(#__sec ".init"))) = fn;

#define __define_initcall(fn, id) ___define_initcall(fn, id, .initcall##id)

#define arch_initcall_sync(fn)      __define_initcall(fn, 3s)

根据 LDS 文件的定义,会将这些 data 存储在指定的位置:

#define INIT_CALLS_LEVEL(level)                             __initcall##level##_start = .;                      KEEP(*(.initcall##level##.init))                    KEEP(*(.initcall##level##s.init))           
#define INIT_CALLS                                  __initcall_start = .;                           KEEP(*(.initcallearly.init))                        INIT_CALLS_LEVEL(0)                         INIT_CALLS_LEVEL(1)                         INIT_CALLS_LEVEL(2)                         INIT_CALLS_LEVEL(3)                         INIT_CALLS_LEVEL(4)                         INIT_CALLS_LEVEL(5)                         INIT_CALLS_LEVEL(rootfs)                        INIT_CALLS_LEVEL(6)                         INIT_CALLS_LEVEL(7)                         __initcall_end = .; 

在内核中,想要调用到这些数据,就会用到

__initcall##level##_start = .;

这个标识。

数据的位置已经搞定了,那么内核又是怎样调用的呢?接下来找到这些函数调用。

调用流程如下:

start_kernel
    -->rest_init
        -->kernel_thread(kernel_init, NULL, CLONE_FS);
            -->kernel_init
                -->kernel_init_freeable
                    -->do_basic_setup
                        -->do_initcalls
                            -->do_initcall_level

do_initcall_level 代码如下:

static void __init do_initcall_level(int level)
{
    initcall_entry_t *fn;

    strcpy(initcall_command_line, saved_command_line);
    parse_args(initcall_level_names[level],
           initcall_command_line, __start___param,
           __stop___param - __start___param,
           level, level,
           NULL, &repair_env_string);

    trace_initcall_level(initcall_level_names[level]);
    for (fn = initcall_levels[level]; fn < initcall_levels[level+1]; fn++)
        do_one_initcall(initcall_from_entry(fn));
}

根据下表,分别调用 do_one_initcall 来执行每一个 initcall。

static initcall_entry_t *initcall_levels[] __initdata = {
    __initcall0_start,
    __initcall1_start,
    __initcall2_start,
    __initcall3_start,
    __initcall4_start,
    __initcall5_start,
    __initcall6_start,
    __initcall7_start,
    __initcall_end,
};

表中的 xxx_start 恰恰就是前面所提到的存储标识,那么这些调用就联系到了一起,实现函数的调用。

分析到这里,相信关于 xxx_initcall 调用的云雾就已经拨开了吧!

以上是关于xxx_initcall 的调用的主要内容,如果未能解决你的问题,请参考以下文章

如何测量代码片段的调用次数和经过时间

如何从片段中调用 getSupportFragmentManager()?

如何从片段 KOTLIN 中调用意图 [重复]

调用模板化成员函数:帮助我理解另一个 *** 帖子中的代码片段

从片段调用 Google Play 游戏服务

Android片段生命周期:onResume调用了两次