??上一節(jié)S3C2440移植uboot之裁剪和修改默認(rèn)參數(shù)裁剪了uboot,修改了默認(rèn)的參數(shù),這一節(jié)開始制作yaffs映像以及補(bǔ)丁文件
燒寫文件系統(tǒng)
??嘗試使用如下命令燒寫JFFS2文件系統(tǒng)
tftp 30000000 fs_mini_mdev.jffs2
nand erase.part rootfs
nand write.jffs2 30000000 0x00260000 5b89a8
??修改啟動參數(shù)
set bootargs console=ttySAC0 root=/dev/mtdblock3 rootfstype=jffs2
??燒寫JFFS2文件系統(tǒng)沒問題
??嘗試使用如下命令燒寫YAFFS文件系統(tǒng)
tftp 30000000 fs_mini_mdev.yaffs2
nand erase.part rootfs
nand write.yaffs 30000000 260000 889bc0
??報(bào)錯(cuò)

??搜索.yaffs ,發(fā)現(xiàn)位于Cmd_nand.c文件中。

??缺少了相關(guān)宏定義
??在smdk2440.h中添加如下代碼
#define CONFIG_CMD_NAND_YAFFS
??使用如下命令重新編譯燒寫
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
燒寫一瞬間完成,不太正常。而且最后啟動文件系統(tǒng)報(bào)錯(cuò)如下

通過對比燒寫的fs_mini_mdev_yaffs2文件內(nèi)容和nand dump 260000顯示的內(nèi)容,發(fā)現(xiàn)OOB區(qū)的內(nèi)容不同。

分析源碼
??首先,每個(gè)命令都會對應(yīng)一個(gè)文件,比如nand命令對應(yīng)的common/cmd_nand.c
??而我們使用nand命令時(shí),便會進(jìn)入do_nand()函數(shù),位于common/cmd_nand.c
??(1)do_nand()函數(shù)代碼如下所示:
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 ... ... ,則進(jìn)入該判斷
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);
//進(jìn)入nand_write_skip_bad,燒寫
#endif
... ...
}
??所以需要在smdk2440.h里,添加CONFIG_CMD_NAND_YAFFS宏定義.
??(2)然后進(jìn)入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) //需要燒寫的塊數(shù)
{
#ifdef CONFIG_CMD_NAND_YAFFS
if (flags & WITH_YAFFS_OOB)
{
... ...
ops.mode = MTD_OOB_AUTO; //這里需要修改
... ...
for (page = 0; page < pages; page++) //for循環(huán)燒寫每一頁
{
... ...
rval = nand->write_oob(nand, offset, &ops); //調(diào)用nand_write_oob()函數(shù)燒寫OOB
if (!rval) //這里需要修改
break; //燒寫失敗,退出for循環(huán)
offset += pagesize;
p_buffer += pagesize_oob;
}
}
... ...
}
??將上面if (!need_skip && !(flags & WITH_DROP_FFS))改為if (!need_skip && !(flags & WITH_DROP_FFS) &&!(flags & WITH_YAFFS_OOB))
??因?yàn)楸苊廨斎雗and write.yaffs時(shí),直接進(jìn)入該判斷,然后不執(zhí)行下面的while (left_to_write > 0) 語句
??將上面的MTD_OOB_AUTO改為MTD_OOB_RAW (表示支持燒寫OOB數(shù)據(jù),用來存放yaffs參數(shù))
??因?yàn)镸TD_OOB_AUTO,使自動填入OOB,不填入yaffs文件里的數(shù)據(jù),從而啟動不了內(nèi)核
??將上面if (!rval) 改為if (rval)
??因?yàn)閚and->write_oob()函數(shù)里面,燒寫正確時(shí),是返回的一個(gè)非整數(shù).
??然后使用nand dump 260000,與yaffs文件對比,可以看到OOB已經(jīng)燒寫成功

??對于64B的OOB而言,數(shù)據(jù)定義如下所示:
??bit0:表示該塊的數(shù)據(jù)是否為壞,若為0xFF表示好的,0x00則是壞的 (一塊=64頁)
??bit1:暫時(shí)沒用到
??bit2~39:表示用來存放oob數(shù)據(jù),若是yaffs文件,則會存放yaffs參數(shù),所以才要修改1.2小節(jié)的代碼
??bit40~63:存放ecc校驗(yàn)值,該頁的每256B字節(jié),就會生成3字節(jié)數(shù)據(jù)存放到ecc里
具體參考nand_oob_64全局結(jié)構(gòu)體變量
燒寫yaffs試驗(yàn)
tftp 30000000 fs_mini_mdev.yaffs2
nand erase.part rootfs
nand write.yaffs 30000000 260000 $filesize
//文件系統(tǒng)太大,所以輸入$filesize,來根據(jù)文件系統(tǒng)真正大小來燒寫
tftp 30000000 fs_mini_mdev.jffs2
boot
(PS:若啟動文件系統(tǒng)失敗,考慮下環(huán)境變量,OOB,內(nèi)核是否正確)
啟動成功

使用part制作補(bǔ)丁
??打補(bǔ)丁之前,首先需要清除make后的編譯文件,以及自己編譯出的反匯編文件等
make distclean //清除生成的所有文件
rm u-boot.dis
cd ..
mv u-boot-2012.04.01 u-boot-2012.04.01_new //重新命名
tar -xjf u-boot-2012.04.01.tar.bz2 //創(chuàng)建原文件
diff -urN u-boot-2012.04.01 u-boot-2012.04.01_new > u-boot-2012.04.01_new.patch //生成補(bǔ)丁
|