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

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

3天内不再提示

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

RTThread物联网操作系统 来源:陈翠 2019-07-21 10:17 次阅读

一、前言

在学RT-Thread时,经常能听到这个词:自动初始化。用起来也非常容易,一个宏就解决了,但是原理是什么呢?官网文档提及到了,

(他们的文档在这里:https://www.rt-thread.org/document/site/programming-manual/basic/basic/#rt-thread_3),但是写的只是概念层面上的,看完后会使用但原理还是不太清楚。之前研究过,今天把它总结下,写出来分享。

1.1、一般情况的初始化调用

一般情况下,系统中的初始化会这样做,应该再熟悉不过了:

1//伪代码 2 3voidmain(void) 4{ 5uart_init(); 6led_init(); 7... 8 910while(1)11{12//func113//func214}15}

这样的显式调用初始化函数,有时可能多达 十几到几十 个,看起来非常非常繁杂。但是好像没啥问题,因为已经看习惯了。

1.2 使用自动初始化后

举例一个自动初始化的用法如下:

1//这是led.c文件23voidled_init(void)4{5//省略6}7INIT_APP_EXPORT(led_init)1//这是main.c文件2intmain(void)3{45}

这样,使用一个宏,初始化函数就会被自动初始化,不用在其他地方显式调用 led_init() 。代码瞬间清爽很多。

二、引入

当然也不用担心一个初始化必须在另一个初始化之前的问题,因为这里有6个自动初始化等级可供选择。

我抠了一张RT-Thread官网文档的图,该图是RT-Thread代码的启动流程图,该图中的蓝色方框部分就是自动初始化的6个等级以及初始化的先后顺序。从图中可以看出这6部分的初始化是由函数 rt_components_board_init() 与 rt_components_init() 完成的。

在一开始的例子中, INIT_APP_EXPORT(led_init) 就位于最后一个方框的位置,属于applications init functions。

那么其他等级分别对应什么宏进行初始化的?看下面的表格:

三、自动初始化原理

3.1 6个自动初始化宏的定义

查看源码,这 6 个宏定义如下:( 不同的段:1 2 3 4 5 6 )

1/*boardinitroutineswillbecalledinboard_init()function*/ 2#defineINIT_BOARD_EXPORT(fn)INIT_EXPORT(fn,"1") 3 4/*pre/device/component/env/appinitroutineswillbecalledininit_thread*/ 5/*componentspre-initialization(puresoftwareinitilization)*/ 6#defineINIT_PREV_EXPORT(fn)INIT_EXPORT(fn,"2") 7/*deviceinitialization*/ 8#defineINIT_DEVICE_EXPORT(fn)INIT_EXPORT(fn,"3") 9/*componentsinitialization(dfs,lwip,...)*/10#defineINIT_COMPONENT_EXPORT(fn)INIT_EXPORT(fn,"4")11/*environmentinitialization(mountdisk,...)*/12#defineINIT_ENV_EXPORT(fn)INIT_EXPORT(fn,"5")13/*appliationinitialization(rtguiapplicationetc...)*/14#defineINIT_APP_EXPORT(fn)INIT_EXPORT(fn,"6")

INIT_EXPORT(fn, level) 表示这个函数 fn 现在属于哪个初始化 level 段, 由 SECTION(".rti_fn."level) 进行定义。

1#defineINIT_EXPORT(fn,level)2RT_USEDconstinit_fn_t__rt_init_##fnSECTION(".rti_fn."level)=fn

而 SECTION(x) 是:

1#defineSECTION(x)__attribute__((section(x)))

__attribute__((section("name"))) :将作用的函数或数据放入指定名为"name"的输入段中。(在不同的编译器中实现的方式也有所不同。)

以上就是整个的宏定义作用就是将函数 fn 的地址赋给一个 __rt_init_fn 的指针,然后放入相应 level 的数据段中。所以函数使用自动初始化宏导出后,这些数据段中就会存储指向各个初始化函数的指针。

举例:INIT_APP_EXPORT(pin_beep_sample);

1//函数pin_beep_sample(),使用INIT_APP_EXPORT()进行自动初始化。 2 3INIT_APP_EXPORT(pin_beep_sample); 4=INIT_EXPORT(pin_beep_sample,"6") 5=constinit_fn_t__rt_init_pin_beep_sampleSECTION(".rti_fn.""6")=pin_beep_sample 6 7/* 8表示把函数pin_beep_sample的地址赋值给常量函数指针__rt_init_pin_beep_sample, 9然后放入名称为".rti_fn.6"的数据段中。10(其中init_fn_t是一个函数指针类型,原型为typedef int (*init_fn_t)(void)。)11*/

表示把函数 pin_beep_sample 的地址赋值给常量函数指针 __rt_init_pin_beep_sample,然后放入名称为 ".rti_fn.6" 的数据段中。( 其中 init_fn_t 是一个函数指针类型,原型为 typedef int (*init_fn_t)(void)。)

注意被自动初始化的函数类型为:int (*init_fn_t)(void) ,无参,int 返回。

3.2 自动初始化过程

那么上面提到,在启动流程中,调用了两个函数 rt_components_board_init() 与 rt_components_init() 就完成了6部分的初始化。从启动流程图中可以看出:rt_components_board_init() 完成了第 1 段, rt_components_init() 完成了第2 到第6 段。

说明:rt_components_board_init()主要board板级的初始化,调度器还未启动,是在系统起来之前做的初始化。所以在使用board级别的初始化时不要使用系统API,如rt_thread_delay()等。rt_components_init()主要是一些组件的初始化及应用初始化,是在main线程中完成的,当调度器启动之后,系统启动开始运行main线程时才会进行的初始化。是线程的运行环境。

3.2.1、两个函数的实现

1、第一个函数 rt_components_board_init() 的实现:

1voidrt_components_board_init(void) 2{ 3constinit_fn_t*fn_ptr; 4 5for(fn_ptr=&__rt_init_rti_board_start;fn_ptr< &__rt_init_rti_board_end; fn_ptr++) 6    { 7        (*fn_ptr)(); 8    } 9#endif10}

非调试模式下rt_components_board_init():for循环会遍历位于__rt_init_rti_board_start 到 __rt_init_rti_board_end 之间保存的函数指针,然后依次执行这些函数。

2、第二个函数 rt_components_init() 的实现:

1voidrt_components_init(void) 2{ 3constinit_fn_t*fn_ptr; 4 5for(fn_ptr=&__rt_init_rti_board_end;fn_ptr< &__rt_init_rti_end; fn_ptr ++) 6    { 7        (*fn_ptr)(); 8    } 9#endif10}

非调试模式下rt_components_init():for循环会遍历位于__rt_init_rti_board_end 到 __rt_init_rti_end 之间保存的函数指针,然后依次执行这些函数 。

那么 __rt_init_rti_board_start、__rt_init_rti_board_end、__rt_init_rti_end 是啥?

3.2.2 划分

在系统中,定义了这几个空函数:rti_start、rti_board_start、rti_board_end、rti_end。不同的段:0、 0.end 、 1.end 、6.end

1staticintrti_start(void) 2{ 3return0; 4} 5INIT_EXPORT(rti_start,"0"); 6 7staticintrti_board_start(void) 8{ 9return0;10}11INIT_EXPORT(rti_board_start,"0.end");1213staticintrti_board_end(void)14{15return0;16}17INIT_EXPORT(rti_board_end,"1.end");1819staticintrti_end(void)20{21return0;22}23INIT_EXPORT(rti_end,"6.end");

这几个函数的导出,加上上面 6 个初始化宏的导出,就有了这样一个表格:

可以看出,这4个空函数所导出的段中间,包含着这6个初始化宏定义的段,而这6个段中分别包含着各自宏导出函数时的函数指针。

rt_components_board_init() 完成了第 1 段, rt_components_init() 完成了第2 到第6 段。

1、rt_components_board_init() 完成了第 1 段,也就是初始化了由INIT_BOARD_EXPORT(fn) 的初始化的所有函数,也就是__rt_init_rti_board_start 到 __rt_init_rti_board_end 之间的函数指针。

2、rt_components_init() 完成了第2 到第6 段,也就是按顺序初始化了由 INIT_PREV_EXPORT(fn) 到 INIT_DEVICE_EXPORT(fn) 到 INIT_COMPONENT_EXPORT(fn)、 INIT_ENV_EXPORT(fn)、 INIT_APP_EXPORT(fn)初始化的所有函数,也就是从 __rt_init_rti_board_end 到 __rt_init_rti_end 之间的函数指针。

所以,当你使用自动初始化导出宏 去初始化一个函数时,是由系统中的这两个函数进行遍历函数指针执行的。

3.2.3、示例

还是上面 INIT_APP_EXPORT(pin_beep_sample); 的例子。

举例:INIT_APP_EXPORT(pin_beep_sample);

1//函数pin_beep_sample(),使用INIT_APP_EXPORT()进行自动初始化。 2 3INIT_APP_EXPORT(pin_beep_sample); 4=INIT_EXPORT(pin_beep_sample,"6") 5=constinit_fn_t__rt_init_pin_beep_sampleSECTION(".rti_fn.""6")=pin_beep_sample 6 7/* 8表示把函数pin_beep_sample的地址赋值给常量函数指针__rt_init_pin_beep_sample, 9然后放入名称为".rti_fn.6"的数据段中。10(其中init_fn_t是一个函数指针类型,原型为typedef int (*init_fn_t)(void)。)11*/

表示把函数 pin_beep_sample 的地址赋值给常量函数指针 __rt_init_pin_beep_sample,然后放入名称为 ".rti_fn.6" 的数据段中。( 其中 init_fn_t 是一个函数指针类型,原型为 typedef int (*init_fn_t)(void)。)

在编译后的.map文件中可以查看到:

常量函数指针 __rt_init_pin_beep_sample 位于 .rti_fn.6 段中。

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

    关注

    0

    文章

    48

    浏览量

    11627
  • RT-Thread
    +关注

    关注

    31

    文章

    1148

    浏览量

    38865

原文标题:RT-Thread 自动初始化详解

文章出处:【微信号:RTThread,微信公众号:RTThread物联网操作系统】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    RT-Thread自动初始化详解

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

    【每日练】RT-Thread Nano-自动初始化及MSH-EXPORT-2(第八节学习视频)

    (第六节学习视频)【每日练】RT-Thread Nano-自动初始化及MSH-EXPORT-1(第七节学习视频)
    发表于 05-24 10:42

    RT-thread初始化过程是怎样进行的

    RT-thread初始化过程是怎样进行的?扩展补丁Sub和super的作用是什么?如何去使用它们呢?
    发表于 11-29 07:42

    如何对RT-Thread系统进行初始化

    RT-Thread是如何启动的?如何对RT-Thread系统进行初始化呢?
    发表于 11-30 07:54

    为什么RT-Thread要采用这种复杂的方式来进行自动初始化操作呢

    在分析之前首先查阅 RT-Thread 的官方文档RT-Thread 自动初始化机制,根据官方文档的讲述在 RTT 源码中共使用了 6 中
    发表于 04-06 17:49

    RT-Thread自动初始化机制简介

    RT-Thread 的时钟管理以时钟节拍为基础,时钟节拍是 RT-Thread 操作系统中最小的RT-Thread 自动初始化机制时钟单位。
    发表于 04-06 18:08

    RT-Thread系统自动初始化机制简介

    RT-Thread 自动初始化机制1、自动初始化机制简介在系统启动流程图中,有两个函数:rt_c
    发表于 04-12 17:43

    RT-Thread系统初始化与启动流程详细描述

    时, 该函数将初始化系统的定时器线程。void rt_application_init( )创建用户线程由此创建个用户main()线程,而 main()函数是RT-Thread的用户
    发表于 08-25 15:15

    如何对RT-Thread系统的线程进行初始化

    , rt_uint32_t tick);rt_thread_init函数用来初始化静态线程对象。而线程句柄(或者说线程控制块指针),线程栈由用户提供。静态线程是指,线程控制块、线程运行栈
    发表于 08-30 14:51

    RT-Thread自动初始化原理分析

    ;}这里我们直接就可以使用 printf 进行打印,而没有进行些其它的初始化,参考这个思路引出了 RT-Thread自动初始化
    发表于 12-05 14:17

    RT-Thread全球技术大会:如何使用组件以及自动初始化流程

    RT-Thread全球技术大会:如何使用组件和自动初始化流程           审核编辑:彭静
    的头像 发表于 05-27 15:16 696次阅读
    <b class='flag-5'>RT-Thread</b>全球技术大会:如何使用组件以及<b class='flag-5'>自动</b><b class='flag-5'>初始化</b>流程

    RT-Thread自动初始化机制

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

    rt-thread 优化系列(六)启动流程重构

    去年此时,笔者刚接触 rt-thread 的时候,被它的初始化过程深深折服了。第一次打开一个 rt-thread 的项目,竟然没找到多线程在哪儿初始化的,"main" 函数里没有!
    的头像 发表于 07-04 15:30 1333次阅读
    <b class='flag-5'>rt-thread</b> 优化系列(六)启动流程重构

    rt-thread线程栈初始化参数分析

    RT-Thread 在线程初始化的代码内有一段初始化线程堆栈的代码
    的头像 发表于 08-14 16:50 991次阅读
    <b class='flag-5'>rt-thread</b>线程栈<b class='flag-5'>初始化</b>参数分析

    RT-Thread使用经验分享:链表未初始化造成死机

    最近在开发调试基于RT-Thread 的驱动时,遇到一个比较奇怪的死机问题,后来经过一步步排查,终于发现是驱动的链表节点没有初始化造成的死机
    的头像 发表于 10-08 14:49 476次阅读
    <b class='flag-5'>RT-Thread</b>使用经验分享:链表未<b class='flag-5'>初始化</b>造成死机