在应用程序中操作NorFlash
相对于操作NandFlash,操作NorFlash相对简单,因为基本不需要考虑坏块,NorFlash也没有OOB区域,也跟ECC没有一毛钱关系。它的读写擦除相对容易。
int dealwithnor()
{ // glob_t mtdbuf;
struct mtd_info_user mtd;
struct erase_info_user erase;
int blocks = ;
int i = ; //用于控制擦除的块的个数
int k = ;
int written = ; //已写入的字节数,只初始化一次
unsigned int size = StateOfImage.st_size; //应该是镜像的实际大小,因为内存中大于镜像的空间的内容不可预知
unsigned int result = ;
unsigned int DevNum = ; //设备的数量
unsigned int StartDev = ; //从第startDev开始擦除
char DevName[] = {};
unsigned int AllSize = ;
#define MAXPARTITIONS 40
struct DeviceInfo
{
int fd;
char dir[];
uint32_t size; // Total size of the MTD
uint32_t erasesize; }DevInfo[MAXPARTITIONS];//用来存储设备信息 bzero(DevInfo, sizeof(struct DeviceInfo)); /* 这是一种方法,但是有一个缺点,当mtd设备大于10个是,通过glob搜索出来的结果
会出问题,下面采用会采用第二种方法 */
// if(searchmtd(&mtdbuf) != 0){ // DEBUG("Sorry! Can not find mtd device\n");
// return 1; //返回大于零的数,表示升级失败
// }
// else
// {
// int fd;
//
// DEBUG("find %d mtd devices \n",mtdbuf.gl_pathc);
// DevNum = mtdbuf.gl_pathc; // for(i=0; i<DevNum; i++)
// {
// fd = safeOpen (mtdbuf.gl_pathv[i],O_SYNC | O_RDWR);
// if(fd < 0)
// {
// printf("failt to open\n");
// return 1;
// }
// bzero(&mtd, sizeof(struct mtd_info_user));
// if (ioctl(fd, MEMGETINFO,&mtd) < 0)
// {
// DEBUG("ioctl(): %m\n");
// DEBUG("This doesn't seem to be a valid MTD flash device!\n");
// return 1;
// }
//
// strcpy(DevInfo[i].dir, mtdbuf.gl_pathv[i]);
// DevInfo[i].fd = fd;
// DevInfo[i].size = mtd.size;
// DevInfo[i].erasesize = mtd.erasesize;
// } // globfree(&mtdbuf);
// for(i=0; i<DevNum; i++)
// {
// printf("\n\tinfo of %s\n",DevInfo[i].dir);
// printf("%s.fd: %d\n",DevInfo[i].dir, DevInfo[i].fd);
// printf("%s.size: %d\n",DevInfo[i].dir,DevInfo[i].size);
// printf("%s.erasesize: %d\n",DevInfo[i].dir,DevInfo[i].erasesize);
// }
// } /* 下面是第二种方法,这种方法克服了第一种方法的缺陷,不受mtd设备数量的限制 */ for(i=; i<MAXPARTITIONS; i++)
{
int fd;
sprintf(DevName, "%s%d", "/dev/mtd",i); if((fd = open (DevName,O_SYNC | O_RDWR)) > )
{
bzero(&mtd, sizeof(struct mtd_info_user));
if (ioctl(fd, MEMGETINFO,&mtd) < )
{
DEBUG("ioctl(): %m\n");
DEBUG("This doesn't seem to be a valid MTD flash device!\n");
return ;
}
strcpy(DevInfo[i].dir, DevName);
DevInfo[i].fd = fd;
DevInfo[i].size = mtd.size;
DevInfo[i].erasesize = mtd.erasesize;
}
else
{
DevNum = i;
break;
} } for(i=; i<DevNum; i++)
{
printf("\n\tinfo of %s\n",DevInfo[i].dir);
printf("%s.fd: %d\n",DevInfo[i].dir, DevInfo[i].fd);
printf("%s.size: %d\n",DevInfo[i].dir,DevInfo[i].size);
printf("%s.erasesize: %d\n",DevInfo[i].dir,DevInfo[i].erasesize);
AllSize += DevInfo[i].size;
} if(AllSize < StateOfImage.st_size)
{
DEBUG("ERROR!! all device size is less than ImageSize\n");
return ;
} for(i=StartDev; i<DevNum; i++)
{ /**
* 先进行擦除操作
*/
int j = ;
g_AllImgSize = DevInfo[i].size;
g_AllImgWrite = ;
erase.start = ;
blocks = DevInfo[i].size / mtd.erasesize; //计算要擦除的块的个数
erase.length = mtd.erasesize;
printf ("\nbegin to erase block %s\n", DevInfo[i].dir);
for (j= ; j <= blocks; j++)
{
fprintf(stderr, "\rErasing blocks: %d/%d (%d%%)", j, blocks, (int)PERCENTAGE (j, blocks));
g_percentage = * ((float)g_AllImgWrite / g_AllImgSize);
if (ioctl(DevInfo[i].fd, MEMERASE, &erase) < )
{
DEBUG("\n");
DEBUG("While erasing blocks 0x%.8x-0x%.8x on %s\n",\
(unsigned int) erase.start, (unsigned int) (erase.start + erase.length), DevInfo[i].dir);
/*return "Error while erasing blocks";*/
return ;
}
g_AllImgWrite += erase.length;
erase.start += mtd.erasesize;
}
printf("\n\rErased blocks: %d/%d (100%%)\n", blocks, blocks); /**
* 再进行写操作
*/
printf ("\nbegin to write block %s\n\n", DevInfo[i].dir);
g_AllImgWrite = ;
k = BUFSIZE;
while (size)
{
if (size < BUFSIZE)
{
k = size;
}
printf("\033[1A");
printf("\r%s usage: %dk/%dk (%d%%)\n",\
DevInfo[i].dir, KB (g_AllImgWrite + k), KB (DevInfo[i].size), (int)PERCENTAGE (g_AllImgWrite + k, DevInfo[i].size));
fprintf(stderr, "Writing data: %dk/%ldk (%d%%)", KB (written + k), KB (StateOfImage.st_size), (int)PERCENTAGE (written + k, StateOfImage.st_size));
result = write(DevInfo[i].fd, &upPack[written], k);
if (k != result)
{
DEBUG ("\n");
if (result < )
{
DEBUG("While writing data to 0x%.8x-0x%.8x on %s\n", written, written + k, DevInfo[i].dir);
return ;
}
DEBUG("Short write count returned while writing to x%.8x-0x%.8x on %s: %d/%d bytes written to flash\n", \
written,written + k, DevInfo[i].dir, written + result, DevInfo[i].size);
return ;
} written += k;
size -= k; g_AllImgWrite += k;
if(g_AllImgWrite >= DevInfo[i].size)
{
g_AllImgWrite = ;
printf("\n");
break;
} }
printf("Wrote %d / %ldk bytes\n", written, (unsigned long int)(StateOfImage.st_size)); } munmap(upPack, UPGRADE_SHM_SIZE);
for(i=; i<DevNum; i++)
{
close (DevInfo[i].fd);
printf("%s is closed!\n",DevInfo[i].dir);
} return ;
}
在应用程序中操作NorFlash的更多相关文章
- c#中操作word文档-四、对象模型
转自:http://blog.csdn.net/ruby97/article/details/7406806 Word对象模型 (.Net Perspective) 本文主要针对在Visual St ...
- 在应用程序中实现对NandFlash的操作
以TC58NVG2S3ETA00 为例: 下面是它的一些物理参数: 图一 图二 图三 图四 图五 图6-0 图6-1 说明一下,在图6-1中中间的那个布局表可以看做是实际的NandFlash一页数据的 ...
- 【转】《深入理解计算机系统》C程序中常见的内存操作有关的典型编程错误
原文地址:http://blog.csdn.net/slvher/article/details/9150597 对C/C++程序员来说,内存管理是个不小的挑战,绝对值得慎之又慎,否则让由上万行代码构 ...
- 《深入理解计算机系统》C程序中常见的内存操作有关的典型编程错误
对C/C++程序员来说,内存管理是个不小的挑战,绝对值得慎之又慎,否则让由上万行代码构成的模块跑起来后才出现内存崩溃,是很让人痛苦的.因为崩溃的位置在时间和空间上,通常是在距真正的错误源一段距离之后才 ...
- PHP程序中使用PDO对象实现对数据库的增删改查操作的示例代码
PHP程序中使用PDO对象实现对数据库的增删改查操作(PHP+smarty) dbconn.php <?php //------------------------使用PDO方式连接数据库文件- ...
- LINQ To SQL在N层应用程序中的CUD操作、批量删除、批量更新
原文:LINQ To SQL在N层应用程序中的CUD操作.批量删除.批量更新 0. 说明 Linq to Sql,以下简称L2S. 以下文中所指的两层和三层结构,分别如下图所示: 准确的说,这里 ...
- websocketj--随时随地在Web浏览器中操作你的服务端程序
0 - 有没有觉得Linux标准终端界面输入输出枯燥无味? 1 - 什么?vmstat命令的输出数据不直观?有没有想过能够可视化该命令的输出? 2 - 尝试过用浏览器操作Windows中的cmd吗? ...
- 在Python程序中的进程操作,multiprocess.Process模块
在python程序中的进程操作 之前我们已经了解了很多进程相关的理论知识,了解进程是什么应该不再困难了,刚刚我们已经了解了,运行中的程序就是一个进程.所有的进程都是通过它的父进程来创建的.因此,运行起 ...
- Python程序中的进程操作
之前我们已经了解了很多进程相关的理论知识,了解进程是什么应该不再困难了,刚刚我们已经了解了,运行中的程序就是一个进程.所有的进程都是通过它的父进程来创建的.因此,运行起来的python程序也是一个进程 ...
随机推荐
- NServiceBus教程-NServiceBus和WCF
WCF中缺少的最主要的事情是发布/订阅,但为什么你必须建立它自己吗?NServiceBus,你把它弄出来. 下一个重要的事情是容错.异常导致WCF代理休息,需要"刷新"的代码,但调 ...
- [翻译]创建ASP.NET WebApi RESTful 服务(8)
本章讨论创建安全的WebApi服务,到目前为止,我们实现的API都是基于未加密的HTTP协议,大家都知道在Web中传递身份信息必须通过HTTPS,接下来我们来实现这一过程. 使用HTTPS 其实可以通 ...
- tomcat的 JNDI 配置
tomcat的conf/server.xml 配置 尽量用简单版 <Context path="/cas" docBase="D:\YC\zqV7\cas\WebR ...
- Redis总结(五)缓存雪崩和缓存穿透等问题
前面讲过一些redis 缓存的使用和数据持久化.感兴趣的朋友可以看看之前的文章,http://www.cnblogs.com/zhangweizhong/category/771056.html .今 ...
- c++中的signal机制
简介 signal是为了解决类之间通信的问题而出现的,更深入的原因是面向对象讲究封装,但是封装必然导致类之间沟通困难,但是使用接口的方式又太重量级--需要写很多代码,而且会导致接口爆炸 比如你需要把一 ...
- EasyUI ComboBox默认值
combobox数据加载完后设置默认值 $('#ck').combobox({ url: '/External/GetAllCk', valueField: 'Ddbh', textField: 'D ...
- SqlServer数据维护
现有两个表:Code和CodeCategory Code表: CodeCategory表: 现要把Code表中的数据如实维护一份数据,但是要设PlantID字段值为2,而ID要按规则自增并且要与Pla ...
- VS2010开发环境最佳字体及配色[转]
从地址:http://www.dev-club.net/xiangxixinxi/42010072906150537/201103010518006.html 获取的,整理如下: 环境:VS2010字 ...
- jquery表格伸展
1:效果图: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org ...
- c++中获取代码运行时间
include<ctime> time_t begin,end; begin=clock(); { .............//被测试的代码 } end=clock(); cout ...