FAT16:
数据按照其不同的特点和作用大致可分为5部分:MBR区、DBR区、FAT区、DIR区和DATA区,相比fat12多了DBR区
Main boot record: MBR(0--1bdh) 磁盘参数存放
DPT(1beh--1fdh) 磁盘分区表
55,aa 分区结束标志
DBR(Dos Boot Record)是操作系统引导记录区的意思
FAT区(有两个,一个备份):对于fat16,每一个fat项16位,所以可寻址的簇项数为65535(2的16次方)。而其每簇大小不超 过32k,所以其每个分区最大容量为2G。fat32,每一个fat项32位,可寻址簇数目为2的32次方。
DIR区(根目录区):紧接着第二FAT表(即备份的FAT表)之后,记录着根目录下每个文件(目录)的起始单元,文件的属性等。定位文件位置时,操作系统根据DIR中的起始单元,结合FAT表就可以知道文件在硬盘中的具体位置和大小了。
DATA区:实际文件内容存放区。
FAT32:
暂时放在这里,不讨论!
Fatfs:嵌入式fat文件系统,支持fat16,fat32。
包含有ff.h,diskio.h,integer.h,ffconf.h 四个头文件以及ff.c 文件系统实现。当然要实现具体的应用移植,自己要根据diskio.h实现其diskio。c 底层驱动。
diskio.h : 底层驱动头文件
ff.h : 文件系统实现头文件,定义有文件系统所需的数据结构
ff.c : 文件系统的具体实现
如下开始逐个文件加以分析:
integer.h :仅实现数据类型重定义,增加系统的可移植性。
ffconf.h : 文件系统配置---逐个配置,先配置实现一个最小的fat文件系统,下面来分析各配置选项:
#define _FFCONF 8255 //版本号
#define _FS_TINY 0 /* 0:Normal or 1:Tiny */ //在这里与先前版本有些许变化,是通过配置头配置两种不同大小的文件系统 ,这里配置为 0。
#define _FS_READONLY 1 //定义文件系统只读,也就不能写修改,在此定义为 1,这样文件系统会大大缩小,简化学习理解过程。
#define _FS_MINIMIZE 3 /* 0 to 3 */ 这个选项是用于过滤掉一些文件系统功能,为0时是全功能,3是功能实现最小
#define _USE_STRFUNC 0 /* 0:Disable or 1/2:Enable */ 是否使用字符串文件接口,为0,
不使用
#define _USE_MKFS 0 /* 0:Disable or 1:Enable */ 制作文件系统,这个功能实现是还要_FS_READONLY=0
#define _USE_FORWARD 0 /* 0:Disable or 1:Enable */ f_forward function 实现还需_FS_TINY =1
#define _USE_FASTSEEK 0 /* 0:Disable or 1:Enable */ 快速查找功能
#define _CODE_PAGE 936 // 936 - Simplified Chinese GBK (DBCS, OEM, Windows)
#define _USE_LFN 0 /* 0 to 3 */ 0:不使用长文件名
#define _MAX_LFN 255 /* Maximum LFN length to handle (12 to 255) */
#define _LFN_UNICODE 0 /* 0:ANSI/OEM or 1:Unicode */
#define _FS_RPATH 0 /* 0 to 2 */ 不使用相对路径
/*---------------------------------------------------------------------------/
/ Physical Drive Configurations
/----------------------------------------------------------------------------*/
#define _VOLUMES 1
/* Number of volumes (logical drives) to be used. */
#define _MAX_SS 512 /* 512, 1024, 2048 or 4096 */ 512字节每扇区
#define _MULTI_PARTITION 0 /* 0:Single partition or 1:Multiple partition */ 只有一个分区
#define _USE_ERASE 0 /* 0:Disable or 1:Enable */
/* To enable sector erase feature, set _USE_ERASE to 1. */
/*---------------------------------------------------------------------------/
/ System Configurations
/----------------------------------------------------------------------------*/
#define _WORD_ACCESS 0 /* 0 or 1 */ 0: Byte-by-byte access.
/* Include a header file here to define sync object types on the O/S */
/* #include #define _FS_REENTRANT 0 /* 0:Disable or 1:Enable */ 同步选项 #define _FS_TIMEOUT 1000 /* Timeout period in unit of time ticks */ #define _SYNC_t HANDLE /* O/S dependent type of sync object. e.g. HANDLE, OS_EVENT*, ID and etc.. */ #define _FS_SHARE 0 /* 0:Disable or >=1:Enable */ 共享选项 如上已经配置成了一个最小的fat文件系统。 Diskio.h:底层驱动头文件,就一些状态宏的定义和底层驱动函数的申明,看源码一目了然。 实现相应的 diskio.c 。 根据我的配置:只需要根据不同的存储介质实现相应的 disk_initialize 个函数就够了, 我在这里实现上s3c2440上的SD卡驱动: DSTATUS disk_initialize (BYTE nDisk) { return 1; } DSTATUS disk_status (BYTE nDisk) { return 1; } disk_status,disk_read三,DRESULT disk_read (BYTE nDisk, BYTE* b, DWORD c, BYTE d) { return 0; } 编译报错 get_fattime 没实现 : DWORD get_fattime (void) { return 0; } 转自博客,供大家相互交流! FATFS文件系统剖析2: 分析下ff.h和ff.c两个文件。 先来分析ff.h中定义的几个结构体: typedef struct { BYTE fs_type; // 系统类型,为0时系统没有被挂载 BYTE drv; // 物理磁盘驱动号 BYTE csize; // 每簇的扇区数目, 簇是文件数据分配的基本单位 BYTE n_fats; // 文件分配表的数目,一般为两个(一个备份fat表) //Fatfs文件系统应该是:引导扇区,文件分配表2个,根目录区,和数据区 BYTE wflag; // 文件活动窗体是否改动标志,为1要回写 BYTE fsi_flag; // 文件信息回写标志 WORD id; // 文件系统加载id号 WORD n_rootdir; // 根目录区目录项数目(针对FAT12/16,FAT32不使用) #if _MAX_SS != 512 WORD ssize; // 每扇区多少字节 #endif #if _FS_REENTRANT _SYNC_t sobj; // 允许重入,则定义同步对象 #endif #if !_FS_READONLY DWORD last_clust; // 最新分配的簇 DWORD free_clust; // 空闲簇 DWORD fsi_sector; // 文件信息扇区(仅用于FAT32) #endif #if _FS_RPATH DWORD cdir; //使用相对路径,文件系统的当前起始路径 0(root 路径) #endif DWORD n_fatent; //文件分配表占用的扇区 n_fatent=数据簇数目+2 DWORD fsize; //每FAT表有多少个扇区 DWORD fatbase; //文件分配表开始扇区 DWORD dirbase; // 如果是FAT32,根目录开始扇区需要首先得到 DWORD database; // 数据起始扇区 DWORD winsect; // win中当前指定的扇区 BYTE win[_MAX_SS]; //扇区操作缓存 } FATFS; typedef struct { FATFS* fs; // 指向的文件系统 WORD id; // 自身文件系统挂载id号 即 fs->id BYTE flag; // 文件状态 BYTE pad1; //文件在簇里面扇区偏移 (0--fs->csize) DWORD fptr; //文件当前读写指针位置,当文件打开时为0 DWORD fsize; //文件大小(按字节计算) DWORD org_clust; // 文件起始簇 (0 when fsize==0) DWORD curr_clust; // 文件当前操作簇 DWORD dsect; // 文件当前操作扇区 #if !_FS_READONLY DWORD dir_sect; // 包含路径入口的扇区号 BYTE* dir_ptr; // 目录入口指针 #endif #if _USE_FASTSEEK DWORD* cltbl; //指向查找映射表的簇 (null on file open) #endif #if _FS_SHARE UINT lockid; // 文件锁 ID号 (index of file semaphore table) #endif #if !_FS_TINY BYTE buf[_MAX_SS]; // 文件读写缓存 #endif } FIL; typedef struct { FATFS* fs; // 对应的文件系统 WORD id; // 自身文件系统挂载id号 即 fs->id WORD index; // 目前读写索引号 /* Current read/write index number */ DWORD sclust; // 目录表起始簇 (0:Root dir) DWORD clust; // 目前处理的簇 DWORD sect; // 目前簇里对应的扇区 BYTE* dir; //指向当前在win[]中的短文件名入口项/* Pointer to the current SFN entry in the win[] BYTE* fn; //指向短文件名 (in/out) {file[8],ext[3],status[1]} #if _USE_LFN WCHAR* lfn; //指向长文件名缓冲 /* Pointer to the LFN working buffer */ WORD lfn_idx; /* Last matched LFN index number (0xFFFF:No LFN) */ #endif } DIR; typedef struct { //文件目录表项 大小=4+2+2+1+13 DWORD fsize; /* File size */ WORD fdate; /* Last modified date */ WORD ftime; /* Last modified time */ BYTE fattrib; /* Attribute */ // 文件属性 TCHAR fname[13]; /* Short file name (8.3 format) */ #if _USE_LFN //长文件名支持 TCHAR* lfname; /* Pointer to the LFN buffer */ UINT lfsize; /* Size of LFN buffer in TCHAR */ #endif } FILINFO; 结构是搞清楚了,但其里面的具体联系怎么也还收理不清楚。只有看ff.c来疏通了! 里面东西还是蛮多的,咋一看,3000多行类(太多,在这里就根据我的配置,进行逐个分析吧),从头到尾,一个一个来。 首先是三个内存操作和以个字符查找处理函数,不说不解释。 然后是: static FRESULT move_window ( FATFS *fs, /* File system object */ DWORD sector /* Sector number to make appearance in the fs->win[] */ ) /* Move to zero only writes back dirty window */ 该函数就是把指定扇区sector中的数据读到fs->win[]里面 DWORD clust2sect ( /* !=0: Sector number, 0: Failed - invalid cluster# */ FATFS *fs, /* File system object */ DWORD clst /* Cluster# to be converted */ ) 计算簇clst在对应文件系统fs里面的扇区号 DWORD get_fat ( /* 0xFFFFFFFF:Disk error, 1:Internal error, Else:Cluster status */ FATFS *fs, /* File system object */ DWORD clst /* Cluster# to get the link information */ ) 获取簇clst在文件系统fs中FAT表里面fat入口 static FRESULT dir_sdi ( DIR *dj, /* Pointer to directory object */ WORD idx /* Directory index number */ ) 根据根目录索引号idx获取相应的目录信息存储到dj结构里面 static FRESULT dir_next ( /* FR_OK:Succeeded, FR_NO_FILE:End of table, FR_DENIED:EOT and could not stretch */ DIR *dj, /* Pointer to directory object */ int stretch /* 0: Do not stretch table, 1: Stretch table if needed */ ) 获取当前目录项的索引值+1 ,对应的目录项信息 static FRESULT dir_find ( DIR *dj /* Pointer to the directory object linked to the file name */ ) 在目录表中查找与dj->fn相同文件名的目录项 static FRESULT create_name ( DIR *dj, /* Pointer to the directory object */ const TCHAR **path /* Pointer to pointer to the segment in the path string */ ) 创建一个文件名为 path指向的 dj目录项 static FRESULT follow_path ( /* FR_OK(0): successful, !=0: error code */ DIR *dj, /* Directory object to return last directory and found object */ const TCHAR *path /* Full-path string to find a file or directory */ ) 获取文件路径path对应的目录项填入dj里面 static BYTE check_fs ( /* 0:The FAT BR, 1:Valid BR but not an FAT, 2:Not a BR, 3:Disk error */ FATFS *fs, /* File system object */ DWORD sect /* Sector# (lba) to check if it is an FAT boot record or not */ ) 读取文件系统fs的一号扇区,进行MBR 检查,文件系统类型区分 static FRESULT chk_mounted ( /* FR_OK(0): successful, !=0: any error occurred */ const TCHAR **path, /* Pointer to pointer to the path name (drive number) */ FATFS **rfs, /* Pointer to pointer to the found file system object */ BYTE chk_wp /* !=0: Check media write protection for write access */ ) 测试文件系统是否已挂在,如没有,就进行挂载,文件系统结构初始化 static FRESULT validate ( /* FR_OK(0): The object is valid, !=0: Invalid */ FATFS *fs, /* Pointer to the file system object */ WORD id /* Member id of the target object to be checked */ ) 检测文件系统是否可用 如下是现配置好的文件系统引出的四个接口函数: FRESULT f_mount ( //挂在一个逻辑的文件系统 BYTE vol, /* Logical drive number to be mounted/unmounted */ FATFS *fs /* Pointer to new file system object (NULL for unmount)*/ ) 系统挂载 FRESULT f_open ( FIL *fp, /* Pointer to the blank file object */ const TCHAR *path, /* Pointer to the file name */ BYTE mode /* Access mode and file open mode flags */ ) 文件打开, 包括(真正的文件系统初始化,系统检测) FRESULT f_read ( FIL *fp, /* Pointer to the file object */ void *buff, /* Pointer to data buffer */ UINT btr, /* Number of bytes to read */ UINT *br /* Pointer to number of bytes read */ ) 文件读 FRESULT f_close ( FIL *fp /* Pointer to the file object to be closed */ ) 文件关闭 转自博客,供大家相互交流! 因篇幅问题不能全部显示,请点此查看更多更全内容