如何为SD卡与NAND Flash的uboot加上menu菜单
亲爱的卡友们,如果看完文章之后还是有疑惑或不懂的地方,请联系我们,自己去理解或猜答案是件很累的事,请把最麻烦的事情交给我们来处理,术业有专攻,闻道有先后,深圳市雷龙发展专注存储行业13年,专业提供小容量存储解决方案。 【SD NAND】用ok6410进行烧写时,每次都需要敲一大堆命令,又费时又费力。 记得以前用TQ2440时,u-boot启动时会有一个菜单,只按一个数字键就把内核烧好了,非常方便。 现在这张SD卡功能就很全面了,不仅能够直接从SD卡启动,而且还可以烧写NAND Flash中的u-boot zImage rootfs,呵呵。 下面就把这个功能加到ok6410的u-boot 中去。 一、修改SD卡的u-boot1.1.6 1. common/main.c中 void main_loop (void) { …… if(bootdelay>=0&&s&&!abortboot(bootdelay)){ } //如果在启动过程中有按键,打断了执行过程的话 run_command("menu",0); //如果menu返回的话,就进入u-boot的shell中 #ifdef CFG_HUSH_PARSER parse_file_outer(); …… } 2. 执行menu的过程 run_command("menu", 0),最终会执行do_menu. do_menu 开始时先打印出命令菜单,然后根据不同的选择,执行不同的命令。 所以添加的文件 common/cmd_menu.c,如下: #include #include #include #include void print_menu_usage(void) { printf("rn##### SD boot Menu#####rn"); printf("[1] Download u-boot bootloader to Nand Flashrn"); printf("[2] Download Linux Kernel to Nand Flashrn"); printf("[3] Download CRAMFS image to Nand Flashrn"); printf("[4] Download YAFFS image to Nand Flashrn"); printf("[5] Boot the systemrn"); printf("[6] Format the Nand Flashrn"); printf("[0] Set the boot parametersrn"); printf("[a] Download User Program (eg: uCOS-II or TQ2440_Test)rn"); printf("[b] Download LOGO Picture (。bin) to Nand Flash rn"); printf("[q] quit from menurn"); printf("Enter your selection: "); } intdo_menu(cmd_tbl_t*cmdtp,intflag,intargc,char*argv[]) { intc; char cmd_buf[200]; while(1) { print_menu_usage(); c=getc(); printf("%cn",c); switch(c) { case'1': { strcpy(cmd_buf,"fatload mmc 0:1 50008000 u-boot.bin_nand; nand erase 0 100000; nand write.uboot 50008000 0 10000"); run_command(cmd_buf,0); break; } case'2': { strcpy(cmd_buf,"fatload mmc 0:1 50008000 zImage_nand; nand erase 100000 500000; nand write.e 50008000 100000 500000"); run_command(cmd_buf,0); break; } case'3': { //strcpy(cmd_buf,"fatload mmc 0:1 50008000 u-boot.bin; nand erase 0 100000; nand write.uboot 50008000 0 10000"); //run_command(cmd_buf,0); break; } case'4': { //strcpy(cmd_buf,"fatload mmc 0:1 50008000 rootfs.yaffs; nand erase 600000 4A4000; nand write.yaffs2 50008000 600000 4A4000"); //strcpy(cmd_buf,"fatload mmc 0:1 50008000 rootfs.yaffs; nand erase 600000 $(filesize); nand write.yaffs2 50008000 600000 $(filesize)"); //注意:nand erase 600000 $(filesize),假设有坏块的话,这样会有数据不能写入, 所以这个地方可以固定一个较大值,如 // 0x1400000=20M,所以rootfs.yaffs的大小不能超过20M, //考虑此处是不是要加上NAND Flash分区: nand erase root,把整个的root分区全部擦除,这样就不必担心是否有坏块的问题了 strcpy(cmd_buf,"fatload mmc 0:1 50008000 rootfs.yaffs; nand erase 600000 1400000; nand write.yaffs2 50008000 600000 $(filesize)"); run_command(cmd_buf,0); break; } case'5': { strcpy(cmd_buf,"bootm 50008000"); run_command(cmd_buf,0); break; } case'q': return; default: printf("command not foundn"); break; } } } U_BOOT_CMD( menu,5,1,do_menu, "menu - manipulate BMP image datan", "menu long help: TNND mu you" ); 3. 添加到Makefile中 最后在common/Makefile中添加一行 COBJS+=cmd_menu.o 注意:在组合命令时,如NAND write不知道烧写的大小怎么办? 没关系,有变量 $(filesize), 只要加上这个任何问题都不再困难。 上面的50008000是从SD卡启动时,要读到的内存地址; 若是从Nand Flash启动要把上面的50008000改为C0008000 二。 测试一下 2.1 sd卡 a. 因为都是从SD卡和第一个分区读数据,所以将SD卡分区时第一个分区格式必须为FAT32. b. 编译u-boot.bin: 代码要用光盘中带的u-boot,这个支持从NAND Flash 启动 make forlinx_nand_ram256_config, 后生成 u-boot.bin 复制到SD卡的第一个分区, 并改名为 u-boot.bin_nand(要与u-boot代码中的名称保持一致) c. 编译zImage 代码要用光盘中带的linux-3.0.1, make后生成zImage 复制到SD卡的第一个分区, 并改名为 zImage_nand(要与u-boot代码中的名称保持一致) d. 生成rootfs.yaffs 可以用光盘中带的FileSystem-Yaffs2.tar.gz, 解压后,嫌太大,把不需要的删掉, 里面的busybox好像是动态链接的,自己编个静态的busybox,最后生成rootfs.yaffs sudo /opt/6410/4.3.2/bin/mkyaffs2image-nand2g FileSystem-Yaffs2 rootfs.yaffs NAND Flash 是2G的所以要用命令 mkyaffs2image-nand2g 将rootfs.yaffs复制到SD卡的第一个分区(要与u-boot代码中的名称保持一致) 注意: 如果不确定rootfs本身有没有问题,可以先从nfs启动,看是否正常。 如果正常再用mkyaffs2image-nand2g,做成yaffs镜像烧入到NAND Flash中。 亲爱的卡友们,如果看完文章之后还是有疑惑或不懂的地方,请联系我们,自己去理解或猜答案是件很累的事,请把最麻烦的事情交给我们来处理,术业有专攻,闻道有先后,深圳市雷龙发展专注存储行业13年,专业提供小容量存储解决方案。
「博文连载」cortex-a8 uboot系列:第一章 uboot引入
一、 Uboot的入口和出口
入口:开机自动启动
出口:启动内核
二、 Uboot的工作方式
Uboot的本质就是一个裸机程序,和在裸机下开发出来得到的xxx.bin并没有本质区别。唯一的区别就是uboot比较大,一般在180k-400k之间。并且功能比较复杂。
Uboot本身是一个开源项目,由若干个.c文件和.h文件组成,配置编译之后会生产一个uboot.bin,这就是uboot这个裸机程序的镜像文件。然后这个镜像文件被合理的烧录到启动介质中去拿给soc启动。也就是说uboot在没有运行时表现为uboot.bin,一般放在启动介质中。
Uboot运行时会被加载到内存中然后一条一条指令的拿给CPU执行。
Uboot的shell命令界面
Uboot启动后,会提供一个shell界面,可以进行人机交互
该shell和操作系统下的shell是很像的,只是命令集不一样
掌握uboot使用的2个关键点:命令和环境变量
Uboot启动后大部分时间和工作都是在shell下完成的(例如uboot要部署系统要在shell下输命令、要设置环境变量也得在命令行下,要启动内核也是在命令行下)
命令就是uboot的shell中可以识别的各种命令。Uboot中有几十个命令,其中有一些是常用另一些不常用(还可以自己给uboot添加命令)
Uboot的环境变量和操作系统的环境变量工作原理和方式几乎完全相同。Uboot在设计时借用了操作系统的环境变量。
环境变量就是运行时的配置属性。系统的全部环境变量是系统内置的。系统可以通过读取环境来指导程序的运行,这样的设计的好处是灵活,使程序更改运行方式,可以不用去修改程序代码进行编译运行,只要修改相应的环境变量就可以了。
Uboot命令
有些命令可以简化,如print和printenv,set和setenv
有些命令要带参数,如setenv后面要带参数
命令中的双引号,uboot的有些命令非常长,为了告诉uboot这个较长的而且中间是有空格的字符串是命令的一个参数
如 set name “weiqi 7777 lujun”,设置一个变量name,变量值是weiqi 7777 lujun
有一些命令是一个命令族(譬如movi)
命令族就是好多个命令的开头都是同一个命令关键字的,但是后面的参数不一样,这个命令的功能和作用也不同。
同一个命令族中的所有的命令都有极大的关联,譬如movi开头的命令族都和moviNand(EMMC,iNand)操作有关。
1. Uboot常见命令
1) printenv/print
Print命令不带参数,作用是打出系统中所有的环境变量
setenv/set
设置(添加/更改)环境变量
后面带一个参数,表示删除这个环境变量
后面带两个参数,表示添加/修改这个环境变量的值为第二个参数
saveenv/save
保存环境变量,将内存中的环境变量的值同步保存到FLASH中环境变量的分区。
环境变量的保存是整体的覆盖保存,也就是说内存中所有的环境变量都会整体的将FLASH中环境变量分区中原来的内容整体覆盖。
ping
开发板网络测试命令,和PC机的命令使用一样
注意,开发板和电脑对插上网线,先ping通主机windows,不过windows的ip地址是本地连接的ip地址,可以将该地址更改为192.168.1.10。确定开发板中uboot的环境变量,ethaddr(随便设置),netmask(255.255.255.0),ipaddr(表示当前开发板的ip地址,要设置为同PC一个网段,可以设置为192.168.1.77)
如果是使用路由器,开发板和PC都连上路由器,只需要将开发板的ipaddr设置和PC同一个网段即可
tftp
下载指令,从网络下载文件到内存中
使用格式 tftf 内存地址 文件名字
如 tftp 0x30000000 zImage-qt 将服务器上zImage-qt文件下载到开发板的内存0x30000000上
要使uboot的serverip为tftp服务器的ip地址
6) movi
SD卡/iNand/EMMC操作指令
movi是一个命令集,有很多子命令
movi指令都是movi read和movi write一组,movi read从iNand中读取内容到内存中,movi write将内存数据写入到iNand中
movi read u-boot 0x30000000,把iNand中的u-boot分区读出到内存的0x30000000起始的位置处。(uboot代码将iNand分成了很多分区,每个分区有地址范围和分区名,uboot程序操作中可以直接使用地址来操作iNand分区,也可以使用分区名来操作分区)
nand
操作nandflash,和movi操作基本一样,不过九鼎的uboot中去掉了这个指令
8) 内存操作指令:mm,mw,md
内存不像FLASH有分区,只有地址,操作时注意不能越界。
Uboot是一个裸机程序,不像操作系统会由系统整体管理所有内存,系统负责分配和管理,系统会保证内存不会越界。但是裸机程序uboot并不管理内存,内存是随便可以使用,所以使用uboot时,内存地址要使用正确。
md,显示内存区域数据, .b 以字节显示, .w 以半字显示, .l 以字显示
mw,memory write将数据写入到内存
mm,memory modify,修改内存中的某一块,地址会自动增加。修改结束,输入y结束。
9) 启动内核指令:bootm,go
Uboot的终极目标就是启动内核,启动内核在uboot中表现为一个指令,uboot命令行中调用这个指令就会启动内核
差别,bootm启动内核同时给内核传参,而go命令启动内核不传参。
go命令内部其实就是一个函数指针指向一个内存地址然后直接调用那个函数。所以go命令的实质就是PC直接跳转到一个内存地址去运行。go命令可以在uboot中执行裸机程序。
环境变量
环境变量有2份,一份在FLASH中,另一份在内存中。uboot开机时一次性从FLASH中读取全部环境变量到内存中作为环境变量的初始值,然后使用过程中是用内存中这一份,用户可以用saveenv命令将内存中的环境变量保存到FLASH的环境变量,实现更新。
环境变量在uboot中是用字符串表示的。uboot按照字符匹配的方式来区分各个环境变量。
变量
自动运行倒数时间:bootdelay
ipaddr,开发板的本地IP地址
serverip是开发板通过tftp指令去tftp服务器下载文件时的tftp服务器的IP地址
gatewayip是开发板的本地网关地址
netmask是子网掩码
ethaddr是开发板的本地网卡的MAC地址
bootcmd,启动运行命令
uboot启动后,会进行倒数bootdelay时间,如果中间没有被打断,uboot会自动执行启动命令。
uboot开机自动启动时机就是在内部执行了bootcmd这个环境变量的值所对应的命令集。
bootargs,uboot给kernel传递的参数
linux内核启动时可以接收uboot给他传递的启动参数,这些启动参数是uboot和内核约定好的形式、内容,linux内核在这些启动参数的指导下完成启动过程。这样的设计是为了灵活,为了内核在不重新编译的情况下可以用不同的方式启动
在uboot的环境变量中设置bootargs,然后bootm命令启动内核是会自动将bootargs传给内核
bootargs=console=ttySAC2,115200 root=/dev/mmcblk0p2 rw init=/linuxrc rootfstype=ext3
console=ttySAC2,115200,控制台使用串口2,波特率115200。
root=/dev/mmcblk0p2 rw 根文件系统在SD卡端口0设置(iNand)第二分区,可读可写
init=/linuxrc linux的进程1(init进程)的路径
rootfstype=ext3 指根文件系统的类型为ext3
d) 内核传参非常重要,内核移植,如果传值不正确,会造成内核启动不起来
七、 Uboot对FLASH和DDR的管理
1. 对FLASH的分区
a) 所谓分区,也就是对FLASH进行分块管理
b) Uboot中没有操作系统,因此需要自己对FLASH进行管理,所以需要实现对FLASH进行分区界定,有了这个界定,在部署系统是按照分区界定方法来部署,uboot和kernel的软件也是按照这个分区界定来工作。
c) 分区方法不是固定的,是可以变动的,但是在一个移植中必须先设计好,然后使用统一的分区。标准:
l Uboot:必须从FLASH起始地址开始存放(也许是扇区0,也许是扇区1,也许是其他,取决于Soc启动设计,如210从扇区1开始),uboot的分区大小必须保证uboot能够放下,一般设置为512KB或1MB。
l 环境变量: 环境变量分区一般紧贴着uboot存放,大小为32KB或者更多
l Kernel: 可以紧贴环境变量,大小一般为3MB或5MB或其他。
l 根文件系统: 紧贴着kernel,大小得看情况
l 自由分区: kernel启动后,将自由分区挂在到rootfs下使用
2. 对DDR的分区
DDR的分区和FLASH的分区不同,因为FLASH是掉电存在的,而DDR是掉电丢失的,所以DDR是每次系统运行时开开始部署的。
内存的分区,主要是在linux启动之前进行分区,linux内核启动后内核的内存管理模块会接管整个内存空间,这个时候我们就不需要管理了。
内存分区关键就在于内存中那一块用来干什么必须分配好,以避免各个不同功能使用了同一块内存造成的数据影响。
招聘信息
相关问答
ubootmlospl是什么东西?当选择从nandflash启动时,spl为第一阶段,copy前4K到iram,然后完成初始化ddr后,重定位,跳转到ddr,copy剩余uboot到ddr,开始启动第二阶段。当选择从nandfl...
Uboot 设备树如何移植?移植方法:你只能通过NandFlash控制器访问NandFlash,即是只要知道Nand控制器的寄存器地址即可。NandFlash不是一个RamLike的器件。Uboot放入nand中,在nand...
飞凌开发板6410 uboot 什么命令擦除nandflash啊?sferase+起始地址+擦查长度在uboot下输入help可以查看所有的命令sferase+起始地址+擦查长度在uboot下输入help可以查看所有的命令