本节主要给大家介绍以下内容:
文件系统简介
FatFs文件系统简介
FatFs文件系统移植
FatFs功能使用
当我们在使用SPI FLASH直接存储数据 当需要记录字符“STM32 SPI FLASH”时。可以把这些文字转化成ASCII码,存储在数组中,然后调用SPI_FLASH_BufferWrite函数,把数组内容写入到SPI Flash芯片的指定地址上,在需要的时候从该地址把数据读取出来,再对读出来的数据以ASCII码的格式进行解读。
但是这样存在以下弊端:
而我们接触到的Windows/Linux系统就是比较常见的文件系统,我们在Windows/Linux中查找文件只需要根据文件的存储路径就可以方便的找到需要的文件,文件内容读取可以通过文件类型(例如txt、exe等)方式去通过特定的打开方式去使用。
Windows上的文件系统:就是对数据进行管理的方式。使用文件系统可有效地管理存储介质。
文件系统在计算机中的表现形式:
磁盘的物理结构:
使用文件系统时,它为了存储和管理数据,在存储介质建立了一些组织结构,这些结构包括操作系统引导区、目录和文件。
常见的windows下的文件系统格式包括FAT32、NTFS、exFAT。在使用文件系统前,要先对存储介质进行格式化。格式化时会在存储介质上新建一个文件分配表和目录。这样,文件系统就可以记录数据存放的物理地址,剩余空间。
Windows操作系统为了便于用户对磁盘的管理。加入了磁盘分区的概念,即将一块磁盘逻辑划分为几块,它会把磁盘的分区信息记录到硬盘分区表中。
在硬盘分区表中,描述了各个逻辑分区的属性,如分区开始和结束位置所在的物理地址(柱面号、扇区号),空间大小等信息。 在硬盘分区表如下图:
使用文件系统时,数据都以文件的形式存储。写入新文件时,先在目录中创建一个文件索引,它指示了文件存放的物理地址,再把数据存储到该地址中。当需要读取数据时,可以从目录中找到该文件的索引,进而在相应的地址中读取出数据。具体还涉及到逻辑地址、簇大小、不连续存储等一系列辅助结构或处理过程。
文件系统的存在使存取数据时,不再是简单地向某物理地址直接读写,而是要遵循它的读写格式。如经过逻辑转换,一个完整的文件可能被分开成多段存储到不连续的物理地址,使用目录或链表的方式来获知下一段的位置。
刚刚介绍了,我们把文件系统格式化之后,它会把一个文件分配表存储在我们磁盘的开头。
(簇:相当于扇区)
存储了A.TXT,B.TXT,C.TXT文件等
记录了文件的开始簇位置、大小等信息。
上面说的目录是记录了某个文件存储在哪个扇区,而文件分配表,记录了这个扇区存储了哪一些内容,然后这个内容的下一部分在哪里。
文件 a.txt 我们根据目录 项中指定的 a.txt 的首簇为 2,然后找到文件分配表的第 2 簇记录,上面登记 的是 3,就能确定下一簇是 3。找到文件分配表的第 3 簇记录,上面登记的 是 4,就能确定下一簇是 4......直到指到第 11 簇,发现下一个指向是 FF,就是结束。文件便读取完毕。
删除B.TXT文件,创建D.TXT文件后的空间示意图
原目录示意图
删除B.TXT文件,创建D.TXT文件后的目录示意图
原文件分配表示意图
删除B.TXT文件,创建D.TXT文件后的文件分配表示意图
首先回顾下C语言中的文件操作:
文件的打开操作 fopen 打开一个文件
文件的关闭操作 fclose 关闭一个文件
文件的读写操作 fgetc 从文件中读取一个字符
fputc 写一个字符到文件中去
fgets 从文件中读取一个字符串
fputs 写一个字符串到文件中去
fprintf 往文件中写格式化数据
fscanf 格式化读取文件中数据
fread 以二进制形式读取文件中的数据
fwrite 以二进制形式写数据到文件中去
getw 以二进制形式读取一个整数
putw 以二进制形式存贮一个整数
文件状态检查函数 feof 文件结束
ferror 文件读/写出错
clearerr 清除文件错误标志
ftell 了解文件指针的当前位置
文件定位函数 rewind 反绕
fseek 随机定位
windows和linux都是系统/平台上支持这些函数,但是STM32不支持。
而FatFs是面向小型嵌入式系统的一种通用的FAT文件系统。它完全是由AISI C语言编写并且完全独立于底层的I/O介质。因此它可以很容易地不加修改地移植到其他的处理器当中,如8051、PIC、AVR、SH、Z80、H8、ARM等。FatFs支持FAT12、FAT16、FAT32等格式。
(他们三个的区别就是单个文件的大小的最大空间,FAT32单个文件最大支持4GB,FAT16单个文件最大支持2GB)
有了这个文件系统之后,把它移植到STM32上以后,然后再把flash进行格式化。格式化之后,就能像我们的电脑里面使用的磁盘一样,再存储介质上建立文件了,以文件格式的形式对flash进行存储数据。
利用前面写好的SPI Flash芯片驱动,把FatFs文件系统代码移植到工程之中,就可以利用文件系统的各种函数,在上层建立一个文件系统的代码层,来实现文件系统,对SPI Flash芯片以“文件”格式进行读写操作了。
FatFs文件系统的源码可以从fatfs官网下载:fatfs官网链接。
网页的上面,这里它也有很多应用函数的一些接口:这些函数跟C语言里面的操作很类似,功能也是完全一样的。
而下面的这些接口需要我们自己去实现。就类似于C++的虚函数,内部需要我们自己实现。
在Resources处进行下载。
下载FatFs源码包解压后,在doc 文件夹里面是一些使用帮助文档;
在src 是FatFs文件系统的源代码。
其中 en 和 ja 这两个文件夹里面是编译好的html文档,讲的是FATFS里面各个函数的使用方法。 00index_e.html和00index_j.html是一些关于FATFS的简介。
建议阅读这些源码的顺序为:integer.h --> diskio.c --> ff.c 。
阅读文件系统源码ff.c文件需要一定的功底,建议先阅读FAT32的文件格式,再去分析ff.c文件。
若仅为了使用文件系统,则只需要理解integer.h及diskio.c文件并会调用ff.c文件中的函数就可以了。
函数 |
条件(ffconf.h) |
备注 |
disk_status |
总是需要 |
底层设备驱动函数 |
disk_write |
_FS_READONLY == 0 |
|
disk_ioctl (GET_SECTOR_COUNT) |
_USE_MKFS == 1 |
|
disk_ioctl (GET_SECTOR_SIZE) |
_MAX_SS != _MIN_SS |
|
disk_ioctl (CTRL_TRIM) |
_USE_TRIM == 1 |
|
ff_convert |
_USE_LFN != 0 |
Unicode支持,为支持简体中文,添加cc936.c到工程即可 |
ff_cre_syncobj |
_FS_REENTRANT == 1 |
FatFs可重入配置,需要多任务系统支持(一般不需要) |
ff_mem_alloc |
_USE_LFN == 3 |
长文件名支持,缓冲区设置在堆空间(一般设置_USE_LFN = 2 ) |
介绍部分就到此结束啦,下一节将给大家聊聊如何让移植的问题~