以下内容源于朱有鹏嵌入式课程的学习与整理,如有侵权请告知删除。
1、uboot的链接脚本里有如下代码片段。
__u_boot_cmd_start = .;.u_boot_cmd : { *(.u_boot_cmd) }
__u_boot_cmd_end = .;
2、uboot中将命令结构体单独存放在某个代码段,是通过以下两个宏来实现的。
#define Struct_Section __attribute__ ((unused,section (".u_boot_cmd")))
#define U_BOOT_CMD(name,maxargs,rep,cmd,usage,help) cmd_tbl_t __u_boot_cmd_##name Struct_Section = {#name, maxargs, rep, cmd, usage, help}
(1)凡是通过U_BOOT_CMD定义的cmd_tbl_t变量(类似说int类型变量)会全部被放在.u_boot_cmd段当中;
(2)##name、#name
- ##name表示将name直接原样替代##name,#name表示以 “name" 的形式替代#name。
- 例如定义一个命令boot,需要写成U_BOOT_CMD(boot, 0, 0, fun, "boot xxx");
- 展开后会变成:cmd_tbl_t __u_boot_cmd_boot __attribute___((unused, section(".u_boot_cmd"))) = {"boot", 0, 0, fun, "boot xxx"}
- 可见Struct_Section展开了,还将##name换成了boot, 将#name换成了"boot"。
(3)总结
- 可以在程序运行时定义一个变量,即通过##xxx来定义一个变量,然后通过上面的形式来使用它。
- 通过宏定义来定义变量,并且把这些同一种结构体的变量放在一个段中,充分的利用连接器的作用,确实很实用。这样开发各个模块的研发人员不必去维护一个全局的结构体数组,值得推广。
(4)实例
#if defined(CONFIG_CMD_ECHO)int do_echo (cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
{int i, putnl = 1;for (i = 1; i < argc; i++) {char *p = argv[i], c;if (i > 1)putc(' ');while ((c = *p++) != '\0') {if (c == '\\' && *p == 'c') {putnl = 0;p++;} else {putc(c);}}}if (putnl)putc('\n');return 0;
}U_BOOT_CMD(echo, CFG_MAXARGS, 1, do_echo,"echo - echo args to console\n","[args..]\n"" - echo args to console; \\c suppresses newline\n"
);//通过宏来定义变量#endif