u-boot是嵌入式系统中广泛使用的1种bootloader。
它不但支持众多的平台和参考板,而且支持丰富的命令,可让用户对开发板进行各种操作。
关于u-boot使用和移植的资料很多,下面用问答的方式介绍u-boot中命令的实现机制。
Q1: u-boot中命令的实现代码在甚么地方?
A1: 在u-boot源码的common目录下。通常1个或多个命令在对应的某个C文件中实现。如果用户要增加1个新命令的话,要在common目录下添加对应的C文件,并在Makefile中指定编译该文件。
Q2: 能不能给个命令的实现模板先?
A2: #include <common.h>
#include <command.h>
int do_hello(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
{
printf(“hello world
”);
return 0;
}
U_BOOT_CMD(hello, CONFIG_SYS_MAXARGS, 1, do_hello, “usage info”, “help info”);
以下是上面代码的解释:
U_BOOT_CMD是u-boot中定义的宏,上面的宏定义展开后以下:
cmd_tbl_t __u_boot_cmd_hello Struct_Section = {“hello”, CONFIG_SYS_MAXARGS, 1, do_hello, “usage info”, “help info”};
这下清楚了,实际上就是定义1个类型为cmd_tbl_t的结构体并对其赋值。
u-boot中每一个命令都用这样的1个结构体来描写,类型定义以下:
typedef struct cmd_tbl_s cmd_tbl_t;
struct cmd_tbl_s {
char *name; // 命令的名称
int maxargs; // 最多支持的参数的个数
int repeatable; // 是不是可重复履行
int (*cmd)(struct cmd_tbl_s *, int, int, char *const[]); // 命令对应的处理函数
char *usage; // 命令扼要使用信息
char *help; // 命令详细帮助信息
}
成员cmd是个函数指针,需要指向该命令对应的处理函数。这也是在前面的模板中先定义1个处理函数的缘由。所有命令的处理函数的接口都是1致的:参数cmdtp指向命令对应的cmt_tbl_t结构体;参数flag可以疏忽不处理;参数argc和参数argv的作用和它们在main函数中是1样的。
Q3: 当用户输入命令后,u-boot是如何履行的?
A3: 前面用U_BOOT_CMD定义结构体时用到了另外1个宏Struct_Section,它表示把该结构体放到1个专门的段。u_boot_cmd,没错,u-boot中所有命令对应的结构体都集中放在这个段里,该段里的内容会被包括在编译好的u-boot里。
这下就容易理解了:u-boot接收到用户的命令后在.u_boot_cmd段中顺次查找每一个cmd_tbl_t,根据成员name比较是不是和用户命令匹配。若成功则履行第3个成员cmd指向的函数,否则查找下个cmd_tbl_t。具体的实现请参考u-boot中的函数find_cmd的代码。
u-boot不但是1个功能强大的bootloader,而且是1个很好的嵌入式编程的学习对象,其中很多编程的思想和实现的方法都值得大家鉴戒与学习。
点击打开链接
下一篇 IOS开发之NSURL