linux内核initcall放置在各个section中函数执行流程

原创 羽林君 2024-01-06 22:04

前言

linux以及嵌入式一些代码,我们看到core_initcall、device_initcall等等需要链接器分配各个section,并且在启动该模块时候执行。下面我们详细追溯一下执行过程。

作者:良知犹存

转载授权以及围观:欢迎关注微信公众号:羽林君

或者添加作者个人微信:become_me


正文

fs_initcall函数介绍:

Linux内核中的fs_initcall函数:用于在引导过程中进行文件系统等初始化。

  1. 初始化注册:

  • 当文件系统模块被加载时,它使用fs_initcall宏注册其初始化函数。
  • 该宏将初始化函数添加到__initcall_fs部分。
  • 内核引导过程:

    • 在引导过程中,在基本硬件初始化和内存设置之后,内核开始执行初始化函数。
  • 执行fs_initcall函数:

    • fs_initcall函数按照其注册顺序依次执行。
    • 这些函数初始化各种文件系统并执行必要的设置任务。
  • 文件系统初始化:

    • 每个fs_initcall函数负责设置和初始化特定的内容。
    • 这可能涉及初始化数据结构、注册文件系统类型、准备缓存和其他相关任务。
  • 完成和交接:

    • 一旦所有fs_initcall函数都执行完毕,内核会继续完成引导过程,包括启动用户空间和初始化设备。

    下面是一个简单的示例代码,展示了fs_initcall函数的使用和文件系统初始化的过程:

    #include 
    #include 

    static int __init my_filesystem_init(void) {
        // 执行文件系统特定的初始化任务
        printk(KERN_INFO "My Filesystem: Initializing\n");
        // 其他初始化操作...

        return 0;
    }

    fs_initcall(my_filesystem_init);

    MODULE_LICENSE("GPL");

    这其中my_filesystem_init函数被注册为fs_initcall函数。当模块加载时,该初始化函数将被执行,完成特定文件系统的初始化任务。实际的文件系统模块会包含更多复杂的初始化逻辑,这个例子只是用来展示fs_initcall函数的基本用法。

    fs_initcall函数调用的层次:

    在Linux内核中,fs_initcall宏实际上是通过__define_initcall来定义的。下面是__define_initcall的定义:

    #define __define_initcall(fn, id) \
        static initcall_t __initcall_##fn##id __used \
        __attribute__((__section__(".initcall" #id ".init"))) = fn

    这段代码展示了__define_initcall的定义方式。在这里,__define_initcall宏创建了一个静态的initcall_t类型变量,并将其放置在特定的.initcall节(section)中。这样,在内核初始化时,这些函数就会按照其在源代码中出现的顺序被依次调用。

    fs_initcall实际上是通过__define_initcall宏来实现的,它们共同构成了Linux内核中初始化调用机制的一部分。

    fs_initcall函数被放置的section的位置

    在Linux内核中,.initcall节(section)是通过链接脚本(linker script)定义的。链接脚本指定了可执行文件的内存布局,包括代码、数据和其他段的放置位置。

    对于.initcall节(section),它通常由链接脚本中的一些规则来定义。这个节用于存放初始化函数的地址,以便在内核启动时按照顺序执行这些初始化函数。

    具体的定义可能会因内核版本和架构而异,但通常可以在内核源代码的arch//kernel/vmlinux.lds.S或类似的文件中找到相关的链接脚本定义。在这些文件中,我们可以看到下面的内容:

    .initcall.init : {
        INIT_CALLS
    }

    文件位置:\linux-xxx\arch\arm\kernel\vmlinux.lds.S

    文件位置:\linux-xxx\include\asm-generic\vmlinux.lds.h

               __initcall_start = .;
               .initcall.init : {
                     *(.initcall1.init)
                    ...
                     *(.initcall7.init)
                     }
               __initcall_end = .;

    上面是INIT_CALLS对应的函数,*(.initcall##level##.init),这个函数就对应了__define_initcall宏里面的__section__(".initcall" #id ".init"),继续查看fs_initcall,对应的level就是这个部分__define_initcall(fn, 5)的5。

    上述示例中的INIT_CALLS通常会包含对.initcall节(section)的定义,规定了将哪些符号放入该节中。这些定义可能会随着不同的内核版本和架构而有所不同,但其基本思想是相似的:将初始化函数的地址放入特定的节(section)中,以便在启动时按顺序执行这些函数。

    __define_initcall这个宏也是可以设置多个初始化函数,并将它们放置在不同的.initcall节(section)中。

    假设我们有两个初始化函数:init_function_1init_function_2,我们可以使用上述宏定义来将它们分别放置在不同的.initcall节(section)中。

    // 定义多个初始化函数
    static void __init init_function_1(void) {
        // 初始化函数1的内容
    }

    static void __init init_function_2(void) {
        // 初始化函数2的内容
    }

    // 使用 __define_initcall 宏定义来设置多个函数
    __define_initcall(init_function_1, 1);
    __define_initcall(init_function_2, 2);

    在这个例子中,init_function_1被放置在.initcall1.init节(section)中,而init_function_2则被放置在.initcall2.init节(section)中。这样,在内核启动时,这些函数就会按照其在源文件中出现的顺序依次被调用。

    通过使用带有不同标识符的宏定义,可以将多个初始化函数放置在不同的.initcall节(section)中,从而实现按顺序执行多个初始化函数的目的。

    以af_inet.c里面的fs_initcall(inet_init);fs_initcall(ipv4_offload_init);介绍放置的情况:怎么在section放置的

    在这个例子中,fs_initcall宏用于将inet_initipv4_offload_init函数放置在.initcall.init节(section)中。这样,在内核启动时,这些函数就会按照其在源文件中出现的顺序依次被调用。

    下面是简化版本的代码:

    // 定义要初始化的函数
    static void __init inet_init(void) {
        // inet_init的初始化内容
    }

    static void __init ipv4_offload_init(void) {
        // ipv4_offload_init的初始化内容
    }

    // 使用 fs_initcall 宏将函数放置在 .initcall.init 节(section)中
    fs_initcall(inet_init);
    fs_initcall(ipv4_offload_init);

    上述代码,inet_initipv4_offload_init函数会被放置在.initcall.init节(section)中,以便在内核启动时按照其在源文件中出现的顺序依次被调用。

    通过这个例子,我们明白了如何使用fs_initcall宏将这两个函数放置在.initcall.init节(section)中.

    每个section空间排布情况是如何的?

    还是上面inet_initipv4_offload_init函数来介绍,由于fs_initcall宏使用了__attribute__((__section__(".initcall.init"))),这将导致这些函数被放置在.initcall.init节(section)中。这样,在链接时,这些函数的地址将按照其在源文件中出现的顺序排布在该特定的节(section)内。

    这些函数位于可执行文件的内存中的某个位置,它们的排布情况如下所示:

    |---------------------|
    |     .text section   |
    |---------------------|
    |    ... other sections ...   |
    |---------------------|
    |  .initcall.init section  |
    |---------------------|
    |       inet_init      |
    |---------------------|
    |  ipv4_offload_init   |
    |---------------------|
    |    ... other functions ...   |
    |---------------------|
    |       .data section  |
    |---------------------|
    |      ... other sections ...    |
    |---------------------|

    在这个示例中,.initcall.init节(section)包含了inet_initipv4_offload_init函数,它们会按照它们在源代码中出现的顺序排布在该节(section)中。这样,在内核启动时,这些函数就会按照它们在.initcall.init节(section)中的排布顺序依次被调用。

    这里section的大小是随机按照大小自动分配还是需要开发者设置好

    在一般情况下,.initcall.init这样的特殊节(section)的大小是由链接器自动分配的,而不是由开发者手动设置的。当链接器处理可执行文件时,它会根据各个节(section)中的内容以及链接脚本中的规则来确定每个节(section)的大小和排布。

    对于.initcall.init节(section),其大小将取决于其中包含的初始化函数的数量和大小。链接器会根据这些函数的地址和大小来动态地分配空间,以便容纳所有的初始化函数。

    因此,开发者通常无需手动设置.initcall.init节(section)的大小。相反,链接器会根据实际情况自动进行分配,确保所有的初始化函数都能被正确地安置在这个特定的节(section)中,并且在内核启动时按照顺序被调用。

    如何自己设置section的大小

    在一般情况下,开发者通常不需要手动设置节(section)的大小。链接器会根据链接脚本中的规则和可执行文件中各个部分的大小自动进行分配。

    如果我们有特殊需求,希望手动设置某个节(section)的大小,可以通过链接脚本来实现。在链接脚本中,我们可以定义节(section)的起始位置、大小以及其他属性。

    以下简单的模板,在链接脚本中手动设置一个名为.my_section的节(section)的大小:

    .my_section : {
    /* 定义节(section)的起始位置 */
    start = .;
    /* 设置节(section)的大小为固定值(例如0x1000)*/
    input_section(.text);
    input_section(.data);
    /* 其他内容... */
    end = .;
    } > RAM

    在这个示例中,.my_section节(section)被手动设置为包含.text.data节(section)的内容,并且其大小被设置为固定值。当链接器处理可执行文件时,它将按照这些规则来分配空间并确定这个特定节(section)的大小。

    需要注意的是,手动设置节(section)的大小可能需要对链接脚本和链接过程有深入的了解,因此在大多数情况下,开发者无需手动设置节(section)的大小,而是依赖于链接器自动进行分配。

    这个是我实际应用的一款芯片的链接修改:

    FUN 0x400 (0x10000-0x400)
    {
        ;cpu.o (+RO)
        xlib.a (+RO)
    }

    上面这部分我使用的链接脚本中一部分内容。为FUN的节(section)中,它的起始地址为0x10000,大小为0x400

    在这个节(section)中包含了两个文件:cpu.oxlib.a,它们都被标记为只读(Read-Only)。链接器会将这两个文件的只读部分放置在由FUN定义的地址范围内。

    链接脚本用于指导链接器如何组织可执行文件的各个部分,包括节(section)的排布和属性。这个片段也是属于链接脚本的一部分,这个里面链接器会将cpu.oxlib.a的只读部分放置在从0x10000开始、大小为0x400的范围内。

    只是一个demo示例,如果进一步操作这个链接脚本,我们要参考特定的链接器文档以及相关的目标平台和工具链的文档,以确保正确地设置节(section)的属性和排布。芯片之间区别挺大的。

    内核执行顺序是?

    介绍完了section片段,我们再来说一下,这些函数的初始化位置以及执行顺序。

    上面我们介绍了vmlinux.lds.S中的INIT_CALLS就是我们定义好的那些函数,那他们怎么被调用的呢 在Linux内核启动过程中,INIT_CALLS(包括subsys_initcallfs_initcalldevice_initcall等)会在不同的阶段被执行。这些初始化调用是通过链接器脚本和特定的内核宏来安排的。

    具体来说,INIT_CALLS的执行时机如下:

    1. 在内核启动的早期阶段,start_kernel函数会调用rest_init
    2. rest_init中,会触发do_basic_setup函数的执行,其中包括对文件系统的基本设置。
    3. do_basic_setup函数中,会调用do_initcalls函数。
    4. do_initcalls函数中,各种初始化函数会按照链接器脚本中的顺序被执行。
    5. fs_initcall函数是其中之一。

    看到了执行过程,其中是按照各个level进行调用的,而__define_initcall(level,fn)的作用就是指示编译器把一些初始化函数的指针(即:函数起始地址)按照顺序放置一个名为 .initcall.init 的section中,这个section又被分成了n个子section,它们按顺序排列。在内核初始化阶段,这些放置到这个section中的函数指针将供do_initcalls() 按顺序依次调用,来完成相应初始化。

    而函数指针放置到的子section由宏定义的level确定,对应level较小的子section位于较前面。而位于同一个子section内的函数指针顺序不定,将由编译器按照编译的顺序随机指定。同理,如果我们想先执行一些定义的函数,那就可以把它们放置于level比较小的定义中。

    结语

    这就是我自己对于linux内核initcall放置在各个section中函数执行流程的一些分享。如果大家有更好的想法,也欢迎大家加我好友交流。


    作者:良知犹存,白天努力工作,晚上原创公号号主。公众号内容除了技术还有些人生感悟,一个认真输出内容的职场老司机,也是一个技术之外丰富生活的人,摄影、音乐 and 篮球。关注我,与我一起同行。

                                  ‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧  END  ‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧

    推荐阅读

    【1】jetson nano开发使用的基础详细分享

    【2】Linux开发coredump文件分析实战分享

    【3】CPU中的程序是怎么运行起来的 必读

    【4】cartographer环境建立以及建图测试

    【5】设计模式之简单工厂模式、工厂模式、抽象工厂模式的对比

    本公众号全部原创干货已整理成一个目录,回复[ 资源 ]即可获得。



    羽林君 某嵌入式程序猿分享技术、生活、人生云云文字。如有诗云:去年今日此门中,人面桃花相映红。人面不知何处去,桃花依旧笑春风。
    评论
    • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
      优思学院 2025-01-06 12:03 113浏览
    •     为控制片内设备并且查询其工作状态,MCU内部总是有一组特殊功能寄存器(SFR,Special Function Register)。    使用Eclipse环境调试MCU程序时,可以利用 Peripheral Registers Viewer来查看SFR。这个小工具是怎样知道某个型号的MCU有怎样的寄存器定义呢?它使用一种描述性的文本文件——SVD文件。这个文件存储在下面红色字体的路径下。    例:南京沁恒  &n
      电子知识打边炉 2025-01-04 20:04 98浏览
    • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
      华普微HOPERF 2025-01-06 15:29 125浏览
    • PLC组态方式主要有三种,每种都有其独特的特点和适用场景。下面来简单说说: 1. 硬件组态   定义:硬件组态指的是选择适合的PLC型号、I/O模块、通信模块等硬件组件,并按照实际需求进行连接和配置。    灵活性:这种方式允许用户根据项目需求自由搭配硬件组件,具有较高的灵活性。    成本:可能需要额外的硬件购买成本,适用于对系统性能和扩展性有较高要求的场合。 2. 软件组态   定义:软件组态主要是通过PLC
      丙丁先生 2025-01-06 09:23 83浏览
    • 村田是目前全球量产硅电容的领先企业,其在2016年收购了法国IPDiA头部硅电容器公司,并于2023年6月宣布投资约100亿日元将硅电容产能提升两倍。以下内容主要来自村田官网信息整理,村田高密度硅电容器采用半导体MOS工艺开发,并使用3D结构来大幅增加电极表面,因此在给定的占位面积内增加了静电容量。村田的硅技术以嵌入非结晶基板的单片结构为基础(单层MIM和多层MIM—MIM是指金属 / 绝缘体/ 金属) 村田硅电容采用先进3D拓扑结构在100um内,使开发的有效静电容量面积相当于80个
      知白 2025-01-07 15:02 66浏览
    • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
      华普微HOPERF 2025-01-06 17:23 141浏览
    • 本文介绍Linux系统更换开机logo方法教程,通用RK3566、RK3568、RK3588、RK3576等开发板,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。制作图片开机logo图片制作注意事项(1)图片必须为bmp格式;(2)图片大小不能大于4MB;(3)BMP位深最大是32,建议设置为8;(4)图片名称为logo.bmp和logo_kernel.bmp;开机
      Industio_触觉智能 2025-01-06 10:43 87浏览
    • 大模型的赋能是指利用大型机器学习模型(如深度学习模型)来增强或改进各种应用和服务。这种技术在许多领域都显示出了巨大的潜力,包括但不限于以下几个方面: 1. 企业服务:大模型可以用于构建智能客服系统、知识库问答系统等,提升企业的服务质量和运营效率。 2. 教育服务:在教育领域,大模型被应用于个性化学习、智能辅导、作业批改等,帮助教师减轻工作负担,提高教学质量。 3. 工业智能化:大模型有助于解决工业领域的复杂性和不确定性问题,尽管在认知能力方面尚未完全具备专家级的复杂决策能力。 4. 消费
      丙丁先生 2025-01-07 09:25 77浏览
    • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球无人机锂电池产值达到2457百万美元,2024-2030年期间年复合增长率CAGR为9.6%。 无人机锂电池是无人机动力系统中存储并释放能量的部分。无人机使用的动力电池,大多数是锂聚合物电池,相较其他电池,锂聚合物电池具有较高的能量密度,较长寿命,同时也具有良好的放电特性和安全性。 全球无人机锂电池核心厂商有宁德新能源科技、欣旺达、鹏辉能源、深圳格瑞普和EaglePicher等,前五大厂商占有全球
      GIRtina 2025-01-07 11:02 63浏览
    • By Toradex 秦海1). 简介嵌入式平台设备基于Yocto Linux 在开发后期量产前期,为了安全以及提高启动速度等考虑,希望将 ARM 处理器平台的 Debug Console 输出关闭,本文就基于 NXP i.MX8MP ARM 处理器平台来演示相关流程。 本文所示例的平台来自于 Toradex Verdin i.MX8MP 嵌入式平台。  2. 准备a). Verdin i.MX8MP ARM核心版配合Dahlia载板并
      hai.qin_651820742 2025-01-07 14:52 40浏览
    • 随着市场需求不断的变化,各行各业对CPU的要求越来越高,特别是近几年流行的 AIOT,为了有更好的用户体验,CPU的算力就要求更高了。今天为大家推荐由米尔基于瑞芯微RK3576处理器推出的MYC-LR3576核心板及开发板。关于RK3576处理器国产CPU,是这些年的骄傲,华为手机全国产化,国人一片呼声,再也不用卡脖子了。RK3576处理器,就是一款由国产是厂商瑞芯微,今年第二季推出的全新通用型的高性能SOC芯片,这款CPU到底有多么的高性能,下面看看它的几个特性:8核心6 TOPS超强算力双千
      米尔电子嵌入式 2025-01-03 17:04 55浏览
    • 根据Global Info Research项目团队最新调研,预计2030年全球封闭式电机产值达到1425百万美元,2024-2030年期间年复合增长率CAGR为3.4%。 封闭式电机是一种电动机,其外壳设计为密闭结构,通常用于要求较高的防护等级的应用场合。封闭式电机可以有效防止外部灰尘、水分和其他污染物进入内部,从而保护电机的内部组件,延长其使用寿命。 环洋市场咨询机构出版的调研分析报告【全球封闭式电机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球封闭式电机总体规
      GIRtina 2025-01-06 11:10 103浏览
    • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
      知白 2025-01-06 12:04 167浏览
    我要评论
    0
    点击右上角,分享到朋友圈 我知道啦
    请使用浏览器分享功能 我知道啦