本文将介绍Fatfs文件系统的常用函数,学过C语言文件函数的小伙伴们就可以跳过了,因为两者之间基本没有什么区别:
下文提到的大多数函数都会返回FRESULT枚举,可以通过返回值来调试程序,找出程序出错的原因,FRESULT枚举如下:
typedef enum {
FR_OK = 0, /* (0) Succeeded */
FR_DISK_ERR, /* (1) A hard error occurred in the low level disk I/O layer */
FR_INT_ERR, /* (2) Assertion failed */
FR_NOT_READY, /* (3) The physical drive cannot work */
FR_NO_FILE, /* (4) Could not find the file */
FR_NO_PATH, /* (5) Could not find the path */
FR_INVALID_NAME, /* (6) The path name format is invalid */
FR_DENIED, /* (7) Access denied due to prohibited access or directory full */
FR_EXIST, /* (8) Access denied due to prohibited access */
FR_INVALID_OBJECT, /* (9) The file/directory object is invalid */
FR_WRITE_PROTECTED, /* (10) The physical drive is write protected */
FR_INVALID_DRIVE, /* (11) The logical drive number is invalid */
FR_NOT_ENABLED, /* (12) The volume has no work area */
FR_NO_FILESYSTEM, /* (13) There is no valid FAT volume */
FR_MKFS_ABORTED, /* (14) The f_mkfs() aborted due to any parameter error */
FR_TIMEOUT, /* (15) Could not get a grant to access the volume within defined period */
FR_LOCKED, /* (16) The operation is rejected according to the file sharing policy */
FR_NOT_ENOUGH_CORE, /* (17) LFN working buffer could not be allocated */
FR_TOO_MANY_OPEN_FILES, /* (18) Number of open files > _FS_LOCK */
FR_INVALID_PARAMETER /* (19) Given parameter is invalid */
} FRESULT;
挂载和取消挂载文件系统
FRESULT f_mount (
FATFS* fs, /* [IN] File system object */
const TCHAR* path, /* [IN] Logical drive number */
BYTE opt /* [IN] Initialization option */
);
fs: 文件系统句柄(注意区别于文件句柄),要提前定义一个FATFS类型变量,然后把它的
地址传给此参数
path: 这个参数决定了你要挂载哪个文件系统,在下文将统一叫它“逻辑驱动器”如果已经在diskio.c文件中有了如下定义:
/* Definitions of physical drive number for each drive */
#define ATA 2 /* Example: Map ATA harddisk to physical drive 0 */
#define MMC 3 /* Example: Map MMC/SD card to physical drive 1 */
#define USB 4 /* Example: Map USB MSD to physical drive 2 */
#define SD_CARD 0
#define FLASH_SPI 1
这个时候如果把path传入"1:"那就表示要挂载FLASH_SPI 这个文件系统,当然前提是你必须提前移植好了FLASH_SPI
opt: 给0会取消挂载path指定的文件系统,给1会挂载path指定的文件系统
格式化文件系统
FRESULT f_mkfs (
const TCHAR* path, /* Logical drive number */
const MKFS_PARM* opt, /* Format options */
void* work, /* Pointer to working buffer (null: use heap memory) */
UINT len /* Size of working buffer [byte] */
)
path : 逻辑驱动器
sfd :分区规则,一般给0即可(给1貌似是不会创建文件分配表,被称为超级软盘(SFD)格式化)
au :以字节数或扇区数指定分配单元(聚集)的大小。当值介于 1 到 128 之间时,它指定扇区数。当值为 >= _MIN_SS 时,它指定字节数。如果给出了任何无效值(零或非 2 的幂),则根据卷大小自动确定簇大小。(所以一般给0即可)
注意!!!:格式化文件系统之后,必须使用f_mount取消挂载,再重新挂载。
打开文件(指定文件权限,完成文件句柄初始化)
FRESULT f_open (
FIL* fp, /* [OUT] Pointer to the file object structure */
const TCHAR* path, /* [IN] File name */
BYTE mode /* [IN] Mode flags */
);
*fp: 文件句柄地址
path:逻辑驱动器
mode :文件模式(权限),详见下表
参数 | 描述 |
---|---|
FA_READ | 读权限 |
FA_WRITE | 写权限 |
FA_OPEN_EXISTING | 如果文件存在,则打开;否则打开失败,返回FR_NO_FILE |
FA_OPEN_ALWAYS | 如果文件存在,则打开,如果不存在则创建一个文件并打开 |
FA_CREATE_NEW | 创建一个文件,如果文件存在,则创建失败,返回FR_EXIST |
FA_CREATE_ALWAYS | 创建一个文件,如果文件存在,则覆盖原文件 |
写文件
FRESULT f_write (
FIL* fp, /* [IN] Pointer to the file object structure */
const void* buff, /* [IN] Pointer to the data to be written */
UINT btw, /* [IN] Number of bytes to write */
UINT* bw /* [OUT] Pointer to the variable to return number of bytes written */
);
*fp:文件句柄地址
*buff:要写入的数据地址(字符数组首地址,普通数组首地址或者字符串),这个参数应该传入const类型的实参
btw:要写入的字节数
*bw:传入一个指针,函数调用之后,将把成功写入的字节数传入该指针指向的变量,可以用来检查是否成功写入
int f_printf (
FIL* fp, /* [IN] File object */
const TCHAR* fmt, /* [IN] Format stirng */
...
);
此函数不必多说了,用法与printf函数一摸一样,只不过多了第一个参数*fp而已.
FRESULT f_read (
FIL* fp, /* [IN] File object */
void* buff, /* [OUT] Buffer to store read data */
UINT btr, /* [IN] Number of bytes to read */
UINT* br /* [OUT] Number of bytes read */
);
* fp: 文件句柄地址
* buff:存放读取结果的数组
btr:要读取的字节数(通常和f_size配合使用)
* br:成功读取到的字节数
补充:在此说明一下br究竟有什么用,想象一下这样的场景:你要读取一个文件,文件很大,buff每次只能存放256个字节,那就只能每次读取256个字节咯!所以你不得不多次读取文件,但是如何确定读取到文件末尾了呢?读取到文件末尾,f_read就不会再读取了,所以此时br必定小于btr,因此,在多次读取的时候,可以用br来判断是否读取到文件末尾。
传入文件句柄地址,返回该文件内容包含的字节数
改变文件指针(光标)的位置
这个函数很简单,但是很不简单,新手经常碰到写入失败,读取失败,很大一部分原因就是不能正确的理解文件指针位置导致的,所以不要小看这个函数
FRESULT f_lseek (
FIL* fp, /* [IN] File object */
DWORD ofs /* [IN] File read/write pointer */
);
* fp:文件句柄地址
ofs 文件指针新的位置
FRESULT f_getfree (
const TCHAR* path, /* [IN] Logical drive number */
DWORD* nclst, /* [OUT] Number of free clusters */
FATFS** fatfs /* [OUT] Corresponding file system object */
);
path:
指向指定逻辑驱动器的空终止字符串的指针。空字符串表示默认驱动器。
*nclst:
指向 DWORD 变量的指针,用于存储剩余簇的数量。
文件系统句柄的csize成员表示每个簇含有几个扇区,可以搭配两者使用来计算剩余空间;而n_fatent成员表示总空间的个数+2,所以还可以通过总空间减去剩余空间来计算已用空间
补充:一个扇区多少个字节,这个由ffconf.h配置文件的_MAX_SS宏决定,比如我的是4096,那就表示一个扇区4096个字节即4KB
** fatfs:
文件系统指针的指针(其实我不太明白为什么这里要用二级指针,感觉完全没必要啊,可能我这个菜鸟还不知道大佬的别有用心吧)
参考官方例程:
FATFS *fs;
DWORD fre_clust, fre_sect, tot_sect;
/* Get volume information and free clusters of drive 1 */
res = f_getfree("1:", &fre_clust, &fs);
/* Get total sectors and free sectors */
tot_sect = (fs->n_fatent - 2) * fs->csize;
fre_sect = fre_clust * fs->csize;
/* Print the free space (assuming 512 bytes/sector) */
printf("%10lu KiB total drive space.\n%10lu KiB available.\n",
tot_sect / 2, fre_sect / 2);
关闭文件,不用多说了,文件使用完之后,肯定要关闭吧,
FRESULT f_close (
FIL* fp /* [IN] Pointer to the file object */
);
打开一个已存在的目录,并创建一个目录对象,如果不存在指定目录,则返回FR_NO_PATH
FRESULT f_opendir (
DIR* dp, /* [OUT] Pointer to the directory object structure */
const TCHAR* path /* [IN] Directory name */
);
*dp:创建的目录对象存放在此指针
* path:目录名字
DIR结构体定义如下:
typedef struct {
FATFS* fs; /*指向文件系统的指针(禁止更改*/
WORD id; /* 文件系统的逻辑驱动器ID(禁止更改) */
WORD index; /* 当前读/写索引号*/
DWORD sclust; /* Table start cluster (0:Root dir) */
DWORD clust; /* Current cluster */
DWORD sect; /* Current sector */
BYTE* dir; /* Pointer to the current SFN entry in the win[] */
BYTE* fn; /* Pointer to the SFN (in/out) {file[8],ext[3],status[1]} */
#if _FS_LOCK
UINT lockid; /* File lock ID (index of file semaphore table Files[]) */
#endif
#if _USE_LFN
WCHAR* lfn; /* 指向长文件名的工作数组 */
WORD lfn_idx; /* Last matched LFN index number (0xFFFF:No LFN) */
#endif
#if _USE_FIND
const TCHAR* pat; /* 指向匹配路径的指针 */
#endif
} DIR;
关闭打开的目录对象。函数成功后,目录对象将不再有效,可以丢弃。
FRESULT f_closedir (
DIR* dp /* [IN] Pointer to the directory object */
);
在指定路径创建目录
FRESULT f_mkdir (
const TCHAR* path /* [IN] Directory name */
);
* path:指定路径和新文件的名字
删除文件
FRESULT f_unlink (
const TCHAR* path /* [IN] Object name */
);
移动,重命名文件或子目录
FRESULT f_rename (
const TCHAR* old_name, /* [IN] Old object name */
const TCHAR* new_name /* [IN] New object name */
)
old_name :旧文件名(包括路径)
new_name:新文件名(包括路径)
检查文件或子目录是否存在,如果存在,则将文件的信息储存在fno指针指向的FILINFO结构体中
FRESULT f_stat (
const TCHAR* path, /* [IN] Object name */
FILINFO* fno /* [OUT] FILINFO structure */
);
*path:指定文件或子目录路径
*fno: 存储文件信息的FILINFO结构体指针
FILINFO结构体如下:
typedef struct {
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;
fattrib成员的所有位的含义如下:
/* File attribute bits for directory entry */
#define AM_RDO 0x01 /* Read only */
#define AM_HID 0x02 /* Hidden */
#define AM_SYS 0x04 /* System */
#define AM_VOL 0x08 /* Volume label */
#define AM_LFN 0x0F /* LFN entry */
#define AM_DIR 0x10 /* Directory */ //文件夹
#define AM_ARC 0x20 /* Archive */ //档案文件(压缩文件)
#define AM_MASK 0x3F /* Mask of defined bits */
输入目录对象,将目录中第一个文件(夹)信息存储在参数二指针指向的地址;如果再次调用,则输出目录中第二个文件(夹)信息,以此类推
FRESULT f_readdir (
DIR* dp, /* [IN] Directory object */
FILINFO* fno /* [OUT] File information structure */
);
DIR和FILINFO结构体如下:
typedef struct {
FATFS* fs; /*指向文件系统的指针(禁止更改*/
WORD id; /* 文件系统的逻辑驱动器ID(禁止更改) */
WORD index; /* 当前读/写索引号*/
DWORD sclust; /* Table start cluster (0:Root dir) */
DWORD clust; /* Current cluster */
DWORD sect; /* Current sector */
BYTE* dir; /* Pointer to the current SFN entry in the win[] */
BYTE* fn; /* Pointer to the SFN (in/out) {file[8],ext[3],status[1]} */
#if _FS_LOCK
UINT lockid; /* File lock ID (index of file semaphore table Files[]) */
#endif
#if _USE_LFN
WCHAR* lfn; /* 指向长文件名的工作数组 */
WORD lfn_idx; /* Last matched LFN index number (0xFFFF:No LFN) */
#endif
#if _USE_FIND
const TCHAR* pat; /* 指向匹配路径的指针 */
#endif
} DIR;
typedef struct {
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;
这个函数通常会被封装成一个递归函数来实现类似Linux系统中的ls命令功能.
官方给出的递归函数如下:
FRESULT scan_files (
char* path /* Start node to be scanned (also used as work area) */
)
{
FRESULT res;
FILINFO fno;
DIR dir;
int i;
char *fn; /* This function assumes non-Unicode configuration */
#if _USE_LFN
static char lfn[_MAX_LFN + 1]; /* Buffer to store the LFN */
fno.lfname = lfn;
fno.lfsize = sizeof lfn;
#endif
res = f_opendir(&dir, path); /* Open the directory */
if (res == FR_OK) {
i = strlen(path);
for (;;) {
res = f_readdir(&dir, &fno); /* Read a directory item */
if (res != FR_OK || fno.fname[0] == 0) break; /* Break on error or end of dir */
if (fno.fname[0] == '.') continue; /* Ignore dot entry */
#if _USE_LFN
fn = *fno.lfname ? fno.lfname : fno.fname;
#else
fn = fno.fname;
#endif
if (fno.fattrib & AM_DIR) { /* It is a directory */
sprintf(&path[i], "/%s", fn);
res = scan_files(path);
path[i] = 0;
if (res != FR_OK) break;
} else { /* It is a file. */
printf("%s/%s\n", path, fn);
}
}
f_closedir(&dir)
}
return res;
}