前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >6.移植uboot-支持yaffs烧写

6.移植uboot-支持yaffs烧写

作者头像
诺谦
发布2019-05-24 20:30:33
1.1K0
发布2019-05-24 20:30:33
举报
文章被收录于专栏:Linux驱动Linux驱动

在上一章,裁剪uboot以及分区后,本章主要使uboot支持yaffs以及制作补丁


1. 修改uboot支持yaffs

首先,每个命令都会对应一个文件,比如nand命令对应的common/cmd_nand.c

而我们使用nand命令时,便会进入do_nand()函数,位于common/cmd_nand.c

1.1do_nand()函数代码如下所示:

代码语言:javascript
复制
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宏定义.

1.2然后进入nand_write_skip_bad(),位于drivers/mtd/nand/nand_util.c

代码语言:javascript
复制
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;
            }
      }
    ... ...
}

1)将上面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) 语句

2)将上面的MTD_OOB_AUTO改为MTD_OOB_RAW (表示支持烧写OOB数据,用来存放yaffs参数)

因为MTD_OOB_AUTO,使自动填入OOB,不填入yaffs文件里的数据,从而启动不了内核

3)将上面第14行if (!rval) 改为if (rval)

因为nand->write_oob()函数里面,烧写正确时,是返回的一个非整数.

1.3然后使用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全局结构体变量

1.4 然后烧写yaffs试验

代码语言:javascript
复制
tftp 30000000 fs_mini_mdev.yaffs2

nand erase.part rootfs

nand write.yaffs 30000000 260000  $filesize    
      //文件系统太大,所以输入$filesize,来根据文件系统真正大小来烧写 

tftp 30000000 fs_mini_mdev.jffs2

boot

(PS:若启动文件系统失败,考虑下环境变量,OOB,内核是否正确)

2. 使用part制作补丁

打补丁之前,首先需要清除make后的编译文件,以及自己编译出的反汇编文件等

步骤如下:

代码语言:javascript
复制
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                       //创建原文件
diff -urN u-boot-2012.04.01 u-boot-2012.04.01_new > u-boot-2012.04.01_new.patch  //生成补丁

移植uboot就此结束了,接下来学习:

1.移植3.4内核-分析内核启动过程,重新分区,烧写jffs2文件系统

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018-01-19 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1.移植3.4内核-分析内核启动过程,重新分区,烧写jffs2文件系统
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档