1、尝试烧写文件系统
烧写内核:
tftp 30000000 uImage_4.3
nand erase.part kernel
nand write 30000000 kernel
烧写JFFS2文件系统:
tftp 30000000 fs_mini_mdev.jffs2
nand erase.part rootfs
nand write.jffs2 30000000 0x00260000 5b89a8
set bootargs console=ttyAC0 root=/dev/mtdblock3 rootfstype=jffs2
烧写YAFFS:
tftp 30000000 fs_mini_mdev.yaffs2
nand erase.part rootfs
nand write.yaffs 30000000 260000 889bc0
烧写JFFS2文件系统没问题,烧写YAFFS文件系统时报错:
搜索.yaffs
,位于Cmd_nand.c
文件中:
缺少了相关宏定义,在include\configs\smdk2440.h
中添加如下代码:
使用如下命令重新编译烧写:
tftp 30000000 u-boot_new.bin; protect off all; erase 0 3ffff; cp.b 30000000 0 40000
tftp 30000000 fs_mini_mdev.yaffs2
nand erase.part rootfs
nand write.yaffs 30000000 260000 889bc0
启动文件系统报错如下:
通过对比烧写的
fs_mini_mdev_yaffs2
文件内容和nand dump 260000
显示的内容,发现OOB
区的内容不同。
2、分析源码查找原因
每个命令都会对应一个文件,比如nand
命令对应的common/cmd_nand.c
,而我们使用nand
命令时便会进入do_nand()
函数,位于common/cmd_nand.c
:
1、do_nand()
函数代码如下所示:
int do_nand(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])
{
... ...
if (strncmp(cmd, "read", 4) == 0 || strncmp(cmd, "write", 5) == 0)
{
... ...
#ifdef CONFIG_CMD_NAND_YAFFS //是否支持YAFFS烧写else if (!strcmp(s, ".yaffs"))
{ //若是nand write.yaffs ... ... ,则进入该判断
if (read) {
printf("Unknown nand command suffix '%s'.\n", s);
return 1; }
ret = nand_write_skip_bad(nand, off, &rwsize, (u_char *)addr, WITH_YAFFS_OOB);
//进入nand_write_skip_bad,烧写
#endif
... ...
}
所以需要在smdk2440.h
里添加CONFIG_CMD_NAND_YAFFS
宏定义。
2、然后进入nand_write_skip_bad()
,位于drivers/mtd/nand/nand_util.c
int nand_write_skip_bad(nand_info_t *nand, loff_t offset, size_t *length,u_char *buffer, int flags)
{
... ...
if (!need_skip && !(flags & WITH_DROP_FFS)) //这里需要修改
{
rval = nand_write (nand, offset, length, buffer); //正常拷贝,不考虑OOB问题
if (rval == 0)
return 0; //拷贝完后,return
*length = 0;
printf ("NAND write to offset %llx failed %d\n",offset, rval);
return rval;
}
while (left_to_write > 0) //需要烧写的块数
{
#ifdef CONFIG_CMD_NAND_YAFFS
if (flags & WITH_YAFFS_OOB)
{
... ...
ops.mode = MTD_OOB_AUTO; //这里需要修改
... ...
for (page = 0; page < pages; page++) //for循环烧写每一页
{
... ...
rval = nand->write_oob(nand, offset, &ops); //调用nand_write_oob()函数烧写OOB
if (!rval) //这里需要修改
break; //烧写失败,退出for循环
offset += pagesize;
p_buffer += pagesize_oob;
}
}
... ...
}
- 将上面
if (!need_skip && !(flags & WITH_DROP_FFS))
改为if (!need_skip && !(flags & WITH_DROP_FFS) &&!(flags & WITH_YAFFS_OOB))
,因为避免输入nand write.yaffs
时直接进入该判断,不执行下面的while (left_to_write > 0)
语句
- 将上面的
MTD_OOB_AUTO
改为MTD_OOB_RAW
(表示支持烧写OOB数据,用来存放yaffs参数),因为MTD_OOB_AUTO
使自动填入OOB,不填入yaffs文件里的数据,从而启动不了内核
- 将上面
if (!rval)
改为if (rval)
,因为nand->write_oob()
函数里面烧写正确时,是返回的一个非整数。
然后使用nand dump 260000
与yaffs
文件对比可以看到OOB已经烧写成功:
对于64B的OOB而言,数据定义如下所示:
- bit0:表示该块的数据是否为坏,若为0xFF表示好的,0x00则是坏的 (一块=64页)
- bit1:暂时没用到
- bit2~39:表示用来存放oob数据,若是yaffs文件,则会存放yaffs参数,所以才要修改1.2小节的代码
- bit40~63:存放ecc校验值,该页的每256B字节,就会生成3字节数据存放到ecc里
具体参考nand_oob_64全局结构体变量
3、烧写yaffs文件系统
tftp 30000000 fs_mini_mdev.yaffs2
nand erase.part rootfs
nand write.yaffs 30000000 260000 $filesize
//文件系统太大,所以输入$filesize,来根据文件系统真正大小来烧写
tftp 30000000 fs_mini_mdev.jffs2
boot
若启动文件系统失败,考虑下环境变量、OOB、内核是否正确。
4、制作补丁
打补丁之前需要清除make后的编译文件、以及自己编译出的反汇编文件等:
制作补丁文件:
make distclean
rm u-boot.dis
cd ../
mv u-boot-2012.04.01 u-boot-2012.04.01_zdy
tar xjf u-boot-2012.04.01.tar.bz2
diff -urN u-boot-2012.04.01 u-boot-2012.04.01_zdy > u-boot-2012.04.01_zdy.patch
打补丁:
cd u-boot-2012.04.01
patch -p1 < ../u-boot-2012.04.01_zdy.patch
编译:
make distclean
make smdk2440_config
make
更新UBOOT:
tftp 30000000 u-boot.bin
protect off all
erase 0 3ffff
cp.b 30000000 0 40000
从nor烧写uboot到nand:
nand erase 0 40000
nand write 0 0 40000
nand read 30000000 0 40000 //比较是否写入nand
cmp.b 0 30000000 40000
烧写内核:
tftp 30000000 uImage_4.3
nand erase.part kernel
nand write 30000000 kernel
烧写YAFFS
tftp 30000000 fs_mini_mdev.yaffs2
nand erase.part rootfs
nand write.yaffs 30000000 260000 889bc0