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 可以看
遍历函数还有一种方法,在 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)(); }