在C51系统上实现YAFFS文件系统

发布者:RainbowJoy最新更新时间:2007-01-18 来源: 单片机及嵌入式系统应用关键字:NAND  存储  嵌入式 手机看文章 扫描二维码
随时随地手机看文章
随着NAND Flash存储器作为大容量数据存储介质的普及,基于NAND闪存的文件系统YAFFS(Yet Another Flash File System)正逐渐被应用到各种嵌入式系统中。本文将详细阐述YAFFS文件系统在C51系统上的实现过程。

1 NAND Flash的特点

非易失性闪速存储器Flash具有速度快、成本低、密度大的特点,被广泛应用于嵌入式系统中。Flash存储器主要有NOR和NAND两种类型。NOR型比较适合存储程序代码;NAND型则可用作大容量数据存储。NAND闪存的存储单元为块和页。本文使用的Samsung公司的K9F5608包括2 048块,每一块又包括32页,一页大小为528字节,依次分为2个256字节的数据区,最后是16字节的备用空间。

K9F5608具有以下特点: 以页为单位进行读/写操作,而擦除操作以块为单位,读、写和擦除操作均通过命令完成;不能字节擦除,在每次改写操作之前需要先擦除一整块;出厂时有一定比例的坏块存在;每一块的擦除次数有限,为10万次左右[1]。

2 YAFFS文件系统简介

YAFFS是第一个专门为NAND Flash存储器设计的嵌入式文件系统,适用于大容量的存储设备;并且是在GPL(General Public License)协议下发布的,可在其网站免费获得源代码。

YAFFS中,文件是以固定大小的数据块进行存储的,块的大小可以是512字节、1 024字节或者2 048字节。这种实现依赖于它能够将一个数据块头和每个数据块关联起来。每个文件(包括目录)都有一个数据块头与之相对应,数据块头中保存了ECC(Error Correction Code)和文件系统的组织信息,用于错误检测和坏块处理。充分考虑了NAND Flash的特点,YAFFS把这个数据块头存储在Flash的16字节备用空间中。当文件系统被挂载时,只须扫描存储器的备用空间就能将文件系统信息读入内存,并且驻留在内存中,不仅加快了文件系统的加载速度,也提高了文件的访问速度,但是增加了内存的消耗。

为了在节省内存的同时提高文件数据块的查找速度,YAFFS利用更高效的映射结构把文件位置映射到物理位置。文件的数据段被组织成树型结构,这个树型结构具有32字节的节点,每个内部节点都包括8个指向其他节点的指针,叶节点包括16个2字节的指向物理地址的指针。YAFFS在文件进行改写时总是先写入新的数据块,然后将旧的数据块从文件中删除。这样即使在修改文件时意外掉电,丢失的也只是这一次修改数据的最小写入单位,从而实现了掉电保护,保证了数据完整性。

结合贪心算法的高效性和随机选择的平均性,YAFFS实现了兼顾损耗平均和减小系统开销的目的。当满足特定的小概率条件时,就会尝试随机选择一个可回收的页面;而在其他情况下,则使用贪心算法来回收最“脏”的块[2]。

YAFFS文件系统是按层次结构设计的,分成以下4部分: yaffs_guts.c,文件系统的主要算法,这部分代码完全是用可移植的C语言编写的;yaffs_fs.c,Linux VFS层的接口;NAND 接口,yaffs_guts 和NAND 内存访问函数之间的包装层,例如调用Linux mtd 层或者RAM模拟层;可移植函数,服务的包装函数。最重要的一点是,为了获得更好的移植性,YAFFS提供直接调用的模式,这才使得我们有机会来实现YAFFS文件系统在C51系统上的移植。

3 移植过程

  可在http://www.aleph1.co.uk/cgi-bin/viewcvs.cgi/获得direct源码,包括以下几个文件及其头文件。

  ◆ yaffscfg.c: 设置各种设备参数和系统参数。
  ◆ yaffsfs.c: 主要实现直接调用的接口函数,如打开文件、写文件和关闭文件等。 使用时在应用程序中包含其头文件即可。
  ◆ yaffs_flashif.c: NAND Flash操作函数接口,就是直接对存储器操作的底层函数。为了测试,此文件中用RAMDISK模拟的方法实现了对Flash存储器的操作。实际应用中,需要修改其中对Flash硬件操作函数的定义,包括yflash_EraseBlockInNAND()、yflash_WriteChunkToNAND()、yflash_ReadChunkFromNAND()和yflash_InitialiseNAND()。
  ◆ yaffs_guts.c: YAFFS文件系统的主要实现算法。
  ◆ nand_ecc.c:: ECC算法。
  ◆ yaffs_ramdisk.c:: RAMDISK支持代码。
  ◆ yaffs_fileem.c: 用主机上的一个文件来模拟Flash存储器,仅用于测试。
  ◆ dtest.c:: 直接调用文件系统的测试函数。

获得源码以后,移植的过程可以分为2步:①? 根据自己的需要进行裁减;②? 将代码向C51风格转化。

3.1 裁减

YAFFS是一个功能强大的文件系统,考虑到C51系统的程序代码存储器和RAM资源都很有限,而应用中可能不需要某些文件操作的功能,所以有必要对这个文件系统进行裁减。裁减包括代码裁减和数据结构的修改。

首先,将用来测试的yaffs_ramdisk.c、yaffs_ramdisk.h、yaffs_fileem.c和interface.h这几个文件去掉,并在yaffscfg.c加上#include yaffs_flashif.h。

本系统中,只是对K9F5608中的3个数据库文件进行读/写,一级目录足够,单用户不存在操作权限问题,简单的文件存储不涉及连接(Linux类操作系统文件间的关系)问题,所以可在系统中删除与目录操作、操作权限以及文件连接相关的操作函数。

在yaffsfs.c及其头文件中包括(省略yaffs_前缀): readlink(), DumpDir(), readdir(), opendir(), lstat(), stat(), freespace(), chmod(), mkdir(), rename(),link(), closedir(), FollowLink(), fstat(), listclear(), fchmod(),sylink()和mknod()。

在yaffs_guts.c及其头文件中包括(省略yaffs_前缀): Renameobject(), mknodedirectoty(), mknodSymLink(), mknodSpecial(), Link(), GetAttributes(), GetSymLinkAlias(), root(), LostNFound(), GutsTest(), DumpObject(), GetNumberofFreeChunk(), GetObjectLInkCount()和GetEquivalentObject()。

然后根据自己的需要进行数据结构修改,与上文提到的目录操作、操作权限以及文件连接相关的数据结构(如Uid、Gid、nlink等)对我们来说就没有意义了,因此需要修改相关的数据结构。为了节省内存,还要修改一些宏定义的数据常量,例如同时在运行的句柄数目和文件名的最大长度等。

裁减工作最好能在一台装有Linux操作系统的机器上进行,可以边裁减,边利用模拟方式来检查是否能实现自己所需的功能。

3.2 向C51风格转化

YAFFS文件系统是在Linux环境下利用开发用户程序的C语言开发的。它与C51是有些差别的,主要有:

  ◇ C51不支持__inline__函数修饰符,可以将其宏定义为空;
  ◇ u8、u16、u32都需要重新宏定义成为C51的数据类型unsigned char、unsigned int和unsigned long;
  ◇ off_t定义为long。

在YAFFS源代码中有用data和bit作为变量的,而在C51中这些都是关键词,须将其替换。

在YAFFS源代码中yaffs_Device结构体的定义中使用了带参数的函数指针,通过调用该指针指向的函数来对Flash硬件进行操作,而C51中通过寄存器函数指针来调用函数不能传递实际参数,除非所得参数可在寄存器间传递。因此这里将yaffs_Device结构体定义中的函数指针去掉,而直接调用yaffs_if.c中的Flash接口函数。

向C51风格转化时,最好是在Keil集成开发环境中一边修改,一边编译,发现错误后再进行修改。当编译器提示有多个错误时,要从第一个错误开始修改;可能前面的错误修改完毕,后面的错误就不再是错误了。

4 C51系统下的YAFFS使用实例

下面的程序代码是设计中的对DBF数据库文件操作的关键代码。

yaffs_StartUp();//设置一些参数,包括各分区在Flash中的起始块和结束块的地址、预留块数等
yaffs_mount("/flash"); //YAFFS支持多个分区,这里选择挂载/flash分区新建文件db1.dbf
f=yaffs_open("/flash/db1.dbf", O_CREAT,S_IREAD | S_IWRITE);
yaffs_close(f);//关闭文件db1.dbf
f=yaffs_open("/flash/db1.dbf", O_RDWR,0);//以读/写的方式打开文件db1.dbf
r=yaffs_write(f,"hello",5);//向文件写数据
yaffs_lseek(f,2,SEEK_SET);//移动文件读/写指针
r=yaffs_write(f,"world",5);
r=yaffs_lseek(f,0,SEEK_SET);
r=yaffs_read(f,buffer,10);//从文件读取数据
r=yaffs_close(f);//关闭文件db1.dbf
r=yaffs_unlink("/flash/db1.dbf");//删除文件db1.dbf

可见, YAFFS的接口函数的使用方法与标准C语言中对文件的操作函数十分相似,简单易用。

5 总结

YAFFS文件系统是第一个专门为NAND Flash存储器编写的嵌入式文件系统。它实现了掉电保护、疲劳均衡和有效的垃圾回收,与JFFS相比占用资源更少,运行速度更快;与FAT相比,更适合用于管理NAND Flash数据存储器。在C51系统中如果需要实现Flash文件系统,那么移植YAFFS是个不错的选择,但是毕竟它是一个在32位机的Linux下开发出来的,要让它能与8位机的C51风格完美结合,还需要进一步的努力。

参考文献
[1] Samsung公司.? K9F5608DataSheet. 20031217.
[2] 毛勇强,黄光明. YAFFS文件系统在嵌入式Linux上的实现. 电子设计应用,2006(3).

关键字:NAND  存储  嵌入式 引用地址:在C51系统上实现YAFFS文件系统

上一篇:SEPTNY256型单片机开关电源及其应用
下一篇:AT89C51单片机在无线数据传输中的应用

推荐阅读最新更新时间:2024-05-13 18:16

基于总线的嵌入式高速图像通信系统设计
   1 引言   随着现代的图形采集技术发展迅速,各种基于ISA,PCI,USB1.1等总线的图形采集卡速度已经不能满足用户的需求,而采用 USB2.0以后就可以解决这个传输速度上的瓶颈,USB2.0的速度是480Mbits/s,完全可以满足图像采集、传输以及后续处理的要求。系统中采用 DSP+CPLD的硬件设计方案,采用现场可编程芯片 CPLD及两片 SRAM构成的图像采集和存储系统,可以根据不同的需要进行现场编程,具有通用性好、价格相对便宜,易于系统调试,升级等特点。系统中 CPLD选择的型号是 ALTERA公司的MAX7000系列低功耗芯片EPM7128A。片外大容量 SRAM是DSP与 CPLD的联系桥梁,系统设计也
[嵌入式]
STM32F107VC的嵌入式远程监控终端设计
摘要:针对处于偏远地区或恶劣环境中的无人值守设备的运行状态的监控问题,本文应用嵌入式技术以及工业以太网技术设计了一种数据采集与监控系统的远程监控终端。围绕核心处理器STM32F107VC构建了终端的硬件系统,介绍了网络接口设计、输入/输出接口设计和本地存储电路设计。监控终端软件基于ARM公司的RL-ARM中间件,实现了多任务并行的网络实时通信、基于WebServer的远程配置服务和临时存储现场数据的文件系统。应用结果表明,系统可靠性高、实时性强,有效降低了远程监控系统成本。 引言 数据采集与监控系统是以计算机为基础的分布式控制系统与电力自动化监控系统,广泛应用于电力、冶金、石油、化工等诸多领域。数据采集与监控系统依靠其数量庞大
[单片机]
STM32F107VC的<font color='red'>嵌入式</font>远程监控终端设计
数据存储在程序存储器(flash)空间的定义
(1)flash常量: #include //须增加的头文件 const prog_uchar FlashConst = 3; //定义uchar型的常量n定义在flash里(flash常量) unsigned char RamVar; //定义无符号整型变量(Ram变量) RamVar = pgm_read_byte(&FlashConst); //读取flash常量到ram变量 (2)flash一维数据: #include const prog_uchar s = { 1, 2, 3, 4, 5 }; unsigned char RamVar; //定义无符号整型变量(Ram变量) RamVar =
[单片机]
嵌入式存储器以及BootLoader的一些总结
我在大学的时候很少关心单片机内部存储器的结构及应用,只是大概的知道RAM和ROM的区别,甚至只是知道程序下载到ROM中就能运行了。其他的如何工作程序如何启动的就不了了知了。因此在接触嵌入式的时候就会出现很多盲区。在此本人将工作三个多月以来接触到的一些关于嵌入式新的理解分享给大家,有什么错误的地方在所难免,希望大家给予指正。 RAM和ROM在单片机中的相关应用 ROM和RAM指的都是半导体存储器,ROM是Read Only Memory的缩写,RAM是Random Access Memory的缩写。ROM在系统停止供电的时候仍然可以保持数据,而RAM通常都是在掉电之后就丢失数据,典型的RAM就是计算机的内存。 RAM有两大类,一种
[单片机]
图像采集综合评估的嵌入式指纹识别系统(一)
  由于具有其持久性和唯一性,指纹被广泛应用在个人身份识别系统。指纹识别在全世界范围内的广泛应用,不仅为防范犯罪,同时也作为处理个人事务和信息安全的一个关键技术。   在指纹识别技术领域,欧美国家在研究和开发中处于领先地位。美国ZK Software在1992年就发布了ZKFinger 1.0版,2008年发布了ZKTime7.0、ADMS解决方案、指纹锁、集成具有美国联邦政府推出的FIPS201认证的Lumidign和Secugen指纹仪;法国Segam公司,是安全解决方案的市场领导者之一,有遍布5大洲超过85个分公司和分支机构,其生产的集成系统部署在100多个国家和地区。亚洲在指纹识别技术上较为先进的公司有:韩国现代、朝鲜的培
[电源管理]
图像采集综合评估的<font color='red'>嵌入式</font>指纹识别系统(一)
基于STM32的嵌入式双目图像采集系统设计
1 引言 随着图像处理技术及嵌入式系统的发展,利用嵌入式系统进行图像处理,已使如视频监控、视频电话和视频会议等应用成为可能。嵌入式系统上进行图像采集则是实现这些应用的前提[1~2]。 双目立体视觉通常由两个摄像头从两个不同的角度,同时获取外界物体的两幅图像,或由单独一个摄像头在不同时间、不同位置获取外界物体的两幅图像,并基于视差的原理来获取外界物体的三维几何信息,复现外界物体的形状和位置。 目前已有很多方案实现在嵌入式平台上的图像采集。本文基于嵌入式的图像采集系统选择了意法半导体(ST)公司生产的STM32F103ZET6芯片为主控芯片,FIFO结构的AL422B芯片实现图像数据缓存,SD卡实现图像存储以及四线制电阻触摸屏实
[单片机]
基于STM32的<font color='red'>嵌入式</font>双目图像采集系统设计
在线ARM仿真器知识--嵌入式系统设计师必备
   本文提供了一些关于在线 ARM 仿真器的信息,以及给作为嵌入式系统设计师的你带来的好处。根据你的需要,你将在产品开发中对开发工具作出更恰当的选择。   一、嵌入式产品的开发周期   典型的嵌入式微控制器开发项目的第一个阶段是用C编译器从源程序生成目标代码,生成的目标代码将包括物理地址和一些调试信息。目前代码可以用软件模拟器、目标Monitor或在线仿真器来执行和调试。软件模拟器是在PC机或工作站平台上,以其CPU(如x86)及其系统资源来模拟目标CPU(如P51XA),并执行用户的目标代码;而目标Monitor则是将生成的目标代码下载到用户目标板的程序存储器中,并在下载的代码中增加一个Monitor任务软件,用
[单片机]
当整车架构加速变革,为什么SSD必须进化?
自1885年卡尔·本茨研制出世界第一辆三轮马车式汽车至今,汽车产业已经正式走过了139年。现如今,汽车已经从简单的交通代步工具转变为高新技术集聚化的智能平台。 百余年发展过程中,行业对于汽车安全性、舒适性和动力性的追求永无止境。这几年,行业通过统合汽车电子电气架构,从而迅速提升整车性能,也就是从域控(Domain)到区域控制(Zonal)。 事实上,想要实现架构的跨越,不光需要强大的SoC芯片,更需要强力的内存和存储芯片。美光作为汽车内存和存储芯片的领头者,日前推出4150AT SSD,它作为全球首款四端口SSD,提供多达四个片上系统(SoC)接口,可实现软件定义智能汽车的集中存储。 帮助汽车架构向集中化靠拢
[汽车电子]
当整车架构加速变革,为什么SSD必须进化?
小广播
最新应用文章
换一换 更多 相关热搜器件

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 安防电子 医疗电子 工业控制

词云: 1 2 3 4 5 6 7 8 9 10

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved