__attribute__之section详解 ------ 把函数指定到具体某个section 之 RT-thread 实例详解

typedef int (*init_fn_t)(void);
#define RT_USED                     __attribute__((used))
#define INIT_EXPORT(fn, level)
  RT_USED const init_fn_t __rt_init_##fn SECTION(".rti_fn." level) = fn #define INIT_BOARD_EXPORT(fn) INIT_EXPORT(fn, "1") #define INIT_PREV_EXPORT(fn) INIT_EXPORT(fn, "2")

 SECTION(".rti_fn." "1") 等效于 SECTION(".rti_fn.1"),可见 level 是前一段的后缀,称".rti_fn.1"为段名

 

以下定义空函数,为了后面遍历函数用,标记起始地址和结束地址

static int rti_start(void)
{
    return 0;
}
INIT_EXPORT(rti_start, "0");

static int rti_board_start(void)
{
    return 0;
}
INIT_EXPORT(rti_board_start, "0.end");

static int rti_board_end(void)
{
    return 0;
}
INIT_EXPORT(rti_board_end, "1.end");

static int rti_end(void)
{
    return 0;
}
INIT_EXPORT(rti_end, "6.end");

查看 rtthread.map 可以看

函数指针变量被分配到不同的 section,然后依次从这些代码段读取函数指针执行函数,通过不同 section 实现了哪些函数先执行,哪些函数后执行

 

 

除了定义空函数记录起始地址和结束地址,也可以在 lds 相应段中定义变量

link.lds

        /* section information for initial. */
        . = ALIGN(4);  //ALIGN()是在.h中定义
        __rt_init_start = .;
        KEEP(*(SORT(.rti_fn*)))
        __rt_init_end = .;

如果使用空函数记录地址,SORT()排序函数必须加 

rtdef.h

    #define ALIGN(n)                    __attribute__((aligned(n)))

 

把相应的函数放入section的段中

INIT_BOARD_EXPORT(esp8266_port_init);
INIT_BOARD_EXPORT(clock_information);

 

通过空函数遍历函数

    const init_fn_t *fn_ptr;

    for (fn_ptr = &__rt_init_rti_board_start; fn_ptr < &__rt_init_rti_board_end; fn_ptr++)
    {
        (*fn_ptr)();
    }

通过空函数遍历的好处是:可以在段名前缀一样的情况下,分几段进行遍历,因为有些函数不是在同一个位置执行

 

通过 lds 定义的变量进行遍历

    extern init_fn_t __rt_init_start;
    extern init_fn_t __rt_init_end;
    
    const init_fn_t *fn_ptr = &__rt_init_start;
    for (fn_ptr = &__rt_init_start; fn_ptr < &__rt_init_end; fn_ptr++)
    {
        (*fn_ptr)();
    }

 

posted @ 2021-06-08 17:54  流水灯  阅读(1415)  评论(0编辑  收藏  举报