__attribute__之section详解 ------ 把函数指定到具体某个section 之 RT-thread 实例详解
Posted 流水灯
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了__attribute__之section详解 ------ 把函数指定到具体某个section 之 RT-thread 实例详解相关的知识,希望对你有一定的参考价值。
typedef int (*init_fn_t)(void);
#define RT_USED __attribute__((used))
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 可以看
遍历函数还有一种方法,在 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)(); }
以上是关于__attribute__之section详解 ------ 把函数指定到具体某个section 之 RT-thread 实例详解的主要内容,如果未能解决你的问题,请参考以下文章
删除使用 __attribute__((section)) 创建的部分
__attribute__((section(”name“)))的一些理解
gcc - 使用 #pragma 将 __attribute__((section(".dflash_code"))) 应用于整个源文件
利用__attribute__((section("name")))构建初始化函数表
利用gcc的__attribute__编译属性section子项构建初始化函数表
mdk keil 指定变量函数存储位置,使用 Scatter-Loading Description File, __attribute__(("section“))