如何实现从SD卡更新NK?——(已经实现)
作者:wogoyixikexie@gliet 2008-12-04
在前段时间,songtitan牛人在论坛说了一下使用SD卡更新NK的方法。如下:
http://topic.youkuaiyun.com/u/20081009/17/4E0F5E66-C7A0-43D2-B33F-14E132280F70.html
在CE下和在bootloader下都可以实现更新NK.
1 在CE下
可以直接用文件系统API读出NK.bin,放到内存buffer中 ,再通过DEVICEIOCTL的方式来调用nand驱动的接一些接口来直接对nand flash进行写操作。
好处:跳过boot section,FAT/FAT32文件系统的解析;对内存大小无要求
缺点:NAND驱动需要暴露读写的IOCTL,必须进入OS才能更新OS,如果OS挂了的话,无法更新。
2 bootloader下
直接对SD CONTROLLER操作来对SD卡进行读写,解析FAT/FAT32格式(fatfs_open() ),找到NK.BIN,并将其读取到内存(该内存在硬件上一般是ddr fatfs_open(0, "xip.nb0", O_RDONLY, &downFile);其中downfile 是内存中开辟的一块空间 )中,然后可以将其从ddr中取出烧写到nand flash中。
好处:只要bootloader不挂,就可以对OS进行更新;
缺点:需要自己解析BOOT SECION,FAT/FAT32文件系统;需要至少和NK.BIN大小相同的剩余RAM
如果BSP本身没有支持bootloader下的SD更新nk,自己研究还是需要不少工作量的,最好的捷径就是能“弄到”别的平台的SD更新代码,改动就只是SD HOST controller部分。
=================现在我的工作就是详细弄出怎么实现这个东西==========================
公司要求在bootloader阶段实现SD更新NK ,并且说要实现用触摸屏操作 ,我觉得这个在裸奔 阶段实现触摸屏菜单太难了,简直是不可能的事情。如果非要实现的话就一定要加上GUI,那bootloader搞的这么大,就没有什么意义了。我现在觉得 有一种方法可行:就是放弃使用SD卡更新NK的办法,而是在flash里面备份一个NK,更新的时候就把备份NK放到启动的NK上。这样实现起来 bootloader修改简单,在wince起来之后实现也不困难。现在的flash都已经1G以上了,那点容量牺牲不成问题,并且SD卡也腾出了空间 ——哈哈,好,就模仿ghost做一个嵌入式还原精灵吧。
====================================================================================
——经过几天的思索,我觉得在flash上做备份有点不妥当,因为flash有时候不稳定,很容易出问题。现在还是决定采用系统起来后更新NK的方法。很荣幸在这里找了个很好的老帖子——--------------------------------------wince下将flash中的内容全部读出来
http://topic.youkuaiyun.com/u/20080530/10/8A86B2C8-1623-4AB9-ACA7-D5DE24994CF6.html 内容经典,摘抄下来。
--------------------------------------------------------------------------
硬件:2410 ,64M nandflash,64M RAM
我现在在想在wince下将nandflash中的内容从地址0开始,到nandflash结束,全部读出来,
该怎么做!
就好像是给台式电脑做GHOST备份一样,不同的是NANDFLASH为空的地方我们也要读取!
紧急,谢谢
--------------------------------------
要看你的文件系统是否支持.因为文件系统只能识别被分区的地方,那没有分区的空白区域是无法通过WINCE API访问到的.一般flash都有专用的读写器直接复制芯片内容的,也就是生产时用来做母片 的工具。
--------------------------------------
恐怕只能自己写一个简单的flash driver,从头到尾读一遍了。wince的fat文件系统肯定是不能保证按flash的物理地址顺序读的。
--------------------------------------
NAND上一层是DISK接口了,不会有相应的API供你读取所有内容的。
如果能修改驱动,那么就让驱动开放一个特殊的IOContrl Code给你,输入地址范围,输出是将地址范围内的数据全部放到指定的buf指针地址去。Nand flash的驱动,修改FMD_OEMIoControl函数
-----------------------------------------------------------------
是标准的流接口,其实还是用DSK驱动就 行,只是你自己设计一个原来没有的IOCONTROL_CODE,通过Device的IOControl()函数传下去时,在NAND FLASH的驱动中的IOControl直接判断,如果满足,就将指定范围的数据都读到指定的地址空间。
9楼的有个小错误,NAND驱动的读函数里已经做了ECC校验了,通过FMD_NAND_READ这样的函数读出来的数据是不需要再ECC校验的。
------------------------------------------------------------------------
是的,需要的
=========================================抄贴结束======================================
——这几天,我首先在学习EVC读写SD卡程序,刚开始以为可以像PC一样malloc几十M的内存,结果试验发现当读400K写字板的时候已经那个非常慢,看来一次性读写30M的想法是行不通的。今天终于分段读写30M内核文件 成功,下面是关键代码。
-
C/C++ code
-
- //我把30M的文件读出来,再存放到一个文件,已经完全可行。该结贴了,是修改flash驱动的时候了。晚上回去结贴。
- //==================从文件中分段读出数据,每次30K ,读1024次,共30M====================
- for (unsigned int count=0;count<READCOUNT;count++)
- {
- SetFilePointer(hFile, m_FilePointer, NULL, FILE_BEGIN); /* 移动文件指针到文件开头 */
- if (ret == 0xFFFFFFFF)
- {
- MessageBox(_T("将文件指针移至文件指定处失败!" ));
- return ;
- }
- pcharbuff=(char *)malloc(READSIZE); //每次申请30K的内存
- ret = ReadFile(hFile, pcharbuff, READSIZE, &actlen, NULL); /* 从文件中分段读出数据,每次30K */
- if (ret == FALSE)
- MessageBox(_T("读文件失败!" ));
- //--------------------write read file to wogo.nb0-----------------------
- //--读出30K,就写30K
- SetFilePointer(hTestFile, m_FilePointer, NULL, FILE_BEGIN); /* 移动文件指针到文件开头 */
- if (ret == 0xFFFFFFFF)
- {
- MessageBox(_T("将文件指针移至文件指定处失败!" ));
- return ;
- }
- ret = WriteFile(hTestFile, pcharbuff, READSIZE, &writelen, NULL); /* 将数据写入文件中 */
- if (ret == FALSE)
- MessageBox(_T("写文件失败!" ));
- free((void *)pcharbuff);//释放内存
- m_FilePointer=m_FilePointer+READSIZE;//每次文件指针下移30K
- //if (count==1023)
- //{
- // MessageBox(_T("读文件成功!"));
- //}
- }
-
使用的读写函数都一样,要想把二进制的数据读出来显示,不能使用显示文本文件的方法。
-
=============================flash驱动修改部分==============================
http://blog.youkuaiyun.com/nanjianhui/archive/2008/03/19/2196466.aspx 微软MVP的文章
——应用程序直接控制flash
===========================采用wince运行下更新NK的方法成功===========================
——现在公布要点。
一、在FMD中增加下面代码
- //------------------------------------------------------------------------------
- //---------------------------应用程序和flash直接通信-------希望大家看了有启发---
- //------------------------------------------------------------------------------
- static DWORD g_dwStartSector=2*64;
- BOOL FMD_OEMIoControl(DWORD dwIoControlCode, PBYTE pInBuf, DWORD nInBufSize, PBYTE pOutBuf, DWORD nOutBufSize, PDWORD pBytesReturned)
- {
- switch (dwIoControlCode)
- {
- case 0x77777777:
- for (unsigned int i=2;i<=242;i++)//30M NK 占用240 block
- FMD_EraseBlock(i); //擦除NK所占区域30M
- RETAILMSG(1, (TEXT("FMD_OEMIoControl: recognized IOCTL.FMD_EraseBlock (0x%x)./r/n" ), dwIoControlCode));
- break ;
- case 0x12345678:
- //写入NK所占区域——这个要看eboot中的函数才知道具体起始block
- //RETAILMSG(1, (TEXT("FMD_OEMIoControl: recognized IOCTL.FMD_WriteSector(0x%x)./r/n"), dwIoControlCode));
- RETAILMSG(1, (TEXT("FMD_OEMIoControl: recognized IOCTL.FMD_WriteSector(0x%x)./r/n" ), nInBufSize));
- RETAILMSG(1, (TEXT("FMD_OEMIoControl: nInBufSize=%d./r/n" ), nInBufSize));
- //for(unsigned int k=0;k<nInBufSize;k++)
- // {
- // FMD_WriteSector(g_dwStartSector,pInBuf,NULL,1);
- // g_dwStartSector=g_dwStartSector+1;
- // }
- //FMD_WriteSector函数的参数很关键,pInBuf是应用程序传入,并且大小只能是一个page
- //FMD_WriteSector的代码推测出,pInBuf只能是一个page
- FMD_WriteSector(g_dwStartSector,pInBuf,NULL,1);
- if (g_dwStartSector<=242*64)
- g_dwStartSector=g_dwStartSector+1;
- break ;
- default :
- RETAILMSG(1, (TEXT("FMD_OEMIoControl: unrecognized IOCTL (0x%x)./r/n" ), dwIoControlCode));
- return (FALSE);
- }
- return TRUE;
- }
转载请标明:作者wogoyixikexie@gliet .桂林电子科技大学一系科协。如有错误,希望能够留言指出;如果你有更加好的方法,也请在博客后面留言,我会感激你的批评和分享。