说来惭愧,又很长时间没更新文章了,原本这篇文章能够春节过来搞定的,结果春节回到公司,大客户一直要求抓紧时间设计DM3730平台的720P宽动态低照度相机产品,和另外两款多网口的DM3730产品的样机,南京老客户也在催DM3730 3G工业级板子样机,刚搞定两个老客户,又有两个新客户找咱们设计DM3730新产品,一个是3G+WIFI 4CIF视频分析产品,另外一个客户人脸识别车载方面的产品,整个桐烨科技都很忙。如今又不敢招人,毕竟冷酷的现实不得不面对:这个掌握无敌的宇宙真理国度经济不断下行,搞得第一季度咱们也没有什么大单子,毕竟给客户作新样机是没什么钱赚的,都是培训客户。虽然如今是春天,但经济的寒冬才来临,通货膨胀形成公司运营成本飞涨,总感受很累,有时本身很想静心钻进本公司新产品的研发去,可是客户那边有些关键的软件问题还得本人亲自解决,累也得坚持下去。linux
上篇文章介绍DM3730的xloader的移植,这边介绍DM3730第2 boot阶段移植:u-boot-2010.06的移植,其实本人已经写过DM6446、DM36X平台UBOOT的文章,可是因为平台和版本不同,因此这里单独一篇,对有些人来讲可能写得比较肤浅,请不要见笑,目的是完善本身的DM3730开发攻略,让有兴趣在这方面开发的生手提供一点点帮助。这个u-boot-2010.06源码存放的位置在《DAVINCIDM3730开发攻略——DVSDK4_03和双核CODEC机制介绍.doc》已经介绍。下面咱们开始DM3730进行u-boot-2010.06的移植工做。android
1、裁剪和交叉编译环境变量设置算法
u-boot-2010.06-psp04.02.00.07.sdk更名u-boot-2010.06,简洁;vim
1、总的Makefile修改api
对u-boot-2010.06/Makefile进行修改,服务器
屏蔽第139和140行:网络
# examples/standalone 架构
# examples/apiide
而后把u-boot-2010.06/examples文件夹删除掉,简洁;工具
第159行的修改成:
CROSS_COMPILE = arm-arago-linux-gnueabi-
这个就是DVSDK4——03自带的交叉编译工具,关于这个交叉编译工具,TI作得很全面了,提供很 多免费的应用程序LIB等等。
第244行修改成:
#LIBS += api/libapi.a
屏蔽,而后把u-boot-2010.06/api,简洁,和项目不相关;
2、涉及到交叉编译环境的u-boot-2010.06\arch\arm\config.mk第24行改成:
CROSS_COMPILE =arm-arago-linux-gnueabi-
二:删除多余文件夹
这个DVSDK4_03自带的u-boot-2010.06-psp04.02.00.07.sdk很占空间,也比较乱。很不便于开发和备份,咱们就是要把68多M字节的变成20M字节简化版(bz2或gz压缩的通常才3.4M的大小)。
注意,下面介绍的有些目录下是删除文件夹,有些是删除文件,不要搞混;
1、删除u-boot-2010.06/arch/除了arm文件夹外其余全部文件夹;
其余非arm平台去掉,占空间,啰嗦。
2、删除u-boot-2010.06/arch/arm除arm_cortexa8外的其余全部文件夹;
DM3730属于arm cortex-A8架构(前几篇文章写成COTEX-A8了,系本人简写错误,有时本人直接跟客户就是ARM-A8的称呼)。DM6446、DM6467、DM6467T、DM365、DM368都是比较差劲的ARM926EJS架构。
3、删除u-boot-2010.06/arch/arm/arm_cortexa8/除omap3文件外,删除其余文件夹,
就是删除mx51,s5pc1xx,ti81xx文件夹,DM3730/DM3725芯片属于OMAP3平台,这里边还包括OMAP3530、2010年拿来作MOTO 智能手机的OMAP3630等芯片。
4、保留u-boot-2010.06/arch/arm/include/asm/下面的三个文件夹:
arch-omap,arch-omap3,proc-armv,同这个目录下的那些.h文件不要删除,这一点要注意一下。
5、删除u-boot-2010.06/board/除ti外的其余全部文件夹;
UBOOT版本愈来愈高,新的厂商的板子不断加进去,不少,很烦,很占空间,对咱们专一开发某个平台很差,因此咱们接把不相关的板子平台去掉。
6、删除u-boot-2010.06/board/ti除evm外的其余全部文件夹和文件;
同时TI文件夹也保留好几家第三方的板子,好比最典型的beagle xM,还有ti 的DM8148 DM8168的。咱们直接使用ti 的evm板,或者参考其余ARM 学习板有关UBOOT移植的移植,从新起个我的的名字或公司名字,修改对应Makefile等等,太多文章介绍了。
7、删除u-boot-2010.06/include/configs除omap3_evm.h其余文件(注意这里是文件);
这个目录下的头文件和平台有关,DM3730使用的是omap3_evm.h,其余头文件能够干掉。
8、删除u-boot-2010.06/ nand_spl和onenand_ipl:
目前在DAVINCI平台还没发觉用到onenand的东西,能够去掉;
好了,通过上面的删除工做,这个u-boot-2010.06已经很简洁了,压缩备份很方便,固然还能够删除更简洁的,这里就不详细说了。
三:编译u-boot-2010.06
在u-boot-2010.06/下生成build-u-boot-all.sh和build-u-boot-tmp.sh的两个文件:
build-u-boot-all.sh内容为:
exportPATH=$PATH:/home/davinci/dm3730/dvsdk4_03/linux-devkit/bin:
make distclean
makeomap3_evm_config
make
cp u-boot.bindm3730_uboot.bin
cp u-boot.bin/tftpboot/dm3730_uboot.bin
上面的会直接读取总的Makefile第3000多行的:
omap3_evm_config : unconfig
@$(MKCONFIG) $(@:_config=) armarm_cortexa8 evm ti omap3
编译参数,这样会自动去选择编译,u-boot-2010.06\arch\arm\cpu\arm_cortexa8\omap3\里边的源码和u-boot-2010.06\board\ti\evm\里边的源码,而且指定include u-boot-2010.06/include/configs/omap3_evm.h的平台头文件。
build-u-boot-tmp.sh内容为
exportPATH=$PATH:/home/davinci/dm3730/dvsdk4_03/linux-devkit/bin:
make
cp u-boot.bindm3730_uboot.bin
cp u-boot.bin/tftpboot/dm3730_uboot.bin
对这两个sh文件进行:
chmod +x build-u-boot-all.sh
和
chmod +x build-u-boot-tmp.sh
而后第一次或者每次作了make distclean动做后,都有先执行./ build-u-boot-all.sh,进行omap3_evm_config。之后改动改动源码的时候能够不要重复make distclean和make omap3_evm_config,直接使用./build-u-boot-tmp.sh编译就能够了。build-u-boot-all.sh或build-u-boot-tmp.sh文件自动帮你copy文件到主机tftp server对应的目录/tftpboot/。
4、修改移植
上面第三点已经讲明如何连接编译omap3_evm_config ,那么咱们如今能够进行对应的移植和修改源码。
1、针对DM3730芯片,咱们先从u-boot-2010.06/include/configs/omap3_evm.h这个文件修改,让他指向对应的芯片平台;
#defineCONFIG_ARMCORTEXA8 1 /* This is an ARM V7 CPU core */
#defineCONFIG_OMAP 1 /*in a TI OMAP core */
#defineCONFIG_OMAP34XX 1 /* which is a 34XX */
#defineCONFIG_OMAP3430 1 /* which is in a 3430 */
#defineCONFIG_OMAP3_EVM 1 /* working with EVM*/
这个头文件最前面的宏定义讲明了DM3730所属的ARM架构,T也属于TI 公司 OMAP系列当中的OMAP34XX家族的芯片,OMAP3430-à3530--à3630-àDM3730是软件硬件架构一脉相承的芯片系列。咱们选择TI EVM板子模式。
/*
* select serial console configuration
*/
#if 1
#defineCONFIG_CONS_INDEX 3
#defineCONFIG_SYS_NS16550_COM3 OMAP34XX_UART3
#defineCONFIG_SERIAL3 3 /* UART3 on TY OMAP3 EVM */
#else
#defineCONFIG_CONS_INDEX 2
#defineCONFIG_SYS_NS16550_COM2 OMAP34XX_UART2
#defineCONFIG_SERIAL2 2 /* UART2 on TY OMAP3 EVM TEST */
#endif
TI EVM定义的LINUX调试串口指定是UART1,DM3730和DM6446同样一共有3个UART,均可以用来作LINUX软件调试的串口,咱们的板子也和其余公司同样使用UART3作为调试串口,而不是TI的EVM指定的UART1,由于上篇文章《DAVINCIDM3730开发攻略——xload-1.51移植》提到过DM3730的BOOT MODE,有种BOOT模式使用了SD-àNAND-àUART3,因此咱们这里使用UART3。而往下看找到有关UBOOT串口调试的BOOTDELAY:
/* Environmentinformation */
#defineCONFIG_BOOTDELAY 1
这个UBOOT的delay等待用户按键调试时间太长了,把10秒改为1秒,有些产品为了加快BOOT 时间,卖出去的产品不须要调试,也能够直接改成0。
继续往下看,
/* commands toinclude */
#include<config_cmd_default.h>
这里边定义了不少CONFIG_CMD_XXX的功能,若是让UBOOT编译出来的文件比较小,可使用#undef把某些不经常使用的功能屏蔽掉。
#defineCONFIG_CMD_EXT2 /* EXT2 Support */
#defineCONFIG_CMD_FAT /* FAT support */
#defineCONFIG_CMD_JFFS2 /* JFFS2 Support */
#define CONFIG_CMD_MTDPARTS /* Enable MTD parts commands */
#define CONFIG_MTD_DEVICE /* needed for mtdparts commands */
#define MTDIDS_DEFAULT "nand0=nand"
#define MTDPARTS_DEFAULT "mtdparts=nand:256k(x-loader),"\
"256k(u-boot-env),1280k(u-boot),"\
"5m(kernel),-(fs)"
上面红色字体是本人添加修改的,使能在UBOOT里边使用MTD分区。
同时咱们打算使用256K字节作为u-boot 参数保存空间,偏移地址是x040000的地址,因此omap3_evm.h后面(第320行左右的地方)提到的
#defineONENAND_ENV_OFFSET 0x00040000 /*environment starts here */
#defineSMNAND_ENV_OFFSET 0x00040000 /*environment starts here */
改为上面的0x00040000。
这里首先声明桐烨科技的DM3730板子NAND是512M BYTE,DDR也是512M BYTE,网口使用DM9000,不一样公司的开发板估计有不一样的配置,这个信息很重要,下面NAND分区定义和程序烧写都须要这了解这方面的信息。
而后到看到#define CONFIG_EXTRA_ENV_SETTINGS 的修改:
定义这个CONFIG_EXTRA_ENV_SETTINGS前,须要补充一些NAND 存放XLAOD,UBOOT,KERNEL等等知识,本人在上篇《DAVINCI DM3730开发攻略——xload-1.51移植》提到UBOOT 编译处理的BIN文件存放在NAND FLASH的地方,这里再更详细描述一下:
//0x000000000000-0x000000040000: "X-Loader"(DM3730 xlaod存放在NAND的位置)
//0x000000040000-0x000000080000: "U-Boot Env"(uboot 参数存放在NAND的位置)
//0x000000080000-0x000000200000: "U-Boot"(uboot 自己存放在NAND的位置)
//0x000000200000-0x000000C00000: "Kernel"(kernel存放在NAND的位置)
//0x000000C00000-0x000008400000: "ubifs0"(主要的文件系统UBIFS存放在NAND的位置)
//0x000008400000-0x00000FC00000: "ubifs1"(备用的UBIFS存放在NAND的位置,也能够不要)
//0x00000FC00000-0x000010000000: "user data"(保存一些用户本身定义的数据)
//loadaddr ==0x80300000(TFTP 下载XLAOD,UBOOT,KERNELBIN文件到内存的起始偏移地址)
//ubifs loadaddr== 0x81000000(TFTP 下载比较大的UBIFSBIN文件到内存起始偏移地址)
//u-boot codeaddress == 0x80E80000(UBOOT自己存放到内存运行的偏移地址)
这是咱们修改后的#define CONFIG_EXTRA_ENV_SETTINGS源码:
#define CONFIG_EXTRA_ENV_SETTINGS\
"loadaddr=0x80300000\0" \
"rdaddr=0x81000000\0"\
"console=ttyS2,115200n8\0" \
"mpurate=1000\0" \
"vram=12M\0" \
"dvimode=1280x720MR-16@60\0" \
"defaultdisplay=dvi\0" \
"nandroot=ubi0:rootfs\0" \
"nandrootfstype=ubifs\0" \
"nandargs=setenv bootargsconsole=${console} rw mem=120M@0x80000000 mem=256M@0xA0000000 " \
"mpurate=${mpurate} " \
"vram=${vram} " \
"ip=192.168.1.188:192.168.1.252:192.168.1.1:255.255.255.0:tgt:eth0:off"\
"omapdss.def_disp=${defaultdisplay}" \
"omapfb.mode=dvi:${dvimode}" \
"ubi.mtd=4 "\
"root=${nandroot} " \
"rootfstype=${nandrootfstype}" \
"init=/initandroidboot.console=ttyS2\0" \
"nfstvargs=setenv bootargsconsole=ttyS2,115200n8 rw mem=120M@0x80000000 mem=256M@0xA0000000 mpurate=1000vram=12M omapfb.mode=tv:720x576@25 omapdss.def_disp=tvip=192.168.1.188:192.168.1.252:192.168.1.1:255.255.255.0:tgt:eth0:offroot=/dev/nfsnfsroot=192.168.1.252:/home/davinci/dm3730/dvsdk4_03/filesystem/dm3730rootfs,nolock\0"\
"nfslcdargs=setenv bootargsconsole=ttyS2,115200n8 rw mem=120M@0x80000000 mem=256M@0xA0000000 mpurate=1000vram=12M omapfb.mode=dvi:${dvimode} omapdss.def_disp=${defaultdisplay}init=/init androidboot.console=ttyS2ip=192.168.1.188:192.168.1.252:192.168.1.1:255.255.255.0:tgt:eth0:offroot=/dev/nfsnfsroot=192.168.1.252:/home/davinci/dm3730/dvsdk4_03/filesystem/dm3730rootfs,nolock\0"\
"tftpboot=tftp 80300000dm3730_kernel.bin; bootm 80300000\0" \
"userboot=nand read ${loadaddr}200000 400000; bootm ${loadaddr}\0" \
"erase_env=nand erase 4000040000\0" \
"eraseall=nand erase\0" \
"updatexload=tftp 80300000dm3730_xload.bin;nand erase 0 40000;nandecc hw;nand write.i 80300000 0${filesize}\0" \
"updateuboot=tftp 80300000dm3730_uboot.bin;nand erase 80000 180000;nandecc sw;nand write.i 80300000 80000${filesize}\0" \
"updatekernel=tftp 80300000dm3730_kernel.bin;nand erase 200000 500000;nandecc sw;nand write.i 80300000200000 ${filesize}\0" \
"updaterootfs=tftp 81000000dm3730_ubifs.bin;nand erase C00000 7800000;nandecc sw;nand write.i 81000000C00000 ${filesize}\0" \
"nandboot=echo Booting from nand...; " \
"run nandargs; " \
"nand read ${loadaddr} 200000400000; " \
"bootm ${loadaddr}\0" \
"nfstvboot=echo Booting from nfs...; " \
"run nfstvargs; " \
"nand read ${loadaddr} 200000400000; " \
"bootm ${loadaddr}\0" \
"nfslcdboot=echo Booting from nfs...; " \
"run nfslcdargs; " \
"nand read ${loadaddr} 200000400000; " \
"bootm ${loadaddr}\0" \
" fact_update =nand erase;mmc init;"\
"fatloadmmc 0 80300000 dm3730_xload.bin;nandecc hw;nand write.i 80300000 0 ${filesize};"\
"fatloadmmc 0 80300000 dm3730_uboot.bin;nandecc sw;nand write.i 80300000 80000${filesize}; "\
"fatloadmmc 0 80300000 dm3730_kernel.bin;nandecc sw;nand write.i 80300000 200000${filesize}; "\
"fatloadmmc 0 81000000 dm3730_ubifs.bin;nandecc sw;nand write.i 81000000 C00000${filesize};"
注意:上面源码格式很讲究,特别是( \ “” \0 ; $)等标点符号,mpurate=1000表示跑1G的A8;自从咱们的linux-2.6.32支持ubifs 文件系统后,咱们不再用所谓的JFFS2和YAFFS2了。咱们公司支持512M 的内存并且CS0片选信号指向0x80000000(前段),第2段的片选信号CS1指向0xA0000000(后段)。为何这样分配呢?前面120M和后面256M都是给LINUX系统的,而前段256M-120M=136M是给DSP用的,固然DSP里边还包括CMEM共享内存。这前段256M的内存分配是动态的,你能够分配80M给LINUX,那么DSP和CMEM就可分配更多的空间了。至于DSP里边如何细分,和应用程序使用的loadmodule.sh这个文件如何配合,这里不重点论述,那是在之后的内核移植再说。还有,有些其余开发板公司的内存若是只有256M字节,那么mem=256M@0xA0000000必须去掉,不然内核根本起不来。
上面有不少BOOT方式,好比nandboot,nfstvboot, tftpboot,咱们直接使用run tftpboot就能够经过网络动态下载TFTPSERVER里边的dm3730_kernel.bin,而后使用默认的bootargs UBOOT参数进行NFS调试,或者run nfstvboot,也能够NFS,run nfstvboot表示视频输出是TV输出,这个和内核有关,之后内核移植有个地方有选择的编译;run nfslcdboot表示从LCD接口输出视频,能够跑android安卓文件系统。
有时客户在uboot命令行对bootargs进行设置,使用到这个刚设置好的bootargs,能够经过使用上面的userboot进行启动。
set bootcmd “run userboot”
saveern
一样使用nandboot也可使用setbootcmd “run nandboot”和saveenv的命令实现。
在测试的时候,咱们和主机TFTP配合,直接经过run updatexload去网络升级xlaod,run updateuboot去升级烧写uboot,run updatekernel去烧写内核,固然了,你的tftpboot目录下要有对应的BIN文件,这里就很少说了,之前的DM6446、DM36X开发攻略都提示过。
继续往下看代码:
#defineCONFIG_BOOTCOMMAND "run nandboot"
#defineCONFIG_BOOTARGS \
"console=ttyS2,115200n8rw mem=120M@0x80000000 mem=256M@0xA0000000 mpurate=1000 vram=12Momapfb.mode=dvi:1280x720@60 omapdss.def_disp=lcdip=192.168.1.188:192.168.1.252:192.168.1.1:255.255.255.0:tgt:eth0:offroot=/dev/nfsnfsroot=192.168.1.252:/home/davinci/dm3730/dvsdk4_03/filesystem/dm3730rootfs,nolock"
这个是默认的bootargs的参数,默认使用nandboot。
U-BOOT保存在NAND FLASH的参数,能够在U-BOOT命令行使用pri命令或pritenv命令看看。
最后面在#if defined(CONFIG_CMD_NET)的后面加上:
/* DM9000 */
#defineCONFIG_NET_MULTI 1
#defineCONFIG_NET_RETRY_COUNT 20
#defineCONFIG_DRIVER_DM9000 1
#defineCONFIG_DM9000_BASE 0x2c000000
#defineDM9000_IO CONFIG_DM9000_BASE
#defineDM9000_DATA (CONFIG_DM9000_BASE + 0x400)
#defineCONFIG_DM9000_USE_16BIT 1
#defineCONFIG_DM9000_NO_SROM 1
#undef CONFIG_DM9000_DEBUG
#defineCONFIG_ETHADDR 88:11:22:33:44:77
#defineCONFIG_IPADDR 192.168.1.188
#defineCONFIG_SERVERIP 192.168.1.252
#defineCONFIG_GATEWAYIP 192.168.1.1
#defineCONFIG_NETMASK 255.255.255.0
咱们本身修改的UBOOT和内核可以把MAC (ETHADDR)地址传给内核;
这个MAC正规的申请途径能够GOOGLE一下,而测试调试能够随便定义;
板子的静态IP是定义192.168.1.188,而服务器主机的地址是192.168.1.252,不一样公司有不一样公司的网段和IP地址,用户能够修改。
好了,omap3_evm.h已经修改完毕。
2、修改u-boot-2010.06\arch\arm\cpu\arm_cortexa8\omap3\
这里边没什么好改动的,估计要改动的就是lowlevel_init.S,去配不一样的PLL,这个参考手册见sprugn4q.pdf。而后就是注意修改mem.c里边的gpmc_init()的BOOT模式,到底内核是从MMC读仍是从NAND读到内存。
3、修改u-boot-2010.06\board\ti\evm\
这里边的Evm.h和Evm.c就是对DM3730管脚复用配置再次进行处理,在上篇《DAVINCI DM3730开发攻略——xload-1.51移植》也提到过,DM3730管脚复用比较复杂,有7种不一样的模式,m0~m7,在evm.h代码里边:
/*
* IEN -Input Enable
* IDIS - Input Disable
* PTD -Pull type Down
* PTU -Pull type Up
* DIS -Pull type selection is inactive
* EN -Pull type selection is active
* M0 -Mode 0
* The commented string gives the final muxconfiguration for that pin
*/
#define MUX_EVM()\
这个就说明了你使用的某个管脚是M0默认模式,仍是M4GPIO模式,而GPIO模式是使用IEN(输入)仍是IDIS(输出 ),GPIO管脚上拉PTU仍是下拉PTD,上拉下拉是否要使能DIS和EN。
MUX_VAL这种格式宏定义在u-boot-2010.06\arch\arm\include\asm\arch-omap3\mux.h定义,指向对应的寄存器。
举个例子:
好比I2C3,DM3730一共有4个I2C总线,咱们板子只须要I2C1和I2C2两个总线就够用了,那么多余的I2C3和I2C4能够看成GPIO使用。
MUX_VAL(CP(I2C3_SCL), (IEN | PTU | EN | M0)) /*I2C3_SCL*/\
MUX_VAL(CP(I2C3_SDA), (IEN | PTU | EN | M0)) /*I2C3_SDA*/\
若是是这样的代码,表示你使用I2C3使用M0模式,那这两个管脚使用I2C的模式,而不是GPIO,
MUX_VAL(CP(I2C3_SCL), (IEN | PTU | EN | M4)) /*GPIO184*/\
MUX_VAL(CP(I2C3_SDA), (IEN | PTU | EN | M4)) /*GPIO185*/\
这样的代码表示I2C3被用来作GPIO,IEN表示用做输入,内部上拉电阻,并且上拉电阻使能,
MUX_VAL(CP(I2C3_SCL), (IDIS | PTU | EN | M4)) /*GPIO184*/\
MUX_VAL(CP(I2C3_SDA), (IDIS | PTU | EN | M4)) /*GPIO185*/\
这样的代码表示I2C3被用来作GPIO,IEN表示用做输出,内部上拉电阻,并且上拉电阻使能。
其实也很好理解。咱们在UBOOT详细定义了管脚复用,那么之后在内核编译的时候,首先内核make menuconfig里边要去掉MUX这个选项,这样内核就不必再作一次管脚配置了。
写到这,对DM3730 UBOOT的移植应该有个头绪了,具体的应用,一些BUG,须要具体问题具体分析,不一样的网口芯片,不一样的NAND和DDR芯片有不一样的驱动,这里就很少说了。U-BOOT的移植在ARM平台来讲,大同小异,虽然一个是ARM926,一个是A8,或者A9,A12,A15,但对于软件工程师来讲,若是不涉及到汇编代码的编写,其实都是同样的C语言,同样的UBOOT软件架构。UBOOT的目的无外乎就是把linux内核给跑起来,把一些参数传给内核,板子启动的时候作些初始化的工做,或者一些测试调试工做。真正体现产品的功能价值就是内核和对应的应用程序。
(声明:
桐烨科技DM3730/DM6446的板子和其余公司的开发板不同,特别是DM3730的板子,目前国内好多家公司都只提供ARM端(CORTEX-A8)的应用例子,不多介绍如何添加客户本身的算法到DSP端的例子,有些须要作DSP算法的人贪便宜,结果买这些便宜的板子回去花大量时间来学习,迟迟搞不清楚整个架构,浪费的这些时间难道不是资金吗?咱们桐烨科技的板子都帮你采集好YUV格式的视频图像,并教会你如何把这个原始的图像数据放到DSP端进行处理,而后再教会你如何传处理过的图像数据和参数到ARM端。同时提醒客户还要注意一些冒牌的公司,特别是杭州有家没道德的公司直接拿咱们桐烨科技的DM3730开发板图片放到他们公司网站上,欺骗其余人,咱们桐烨科技历来没有想到让其余公司作代理。)