以下内容源于朱有鹏嵌入式课程的学习与整理,如有侵权请告知删除。
一、在链接脚本中定义.u_boot_cmd段
如下所示,uboot的链接脚本u-boot.lds中定义了一个名为.u_boot_cmd 的段:
__u_boot_cmd_start = .;
.u_boot_cmd : { *(.u_boot_cmd) }
__u_boot_cmd_end = .;
二、将所有命令结构体变量放在.u_boot_cmd段
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}
这样一来,通过U_BOOT_CMD定义的cmd_tbl_t类型的变量,全部被存放在.u_boot_cmd段中。
(1)##name、#name的含义
##name表示将name直接原样替代##name;#name表示以 “name"(加个双引号) 的形式替代#name。
比如定义一个命令boot,需要写成U_BOOT_CMD(boot, 0, 0, fun, "boot xxx");
展开后会变成(把Struct_Section展开,将##name换成boot,将#name换成"boot"):
cmd_tbl_t __u_boot_cmd_boot __attribute___((unused, section(".u_boot_cmd"))) = {"boot", 0, 0, fun, "boot xxx"}
(2)启发与总结
我们可以在程序运行时定义一个变量,即通过##xxx来定义一个变量,然后通过上面的形式来使用这个变量。
另外通过宏定义来定义变量,并且把这些同一种结构体的变量放在一个段中,可以充分地利用连接器的作用,这样开发各个模块的研发人员就不必去维护一个全局的结构体数组。
(3)实例演示
#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