问题描述:
使用stm32cubemx生成的sdio和usb Mass Storage Class的代码后,在USB_DEVICE\App\usbd_storage_if.c文件里面的接口调用以下函数出现卡死问题:
SD_Driver.disk_initialize(0);
SD_Driver.disk_read(lun, buf, blk_addr, blk_len)
SD_Driver.disk_write(lun, buf, blk_addr, blk_len);
解决办法:
1、STORAGE_Init_FS里面不直接调用disk_initialize,在MX_SDIO_SD_Init里面直接初始化sd卡,
2、读写sd卡的函数改为dma的读写接口:
HAL_SD_ReadBlocks_DMA(&hsd, buf, blk_addr, blk_len)
HAL_SD_WriteBlocks_DMA(&hsd, buf, blk_addr, blk_len)
USB_DEVICE\App\usbd_storage_if.c文件的具体代码如下:
/* USER CODE BEGIN Header */
/********************************************************************************* @file : usbd_storage_if.c* @version : v1.0_Cube* @brief : Memory management layer.******************************************************************************* @attention** Copyright (c) 2025 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 "sdio.h"
#include "sd_diskio.h"
#include "log_debug.h"
/* USER CODE END INCLUDE *//* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
/* Private macro -------------------------------------------------------------*//* USER CODE BEGIN PV */
/* Private variables ---------------------------------------------------------*//* 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/* 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 *///需要提取初始化sd卡,这里初始化sd卡会卡死if(HAL_SD_GetCardState(&hsd) == HAL_SD_CARD_TRANSFER){log_debug("STORAGE_Init_FS\n");return USBD_OK;}log_debug("STORAGE_Init_FS fail\n");return USBD_FAIL;/* 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 */HAL_SD_CardInfoTypeDef info;if(HAL_SD_GetCardState(&hsd) == HAL_SD_CARD_TRANSFER){HAL_SD_GetCardInfo(&hsd, &info);*block_num = info.LogBlockNbr;*block_size = info.LogBlockSize;//log_debug("SD_GetCardInfo: %d, %d\r\n", *block_num, *block_size);return USBD_OK;}log_debug("STORAGE_GetCapacity_FS Fail !!!\n");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 */if(HAL_SD_GetCardState(&hsd) == HAL_SD_CARD_TRANSFER){return USBD_OK;}log_debug("STORAGE_IsReady_FS Fail !!!\n");return USBD_FAIL;/* 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 *///log_debug("STORAGE_IsWriteProtected_FS: %d\r\n", 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 */int8_t ret = USBD_FAIL; if( HAL_SD_ReadBlocks_DMA(&hsd, buf, blk_addr, blk_len) == HAL_OK )// if( HAL_SD_ReadBlocks(&hsd, buf, blk_addr, blk_len, HAL_MAX_DELAY) == HAL_OK ){ret = USBD_OK;//while(HAL_SD_GetState(&hsd) == HAL_SD_STATE_BUSY){};while( HAL_SD_GetCardState(&hsd) != HAL_SD_CARD_TRANSFER ){};} else {ret = USBD_FAIL;log_debug("STORAGE_Read_FS Fail !!!\n");} return ret;/* 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 */int8_t ret = USBD_FAIL; if( HAL_SD_WriteBlocks_DMA(&hsd, buf, blk_addr, blk_len) == HAL_OK )// if( HAL_SD_WriteBlocks(&hsd, buf, blk_addr, blk_len, HAL_MAX_DELAY) == HAL_OK ){ret = USBD_OK;//while(HAL_SD_GetState(&hsd) == HAL_SD_STATE_BUSY){};while( HAL_SD_GetCardState(&hsd) != HAL_SD_CARD_TRANSFER ){};} else {ret = USBD_FAIL;log_debug("STORAGE_Write_FS Fail !!!\n");}return ret;/* 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 */
// log_debug("STORAGE_GetMaxLun_FS: %d\r\n", STORAGE_LUN_NBR);return (STORAGE_LUN_NBR - 1);/* USER CODE END 8 */
}/* USER CODE BEGIN PRIVATE_FUNCTIONS_IMPLEMENTATION *//* USER CODE END PRIVATE_FUNCTIONS_IMPLEMENTATION *//*** @}*//*** @}*/