单片机模拟Linux自动初始化流程

关注+星标公众,不错过精彩内容

4f147cd514abbf9c9d9225f9549150fd.gif

来源 | 嵌入式大杂烩

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

b140c90216f36c4dccf3991078069967.jpeg

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

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

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

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

1,程序代码的组织

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

3,函数指针的应用。

8ee3f141c349db61ecf1b665b50589ee.jpeg

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

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

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

02101744ba478ddf83ae8129672e5f71.png

先加段代码:

3bafdec496cc5305f7011f3bf77a5e81.png

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

4ee981173b2f1f24508d86b96606ef75.jpeg

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

如图片:

e5850e74a3933ffea46dcd1bc3e3102e.png

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

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

d2f15f2be6b4bace73e94ec2aa198359.png

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

cf5acabcb2d863f52785b898681a425e.png

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

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

6c6accc5f749cae2ae7b7aeed522d05c.png a1b32c603bf711b9ac6d2918777eb84b.png

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

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

这个也一样。

53d04746a28f9b0a3b539a25cf8e825a.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的顺序执行了。

1ee37af4ddd1d682960efd09f5fa403d.png e16a9664417816d8fb3e258a34e58e1a.png

最后在来看一下initcall 段:

f46b7f757633e81c6e7153040215eb45.png 7705de425e5370246b724895cf759978.png 8da8e2b9597cd8391983dcf53b77ea21.jpeg 0f9db6626c93de0e0dacc0d8d79b6acc.jpeg

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

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

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

https://gitee.com/android_life/stm32_freertos_opensource/tree/master/bareos/initcall

感兴趣的可以看看,及关注,转发。

原文:https://www.toutiao.com/a6906858875912307203/?log_from=16f51be0430a9_1647178471075

声明:本文素材来源网络,版权归原作者所有。如涉及作品版权问题,请与我联系删除。

------------ END ------------

d1dade8d2e370d9994231b81af498962.gif

●专栏《嵌入式工具

●专栏《嵌入式开发》

●专栏《Keil教程》

●嵌入式专栏精选教程

关注公众号回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。

9cf3a14f14fc96031907535fc9fe40e2.jpeg

7dde13dc19089b7ce4c3b9c901af65a8.png

点击“阅读原文”查看更多分享。

猜你喜欢

转载自blog.csdn.net/ybhuangfugui/article/details/135007758
今日推荐