1、目的
在移植之前,先将源代码大概的阅读一遍,主要是了解文件系统的结构、各个函数的功能和接口、与移植
相关的代码等等。
2、准备工作
在官方网站下载了0.07c版本的源代码,利用记事本进行阅读。
二、源代码的结构
1、源代码组成
件夹。文件夹是option,还有00readme.txt、diskio.c、diskio.h、ff.c、ff.h、integer.h。对比网上
的文章,版本已经不同了,已经没有所谓的tff.c和tff.h了,估计现在都采用条件编译解决这个问题了,
当然文件更少,可能编译选项可能越复杂。
2、00readme.txt的说明
上使用。我们移植时针对具体存储设备提供底层代码。
3、源代码阅读次序
后是diskio.h,了解与介质相关的数据结构和操作函数。再把ff.c和diskio.c两个文件所实现的函数大致
扫描一遍。最后根据用户应用层程序调用函数的次序仔细阅读相关代码。
三、源代码阅读
1、integer.h头文件
这个文件主要是类型声明。以下是部分代码。
typedef int
typedef unsigned int UINT;
typedef signed char
都是用typedef做类型定义。移植时可以修改这部分代码,特别是某些定义与你所在工程的类型定义有冲
突的时候。
2、ff.h头文件
以下是部分代码的分析
#include "integer.h" 使用integer.h的类型定义
#ifndef _FATFS
#define _FATFS 0x007C
#define _WORD_ACCESS 0 //如果定义为1,则可以使用word访问。
中间有一些看着说明很容易弄清楚意思。这里就不例举了。
#define _CODE_PAGE 936
#define _FS_REENTRANT 0
#define _TIMEOUT
#define _SYNC_t
#elif _CODE_PAGE == 936
#define _DF1S 0x81
#define _DF1E 0xFE
#define _DS1S 0x40
#define _DS1E 0x7E
#define _DS2S 0x80
#define _DS2E 0xFE
接下来很大一部分都是与语言相关的因素,略过。
三个宏判断是否大写、小写、数字。
#define IsUpper(c) (((c)>='A')&&((c)<='Z'))
#define IsLower(c) (((c)>='a')&&((c)<='z'))
#define IsDigit(c) (((c)>='0')&&((c)<='9'))
#if _DF1S
#if _MULTI_PARTITION
//该变量定义为1时,支持一个磁盘的多个分区。
typedef struct _PARTITION {
} PARTITION;
Extern
#define LD2PD(drv) (Drives[drv].pd)
#define LD2PD(drv) (drv)
#define LD2PT(drv) 0
#if _MAX_SS == 512
#define
#if _LFN_UNICODE && _USE_LFN
typedef WCHAR XCHAR;
#else
typedef char XCHAR;
#endif
typedef struct _FATFS_ {
先查一下簇的含义:应该是文件数据分配的基本单位。
FAT文件系统依次应该是:引导扇区、文件分配表两个、根目录区和数据区。
//文件是否改动的标志,为1时要回写。
#if _FS_REENTRANT
#endif
#if _MAX_SS != 512
#endif
#if !_FS_READONLY
//文件需要回写的标志
#endif
#if _FS_RPATH
//目前的扇区在win[]里面,这个win[]数组暂时还不知道含义。
//这是一个win[512]数组,存储着一个扇区,好像作为扇区缓冲使用。
} FATFS;
typedef struct _DIR_ {
#if _USE_LFN
#endif
} DIR;
typedef struct _FIL_ {
#if !_FS_READONLY
#endif
#if !_FS_TINY
#endif
} FIL;
typedef struct _FILINFO_ {
#if _USE_LFN
#endif
} FILINFO; 这个结构主要描述文件的状态信息,包括文件名13个字符(8+.+3+ )、属性、修改时间等。
接下来是函数的定义,先大概浏览一遍。
FRESULT f_mount (BYTE, FATFS*);
FRESULT f_open (FIL*, const XCHAR*, BYTE);//打开文件,第一个参数是文件信息结构,第二个参数是文件名,第三是文件打开模式
FRESULT f_read (FIL*, void*, UINT, UINT*);
FRESULT f_write (FIL*, const void*, UINT, UINT*);//写文件,参数跟读差不多
FRESULT f_lseek (FIL*, DWORD); //移动文件的读写指针,参数2应该是移动的数目。
FRESULT f_close (FIL*);
FRESULT f_opendir (DIR*, const XCHAR*);
FRESULT f_readdir (DIR*, FILINFO*);
FRESULT f_stat (const XCHAR*, FILINFO*);
FRESULT f_getfree (const XCHAR*, DWORD*, FATFS**);
FRESULT f_truncate (FIL*);
FRESULT f_sync (FIL*);
FRESULT f_unlink (const XCHAR*);
FRESULT
FRESULT f_chmod (const XCHAR*, BYTE, BYTE);
FRESULT f_utime (const XCHAR*, const FILINFO*);
FRESULT f_rename (const XCHAR*, const XCHAR*);
FRESULT f_forward (FIL*, UINT(*)(const BYTE*,UINT), UINT, UINT*); 这个函数还要提供一个回调函数。
FRESULT f_mkfs (BYTE, BYTE, WORD);
FRESULT f_chdir (const XCHAR*);
FRESULT f_chdrive (BYTE);
应该说基本能明白这些函数用于干什么。
[page]
#if _USE_STRFUNC
int f_putc (int, FIL*);
int f_puts (const char*, FIL*);
int f_printf (FIL*, const char*, ...);
char* f_gets (char*, int, FIL*);
#define f_eof(fp) (((fp)->fptr == (fp)->fsize) ? 1 : 0)
#define f_error(fp) (((fp)->flag & FA__ERROR) ? 1 : 0)
#if _FS_REENTRANT
BOOL ff_cre_syncobj(BYTE, _SYNC_t*); 创建同步对象
BOOL ff_del_syncobj(_SYNC_t);
BOOL ff_req_grant(_SYNC_t);
void ff_rel_grant(_SYNC_t); 释放同步对象。
#endif
3、diskio.h文件
typedef BYTE
typedef
BOOL assign_drives (int argc, char *argv[]); //这个函数不知道干吗
DSTATUS disk_initialize (BYTE); //磁盘初始化
DSTATUS disk_status (BYTE); //获取磁盘状态
DRESULT disk_read (BYTE, BYTE*, DWORD, BYTE);
#if
DRESULT disk_write (BYTE, const BYTE*, DWORD, BYTE);
#endif
DRESULT disk_ioctl (BYTE, BYTE, void*); //磁盘控制
接下来还有一些常数的定义,具体用到时在看。
4、diskio.c的结构
DSTATUS disk_initialize (
{
}
函数基本都像这样,drv表示磁盘的类型。没有实现,用户必须实现这部分代码。
5、ff.c文件简单浏览
#include "ff.h"
#include "diskio.h"
#define
#define
Static
Static WORD LfnBuf[_MAX_LFN + 1];
#define
#define INITBUF(dj,sp,lp) dj.fn = sp; dj.lfn = lp
下面都是函数的定义,很多只在内部使用。
Static
} //接下来还定义了几个内存操作的函数,这个函数实现了从一块内存到另一块的复制,下面还有mem_set()对一块内存进行清0或设置操作;mem_cmp()比较内存的多个字节是否相同,相同返回0;chk_chr()检测字符串中是否存在某个字符,存在则返回该字符。
FRESULT move_window (
)//简单阅读了一下源代码,应该是改变文件系统的当前工作扇区,如果想要操作的扇区就是当前扇区,什么事不做;如果不是,则将原扇区写回;如果是FAT表,还得写入备份区。
这个函数内部使用,外部无法引用。
FRESULT sync (
)//这个函数用于更新FAT32文件系统的FSI_Sector。什么含义还不太清楚。
DWORD get_fat (
)
综合起来,这个函数应该是获取下一簇,感觉这个函数名起得不太好。get_nextcluster感觉更好一点。
FRESULT put_fat (
)//上个函数是获取连接簇,这个是写入新的连接信息。
FRESULT remove_chain (
)//将下一簇号写为0,也就是该文件的簇到此为止,同时系统的自由簇增加1.
DWORD create_chain (
)//跟上一个相反,在该簇的位置写入新的下一簇簇号。
DWORD clust2sect (
) //这个函数是将簇号转变为对应的扇区号。
clst * fs->csize + fs->database; //这个是算法
FRESULT dir_seek (
)//这个函数的最终目的是根据索引号找到目录项所在簇、所在扇区、并是目录对象的对象指针指向文件系统对象窗口扇区的对应位置。
FRESULT dir_next (
)//
FRESULT dir_read (
)
FRESULT dir_register (
)
FRESULT dir_remove (
)
//以上这些函数都是对目录项的操作函数。
FRESULT create_name (
//这个函数太长了,具体用到的时候再说吧。
void get_fileinfo (
该函数用于获取文件状态信息。主要是从文件的目录项中获取信息。
FRESULT follow_path (
)
该函数给定一个全路径,得到相应的目录对象。
BYTE check_fs (
该函数用于读取BOOT扇区,检查是否FAT文件系统。
FRESULT auto_mount (
这个函数的功能不太明白。
FRESULT validate (
)//检查是否合法的文件系统。
FRESULT f_mount (
这是一个很重要的函数,装载文件系统。也是从这个函数开始,对外输出供用户调用。
if (vol >= _DRIVES)现在只支持卷号0.
FatFs[vol] = fs;将参数文件系统对象指针赋给全局文件对象指针。
后面的函数主要是对文件和目录进行操作,这里就不一一例举了。
上一篇:TQ2440复制文件无法编译问题
下一篇:STM32启动代码分析
推荐阅读最新更新时间:2024-03-16 14:36
设计资源 培训 开发板 精华推荐
- 谁是下一个智能底盘的“宁王”?
- 自动驾驶独角兽进入生死局:纵目科技停发工资,毫末智行裁员
- PC产业驶入创新超车道,英特尔蓉城撬动AI新引擎
- 与产业聚力共赢,英特尔举行新质生产力技术生态大会
- “新”享5G-A万兆网络前沿体验 高通携手产业伙伴亮相第二届链博会
- 英飞凌推出符合ASIL-D标准的新型汽车制动系统和电动助力转向系统三相栅极驱动器 IC
- 南芯科技推出80V升降压转换器,持续深耕工业储能市场
- 法雷奥与罗姆联合开发新一代功率电子领域
- 贸泽电子开售能为电动汽车牵引逆变器提供可扩展性能的 英飞凌HybridPACK Drive G2模块
- 德州仪器新型 MCU 可实现边缘 AI 和先进的实时控制, 提高系统效率、安全性和可持续性