在u-boot中提供了一个go命令,是可以跳到指定地址执行代码的命令,这实际上是执行了下面的函数:
int do_go(cmd_tbl_t * cmdtp,int flag,int argc,char *argv[])
{
ulong addr, rc;
int rcode = 0;
if (argc < 2) {
printf ("Usage:\n%s\n", cmdtp->usage);
return 1;
}
addr = simple_strtoul(argv[1], NULL, 16);
printf ("## Starting application at 0x%08lX ...\n", addr);
/*
* pass address parameter as argv[0] (aka command name),
* and all remaining args
*/
rc = ((ulong (*)(int, char *[]))addr) (--argc, &argv[1]);
if (rc != 0) rcode = 1;
printf ("## Application terminated, rc = 0x%lX\n", rc);
return rcode;
}
这个函数很简单,首先把参数1转换为ulong类型,我们输入的执行命令是go 0xc0008000,那么下面的代码将做此事:
addr = simple_strtoul(argv[1], NULL, 16);
关键代码如下所示:
rc = ((ulong (*)(int, char *[]))addr) (--argc, &argv[1]);
上面这行代码正是执行go命令的核心,tc是存返回码的,右边的刚开始看是有点晕,我们把它分开来看就不晕了:
( ulong (*) (int, char *[]) )
这是什么?它是一个函数指针,addr就是我们上面转还为ulong 类型的地址了,至于右边的就是传给函数的参数,实际上我们一般都不会使用这个参数了.
总结:这个总的函数其实最重要的就是上面那行代码了,它的意思是把addr强制转换为 ( ulong (*) (int, char *[]) ) 类型的函数指针,然后执行这个函数, 这个函数使用的是绝对地址跳转,也就是传入的地址处必须包含实际的可执行有效代码。所以这个函数不能直接加载内核img。