0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

在STM32上模拟Linux自动初始化

STM32嵌入式开发 来源:STM32嵌入式开发 2023-05-09 09:25 次阅读

Linux中有很多编程思想可以学习,很多大佬把这些思想、机制运用到单片机的编程上。

下文,在STM32模拟Linux kernel自动初始化流程。

通常我们写程序都是按照这个套路,一个函数一个函数按照顺序逻辑一个一个的执行下去。

wKgaomRZoXqAX6rbAAEIxGySr_E226.jpg

如果逻辑非常复杂,涉及的模块比较多,那么这种顺序执行的代码就会比较臃肿,各模块耦合非常紧密。Linux kernel 中,有各种外设驱动,想按照一个顺序逻辑执行下去,几乎是不可能的。

而kenrel 代码能有这么大的代码量,大而不乱,把各层次,各模块有效的分离,而大量的代码又有逻辑的组织在一起,和这个initcall 有至关重要的作用。

通过模仿这种方式,最后把图片中main函数代码清空,分离这种逻辑,又实现同样的功能。

如何能实现这样的功能了,需要一些背景知识:

1,程序代码的组织

2,链接脚本相关的知识。

3,函数指针的应用。

wKgZomRZoXqAGB9mAAB5N68kDh0447.jpg

代码的组织,如图片需要知道变量a,b及函数指针 f,f2是存放在程序的哪些段中,可以去看一下这篇stm32 启动代码 实现|C语言,上述的a,f都是存放在bss 段中,b,f2是存放在data段中,因为已经给定了初始值,而实现这个intcall会把需要自动初始化的数据放到一个自定义的段中去,如.initcall。

如何放到特定的段中了,就需要用到了attribute((section)) 关键字来改变的数据存放段了。

目前的程序编译出来用到了这些个段,除了.isr_vector也是添加的,其他都是编译器默认的。

wKgaomRZoXqAYgguAALDZ7tSpOo994.png

先加段代码:

wKgZomRZoXqAYaVoAABqFGapjxQ792.png

当然这还不够,还需要告诉连接器(LD) 要把 .initcall 段也链接到程序中,所以也需要这段修改。

wKgaomRZoXqAKTtAAAEV4ZDxaz4342.jpg

这段按8字节对齐,定义两个全局变量,及按0-5顺序的链接这些数据,这样的两处修改,再来看一下程序各段的情况。

如图片:

wKgaomRZoXqAAOXZAAIqQD5KTys550.png

已经多出红色框框为.initcalls段,这段总共是8个字节,从0x80005a8除开始。

在来看一下具体的这一段的情况,用readelf 工具。

wKgaomRZoXqAO4SyAACwT_haxtc325.png

和上面的size工具是匹配的,而绿色框框的地址就是SystemInit(0x08000231,小端模式。)

wKgZomRZoXqABp0rAAEJjXL8JG8510.png

所以通过attribute及修改链接脚本,就把函数指针变量放到了.initcall 段中。

那么如何来调用这个函数了,和之前的初始化data段数据类似,遍历这个段,然后取出这个函数地址,然后强制把段中的地址,转成函数指针,再直接调用即可。

wKgaomRZoXqAJnrlAACJK4aTfXI240.png

wKgZomRZoXqAMgNBAAFWSNbCAg4585.png

实现的这张图片,就是从.initcall段中取出函数地址,然后直接调用,非常容易把函数的地址及这个函数指针变量的地址搞混。

代码这么修改,需要自动初始化函数的确是可以调到了,但是每次都写这么长长的一段static initcall_t __ attribute__(( __ used__,__ section__(".initcall.0.init"))),就是不舒服. linux kernel中通过宏来修改。

这个也一样。

wKgZomRZoXqAAwZjAAHF0FeQdh0722.png

添加 按照程序逻辑顺序执行的一些宏

0,low_level_init 比如放始化系统基本时钟

1,arch_init 比如放CPU架构d如初始化NVIC的一些初始化。

2,dev_init 外设模块初始化,比 i2c, flash, spi等。

3,board_init 做具体硬件板及的一些设置。

4,os_init 操作系统的一些设置如,文件系统,网络协议栈等。

5,app_init 最后跑用户程序。

把自己的程序也做一下修改,用宏代替。这样子掉调用do_initcalls 就会按照0,1-到5的顺序执行了。

wKgaomRZoXqAR4AIAADA5rrXPBo591.png

wKgaomRZoXqANQbXAADaxBlZWoc941.png

最后在来看一下initcall 段:

wKgaomRZoXqAee7BAAF4g331D14969.png

wKgZomRZoXqAF2OkAAAPYn2Y9cE421.png

wKgZomRZoXqAIQ7QAAJ4F9IxZcE643.jpg

wKgZomRZoXqAa-9rAAEBcgMElwo640.jpg

这样只要在需要自动初始化函数加上类似于dev_init(),app_init() 就可以了,就会自动调用到,而不需要main 函数中一个一个的顺序执行。

比如i2c控制的初始化放到dev_init 中,下面挂了很多i2c的从设备,只要分别给个从设备用app_init 初始化就行,即使来了一个新的,也用这app_init初始化就行,也不需要更改原来的,高度的分离模块间的耦合度。

这样模拟Linux kenerl 初始化验证成功,最后上库。

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 单片机
    +关注

    关注

    6030

    文章

    44482

    浏览量

    631705
  • Linux
    +关注

    关注

    87

    文章

    11194

    浏览量

    208650
  • STM32
    +关注

    关注

    2264

    文章

    10852

    浏览量

    354148
  • 函数
    +关注

    关注

    3

    文章

    4273

    浏览量

    62299
  • 函数指针
    +关注

    关注

    2

    文章

    56

    浏览量

    3768

原文标题:在STM32上模拟Linux自动初始化

文章出处:【微信号:c-stm32,微信公众号:STM32嵌入式开发】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    RT-Thread自动初始化详解

    我们知道,写裸机程序时,当我们完成硬件初始化后,就需要在主函数中进行调用。当我们使用RT-Thread后,完全不需要这样做了,我们可以将硬件等自动初始化。 RT-Thread 
    的头像 发表于 06-25 21:38 1.1w次阅读
    RT-Thread<b class='flag-5'>自动</b><b class='flag-5'>初始化</b>详解

    自动初始化机制原理详解

    自动初始化机制是指初始化函数不需要被显式调用,只需要在函数定义处通过宏定义的方式进行申明,就会在系统启动过程中被执行。这篇文章就来探索一下其中的奥秘, 简单理解其原理!
    的头像 发表于 12-16 09:33 939次阅读
    <b class='flag-5'>自动</b><b class='flag-5'>初始化</b>机制原理详解

    LINUX系统引导和初始化-LINUX内核解读

    Linux 的系统引导和初始化 ----------Linux2.4.22内核解读之一 一、 系统引导和初始化概述 相关代码(引导扇区的程序及其辅助程序,以 x86体系为例): \
    发表于 11-03 22:31 53次下载

    CR95HF 的初始化步骤

    STM32F103的CR95HF 的初始化步骤
    发表于 12-14 16:01 0次下载

    Linux内存初始化

    之前有几篇博客详细介绍了Xen的内存初始化,确实感觉这部分内容蛮复杂的。这两天在看Linux内核启动中内存的初始化,也是看的云里雾里的,想尝试下边看边写,写博客的过程中慢慢思考,最后
    发表于 10-12 11:16 0次下载

    ds1302时钟芯片初始化自动决定DS1302是否需要初始化程序

    ds1302芯片时钟芯片大家都在问到底需要不需要初始化?这篇文章将会给大家一个程序,可以自动决定DS1302是否需要初始化
    发表于 10-19 19:19 8442次阅读

    stm32初始化流程图解析

    STM32系列基于专为要求高性能、低成本、低功耗的嵌入式应用专门设计的ARM Cortex-M3内核。本文主要以stm32初始化流程而展开的讨论。
    发表于 11-16 11:39 1.9w次阅读
    <b class='flag-5'>stm32</b><b class='flag-5'>初始化</b>流程图解析

    51平台下初始化文件的引入导致全局变量无法初始化的问题如何解决

    本文档的主要内容详细介绍的是51平台下初始化文件的引入导致全局变量无法初始化的问题如何解决。
    发表于 08-20 17:31 0次下载
    <b class='flag-5'>在</b>51平台下<b class='flag-5'>初始化</b>文件的引入导致全局变量无法<b class='flag-5'>初始化</b>的问题如何解决

    一文详解RT-Thread自动初始化

    在学RT-Thread时,经常能听到这个词:自动初始化。用起来也非常容易,一个宏就解决了,但是原理是什么呢?
    的头像 发表于 07-21 10:17 7507次阅读
    一文详解RT-Thread<b class='flag-5'>自动</b><b class='flag-5'>初始化</b>

    STM32执行代码初始化卡住,或者电卡住,或者复位卡住,导致代码不执行

    STM32的板子电或者复位,接有显示屏或者LED指示灯的都会卡住解决:1、检查自己的代码是否有中断,有中断的话,其初始化放在其他硬件初始化之后 即:中断的
    发表于 12-09 09:21 22次下载
    <b class='flag-5'>STM32</b>执行代码<b class='flag-5'>初始化</b>卡住,或者<b class='flag-5'>上</b>电卡住,或者复位卡住,导致代码不执行

    RT-Thread自动初始化机制

      分析之前首先查阅 RT-Thread 的官方文档 [RT-Thread 自动初始化机制](https://www.rt-thread.org/document/site
    的头像 发表于 06-17 08:52 2486次阅读
    RT-Thread<b class='flag-5'>自动</b><b class='flag-5'>初始化</b>机制

    如何对PNET 模拟器进行初始化安装

    今天给大家带来一期干货内容,关于模拟器, 这个模拟器就是PNET,今天给大家分享一下如何对PNET 模拟器进行初始化安装。
    的头像 发表于 10-28 09:15 1923次阅读

    DB2163_STM32配置和初始化C代码生成

    DB2163_STM32配置和初始化C代码生成
    发表于 11-23 20:29 0次下载
    DB2163_<b class='flag-5'>STM32</b>配置和<b class='flag-5'>初始化</b>C代码生成

    使用STM32CubeMX生成初始化代码

    我使用STM32CubeMX生成初始化代码,使用LL库,这里只介绍跟i2c相关的部分,其他必要的初始化需要自己完成。芯片使用stm32f042。本文的代码不能到手即用,只提供思路。
    的头像 发表于 03-22 15:26 2802次阅读

    STM32 模拟Linux kernel自动初始化流程

    : 本次给大家转来一篇:STM32 模拟Linux kernel自动初始化流程,这个机制 嵌入式大杂烩周记 | 第 4 期 中也有用到。文末
    的头像 发表于 06-22 10:03 569次阅读
    <b class='flag-5'>STM32</b> <b class='flag-5'>模拟</b><b class='flag-5'>Linux</b> kernel<b class='flag-5'>自动</b><b class='flag-5'>初始化</b>流程