【STM32】文件系统FATFS与Flash的初步使用

这篇具有很好参考价值的文章主要介绍了【STM32】文件系统FATFS与Flash的初步使用。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

文件系统简介

简介可以不看,直接看移植步骤
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件
文件系统是介于应用层和底层间的模糊层。底层提供API,比如说使用SDIO或者SPI等读写一个字节。文件系统把这些API组合包装起来,并且提供一些列函数,我们可以使用这些函数进行更进一步的对存储设备的操作。

底层:操作单片机外设,读写。需要我们进行配置。
中层:中间层 FATFS 模块,实现了 FAT 文件读/写协议。一般不管
顶层:文件系统提供给我们的函数库,我们就是要使用这些。
FATFS的实现过程(白色框中的内容是需要我们自己实现的也就是底层设备的输入输出以及最高层的用户应用程序,蓝色框由fatfs提供)
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件

源码获取

FATFS 的源码及英文详述,大家可以在:http://elm-chan.org/fsw/ff/00index_e.html 这个网站下载。下载解压之后的根目录下有这么两个文件夹,源码和帮助文档。
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件
DOC文件夹下有Fatfs提供的具体函数的使用方法
SRC文件的构成如下
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件

【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件
Fatfs的源码阅读可以参考《零死角玩转 STM32F103—指南者》中的阅读提示,如果只是想使用,那么只需要看如何移植

移植FATFS 准备工作

背景:两个文件 ffconf.h diskio.c

FATFS 模块在移植的时候,我们一般只需要修改 2 个文件,即 ffconf.hdiskio.c
ffconf.h :FATFS模块的所有配置项都是存放在 ffconf.h 里面,我们可以通过配置里面的一些选项,来满足自己的需求.FATFS 的说明文档里面有很详细的介绍
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件

以spi读写flash为例,在实现了spi往flash中读写单个字节的底层代码之后,就可以进行fatfs的移植了。

0.创建一个新的工程,并且实现简单的点灯,spi操作等功能。

1.准备一份工程源码称为SPI—FatFs。将。将 FatFs 源码中的 src 文件夹整个文件夹拷贝一份至“SPI—FatFs 文件系统\USER\”文件夹下并修改名为“FATFS”。文件夹名字没有固定名称,这里只是方便书写。

2.打开工程文件,并将 FatFs 组件文件添加到工程中,需要添加有 ff.c、diskio.c 和cc936.c 三个文件.
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件
3.添加 FATFS 文件夹到工程的 include 选项中。打开工程选项对话框,选择“C/C++”选项下的“Include Paths”项目,在弹出路径设置对话框中选择添加“FATFS”文件夹
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件
4.修改 diskio.c 文件和 ffconf.h 文件,diskio.c 文件内容是与底层设备输入输出接口函数文件,不同硬件设计驱动就不同,需要的文件也不同;FatFs默认使用日语,我们想要支持简体中文需要
修改 FatFs 的配置,即修改 ffconf.h 文件。

FatFs 底层设备驱动函数

文件系统实现中间模糊层,提供上层函数给用户使用。需要用户实现底层接口。diskio.c 文件内容就是与底层接口相关的。

移植需要用户支持函数,一般只有前六个需要使用
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件
disk_status,disk_initialize,disk_read是必要配置的。
disk_write,get_fattime,disk_ioctl (CTRL_SYNC)是实现创建文件、修改文件需要的。
为支持简体中文长文件名称需要添加 ff_convert 和 ff_wtoupper 函数,实际这两个已经在 cc936.c 文件中实现,我们只要直接把 cc936.c 文件添加到工程中就可以。

移植FATFS 主要步骤

1.配置数据类型:在 integer.h 里面去定义好数据的类型。这里需要了解你用的编译器的数据类型,并根据编译器定义好数据类型。

2.配置:通过 ffconf.h 配置 FATFS 的相关功能,以满足你的需要。

3.函数编写:打开 diskio.c,进行底层驱动编写,需要编写 5 个接口函数。
【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件

配置integer.h 以定义数据类型(一般不需要)

我们使用的是 MDK5.34 编译器,数据类型和 integer.h 里面定义的一致,所以此步,我们不需要做任何改动。

配置 ffconf.h 选择模式

关于 ffconf.h 里面的相关配置,配置修改为我们需要的值即可,其他的配置用默认配置。

1 #define _USE_MKFS 1	//格式化功能选择,为使用 FatFs 格式化功能,需要把它设置为 1。
2 #define _CODE_PAGE 936	//语言功能选择,并要求把相关语言文件添加到工程宏。为支持简体中文文件名需要使用“936” 指的是把 cc936.c 文件添加到工程中
3 #define _USE_LFN 2	//长文件名支持,默认不支持长文件名,这里配置为 2,支持长文件名,并指定使用栈空间为缓冲区。
4 #define _VOLUMES 2	//指定物理设备数量
5 #define _MIN_SS 512	//指定扇区大小的最小值和最大值。SD 卡扇区大小一般都为 512字节,SPI Flash芯片扇区大小一般设置为 4096字节
6 #define _MAX_SS 4096	//指定扇区大小的最小值和最大值

【STM32】文件系统FATFS与Flash的初步使用,STM32,stm32,数据库,嵌入式硬件

为每个设备定义一个物理编号

 #define ATA 	0 	// 预留 SD 卡使用
 #define SPI_FLASH 	1 // 外部 SPI Flash

实现五个函数

设备状态获取(disk_status)、设备初始化(disk_initialize)、扇区读取(disk_read)、扇区写(disk_write)、其他控制(disk_ioctl)。

disk_initialize

函数名称 disk_initialize
函数原型 DSTATUS disk_initialize(BYTE Drive)
功能描述 初始化磁盘驱动器
函数参数 Drive:指定要初始化的逻辑驱动器号,即盘符,应当取值 0~9
返回值 函数返回一个磁盘状态作为结果,对于磁盘状态的细节信息,请参考 disk_status函数
所在文件 ff.c
实例 disk_initialize(0); /* 初始化驱动器 0 */
注意事项 disk_initialize 函数初始化一个逻辑驱动器为读/写做准备,函数成功时,返回值的 STA_NOINIT 标志被清零;应用程序不应调用此函数,否则卷上的 FAT 结构可能会损坏;如果需要重新初始化文件系统,可使用 f_mount 函数;在 FatFs 模块上卷注册处理时调用该函数可控制设备的改变;此函数在 FatFs 挂在卷时调用,应用程序不应该在 FatFs 活动时使用此函数
DSTATUS disk_status (BYTE pdrv /* 物理编号 */)
{
DSTATUS status = STA_NOINIT;
switch (pdrv) {
	case ATA: /* SD CARD 预留,也可以*/
		 break;
 	case SPI_FLASH: /* SPI Flash 状态检测:读取 SPI Flash 设备 ID */
		 if (sFLASH_ID == SPI_FLASH_ReadID()) {/* 设备 ID 读取结果正确 */
 				status &= ~STA_NOINIT;
 			} else { /* 设备 ID 读取结果错误 */
				 status = STA_NOINIT;
			 }
		 break;
 	default:
		 status = STA_NOINIT;
		 }
	 return status;
	  }

注意点
SPI_FLASH_ReadID函数由用户自己实现,目的是检测设备是否已经就绪。

disk_status

函数名称 disk_status
函数原型 DRESULT disk_status (BYTE Drive)
功能描述 返回当前磁盘驱动器的状态
函数参数 Drive:指定要确认的逻辑驱动器号,即盘符,应当取值 0~9
返回值 磁盘状态返回下列标志的组合,FatFs 只使用 STA_NOINIT 和 STA_PROTECTEDSTA_NOINIT: 表明磁盘驱动未初始化,下面列出了产生该标志置位或清零的原因:置位:系统复位,磁盘被移除和磁盘初始化函数失败。清零:磁盘初始化函数成功.STA_NODISK:表明驱动器中没有设备,安装磁盘驱动器后总为0 STA_PROTECTED:表明设备被写保护,不支持写保护的设备总为 0,当STA_NODISK 置位时非法
所在文件 ff.c
实例 disk_status(0); /* 获取驱动器 0 的状态 */
DSTATUS disk_initialize (BYTE pdrv /* 物理编号 */)
{	uint16_t i;
	DSTATUS status = STA_NOINIT;
	switch (pdrv) {
		case ATA: /* SD CARD */
			break;
 
	case SPI_FLASH: /* SPI Flash */
				SPI_FLASH_Init(); /* 初始化 SPI Flash */
		 		i=500;/* 延时一小段时间 */
				while (--i);
				SPI_Flash_WAKEUP();/* 唤醒 SPI Flash */
				status = disk_status(SPI_FLASH);/* 获取 SPI Flash 芯片状态 */
				break;
	default:
		 status = STA_NOINIT;
		}
 return status;
}

注意点
SPI_FLASH_Init();SPI_Flash_WAKEUP();都是由用户自己实现的底层功能,具体需要参照Flash对应的要求进行编写(一般也就是按着一定的规则发送数据字节过去,所以一般编程顺序是,线实现单字节的spi收发,再按照要求编写多字节的功能,再嵌入文件系统)

disk_read

函数名称 disk_read
函数原型 DRESULT disk_read (BYTE Drive, BYTE* Buffer, DWORD SectorNumber, BYTE SectorCount)
功能描述 从磁盘驱动器上读取扇区
函数参数 Drive:指定逻辑驱动器号,即盘符,应当取值 0~9 Buffer:指向存储读取数据字节数组的指针,需要为所读取字节数的大小,扇区统计的扇区大小是需要的(注:FaFts 指定的内存地址并不总是字对齐的,如果硬件不支持不对齐的数据传输,函数里需要进行处理)SectorNumber:指定起始扇区的逻辑块(LBA)上的地址SectorCount:指定要读取的扇区数,取值 1~128
返回值 RES_OK(0):函数成功RES_ERROR:读操作期间产生了任何错误且不能恢复它RES_PARERR:非法参数RES_NOTRDY:磁盘驱动器没有初始化
所在文件 ff.c
DRESULT disk_read (
	BYTE pdrv, /* 设备物理编号(0..) */
	BYTE *buff, /* 数据缓存区 */
	DWORD sector, /* 扇区首地址 */
	UINT count /* 扇区个数(1..128) */
	)
{
	DRESULT status = RES_PARERR;
	switch (pdrv) {
		 case ATA: /* SD CARD */
			break;
 
		 case SPI_FLASH:
			sector+=512;/* 扇区偏移 2MB,外部 Flash 文件系统空间放在 SPI Flash 后面 6MB 空间 */
			SPI_FLASH_BufferRead(buff, sector <<12, count<<12);
			status = RES_OK;
			break;
 
		default:
			status = RES_PARERR;
				 }
		 return status;
 }

注意点
参数pdrv 为设备物理编号,用户自定义,buff:BYTE 类型指针变量,buff指向用来存放读取到数据的存储区首地址,sector 是一个 DWORD 类型变量,指定要读取数据的扇区首地址。count 是一个 UINT 类型变量,指定扇区数量。
数据类型:BYTE 类型实际是 unsigned char 类型,DWORD 类型实际是 unsigned long 类型,UINT类型实际是 unsigned int 类型,类型定义在 integer.h 文件中。
偏移 :板使用的 SPI Flash 芯片型号为 W25Q64FV,每个扇区大小为 4096 个字节(4KB),总共有8M字节空间若。要从Flash的首地址开始存放文件系统则不需要偏移。 只将后部分6MB空间分配给FatFs使用,即 FatFs 是从 2MB 空间开始,为实现这个效果需要将所有的读写地址都偏移 512 个扇区空间
SPI_FLASH_BufferRead():读flash的底层程序,实现在指定地址读取指定长度的数据,示例

/**
* @brief 读取 FLASH 数据
* @param pBuffer,存储读出数据的指针
* @param ReadAddr,读取地址
* @param NumByteToRead,读取数据长度
* @retval 无
*/
void SPI_FLASH_BufferRead(u8* pBuffer, u32 ReadAddr, u16 NumByteToRead)
{

 SPI_FLASH_CS_LOW(); /* 选择 FLASH: CS 低电平 */
 
 SPI_FLASH_SendByte(W25X_ReadData); /* 发送 读 指令 */
 SPI_FLASH_SendByte((ReadAddr & 0xFF0000) >> 16) /* 发送 读 地址高位 */
 SPI_FLASH_SendByte((ReadAddr& 0xFF00) >> 8); /* 发送 读 地址中位 */
 SPI_FLASH_SendByte(ReadAddr & 0xFF); /* 发送 读 地址低位 */
  
 while (NumByteToRead--)/* 读取数据 */
 {
	  *pBuffer = SPI_FLASH_SendByte(Dummy_Byte);/* 读取一个字节*/
	  pBuffer++;/* 指向下一个字节缓冲区 */
 }
	SPI_FLASH_CS_HIGH();/* 停止信号 FLASH: CS 高电平 */
}

disk_write

函数名称 disk_write
函数原型 DRESULT disk_write (BYTE Drive, const BYTE* Buffer, DWORD SectorNumber, BYTE SectorCount)
功能描述 向磁盘写入一个或多个扇区
函数参数 Drive:指定逻辑驱动器号,即盘符,应当取值 0~9Buffer:指向要写入字节数组的指针(注:FaFts 指定的内存地址并不总是字对齐的,如果硬件不支持不对齐的数据传输,函数里需要进行处理)SectorNumber:指定起始扇区的逻辑块(LBA)上的地址SectorCount:指定要写入的扇区数,取值 1~128
返回值 RES_OK(0):函数成功RES_ERROR:写操作期间产生了任何错误且不能恢复它RES_WRPER:媒体被写保护RES_PARERR:非法参数RES_NOTRDY:磁盘驱动器没有初始化
所在文件 ff.c
注意事项 只读配置中不需要此函数
DRESULT disk_write (
	BYTE pdrv, /* 设备物理编号(0..) */
	const BYTE *buff, /* 欲写入数据的缓存区 */
	DWORD sector, /* 扇区首地址 */
	UINT count /* 扇区个数(1..128) */
)
{
	uint32_t write_addr;
	DRESULT status = RES_PARERR;
 if (!count) {
		 return RES_PARERR; /* Check parameter */
			}
 
		switch (pdrv) {
			case ATA: /* SD CARD */
			 break;
			case SPI_FLASH:
			/* 扇区偏移 2MB,外部 Flash 文件系统空间放在 SPI Flash 后面 6MB 空间 */
			 sector+=512;
			 write_addr = sector<<12;
			 SPI_FLASH_SectorErase(write_addr);//对于Flash 先擦除再写
			 SPI_FLASH_BufferWrite((u8 *)buff,write_addr,count<<12);//写
			 status = RES_OK;
			break;
 
			default:
			status = RES_PARERR;
				}
			return status;
 }

SPI_FLASH_SectorErase SPI_FLASH_BufferWrite用户实现

disk_ioctl

函数名称 disk_ioctl
函数原型 DRESULT disk_ioctl (BYTE Drive, BYTE Command, void* Buffer)
功能描述 控制设备指定特性和除了读/写外的杂项功能
函数参数 Drive:指定逻辑驱动器号,即盘符,应当取值 0~9Command:指定命令代码Buffer:指向参数缓冲区的指针,取决于命令代码,不使用时,指定一个 NULL指针
返回值 RES_OK(0):函数成功RES_ERROR:写操作期间产生了任何错误且不能恢复它RES_PARERR:非法参数RES_NOTRDY:磁盘驱动器没有初始化
所在文件 ff.c
注意事项 CTRL_SYNC:确保磁盘驱动器已经完成了写处理,当磁盘 I/O 有一个写回缓存,立即刷新原扇区,只读配置下不适用此命令GET_SECTOR_SIZE:返回磁盘的扇区大小,只用于 f_mkfs()GET_SECTOR_COUNT:返回可利用的扇区数,_MAX_SS ≥ 1024 时可用GET_BLOCK_SIZE:获得擦除块大小,只用于 f_mkfs()CTRL_ERASE_SECTOR:强制擦除一块的扇区,_USE_ERASE > 0 时可用
DRESULT disk_ioctl (
BYTE pdrv, /* 物理编号 */
BYTE cmd, /* 控制指令 */
void *buff /* 写入或者读取数据地址指针 */
)
{
	DRESULT status = RES_PARERR;
	switch (pdrv) {
		case ATA: /* SD CARD */
			break;
 
		case SPI_FLASH:
			switch (cmd) {
			/* 扇区数量:1536*4096/1024/1024=6(MB) */
			case GET_SECTOR_COUNT:
					*(DWORD * )buff = 1536;
					break;
			/* 扇区大小 */
			case GET_SECTOR_SIZE :
					*(WORD * )buff = 4096;
					 break;
			/* 同时擦除扇区个数 */
		    case GET_BLOCK_SIZE :
					*(DWORD * )buff = 1;
					break;
				 }
			status = RES_OK;
				break;
 
			default:
			status = RES_PARERR;
		 }
		return status;
 }

参数pdrv 为设备物理编号,cmd 为控制指令,包括发出同步信号、获取扇区数目、获取扇区大小、获取擦除块数量等等指令,buff 为指令对应的数据指针。
cmd:对 于 SPI Flash 芯 片 , 为 支 持 FatFs 格 式 化 功 能 , 需 要 用 到 获 取 扇 区 数 量(GET_SECTOR_COUNT)指令和获取擦除块数量(GET_BLOCK_SIZE)。另外,SD 卡扇区大小为 512 字节,SPI Flash 芯片一般设置扇区大小为 4096 字节,所以需要用到获取扇区大小(GET_SECTOR_SIZE)指令。文章来源地址https://www.toymoban.com/news/detail-695703.html

到了这里,关于【STM32】文件系统FATFS与Flash的初步使用的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • STM32CubeMX教程29 USB_HOST - 使用FatFs文件系统读写U盘

    正点原子stm32f407探索者开发板V2.4 STM32CubeMX软件(Version 6.10.0) keil µVision5 IDE(MDK-Arm) ST-LINK/V2驱动 野火DAP仿真器 XCOM V2.6串口助手 使用STM32CubeMX软件配置STM32F407开发板 USB_OTG_FS为工作在Mass Storage Host Class(大容量存储主机类)模式下的USB_HOST(USB主机),并使用FatFs文件系统对

    2024年02月19日
    浏览(45)
  • STM32-基本知识梳理8-FATFS文件系统移植

    一、基本概念 1,为什么需要文件系统? 在没有文件系统的存储器中,一般通过手工记录,那些变量存放在那些位置,很难有系统的管理 难以记录有效数据的位置 、难以确定存储介质的剩余空间、 不明确应以何种格式来解读数据 加入文件系统后,就可以引导区、目录等对内

    2024年02月03日
    浏览(43)
  • STM32挂载SD卡基于Fatfs文件系统读取文件信息

    🔖本例程基于正点原子标准库修改而来。 📍FatFs 相关API函数网站: http://elm-chan.org/fsw/ff/00index_e.html 🌴分别测试了SD卡模块以及Mini SD卡模块。 🌿STM32f1单片机和TF卡、SD卡连接采用SPI通讯方式。 SD模块: Mini SD卡模块: 👉🏻如果需要采用SDIO驱动方式,那么需要采用下面这种

    2024年02月10日
    浏览(42)
  • 【STM32】SPI初步使用 读写FLASH W25Q64

    (1) SS( Slave Select):从设备选择信号线,常称为片选信号线,每个从设备都有独立的这一条 NSS 信号线,当主机要选择从设备时,把该从设备的 NSS 信号线设置为低电平,该从设备即被选中,即片选有效,接着主机开始与被选中的从设备进行 SPI通讯。所以 SPI通讯以 NSS 线置低电

    2024年02月10日
    浏览(57)
  • 【STM32】FATFS文件系统f_mkfs卡死

    最近调试FATFS,通过STM32对SPI Flash进行读写操作。昨天晚上调通了,今天早上再调试,发现f_mount返回FR_NOFILESYSTEM。 依据网上所说,对磁盘格式化f_mkfs,卡死在FATFS库的ff.c-f_mkfs()的某个for循环里。 f_mkfs卡死的for循环代码位置 最后对SPI Flash进行了全片擦除命令,之后就可以正常

    2024年02月02日
    浏览(42)
  • STM32使用HAL库SPI驱动W25Q16 使用FATFS文件系统+USB虚拟U盘

    使用stm32F407驱动W25Q16,使用FATFS文件系统,USB虚拟优盘功能,W25Q16一共512个扇区,其中128作为flash存取相关数据,其他的384个扇区用作虚拟U盘使用 W25Q16.c W25Q16.h user_diskio.c usbd_storage_if.c main.c STORAGE_BLK_NBR 表示扇区数量 STORAGE_BLK_SIZ 表示扇区大小 电脑上的U盘容量跟这两个参数密

    2024年04月28日
    浏览(45)
  • STM32F103C8用内部Flash做一个优盘(USB+MSC+FATFS)

    STM32F103C8用内部Flash做一个优盘(USB+MSC+FATFS),轻松实现APP升级、数据存储。 直接使用STM32CubeMX生成基本的工程,省得我们去调底层。 时钟配置为外部8MHz晶振,这个需要根据自己开发板的晶振选择。  启用SWD下载和滴答定时器  启用USB  启用FATFS,MAX_SS和MIN_SS设置为1024。  配

    2024年02月14日
    浏览(46)
  • STM32CubeMX系列09——SDIO(SD卡读写、SD卡移植FATFS文件系统)

    ==== 文章汇总(有代码汇总) ==== 准备看看这方面的知识,一时间还没不清有什么区别,先补补课,不需要的跳过。 参考文章(内容来源):http://www.360doc.com/content/21/1125/22/59057945_1005908465.shtml 主要写这两个:SD卡、TF卡 共同点:SD、TF、MMC都是在MMC基础上演化发展不同的规范,

    2024年02月09日
    浏览(49)
  • STM32CubeMX教程26 FatFs 文件系统 - W25Q128读写

    正点原子stm32f407探索者开发板V2.4 STM32CubeMX软件(Version 6.10.0) keil µVision5 IDE(MDK-Arm) ST-LINK/V2驱动 野火DAP仿真器 XCOM V2.6串口助手 使用STM32CubeMX软件配置STM32F407开发板 使用FatFs中间件通过SPI通信协议对W25Q128芯片进行读写等操作 关于STM32F407使用SPI通信协议对W25Q128 FLASH芯片读写

    2024年02月19日
    浏览(55)
  • STM32F429 Discovery开发板应用:实现SPI-SD Card文件写入(搭载FatFS文件系统)

    MCU:STM32F429ZIT6 开发环境:STM32CubeMX+MDK5   外购了一个SPI接口的SD Card模块,想要实现SD卡存储数据的功能。 首先需要打开STM32CubeMX工具。输入开发板MCU对应型号,找到开发板对应封装的MCU型号,双击打开(图中第三)。   此时,双击完后会关闭此界面,然后打开一个新界面。

    2024年02月08日
    浏览(59)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包