- 新增usb连接状态检测函数,根据usb连接状态自动切换数据采集模式 - 新增文件系统动态挂载/卸载功能,usb连接时卸载文件系统,断开时重新挂载 - 修改系统启动逻辑,根据初始usb连接状态决定是否开始数据采集 - 新增usb模式系统状态,完善系统状态机 🐛 fix(sdio): 修复sdio配置问题并启用中断 - 修改sdio时钟分频为2,优化sd nand通信时序 - 启用sdio数据线内部上拉电阻,提高信号稳定性 - 提高sdio dma传输优先级至最高,确保数据传输实时性 - 启用sdio全局中断并设置优先级为9 🔧 chore(config): 优化系统配置参数 - 修改系统滴答定时器中断优先级为0(最高优先级) - 增加堆栈大小至0x1000,增加堆大小至0x800 - 修改usb msc媒体数据包大小至32768,提高usb传输效率 - 修改fatfs配置,设置最大最小扇区大小为512字节 ♻️ refactor(usb): 重构usb存储接口实现 - 修改usb存储初始化逻辑,避免重复初始化sd卡 - 优化usb存储容量报告机制,强制报告512字节扇区 - 增加sd nand读写超时等待机制,确保数据传输完成 - 修改usb中断优先级为11,避免与sdio中断冲突 📝 docs(headers): 更新头文件声明 - 在stm32f4xx_it.h中添加sdio中断处理函数声明 - 在system_monitor.h中添加usb模式系统状态定义 - 更新data_storage.h中的数据存储路径配置
364 lines
8.9 KiB
C
364 lines
8.9 KiB
C
/* USER CODE BEGIN Header */
|
||
/**
|
||
******************************************************************************
|
||
* @file : usbd_storage_if.c
|
||
* @version : v1.0_Cube
|
||
* @brief : Memory management layer.
|
||
******************************************************************************
|
||
* @attention
|
||
*
|
||
* Copyright (c) 2026 STMicroelectronics.
|
||
* All rights reserved.
|
||
*
|
||
* This software is licensed under terms that can be found in the LICENSE file
|
||
* in the root directory of this software component.
|
||
* If no LICENSE file comes with this software, it is provided AS-IS.
|
||
*
|
||
******************************************************************************
|
||
*/
|
||
/* USER CODE END Header */
|
||
|
||
/* Includes ------------------------------------------------------------------*/
|
||
#include "usbd_storage_if.h"
|
||
|
||
/* USER CODE BEGIN INCLUDE */
|
||
#include "fatfs.h"
|
||
#include "ff.h"
|
||
/* USER CODE END INCLUDE */
|
||
|
||
/* Private typedef -----------------------------------------------------------*/
|
||
/* Private define ------------------------------------------------------------*/
|
||
/* Private macro -------------------------------------------------------------*/
|
||
|
||
/* USER CODE BEGIN PV */
|
||
/* Private variables ---------------------------------------------------------*/
|
||
extern SD_HandleTypeDef hsd;
|
||
extern FATFS SDFatFS;
|
||
extern char SDPath[4];
|
||
|
||
static uint8_t sd_initialized = 0;
|
||
/* USER CODE END PV */
|
||
|
||
/** @addtogroup STM32_USB_OTG_DEVICE_LIBRARY
|
||
* @brief Usb device.
|
||
* @{
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE
|
||
* @brief Usb mass storage device module
|
||
* @{
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE_Private_TypesDefinitions
|
||
* @brief Private types.
|
||
* @{
|
||
*/
|
||
|
||
/* USER CODE BEGIN PRIVATE_TYPES */
|
||
|
||
/* USER CODE END PRIVATE_TYPES */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE_Private_Defines
|
||
* @brief Private defines.
|
||
* @{
|
||
*/
|
||
|
||
#define STORAGE_LUN_NBR 1
|
||
#define STORAGE_BLK_NBR 0x10000
|
||
#define STORAGE_BLK_SIZ 0x200
|
||
|
||
/* USER CODE BEGIN PRIVATE_DEFINES */
|
||
|
||
/* USER CODE END PRIVATE_DEFINES */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE_Private_Macros
|
||
* @brief Private macros.
|
||
* @{
|
||
*/
|
||
|
||
/* USER CODE BEGIN PRIVATE_MACRO */
|
||
|
||
/* USER CODE END PRIVATE_MACRO */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE_Private_Variables
|
||
* @brief Private variables.
|
||
* @{
|
||
*/
|
||
|
||
/* USER CODE BEGIN INQUIRY_DATA_FS */
|
||
/** USB Mass storage Standard Inquiry Data. */
|
||
const int8_t STORAGE_Inquirydata_FS[] = {/* 36 */
|
||
|
||
/* LUN 0 */
|
||
0x00,
|
||
0x80,
|
||
0x02,
|
||
0x02,
|
||
(STANDARD_INQUIRY_DATA_LEN - 5),
|
||
0x00,
|
||
0x00,
|
||
0x00,
|
||
'S', 'T', 'M', ' ', ' ', ' ', ' ', ' ', /* Manufacturer : 8 bytes */
|
||
'P', 'r', 'o', 'd', 'u', 'c', 't', ' ', /* Product : 16 Bytes */
|
||
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
||
'0', '.', '0' ,'1' /* Version : 4 Bytes */
|
||
};
|
||
/* USER CODE END INQUIRY_DATA_FS */
|
||
|
||
/* USER CODE BEGIN PRIVATE_VARIABLES */
|
||
|
||
/* USER CODE END PRIVATE_VARIABLES */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE_Exported_Variables
|
||
* @brief Public variables.
|
||
* @{
|
||
*/
|
||
|
||
extern USBD_HandleTypeDef hUsbDeviceFS;
|
||
|
||
/* USER CODE BEGIN EXPORTED_VARIABLES */
|
||
|
||
/* USER CODE END EXPORTED_VARIABLES */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
/** @defgroup USBD_STORAGE_Private_FunctionPrototypes
|
||
* @brief Private functions declaration.
|
||
* @{
|
||
*/
|
||
|
||
static int8_t STORAGE_Init_FS(uint8_t lun);
|
||
static int8_t STORAGE_GetCapacity_FS(uint8_t lun, uint32_t *block_num, uint16_t *block_size);
|
||
static int8_t STORAGE_IsReady_FS(uint8_t lun);
|
||
static int8_t STORAGE_IsWriteProtected_FS(uint8_t lun);
|
||
static int8_t STORAGE_Read_FS(uint8_t lun, uint8_t *buf, uint32_t blk_addr, uint16_t blk_len);
|
||
static int8_t STORAGE_Write_FS(uint8_t lun, uint8_t *buf, uint32_t blk_addr, uint16_t blk_len);
|
||
static int8_t STORAGE_GetMaxLun_FS(void);
|
||
|
||
/* USER CODE BEGIN PRIVATE_FUNCTIONS_DECLARATION */
|
||
|
||
/* USER CODE END PRIVATE_FUNCTIONS_DECLARATION */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
USBD_StorageTypeDef USBD_Storage_Interface_fops_FS =
|
||
{
|
||
STORAGE_Init_FS,
|
||
STORAGE_GetCapacity_FS,
|
||
STORAGE_IsReady_FS,
|
||
STORAGE_IsWriteProtected_FS,
|
||
STORAGE_Read_FS,
|
||
STORAGE_Write_FS,
|
||
STORAGE_GetMaxLun_FS,
|
||
(int8_t *)STORAGE_Inquirydata_FS
|
||
};
|
||
|
||
/* Private functions ---------------------------------------------------------*/
|
||
/**
|
||
* @brief Initializes the storage unit (medium) over USB FS IP
|
||
* @param lun: Logical unit number.
|
||
* @retval USBD_OK if all operations are OK else USBD_FAIL
|
||
*/
|
||
int8_t STORAGE_Init_FS(uint8_t lun)
|
||
{
|
||
/* USER CODE BEGIN 2 */
|
||
UNUSED(lun);
|
||
|
||
// 1. 检查 SD 句柄状态
|
||
// 如果之前在 main.c 里已经 HAL_SD_Init 过了,这里只要确认状态即可
|
||
HAL_SD_StateTypeDef state = HAL_SD_GetState(&hsd);
|
||
|
||
if(state == HAL_SD_STATE_RESET)
|
||
{
|
||
// 只有在从未初始化的情况下才初始化
|
||
if (HAL_SD_Init(&hsd) != HAL_OK) {
|
||
return (USBD_FAIL);
|
||
}
|
||
}
|
||
|
||
// 2. 绝对不要在这里 f_mount 或 f_mkfs !!!
|
||
// USB 只是搬运工,文件系统由电脑端管理。
|
||
|
||
sd_initialized = 1;
|
||
return (USBD_OK);
|
||
/* USER CODE END 2 */
|
||
}
|
||
|
||
/**
|
||
* @brief Returns the medium capacity.
|
||
* @param lun: Logical unit number.
|
||
* @param block_num: Number of total block number.
|
||
* @param block_size: Block size.
|
||
* @retval USBD_OK if all operations are OK else USBD_FAIL
|
||
*/
|
||
int8_t STORAGE_GetCapacity_FS(uint8_t lun, uint32_t *block_num, uint16_t *block_size)
|
||
{
|
||
/* USER CODE BEGIN 3 */
|
||
UNUSED(lun);
|
||
|
||
HAL_SD_CardInfoTypeDef cardinfo;
|
||
|
||
if (HAL_SD_GetCardInfo(&hsd, &cardinfo) == HAL_OK)
|
||
{
|
||
// 修正 2: 欺骗电脑,强制报告 512 字节扇区
|
||
*block_size = 512;
|
||
|
||
if (cardinfo.LogBlockSize == 512) {
|
||
*block_num = cardinfo.LogBlockNbr - 1;
|
||
}
|
||
else {
|
||
// 如果 SD NAND 是 2048/4096 字节页,需要转换块数量
|
||
// 例如:实际 4096,那 USB 看到的块数就是 实际块数 * 8
|
||
uint32_t ratio = cardinfo.LogBlockSize / 512;
|
||
*block_num = (cardinfo.LogBlockNbr * ratio) - 1;
|
||
}
|
||
return (USBD_OK);
|
||
}
|
||
|
||
return (USBD_FAIL);
|
||
/* USER CODE END 3 */
|
||
}
|
||
|
||
/**
|
||
* @brief Checks whether the medium is ready.
|
||
* @param lun: Logical unit number.
|
||
* @retval USBD_OK if all operations are OK else USBD_FAIL
|
||
*/
|
||
int8_t STORAGE_IsReady_FS(uint8_t lun)
|
||
{
|
||
/* USER CODE BEGIN 4 */
|
||
UNUSED(lun);
|
||
|
||
// 简单检查硬件状态
|
||
if (hsd.State == HAL_SD_STATE_RESET) {
|
||
return (USBD_FAIL);
|
||
}
|
||
|
||
// 对于 SD NAND,即使内部 BUSY,为了不让 USB 掉线,通常也返回 OK
|
||
// 实际的等待放到 Read/Write 函数里去做
|
||
return (USBD_OK);
|
||
/* USER CODE END 4 */
|
||
}
|
||
|
||
/**
|
||
* @brief Checks whether the medium is write protected.
|
||
* @param lun: Logical unit number.
|
||
* @retval USBD_OK if all operations are OK else USBD_FAIL
|
||
*/
|
||
int8_t STORAGE_IsWriteProtected_FS(uint8_t lun)
|
||
{
|
||
/* USER CODE BEGIN 5 */
|
||
UNUSED(lun);
|
||
|
||
return (USBD_OK);
|
||
/* USER CODE END 5 */
|
||
}
|
||
|
||
/**
|
||
* @brief Reads data from the medium.
|
||
* @param lun: Logical unit number.
|
||
* @param buf: data buffer.
|
||
* @param blk_addr: Logical block address.
|
||
* @param blk_len: Blocks number.
|
||
* @retval USBD_OK if all operations are OK else USBD_FAIL
|
||
*/
|
||
int8_t STORAGE_Read_FS(uint8_t lun, uint8_t *buf, uint32_t blk_addr, uint16_t blk_len)
|
||
{
|
||
/* USER CODE BEGIN 6 */
|
||
UNUSED(lun);
|
||
|
||
// 使用较长的超时时间 (SD NAND 读写较慢)
|
||
uint32_t timeout = 2000;
|
||
|
||
if (HAL_SD_ReadBlocks(&hsd, buf, blk_addr, blk_len, timeout) != HAL_OK) {
|
||
return (USBD_FAIL);
|
||
}
|
||
|
||
// 等待传输完成 (重要: 确保 DMA/中断搬运完毕)
|
||
uint32_t tickstart = HAL_GetTick();
|
||
while(HAL_SD_GetCardState(&hsd) != HAL_SD_CARD_TRANSFER)
|
||
{
|
||
if((HAL_GetTick() - tickstart) > timeout) return USBD_FAIL;
|
||
}
|
||
|
||
return (USBD_OK);
|
||
/* USER CODE END 6 */
|
||
}
|
||
|
||
/**
|
||
* @brief Writes data into the medium.
|
||
* @param lun: Logical unit number.
|
||
* @param buf: data buffer.
|
||
* @param blk_addr: Logical block address.
|
||
* @param blk_len: Blocks number.
|
||
* @retval USBD_OK if all operations are OK else USBD_FAIL
|
||
*/
|
||
int8_t STORAGE_Write_FS(uint8_t lun, uint8_t *buf, uint32_t blk_addr, uint16_t blk_len)
|
||
{
|
||
/* USER CODE BEGIN 7 */
|
||
UNUSED(lun);
|
||
|
||
// 修正 3: 增加超时时间,并等待 Busy 结束
|
||
uint32_t timeout = 5000; // 给 5秒,SD NAND 写入由于要擦除/编程,很慢
|
||
|
||
if (HAL_SD_WriteBlocks(&hsd, buf, blk_addr, blk_len, timeout) != HAL_OK) {
|
||
return (USBD_FAIL);
|
||
}
|
||
|
||
// 【最关键的一步】等待 SD NAND 内部编程结束
|
||
// 如果没有这一步,文件就会丢失!
|
||
uint32_t tickstart = HAL_GetTick();
|
||
while(HAL_SD_GetCardState(&hsd) != HAL_SD_CARD_TRANSFER)
|
||
{
|
||
if((HAL_GetTick() - tickstart) > timeout) return USBD_FAIL;
|
||
}
|
||
|
||
return (USBD_OK);
|
||
/* USER CODE END 7 */
|
||
}
|
||
|
||
/**
|
||
* @brief Returns the Max Supported LUNs.
|
||
* @param None
|
||
* @retval Lun(s) number.
|
||
*/
|
||
int8_t STORAGE_GetMaxLun_FS(void)
|
||
{
|
||
/* USER CODE BEGIN 8 */
|
||
return (STORAGE_LUN_NBR - 1);
|
||
/* USER CODE END 8 */
|
||
}
|
||
|
||
/* USER CODE BEGIN PRIVATE_FUNCTIONS_IMPLEMENTATION */
|
||
|
||
/* USER CODE END PRIVATE_FUNCTIONS_IMPLEMENTATION */
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|
||
/**
|
||
* @}
|
||
*/
|
||
|