Fork me on GitHub

FreeRTOS移植SFUD

1. SFUD

SFUD是一款串行 Flash 通用驱动库,GitHub:https://link.zhihu.com/?target=https%3A//github.com/armink/SFUD  Gitee:https://gitee.com/Armink/SFUD

2. SFUD的移植

① 使用CubeMX配置spi

② 复制sfud源码文件到工程中,在keil中建立工程对应的目录,包含对应的头文件

③ 改写sfud_port.c中的函数

/*
 * This file is part of the Serial Flash Universal Driver Library.
 *
 * Copyright (c) 2016-2018, Armink, <armink.ztl@gmail.com>
 *
 * Permission is hereby granted, free of charge, to any person obtaining
 * a copy of this software and associated documentation files (the
 * 'Software'), to deal in the Software without restriction, including
 * without limitation the rights to use, copy, modify, merge, publish,
 * distribute, sublicense, and/or sell copies of the Software, and to
 * permit persons to whom the Software is furnished to do so, subject to
 * the following conditions:
 *
 * The above copyright notice and this permission notice shall be
 * included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 *
 * Function: Portable interface for each platform.
 * Created on: 2016-04-23
 */
#include <sfud.h>
#include <stdarg.h>
#include "main.h"
#include "spi.h"
#include "FreeRTOS.h"
#include "task.h"
#include "semphr.h"
#include "BSP_SPIFlash.h"
#include "BSP_DebugUART.h"
//#define SFUD_USE_ELOG

#ifdef SFUD_USE_ELOG
#include <elog.h>
#endif

static char log_buf[256];

void sfud_log_debug(const char *file, const long line, const char *format, ...);

static void spi_lock(const sfud_spi *spi) {
	/* 裸机中使用开关中断方法 */ 
//    __disable_irq(); 
	/* OS 中使用临界区或互斥量,推荐互斥量方式 */ 
//	taskENTER_CRITICAL();  /* 进入临界区 */
	xSemaphoreTake(xMutexSPIFlash, portMAX_DELAY);  /* 互斥信号量 */
}

static void spi_unlock(const sfud_spi *spi) {
	/* 裸机中使用开关中断方法 */ 
//    __enable_irq(); 
	/* OS 中使用临界区或互斥量,推荐互斥量方式 */ 
//	taskEXIT_CRITICAL();  /* 退出临界区 */
	xSemaphoreGive(xMutexSPIFlash);  /* 互斥信号量 */
}

/**
 * SPI write data then read data
 */
static sfud_err spi_write_read(const sfud_spi *spi, const uint8_t *write_buf, size_t write_size, uint8_t *read_buf,
        size_t read_size) {
    sfud_err result = SFUD_SUCCESS;
//    uint8_t send_data, read_data;

    /**
     * add your spi write and read code
     */
    if (write_size) {
        SFUD_ASSERT(write_buf);
    }
    if (read_size) {
        SFUD_ASSERT(read_buf);
    }

//	HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_RESET);
//	if( HAL_OK != HAL_SPI_TransmitReceive(&hspi2, (uint8_t *)write_buf, (uint8_t *)read_buf, write_size+read_size, 1000) )
//		result = SFUD_ERR_TIMEOUT;
//	HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_SET);

	if(read_size != NULL)
	{
		HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_RESET);
		if( HAL_OK != HAL_SPI_Transmit(&hspi1, (uint8_t *)write_buf, write_size, 1000) )
			result = SFUD_ERR_TIMEOUT;
//		HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_SET);
//		HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_RESET);
		if( HAL_OK != HAL_SPI_Receive(&hspi1, (uint8_t *)read_buf, read_size, 1000) )
			result = SFUD_ERR_TIMEOUT;
		HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_SET);
	}
	else
	{
		HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_RESET);
		if( HAL_OK != HAL_SPI_Transmit(&hspi1, (uint8_t *)write_buf, write_size, 1000) )
			result = SFUD_ERR_TIMEOUT;
		HAL_GPIO_WritePin(SPIFlashCS_GPIO_Port, SPIFlashCS_Pin, GPIO_PIN_SET);
	}
	
    return result;
}

#ifdef SFUD_USING_QSPI
/**
 * read flash data by QSPI
 */
static sfud_err qspi_read(const struct __sfud_spi *spi, uint32_t addr, sfud_qspi_read_cmd_format *qspi_read_cmd_format,
        uint8_t *read_buf, size_t read_size) {
    sfud_err result = SFUD_SUCCESS;

    /**
     * add your qspi read flash data code
     */

    return result;
}
#endif /* SFUD_USING_QSPI */

/* about 100 microsecond delay */
static void retry_delay_100us(void) {
    uint32_t delay = 1700*2;
    while(delay--);
}

sfud_err sfud_spi_port_init(sfud_flash *flash) {
    sfud_err result = SFUD_SUCCESS;

    /**
     * add your port spi bus and device object initialize code like this:
     * 1. rcc initialize
     * 2. gpio initialize
     * 3. spi device initialize
     * 4. flash->spi and flash->retry item initialize
     *    flash->spi.wr = spi_write_read; //Required
     *    flash->spi.qspi_read = qspi_read; //Required when QSPI mode enable
     *    flash->spi.lock = spi_lock;
     *    flash->spi.unlock = spi_unlock;
     *    flash->spi.user_data = &spix;
     *    flash->retry.delay = null;
     *    flash->retry.times = 10000; //Required
     */

    switch (flash->index) {
		case SFUD_W25_DEVICE_INDEX: {
			/* 同步 Flash 移植所需的接口及数据 */
			flash->spi.wr = spi_write_read;
			flash->spi.lock = spi_lock;
			flash->spi.unlock = spi_unlock;
			/* about 100 microsecond delay */
			flash->retry.delay = retry_delay_100us;
			/* adout 60 seconds timeout */
			flash->retry.times = 60 * 10000;
			break;
		}
    }

    return result;
}

/**
 * This function is print debug info.
 *
 * @param file the file which has call this function
 * @param line the line number which has call this function
 * @param format output format
 * @param ... args
 */
void sfud_log_debug(const char *file, const long line, const char *format, ...) {
    va_list args;

    /* args point to the first variable parameter */
    va_start(args, format);
#ifdef SFUD_USE_ELOG
    elog_raw("[SFUD](%s:%ld) ", file, line);
#else
    BSP_Printf("[SFUD](%s:%ld) ", file, line);
#endif
    /* must use vprintf to print */
    vsnprintf(log_buf, sizeof(log_buf), format, args);
#ifdef SFUD_USE_ELOG
    elog_raw("%s\n", log_buf);
#else
    BSP_Printf("%s\n", log_buf);
#endif
    va_end(args);
}

/**
 * This function is print routine info.
 *
 * @param format output format
 * @param ... args
 */
void sfud_log_info(const char *format, ...) {
    va_list args;

    /* args point to the first variable parameter */
    va_start(args, format);
#ifdef SFUD_USE_ELOG
    elog_raw("[SFUD]");
#else
    BSP_Printf("[SFUD]");
#endif
    /* must use vprintf to print */
    vsnprintf(log_buf, sizeof(log_buf), format, args);
#ifdef SFUD_USE_ELOG
    elog_raw("%s\n", log_buf);
#else
    BSP_Printf("%s\n", log_buf);
#endif
    va_end(args);
}

④ 改写sfud_cfg.h

/*
 * This file is part of the Serial Flash Universal Driver Library.
 *
 * Copyright (c) 2016-2018, Armink, <armink.ztl@gmail.com>
 *
 * Permission is hereby granted, free of charge, to any person obtaining
 * a copy of this software and associated documentation files (the
 * 'Software'), to deal in the Software without restriction, including
 * without limitation the rights to use, copy, modify, merge, publish,
 * distribute, sublicense, and/or sell copies of the Software, and to
 * permit persons to whom the Software is furnished to do so, subject to
 * the following conditions:
 *
 * The above copyright notice and this permission notice shall be
 * included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 *
 * Function: It is the configure head file for this library.
 * Created on: 2016-04-23
 */
#ifndef _SFUD_CFG_H_
#define _SFUD_CFG_H_
#define SFUD_DEBUG_MODE
#define SFUD_USING_SFDP
//#define SFUD_USING_FLASH_INFO_TABLE
enum {
    SFUD_W25_DEVICE_INDEX = 0,
};
#define SFUD_FLASH_DEVICE_TABLE                                                \
{                                                                              \
    [SFUD_W25_DEVICE_INDEX] = {.name = "W25Q128F", .spi.name = "SPI1"},           \
}
//#define SFUD_USING_QSPI
#endif /* _SFUD_CFG_H_ */

3. SFUD的使用

可单独使用,查看说明文档中的API。此处配合后面的easyflash使用

posted @ 2021-06-22 21:51  一地鸡毛-  阅读(1274)  评论(0编辑  收藏  举报