基于W25QXX的fatfs文件系统

材料

  1. W25QXX驱动参考:https://github.com/iammingge/Driver_W25Qxx.git
  2. FATFS文件系统下载地址:https://github.com/abbrev/fatfs.git
  3. 基于上述两个库和合宙csdk构建fatfs文件系统📎usr_fatsfs_w25Q128_test.rar

工程目录

  • W25Qxx驱动(W25Qxx.c、W25Qxx.h)
  • fatfs库(diskio.c、diskio.h、ff.c、ff.h、ffconf.h、ffsystem.c、ffunicode.c)
  • 中间调用 (usr_lib_fatfs.c、usr_lib_fatfs.h)
  • 测试应用(example_main.c)

如何适配具体芯片

  1. usr_lib_fatfs.c,在spi_delay_ms、spi_readWrite、spi_cs_h、spi_cs_l、spi_init这几个函数中添加具体的操作函数,根据具体芯片确定
  1. diskio.c中的disk_status、disk_read、disk_write、disk_ioctl这几个函数中添加具体W25Qxx的操作函数
DSTATUS disk_status (BYTE pdrv) {
	DSTATUS status = STA_NOINIT;             // 设备状态   

    switch(pdrv)
    {
	case DISK_SPI:
	    if(W25Qxx_ID_Manufacturer(&testdev) == sFLASH_ID)
	    {
	        status = 0;                  // 设备ID读取成功
	    }
	    else
	    {
			status = STA_NOINIT;         // 设备ID读取失败
	    }
	    break;

	default:
	    status = STA_NOINIT;
    }
    return status;
}

DRESULT disk_read (BYTE pdrv, BYTE* buff, LBA_t sector, UINT count) {
	
	DRESULT status = RES_PARERR;
    switch(pdrv)
    {
	case DISK_SPI:
		for(;count>0;count--)
		{
			W25Qxx_Read(&testdev, buff, sector*FLASH_SECTOR_SIZE, FLASH_SECTOR_SIZE, &err);
			if (err != W25Qxx_ERR_NONE){
				status = RES_ERROR;
				return status;
			}
			sector++;
			buff+=FLASH_SECTOR_SIZE;
		}                              
            
            status = RES_OK;
	    break;

	default:
	    status = RES_PARERR;
	}

	return status;
}

DRESULT disk_write (BYTE pdrv, const BYTE* buff, LBA_t sector, UINT count) {
	
	DRESULT status = RES_PARERR;

    switch(pdrv)
    {
	case DISK_SPI:
		for(;count>0;count--){
			W25Qxx_Program(&testdev, (uint8_t *)buff, sector*FLASH_SECTOR_SIZE, FLASH_SECTOR_SIZE, &err);
			if (err != W25Qxx_ERR_NONE){
				status = RES_ERROR;
				return status;
			}
			sector++;
			buff+=FLASH_SECTOR_SIZE;
		}
	    
	    status = RES_OK;
	    break;

	default:
	    status = RES_PARERR;
    }

    return status;
}

DRESULT disk_ioctl (BYTE pdrv, BYTE cmd, void* buff) {
	DRESULT status = RES_PARERR;
    switch (pdrv) 
    {
	case DISK_SPI:
	    switch(cmd)
		{
			case CTRL_SYNC:
				status = RES_OK; 
				break;	 
			/* 扇区数量:6MB*1024*1024/4096=1536 */
			case GET_SECTOR_COUNT:
				*(DWORD *)buff = FLASH_SECTOR_COUNT;
				break;

					/* 扇区大小 */
			case GET_SECTOR_SIZE:
				*(WORD *) buff = FLASH_SECTOR_SIZE;
				break;

					/* 擦除的扇区个数 */
			case GET_BLOCK_SIZE:
				*(DWORD *)buff = FLASH_BLOCK_SIZE;
				break;
			case CTRL_TRIM:
				status = RES_OK; 
				break;	 
	    }
	    status = RES_OK;
	    break;

        default:
	    status = RES_PARERR;
    }
    return status;
}

DWORD get_fattime (void) {

}

应用

#include "usr_lib_fatfs.h"

luat_rtos_task_handle fatfs_task_handle;

FATFS fs;                         // FATFS文件系统结构体对象
FIL fnew;                         // FIL文件结构体对象
FRESULT res_flash;                // 文件操作结果结构体对象
UINT fnum;                        // 文件读写数量
BYTE ReadBuffer[1024] = {0};      // 读缓冲区
BYTE WriteBuffer[] = "FATFS测试"; // 写缓冲区

static void task_test_fatfs(void *param)
{
    w25qxx_init();

    res_flash = f_mount(&fs, "1:", 1); // 挂载时会对SPI设备初始化
    LUAT_DEBUG_PRINT("res_flash %d\r\n", res_flash);
    if (res_flash == FR_NO_FILESYSTEM) // 判断是否有文件系统,没有则格式化并创建
    {
        LUAT_DEBUG_PRINT("没有检测到文件系统...开始进行格式化...\r\n");
        res_flash = f_mkfs("1:", 1, 0, 4096);
        LUAT_DEBUG_PRINT("res_flash %d\r\n", res_flash);

        if (res_flash == FR_OK) // 格式化成功
        {
            LUAT_DEBUG_PRINT("已成功格式化\r\n");
            res_flash = f_mount(NULL, "1:", 1); // 格式化后先取消挂载
            res_flash = f_mount(&fs, "1:", 1);  // 重新挂载
            LUAT_DEBUG_PRINT("res_flash %d\r\n", res_flash);
        }
        else
        {
            LUAT_DEBUG_PRINT("格式化失败\r\n");
        }
    }
    else if (res_flash != FR_OK)
    {
        LUAT_DEBUG_PRINT("FatFS文件系统挂载失败(%d)\r\n", res_flash);
    }
    else
    {
        LUAT_DEBUG_PRINT("FatFS文件系统挂载成功,可以开始读写测试。\r\n");
        usr_get_fatfs_free("1:");
        usr_get_fatfs_list_files("1:");
    }

    res_flash = f_open(&fnew, "1:/FatFS读写测试文件.txt", FA_CREATE_NEW | FA_WRITE | FA_READ);
    LUAT_DEBUG_PRINT("res_flash %d\r\n", res_flash);
    if (res_flash == FR_OK)
    {
        LUAT_DEBUG_PRINT("打开/创建 FatFS读写测试文件.txt 成功。\r\n\r\n");
        res_flash = f_write(&fnew, WriteBuffer, sizeof(WriteBuffer), &fnum); // 将写缓冲区内容写入到文件内
        if (res_flash == FR_OK)
        {
            LUAT_DEBUG_PRINT("写入的字节数据大小:%d\n", fnum);
            LUAT_DEBUG_PRINT("写入的内容为:%s\r\n\r\n", WriteBuffer);
        }
        else
        {
            LUAT_DEBUG_PRINT("写入失败(%d)\n", res_flash);
        }
    }
    else
    {
        LUAT_DEBUG_PRINT("打开/创建文件失败!!!\r\n");
    }

    f_close(&fnew);

    while (1)
    {
        luat_rtos_task_sleep(1000);
    }
}

static void task_demo_fatfs(void)
{
    luat_rtos_task_create(&fatfs_task_handle, 4096, 20, "fatfs", task_test_fatfs, NULL, NULL);
}

INIT_TASK_EXPORT(task_demo_fatfs, "1");
  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要将w25q128移植到fatfs文件系统中,需要按照以下步骤进行操作: 1. 首先,需要在系统中添加w25q128驱动程序,以便能够与芯片进行通信。 2. 接下来,需要在fatfs文件系统中添加w25q128驱动程序的支持,以便能够在文件系统中读写数据。 3. 然后,需要在系统中初始化fatfs文件系统,并将其与w25q128驱动程序进行关联。 4. 最后,可以在应用程序中使用fatfs文件系统来读写w25q128芯片中的数据。 需要注意的是,在移植过程中,需要根据具体的系统和芯片进行相应的调整和修改,以确保系统能够正常运行。 ### 回答2: W25Q128是一款Flash存储器,可以在嵌入式系统中使用。而FATFS是一个在嵌入式系统中常用的文件系统,本文将介绍如何将FATFS文件系统移植到W25Q128中。 一般来说,将文件系统移植到Flash存储器需要考虑以下几个方面: 1. 块大小 Flash存储器有特定的块大小。如果文件系统使用的块大小与Flash存储器不匹配,可能会导致空间浪费或读写效率低下。因此,需要在移植时考虑Flash存储器的块大小,并根据实际情况对文件系统进行适配。 2. 页大小 与块大小类似,Flash存储器还有特定的页大小。如果文件系统使用的页大小与Flash存储器不匹配,则可能会导致写入数据时无法正确对齐,进而导致数据损坏。因此,需要在移植时考虑Flash存储器的页大小,与文件系统的页大小进行适配。 3. 坏块管理 Flash存储器有可能出现坏块,需对其进行管理。一般有2种方式:动态坏块管理和静态坏块管理。动态坏块管理是指在运行时对坏块进行检测和处理;静态坏块管理是指在生产时对坏块进行识别和标记。一般来说,静态坏块管理的效率更高,但在使用过程中可能会出现新的坏块,需要对其进行动态管理。 4. 数据完整性 Flash存储器是一种非易失性存储器,但它的写入过程不具有原子性。因此,在写入数据时需要考虑数据的完整性。一般有2种方式:冗余备份和写入前检验。冗余备份是指将同一数据存储在2个地方,如果读取时发现其中一个数据有损坏,可以使用另一个数据进行恢复;写入前检验则是在写入数据时先检验数据的完整性,如果检验失败则不写入,避免数据损坏。 综上所述,将FATFS文件系统移植到W25Q128中,需要考虑以上几个方面,并根据实际情况进行适配,以确保文件系统的稳定性和可靠性。 ### 回答3: W25Q128是一款SPI串行Flash存储器,可以用于嵌入式系统中存储数据。FATFS文件系统是一款非常流行的文件系统,被广泛应用于各种嵌入式系统中。本文将介绍如何将FATFS文件系统移植到W25Q128串行Flash存储器上。 1. 硬件准备 首先需要准备好一款支持SPI接口的MCU和一块W25Q128串行Flash存储器。常用的MCU有STM32、ATmega等,根据MCU不同,代码实现有所不同。本文以STM32为例。同时需要准备好SPI的时钟、引脚和片选等参数。还需要购买一张SD卡,用于模拟W25Q128串行Flash存储器。 2. 软件准备 在将FATFS文件系统移植到W25Q128上之前,需要先编写SPI驱动程序,确保MCU能够与W25Q128串行Flash存储器通信。可以选择使用标准库或HAL库来编写SPI驱动程序。接着,需要下载FatFs库,并将源代码拷贝到工程文件夹中。 3. 移植FATFS文件系统 移植FATFS文件系统需要完成以下几个步骤: 1)定义磁盘接口结构体,并实现初始化、读写扇区等自定义函数,用于与W25Q128串行Flash存储器交互。 2)调用f_mount函数,将磁盘接口与FATFS文件系统进行绑定,可以通过该接口读写文件及目录。 3)编写测试代码,调用FATFS的API函数,读写文件等。 4. 样例代码 以下是一份基于STM32的代码示例,用于演示如何将FATFS文件系统移植到W25Q128串行Flash存储器上。 ```c #include "stm32f10x.h" #include "diskio.h" #include "ff.h" FATFS fs; FIL fil; BYTE work[FF_MAX_SS]; void SysTick_Handler(void) { static WORD cnt; disk_timerproc(); if (cnt++ >= 10) { cnt = 0; GPIOA->ODR ^= (1<<8); } } DSTATUS disk_initialize(BYTE pdrv) { return 0; } DRESULT disk_read(BYTE pdrv, BYTE* buff, DWORD sector, UINT count) { for (int i = 0; i < count; i++) { SPI_Read(buff+i*512, sector+i, 512); } return RES_OK; } DRESULT disk_write(BYTE pdrv, const BYTE* buff, DWORD sector, UINT count) { for (int i = 0; i < count; i++) { SPI_Write(buff+i*512, sector+i, 512); } return RES_OK; } DRESULT disk_ioctl(BYTE pdrv, BYTE cmd, void* buff) { return RES_OK; } int main(void) { SystemInit(); SysTick_Config(SystemCoreClock/100); RCC->APB2ENR |= RCC_APB2ENR_IOPAEN; GPIOA->CRL &= ~GPIO_CRL_CNF8; GPIOA->CRL |= GPIO_CRL_MODE8_1; GPIOA->ODR |= (1<<8); //SPI初始化 SPI_InitTypeDef spi; spi.SPI_Direction = SPI_Direction_2Lines_FullDuplex; spi.SPI_Mode = SPI_Mode_Master; spi.SPI_DataSize = SPI_DataSize_8b; spi.SPI_CPOL = SPI_CPOL_Low; spi.SPI_CPHA = SPI_CPHA_1Edge; spi.SPI_NSS = SPI_NSS_Soft; spi.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4; spi.SPI_FirstBit = SPI_FirstBit_MSB; spi.SPI_CRCPolynomial = 0; SPI_Init(SPI1, &spi); SPI_Cmd(SPI1, ENABLE); //SD卡初始化 SD_Init(); //磁盘接口结构体 static const Diskio_drvTypeDef pdrv = { disk_initialize, disk_read, disk_write, disk_ioctl, }; //挂载文件系统 f_mount(&fs, "", 0); //打开文件 f_open(&fil, "hello.txt", FA_WRITE|FA_CREATE_ALWAYS); //写入文件 char buf[] = "Hello FATFS!"; UINT bw; f_write(&fil, buf, strlen(buf), &bw); f_close(&fil); //读取文件 f_open(&fil, "hello.txt", FA_READ); char read_buf[100]; f_read(&fil, read_buf, fil.fsize, &bw); f_close(&fil); //复制文件 f_open(&fil, "hello2.txt", FA_WRITE|FA_CREATE_ALWAYS); f_lseek(&fil, 0); f_truncate(&fil); f_copy("hello.txt", "hello2.txt"); f_close(&fil); while(1); } ``` 5. 结论 本文介绍了如何将FATFS文件系统移植到W25Q128串行Flash存储器上,并提供了一份基于STM32的代码示例。移植过程需要完成SPI驱动程序的编写、磁盘接口结构体的定义和实现、以及文件系统的绑定和API调用等步骤。对于初学者而言,需要具备一定的MCU编程和SPI协议的基础知识。移植过程中需要注意时序和命令格式等细节问题,以保证与W25Q128串行Flash存储器的正常通信。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值