You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
683 lines
18 KiB
683 lines
18 KiB
/* USER CODE BEGIN Header */ |
|
/** |
|
****************************************************************************** |
|
* @file sd_diskio.c |
|
* @brief SD Disk I/O driver |
|
****************************************************************************** |
|
* @attention |
|
* |
|
* Copyright (c) 2022 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 */ |
|
|
|
/* Note: code generation based on sd_diskio_dma_rtos_template_bspv1.c v2.1.4 |
|
as FreeRTOS is enabled. */ |
|
|
|
/* USER CODE BEGIN firstSection */ |
|
/* can be used to modify / undefine following code or add new definitions */ |
|
/* USER CODE END firstSection*/ |
|
|
|
/* Includes ------------------------------------------------------------------*/ |
|
#include "ff_gen_drv.h" |
|
#include "sd_diskio.h" |
|
|
|
#include <string.h> |
|
#include <stdio.h> |
|
|
|
/* Private typedef -----------------------------------------------------------*/ |
|
/* Private define ------------------------------------------------------------*/ |
|
|
|
#define QUEUE_SIZE (uint32_t) 10 |
|
#define READ_CPLT_MSG (uint32_t) 1 |
|
#define WRITE_CPLT_MSG (uint32_t) 2 |
|
/* |
|
================================================================== |
|
enable the defines below to send custom rtos messages |
|
when an error or an abort occurs. |
|
Notice: depending on the HAL/SD driver the HAL_SD_ErrorCallback() |
|
may not be available. |
|
See BSP_SD_ErrorCallback() and BSP_SD_AbortCallback() below |
|
================================================================== |
|
|
|
#define RW_ERROR_MSG (uint32_t) 3 |
|
#define RW_ABORT_MSG (uint32_t) 4 |
|
*/ |
|
/* |
|
* the following Timeout is useful to give the control back to the applications |
|
* in case of errors in either BSP_SD_ReadCpltCallback() or BSP_SD_WriteCpltCallback() |
|
* the value by default is as defined in the BSP platform driver otherwise 30 secs |
|
*/ |
|
#define SD_TIMEOUT 30 * 1000 |
|
|
|
#define SD_DEFAULT_BLOCK_SIZE 512 |
|
|
|
/* |
|
* Depending on the use case, the SD card initialization could be done at the |
|
* application level: if it is the case define the flag below to disable |
|
* the BSP_SD_Init() call in the SD_Initialize() and add a call to |
|
* BSP_SD_Init() elsewhere in the application. |
|
*/ |
|
/* USER CODE BEGIN disableSDInit */ |
|
/* #define DISABLE_SD_INIT */ |
|
/* USER CODE END disableSDInit */ |
|
|
|
/* |
|
* when using cacheable memory region, it may be needed to maintain the cache |
|
* validity. Enable the define below to activate a cache maintenance at each |
|
* read and write operation. |
|
* Notice: This is applicable only for cortex M7 based platform. |
|
*/ |
|
/* USER CODE BEGIN enableSDDmaCacheMaintenance */ |
|
/* #define ENABLE_SD_DMA_CACHE_MAINTENANCE 1 */ |
|
/* USER CODE END enableSDDmaCacheMaintenance */ |
|
|
|
/* |
|
* Some DMA requires 4-Byte aligned address buffer to correctly read/write data, |
|
* in FatFs some accesses aren't thus we need a 4-byte aligned scratch buffer to correctly |
|
* transfer data |
|
*/ |
|
/* USER CODE BEGIN enableScratchBuffer */ |
|
/* #define ENABLE_SCRATCH_BUFFER */ |
|
/* USER CODE END enableScratchBuffer */ |
|
|
|
/* Private variables ---------------------------------------------------------*/ |
|
#if defined(ENABLE_SCRATCH_BUFFER) |
|
#if defined (ENABLE_SD_DMA_CACHE_MAINTENANCE) |
|
ALIGN_32BYTES(static uint8_t scratch[BLOCKSIZE]); // 32-Byte aligned for cache maintenance |
|
#else |
|
__ALIGN_BEGIN static uint8_t scratch[BLOCKSIZE] __ALIGN_END; |
|
#endif |
|
#endif |
|
/* Disk status */ |
|
static volatile DSTATUS Stat = STA_NOINIT; |
|
|
|
#if (osCMSIS <= 0x20000U) |
|
static osMessageQId SDQueueID = NULL; |
|
#else |
|
static osMessageQueueId_t SDQueueID = NULL; |
|
#endif |
|
/* Private function prototypes -----------------------------------------------*/ |
|
static DSTATUS SD_CheckStatus(BYTE lun); |
|
DSTATUS SD_initialize (BYTE); |
|
DSTATUS SD_status (BYTE); |
|
DRESULT SD_read (BYTE, BYTE*, DWORD, UINT); |
|
#if _USE_WRITE == 1 |
|
DRESULT SD_write (BYTE, const BYTE*, DWORD, UINT); |
|
#endif /* _USE_WRITE == 1 */ |
|
#if _USE_IOCTL == 1 |
|
DRESULT SD_ioctl (BYTE, BYTE, void*); |
|
#endif /* _USE_IOCTL == 1 */ |
|
|
|
const Diskio_drvTypeDef SD_Driver = |
|
{ |
|
SD_initialize, |
|
SD_status, |
|
SD_read, |
|
#if _USE_WRITE == 1 |
|
SD_write, |
|
#endif /* _USE_WRITE == 1 */ |
|
|
|
#if _USE_IOCTL == 1 |
|
SD_ioctl, |
|
#endif /* _USE_IOCTL == 1 */ |
|
}; |
|
|
|
/* USER CODE BEGIN beforeFunctionSection */ |
|
/* can be used to modify / undefine following code or add new code */ |
|
/* USER CODE END beforeFunctionSection */ |
|
|
|
/* Private functions ---------------------------------------------------------*/ |
|
|
|
static int SD_CheckStatusWithTimeout(uint32_t timeout) |
|
{ |
|
uint32_t timer; |
|
/* block until SDIO peripheral is ready again or a timeout occur */ |
|
#if (osCMSIS <= 0x20000U) |
|
timer = osKernelSysTick(); |
|
while( osKernelSysTick() - timer < timeout) |
|
#else |
|
timer = osKernelGetTickCount(); |
|
while( osKernelGetTickCount() - timer < timeout) |
|
#endif |
|
{ |
|
if (BSP_SD_GetCardState() == SD_TRANSFER_OK) |
|
{ |
|
return 0; |
|
} |
|
} |
|
|
|
return -1; |
|
} |
|
|
|
static DSTATUS SD_CheckStatus(BYTE lun) |
|
{ |
|
Stat = STA_NOINIT; |
|
|
|
if(BSP_SD_GetCardState() == SD_TRANSFER_OK) |
|
{ |
|
Stat &= ~STA_NOINIT; |
|
} |
|
|
|
return Stat; |
|
} |
|
|
|
/** |
|
* @brief Initializes a Drive |
|
* @param lun : not used |
|
* @retval DSTATUS: Operation status |
|
*/ |
|
DSTATUS SD_initialize(BYTE lun) |
|
{ |
|
Stat = STA_NOINIT; |
|
|
|
/* |
|
* check that the kernel has been started before continuing |
|
* as the osMessage API will fail otherwise |
|
*/ |
|
#if (osCMSIS <= 0x20000U) |
|
if(osKernelRunning()) |
|
#else |
|
if(osKernelGetState() == osKernelRunning) |
|
#endif |
|
{ |
|
#if !defined(DISABLE_SD_INIT) |
|
|
|
if(BSP_SD_Init() == MSD_OK) |
|
{ |
|
Stat = SD_CheckStatus(lun); |
|
} |
|
|
|
#else |
|
Stat = SD_CheckStatus(lun); |
|
#endif |
|
|
|
/* |
|
* if the SD is correctly initialized, create the operation queue |
|
* if not already created |
|
*/ |
|
|
|
if (Stat != STA_NOINIT) |
|
{ |
|
if (SDQueueID == NULL) |
|
{ |
|
#if (osCMSIS <= 0x20000U) |
|
osMessageQDef(SD_Queue, QUEUE_SIZE, uint16_t); |
|
SDQueueID = osMessageCreate (osMessageQ(SD_Queue), NULL); |
|
#else |
|
SDQueueID = osMessageQueueNew(QUEUE_SIZE, 2, NULL); |
|
#endif |
|
} |
|
|
|
if (SDQueueID == NULL) |
|
{ |
|
Stat |= STA_NOINIT; |
|
} |
|
} |
|
} |
|
|
|
return Stat; |
|
} |
|
|
|
/** |
|
* @brief Gets Disk Status |
|
* @param lun : not used |
|
* @retval DSTATUS: Operation status |
|
*/ |
|
DSTATUS SD_status(BYTE lun) |
|
{ |
|
return SD_CheckStatus(lun); |
|
} |
|
|
|
/* USER CODE BEGIN beforeReadSection */ |
|
/* can be used to modify previous code / undefine following code / add new code */ |
|
/* USER CODE END beforeReadSection */ |
|
/** |
|
* @brief Reads Sector(s) |
|
* @param lun : not used |
|
* @param *buff: Data buffer to store read data |
|
* @param sector: Sector address (LBA) |
|
* @param count: Number of sectors to read (1..128) |
|
* @retval DRESULT: Operation result |
|
*/ |
|
|
|
DRESULT SD_read(BYTE lun, BYTE *buff, DWORD sector, UINT count) |
|
{ |
|
uint8_t ret; |
|
DRESULT res = RES_ERROR; |
|
uint32_t timer; |
|
#if (osCMSIS < 0x20000U) |
|
osEvent event; |
|
#else |
|
uint16_t event; |
|
osStatus_t status; |
|
#endif |
|
#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1) |
|
uint32_t alignedAddr; |
|
#endif |
|
/* |
|
* ensure the SDCard is ready for a new operation |
|
*/ |
|
|
|
if (SD_CheckStatusWithTimeout(SD_TIMEOUT) < 0) |
|
{ |
|
return res; |
|
} |
|
|
|
#if defined(ENABLE_SCRATCH_BUFFER) |
|
if (!((uint32_t)buff & 0x3)) |
|
{ |
|
#endif |
|
/* Fast path cause destination buffer is correctly aligned */ |
|
ret = BSP_SD_ReadBlocks_DMA((uint32_t*)buff, (uint32_t)(sector), count); |
|
|
|
if (ret == MSD_OK) { |
|
#if (osCMSIS < 0x20000U) |
|
/* wait for a message from the queue or a timeout */ |
|
event = osMessageGet(SDQueueID, SD_TIMEOUT); |
|
|
|
if (event.status == osEventMessage) |
|
{ |
|
if (event.value.v == READ_CPLT_MSG) |
|
{ |
|
timer = osKernelSysTick(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
while(osKernelSysTick() - timer <SD_TIMEOUT) |
|
#else |
|
status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT); |
|
if ((status == osOK) && (event == READ_CPLT_MSG)) |
|
{ |
|
timer = osKernelGetTickCount(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
while(osKernelGetTickCount() - timer <SD_TIMEOUT) |
|
#endif |
|
{ |
|
if (BSP_SD_GetCardState() == SD_TRANSFER_OK) |
|
{ |
|
res = RES_OK; |
|
#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1) |
|
/* |
|
the SCB_InvalidateDCache_by_Addr() requires a 32-Byte aligned address, |
|
adjust the address and the D-Cache size to invalidate accordingly. |
|
*/ |
|
alignedAddr = (uint32_t)buff & ~0x1F; |
|
SCB_InvalidateDCache_by_Addr((uint32_t*)alignedAddr, count*BLOCKSIZE + ((uint32_t)buff - alignedAddr)); |
|
#endif |
|
break; |
|
} |
|
} |
|
#if (osCMSIS < 0x20000U) |
|
} |
|
} |
|
#else |
|
} |
|
#endif |
|
} |
|
|
|
#if defined(ENABLE_SCRATCH_BUFFER) |
|
} |
|
else |
|
{ |
|
/* Slow path, fetch each sector a part and memcpy to destination buffer */ |
|
int i; |
|
|
|
for (i = 0; i < count; i++) |
|
{ |
|
ret = BSP_SD_ReadBlocks_DMA((uint32_t*)scratch, (uint32_t)sector++, 1); |
|
if (ret == MSD_OK ) |
|
{ |
|
/* wait until the read is successful or a timeout occurs */ |
|
#if (osCMSIS < 0x20000U) |
|
/* wait for a message from the queue or a timeout */ |
|
event = osMessageGet(SDQueueID, SD_TIMEOUT); |
|
|
|
if (event.status == osEventMessage) |
|
{ |
|
if (event.value.v == READ_CPLT_MSG) |
|
{ |
|
timer = osKernelSysTick(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
while(osKernelSysTick() - timer <SD_TIMEOUT) |
|
#else |
|
status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT); |
|
if ((status == osOK) && (event == READ_CPLT_MSG)) |
|
{ |
|
timer = osKernelGetTickCount(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
ret = MSD_ERROR; |
|
while(osKernelGetTickCount() - timer < SD_TIMEOUT) |
|
#endif |
|
{ |
|
ret = BSP_SD_GetCardState(); |
|
|
|
if (ret == MSD_OK) |
|
{ |
|
break; |
|
} |
|
} |
|
|
|
if (ret != MSD_OK) |
|
{ |
|
break; |
|
} |
|
#if (osCMSIS < 0x20000U) |
|
} |
|
} |
|
#else |
|
} |
|
#endif |
|
#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1) |
|
/* |
|
* |
|
* invalidate the scratch buffer before the next read to get the actual data instead of the cached one |
|
*/ |
|
SCB_InvalidateDCache_by_Addr((uint32_t*)scratch, BLOCKSIZE); |
|
#endif |
|
memcpy(buff, scratch, BLOCKSIZE); |
|
buff += BLOCKSIZE; |
|
} |
|
else |
|
{ |
|
break; |
|
} |
|
} |
|
|
|
if ((i == count) && (ret == MSD_OK )) |
|
res = RES_OK; |
|
} |
|
#endif |
|
return res; |
|
} |
|
|
|
/* USER CODE BEGIN beforeWriteSection */ |
|
/* can be used to modify previous code / undefine following code / add new code */ |
|
/* USER CODE END beforeWriteSection */ |
|
/** |
|
* @brief Writes Sector(s) |
|
* @param lun : not used |
|
* @param *buff: Data to be written |
|
* @param sector: Sector address (LBA) |
|
* @param count: Number of sectors to write (1..128) |
|
* @retval DRESULT: Operation result |
|
*/ |
|
#if _USE_WRITE == 1 |
|
|
|
DRESULT SD_write(BYTE lun, const BYTE *buff, DWORD sector, UINT count) |
|
{ |
|
DRESULT res = RES_ERROR; |
|
uint32_t timer; |
|
|
|
#if (osCMSIS < 0x20000U) |
|
osEvent event; |
|
#else |
|
uint16_t event; |
|
osStatus_t status; |
|
#endif |
|
|
|
#if defined(ENABLE_SCRATCH_BUFFER) |
|
int32_t ret; |
|
#endif |
|
|
|
/* |
|
* ensure the SDCard is ready for a new operation |
|
*/ |
|
|
|
if (SD_CheckStatusWithTimeout(SD_TIMEOUT) < 0) |
|
{ |
|
return res; |
|
} |
|
|
|
#if defined(ENABLE_SCRATCH_BUFFER) |
|
if (!((uint32_t)buff & 0x3)) |
|
{ |
|
#endif |
|
#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1) |
|
uint32_t alignedAddr; |
|
/* |
|
the SCB_CleanDCache_by_Addr() requires a 32-Byte aligned address |
|
adjust the address and the D-Cache size to clean accordingly. |
|
*/ |
|
alignedAddr = (uint32_t)buff & ~0x1F; |
|
SCB_CleanDCache_by_Addr((uint32_t*)alignedAddr, count*BLOCKSIZE + ((uint32_t)buff - alignedAddr)); |
|
#endif |
|
|
|
if(BSP_SD_WriteBlocks_DMA((uint32_t*)buff, |
|
(uint32_t) (sector), |
|
count) == MSD_OK) |
|
{ |
|
#if (osCMSIS < 0x20000U) |
|
/* Get the message from the queue */ |
|
event = osMessageGet(SDQueueID, SD_TIMEOUT); |
|
|
|
if (event.status == osEventMessage) |
|
{ |
|
if (event.value.v == WRITE_CPLT_MSG) |
|
{ |
|
#else |
|
status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT); |
|
if ((status == osOK) && (event == WRITE_CPLT_MSG)) |
|
{ |
|
#endif |
|
#if (osCMSIS < 0x20000U) |
|
timer = osKernelSysTick(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
while(osKernelSysTick() - timer < SD_TIMEOUT) |
|
#else |
|
timer = osKernelGetTickCount(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
while(osKernelGetTickCount() - timer < SD_TIMEOUT) |
|
#endif |
|
{ |
|
if (BSP_SD_GetCardState() == SD_TRANSFER_OK) |
|
{ |
|
res = RES_OK; |
|
break; |
|
} |
|
} |
|
#if (osCMSIS < 0x20000U) |
|
} |
|
} |
|
#else |
|
} |
|
#endif |
|
} |
|
#if defined(ENABLE_SCRATCH_BUFFER) |
|
else { |
|
/* Slow path, fetch each sector a part and memcpy to destination buffer */ |
|
int i; |
|
|
|
#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1) |
|
/* |
|
* invalidate the scratch buffer before the next write to get the actual data instead of the cached one |
|
*/ |
|
SCB_InvalidateDCache_by_Addr((uint32_t*)scratch, BLOCKSIZE); |
|
#endif |
|
for (i = 0; i < count; i++) |
|
{ |
|
memcpy((void *)scratch, buff, BLOCKSIZE); |
|
buff += BLOCKSIZE; |
|
|
|
ret = BSP_SD_WriteBlocks_DMA((uint32_t*)scratch, (uint32_t)sector++, 1); |
|
if (ret == MSD_OK ) |
|
{ |
|
/* wait until the read is successful or a timeout occurs */ |
|
#if (osCMSIS < 0x20000U) |
|
/* wait for a message from the queue or a timeout */ |
|
event = osMessageGet(SDQueueID, SD_TIMEOUT); |
|
|
|
if (event.status == osEventMessage) |
|
{ |
|
if (event.value.v == READ_CPLT_MSG) |
|
{ |
|
timer = osKernelSysTick(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
while(osKernelSysTick() - timer <SD_TIMEOUT) |
|
#else |
|
status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT); |
|
if ((status == osOK) && (event == READ_CPLT_MSG)) |
|
{ |
|
timer = osKernelGetTickCount(); |
|
/* block until SDIO IP is ready or a timeout occur */ |
|
ret = MSD_ERROR; |
|
while(osKernelGetTickCount() - timer < SD_TIMEOUT) |
|
#endif |
|
{ |
|
ret = BSP_SD_GetCardState(); |
|
|
|
if (ret == MSD_OK) |
|
{ |
|
break; |
|
} |
|
} |
|
|
|
if (ret != MSD_OK) |
|
{ |
|
break; |
|
} |
|
#if (osCMSIS < 0x20000U) |
|
} |
|
} |
|
#else |
|
} |
|
#endif |
|
} |
|
else |
|
{ |
|
break; |
|
} |
|
} |
|
|
|
if ((i == count) && (ret == MSD_OK )) |
|
res = RES_OK; |
|
} |
|
|
|
} |
|
#endif |
|
|
|
return res; |
|
} |
|
#endif /* _USE_WRITE == 1 */ |
|
|
|
/* USER CODE BEGIN beforeIoctlSection */ |
|
/* can be used to modify previous code / undefine following code / add new code */ |
|
/* USER CODE END beforeIoctlSection */ |
|
/** |
|
* @brief I/O control operation |
|
* @param lun : not used |
|
* @param cmd: Control code |
|
* @param *buff: Buffer to send/receive control data |
|
* @retval DRESULT: Operation result |
|
*/ |
|
#if _USE_IOCTL == 1 |
|
DRESULT SD_ioctl(BYTE lun, BYTE cmd, void *buff) |
|
{ |
|
DRESULT res = RES_ERROR; |
|
BSP_SD_CardInfo CardInfo; |
|
|
|
if (Stat & STA_NOINIT) return RES_NOTRDY; |
|
|
|
switch (cmd) |
|
{ |
|
/* Make sure that no pending write process */ |
|
case CTRL_SYNC : |
|
res = RES_OK; |
|
break; |
|
|
|
/* Get number of sectors on the disk (DWORD) */ |
|
case GET_SECTOR_COUNT : |
|
BSP_SD_GetCardInfo(&CardInfo); |
|
*(DWORD*)buff = CardInfo.LogBlockNbr; |
|
res = RES_OK; |
|
break; |
|
|
|
/* Get R/W sector size (WORD) */ |
|
case GET_SECTOR_SIZE : |
|
BSP_SD_GetCardInfo(&CardInfo); |
|
*(WORD*)buff = CardInfo.LogBlockSize; |
|
res = RES_OK; |
|
break; |
|
|
|
/* Get erase block size in unit of sector (DWORD) */ |
|
case GET_BLOCK_SIZE : |
|
BSP_SD_GetCardInfo(&CardInfo); |
|
*(DWORD*)buff = CardInfo.LogBlockSize / SD_DEFAULT_BLOCK_SIZE; |
|
res = RES_OK; |
|
break; |
|
|
|
default: |
|
res = RES_PARERR; |
|
} |
|
|
|
return res; |
|
} |
|
#endif /* _USE_IOCTL == 1 */ |
|
|
|
/* USER CODE BEGIN afterIoctlSection */ |
|
/* can be used to modify previous code / undefine following code / add new code */ |
|
/* USER CODE END afterIoctlSection */ |
|
|
|
/* USER CODE BEGIN callbackSection */ |
|
/* can be used to modify / following code or add new code */ |
|
/* USER CODE END callbackSection */ |
|
/** |
|
* @brief Tx Transfer completed callbacks |
|
* @param hsd: SD handle |
|
* @retval None |
|
*/ |
|
void BSP_SD_WriteCpltCallback(void) |
|
{ |
|
|
|
/* |
|
* No need to add an "osKernelRunning()" check here, as the SD_initialize() |
|
* is always called before any SD_Read()/SD_Write() call |
|
*/ |
|
#if (osCMSIS < 0x20000U) |
|
osMessagePut(SDQueueID, WRITE_CPLT_MSG, 0); |
|
#else |
|
const uint16_t msg = WRITE_CPLT_MSG; |
|
osMessageQueuePut(SDQueueID, (const void *)&msg, NULL, 0); |
|
#endif |
|
} |
|
|
|
/** |
|
* @brief Rx Transfer completed callbacks |
|
* @param hsd: SD handle |
|
* @retval None |
|
*/ |
|
void BSP_SD_ReadCpltCallback(void) |
|
{ |
|
/* |
|
* No need to add an "osKernelRunning()" check here, as the SD_initialize() |
|
* is always called before any SD_Read()/SD_Write() call |
|
*/ |
|
#if (osCMSIS < 0x20000U) |
|
osMessagePut(SDQueueID, READ_CPLT_MSG, 0); |
|
#else |
|
const uint16_t msg = READ_CPLT_MSG; |
|
osMessageQueuePut(SDQueueID, (const void *)&msg, NULL, 0); |
|
#endif |
|
} |
|
|
|
/* USER CODE BEGIN ErrorAbortCallbacks */ |
|
/* |
|
void BSP_SD_AbortCallback(void) |
|
{ |
|
#if (osCMSIS < 0x20000U) |
|
osMessagePut(SDQueueID, RW_ABORT_MSG, 0); |
|
#else |
|
const uint16_t msg = RW_ABORT_MSG; |
|
osMessageQueuePut(SDQueueID, (const void *)&msg, NULL, 0); |
|
#endif |
|
} |
|
*/ |
|
/* USER CODE END ErrorAbortCallbacks */ |
|
|
|
/* USER CODE BEGIN lastSection */ |
|
/* can be used to modify / undefine previous code or add new code */ |
|
/* USER CODE END lastSection */
|
|
|