RTthread挂载外部Flash W25Q64并加入FATFS文件系统

     之前尝试移植FATFS以及在cubemx上使用FATFS组件但都失败了,痛苦了两星期,最近尝试用RTthread Studio对FATFS进行移植,发现过程还是很顺利的,因此决定记录一下:

硬件:正点原子miniV3.0开发板

软件:RTthread Studio

步骤按照这里的来就可以了:

4.RT-thread 项目实战--fatfs文件系统移植_rtthread fatfs-CSDN博客

我的由于是W25q64,程序有点不一样,如下:

#include "drv_spi.h"
#include "spi_flash_sfud.h"
#include "dfs_fs.h"
#include <dfs_posix.h>
#include <BSP_w25q64.h>

static int rt_hw_spi_flash_init(void)
{
    __HAL_RCC_GPIOA_CLK_ENABLE();
    rt_hw_spi_device_attach("spi1", "spi10", GPIOA, GPIO_PIN_2);// spi10 表示挂载在 spi1总线上的 0 号设备,PA2是片选,这一步就可以将从设备挂在到总线中。
    if (RT_NULL == rt_sfud_flash_probe("W25Q64", "spi10"))  //注册块设备,这一步可以将外部 flash抽象为系统的块设备
    {
        return -RT_ERROR;
    };

    return RT_EOK;
}
/* 导出到自动初始化 */
INIT_COMPONENT_EXPORT(rt_hw_spi_flash_init);



int dfs_mount_init(void)
{
    dfs_mkfs("elm","W25Q64");
    if(dfs_mount("W25Q64", "/","elm",0,0) == 0)  //注册块设备,这一步可以将外部flash抽象为系统的块设备
    {
        return -RT_ERROR;
    }
    return RT_EOK;
}
/* 导出到自动初始化 */
INIT_COMPONENT_EXPORT(dfs_mount_init);

这里说一下如果打印串口报如下错误的解决方法:

Warning: Read SFDP parameter header information failed. The W25Q64 is not support JEDEC SFDP.[0m
[32m[I/SFUD] Warning: This flash device is not found or not support.[0m
[32m[I/SFUD] Error: W25Q64 flash device is initialize fail.[0m
[31m[E/SFUD] ERROR: SPI flash probe failed by SPI device spi10.[0m

   这个说明RTthread的SFUD 组件在尝试初始化和识别 W25Q64 设备时遇到了问题,特别是与 JEDEC SFDP(Serial Flash Discoverable Parameters)相关的参数读取失败。这可能导致 SFUD 无法正确识别和初始化该 SPI Flash 设备。我一开始以为是W25Q64 设备不支持支持 SFDP。如果设备不支持 SFDP,就需要手动配置设备的参数,但是百度了一下是支持的,然后我就在RTthread里的cubemx中把片选引脚CS即PA2也初始化了,然后再次生成代码就可以了;(有个问题就是后面再取消cubemx的PA2初始化,生成代码后,仍然可以正常使用FATFS)

测试代码如下:

BSP_w25q64.c

// 测试文件操作
void test_file_operations(void)
{
   const char *dir_path = "/Test";

    // 创建并写入文件1
   create_and_write_file(FILE1_PATH, "This is file 1.\n");
   // 创建并写入文件2
   create_and_write_file(FILE2_PATH, "This is file 2.\n");

   // 读取文件1的内容
   read_file_content(FILE1_PATH);
   // 读取文件2的内容
   read_file_content(FILE2_PATH);

   //在根目录下创建一个文件夹Test
   create_folder(dir_path);
   // 创建并写入文件3
   create_and_write_file(FILE3_PATH, "This is file 3.\n");
   // 读取文件3的内容
   read_file_content(FILE3_PATH);
}



// 创建并写入文件
static void create_and_write_file(const char *file_path, const char *content)
{
    int fd;
    int bytes_written;

    // 打开文件,如果文件不存在则创建,如果文件存在则截断文件长度为0
    fd = open(file_path, O_WRONLY | O_CREAT | O_TRUNC);
    if (fd < 0)
    {
        rt_kprintf("Failed to open file for writing: %s\n", file_path);
        return;
    }

    // 写入文件
    bytes_written = write(fd, content, strlen(content));
    if (bytes_written < 0)
    {
        rt_kprintf("Failed to write to file: %s\n", file_path);
    }
    else
    {
        rt_kprintf("Wrote %d bytes to file: %s\n", bytes_written, file_path);
    }

    // 关闭文件
    close(fd);


}

// 读取文件内容
static void read_file_content(const char *file_path)
{
    int fd;
    char read_buffer[64];
    int bytes_read;

    // 打开文件
    fd = open(file_path, O_RDONLY);
    if (fd < 0)
    {
        rt_kprintf("Failed to open file for reading: %s\n", file_path);
        return;
    }

    // 读取文件内容
    //最多只能读sizeof(read_buffer) - 1是因为当数据超过sizeof(read_buffer) - 1大小时,read_buffer还能有一字节空间放'\0'
    bytes_read = read(fd, read_buffer, sizeof(read_buffer) - 1);
    if (bytes_read < 0)
    {
        rt_kprintf("Failed to read from file: %s\n", file_path);
    }
    else
    {
        read_buffer[bytes_read] = '\0'; // Null-terminate the string
        rt_kprintf("Read %d bytes from file: %s\n", bytes_read, file_path);
        rt_kprintf("File content: %s\n", read_buffer);
    }

    // 关闭文件
    close(fd);
}

// 创建新文件夹
int create_folder(const char *folder_path)
{

    if (mkdir(folder_path, S_IRWXU | S_IRWXG | S_IRWXO) == 0)
    {
        rt_kprintf("Folder created successfully: %s\n", folder_path);
        return 0; // 成功返回 0
    }
    else
    {
        rt_kprintf("Failed to create folder: %s\n", folder_path);
        return -1; // 失败返回 -1
    }
}

BSP_w25q64.h

#ifndef BSP_BSP_W25Q64_H_
#define BSP_BSP_W25Q64_H_

#define FILE1_PATH "/file1.txt"
#define FILE2_PATH "/file2.txt"
#define FILE3_PATH "/Test/file3.txt"

void test_file_operations(void);

static void create_and_write_file(const char *file_path, const char *content);
static void read_file_content(const char *file_path);
int create_folder(const char *folder_path);

#endif /* BSP_BSP_W25Q64_H_ */

测试结果如下:

参考文章:4.RT-thread 项目实战--fatfs文件系统移植_rtthread fatfs-CSDN博客

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值