uboot自定义命令

描述

uboot自定义命令

      U-Boot 是一个主要用于嵌入式系统的引导加载程序,可以支持多种不同的计算机系统结构,包括PPC、ARM、AVR32、MIPS、x86、68k、Nios与MicroBlaze。这也是一套在GNU通用公共许可证之下发布的自由软件。

      U-Boot本质是一个裸机程序,是一种普遍用于嵌入式系统中的开源的Bootloader,作用是用来引导操作系统,以及给开发人员提供测试调试工具。主要负责基本硬件初始化,导启动内核启动。

      U-Boot命令众多,通过uboot命令可完成系统环境变量设置。U-Boot本质就是一份裸机程序,这样可以在U-Boot命令下调试一些硬件设备问题。U-Boot命令行也是可以实现自动补全功能。

1.uboot命令格式分析

  uboot命令存放在uboot目录下的common目录下。
  do_help命令功能实现如下:

 

extern cmd_tbl_t  __u_boot_cmd_bdinfo;
extern cmd_tbl_t  __u_boot_cmd_showvar;

int do_help(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])
{
//	return _do_help(&__u_boot_cmd_start,
//			&__u_boot_cmd_end - &__u_boot_cmd_start,
//			cmdtp, flag, argc, argv);
	return _do_help(&__u_boot_cmd_bdinfo,
			&__u_boot_cmd_showvar - &__u_boot_cmd_bdinfo + 1,
			cmdtp, flag, argc, argv);
}

U_BOOT_CMD(
	help,	CONFIG_SYS_MAXARGS,	1,	do_help,
	"print command description/usage",
	"\n"
	"	- print brief description of all commands\n"
	"help command ...\n"
	"	- print detailed usage of 'command'"
);

/* This does not use the U_BOOT_CMD macro as ? can't be used in symbol names */
cmd_tbl_t __u_boot_cmd_question_mark Struct_Section = {
	"?",	CONFIG_SYS_MAXARGS,	1,	do_help,
	"alias for 'help'",
#ifdef  CONFIG_SYS_LONGHELP
	""
#endif /* CONFIG_SYS_LONGHELP */
};

 

U_BOOT_CMD为命令参数传递:

      help 命令标志符;
      CONFIG_SYS_MAXARGS命令传递的最大参数个数;
     1 命令重复执行次数;

 do_help命令实现函数;

  “alias for ‘help’”命令用法说明;
     最后一个参数为命令用法详细说明;
     int do_help(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])函数参数说明:
     cmd_tbl_t * cmdtp结构体指针。

 

struct cmd_tbl_s {
	char		*name;		/* 命令名字*/
	int		maxargs;	/*命令传递的最大参数个数*/
	int		repeatable;	/*命令重复执行次数*/
	/*命令功能实现函数*/
	int		(*cmd)(struct cmd_tbl_s *, int, int, char * const []);
	char		*usage;		/*命令基本用法说明*/
#ifdef	CONFIG_SYS_LONGHELP
	char		*help;		/*命令详细用法说明*/
#endif
#ifdef CONFIG_AUTO_COMPLETE
	/* do auto completion on the arguments */
	int		(*complete)(int argc, char * const argv[], char last_char, int maxv, char *cmdv[]);
#endif
};

 

  argc 命令传入的参数个数,和main函数argc相同;
  argv 存放命令参数内容,和main函数argv相同;

2.tiny4412下编写uboot命令控制蜂鸣器

2.1 tiny4412开发板beep硬件接口

UbootUboot

2.2 GPD0端口寄存器地址

Uboot

  配置寄存器:GPD0CON_ADDR = 0x1140_0000 + 0x00A0

Uboot

2.3 参考help命令模板实现beep命令

 

#include 
#include 

/*蜂鸣器*/
#define GPD0_CON *((volatile unsigned int *)0x114000A0)//控制寄存器
#define GPD0_DAT *((volatile unsigned int *)0x114000A4)//数据寄存器

int do_beep(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])
{
	/*蜂鸣器*/
	GPD0_CON&=0xfffffff0;//清除当前GPD0_0的配置
	GPD0_CON|=0x00000001;//设置为输出模式
	//beep on 或者 beep off
	if(argc!=2)
	{
		printf("格式:beep on/off\n");
		return 0;
	}
	if(strcmp(argv[1],"on")==0)//开蜂鸣器
	{
		GPD0_DAT|=1<<0;//开蜂鸣器
	}
	else if(strcmp(argv[1],"off")==0)//关蜂鸣器
	{
		GPD0_DAT&=~(1<<0);//关蜂鸣器
	}	
	return 0;
}

U_BOOT_CMD(
	beep,	CONFIG_SYS_MAXARGS,	1,	do_beep,
	"beep ",
	"\n"
	"beep用法\n"
	"开蜂鸣器:beep on\n"
	"关蜂鸣器:beep_off"
);

 

2.4 程序编译

  1. 将cmd_beep.c拷贝到u_boot目录下的common目录下

 

[wbyq@wbyq common]$ cp /mnt/hgfs/ubuntu/cmd_beep.c ./ 

 

  2. 修改common目录下的Makefile文件。

 

[wbyq@wbyq common]$ vim  Makefile 
Uboot

 

  3.在uoot顶层目录下执行make,重新编译uboot

 

[wbyq@wbyq uboot_tiny4412-sdk1506]$ make

 

  4.烧写uboot

 

[wbyq@wbyq uboot_tiny4412-sdk1506]$ cd sd_fuse/
[wbyq@wbyq tiny4412]$ sudo ./sd_fusing.sh /dev/sdb

 

  beep命令编写成功。

Uboot

  5. beep命令运行测试

 

TINY4412 # beep
格式:beep on/off
TINY4412 # ? beep
flag=0
beep - beep 

Usage:
beep 
beep用法
开蜂鸣器:beep on
关蜂鸣器:beep_off
TINY4412 # beep on
TINY4412 # beep off

 

3.分析uboot保存环境变量位置

  在Linux操作系统启动前,uboot需要完成系统环境变量配置。在uboot目录下有tiny4412.h中保存的系统的默认环境变量参数。

Uboot

  1.当我们修改了系统环境变量,执行了saveenv命令后,则环境变量将写入到磁盘中,接下来我们将分析一下uboot源码中环境变量写入磁盘的扇区位置。
 在uboot命令中,并没有直接脚cmd_save.c的函数,因此,我们可以根据saveenv命令执行后的提示信息来分析。在uboot命令行下执行saveenv命令时,会看到有输出提示信息,这样我们可以直接在uboot工程中搜索一下输出信息的位置。
 

 

TINY4412 # save
Saving Environment to SMDK bootable device...
done
UbootUbootUboot

 

  2.接下来跳入saveenv函数,查看一下该函数执行过程。

Uboot

  当我们跳到定义处时会看到有多重位置,要想准确跳转到正确位置,我们可以再查找参考saveenv命令的后半段提示信息赋值位置。

UbootUboot

  3.经过查找参考我们发现,后半段提示信息的定义位置是在env_auto.c中,而我们saveenv函数右键跳到定义处时在env_auto.c中也有定义,说明我们要跳转的文件即为env_atuo.c文件。

      在 saveenv函数中,并没有看到有关于mmc写扇区的函数,说明我们还需要进一步跳转到对应的子函数中去分析。
 

Uboot

  4.我们可以再次根据saveenv命令的打印提示,在最后还会打印“done”这个信息,这样我们就可以更新这个信息,对当前函数中的三个子函数进行分析,看下哪一个子函数中有这个字符串的打印。经过查看代码可知最终调用的函数即为saveenv_movinand()函数。

Uboot

  5.再次跳到movi_write_env()函数中查看一下写入到mmc中的地址。

Uboot

  6.在这个函数中即为实现环境变量写入作用,再次跳到movi_write()函数中查看一下这个函数参数作用。

Uboot

  可以看到,movi_write()函数参数作用分别为:mmc设备编号、写入mmc的起始扇区、写入的扇区个数、写入的环境变量内容。
  我们可以在这个函数中加入printf函数打印一下写入的扇区地址和扇区数量。然后再重新编译uboot,烧写到SD卡中再次启动查看效果。

 

TINY4412 # save
Saving Environment to SMDK bootable device...
start=1025,blkcnt=32
done
TINY4412 # 

 

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分