cueOS  2.4
cueOS - Universal Show Control OS for ARM
sd_diskio.c
Go to the documentation of this file.
1/* USER CODE BEGIN Header */
19/* USER CODE END Header */
20
21/* Note: code generation based on sd_diskio_dma_rtos_template_bspv1.c v2.1.4
22 as FreeRTOS is enabled. */
23
24/* USER CODE BEGIN firstSection */
25/* can be used to modify / undefine following code or add new definitions */
26/* USER CODE END firstSection*/
27
28/* Includes ------------------------------------------------------------------*/
29#include "ff_gen_drv.h"
30#include "sd_diskio.h"
31
32#include <string.h>
33#include <stdio.h>
34
35/* Private typedef -----------------------------------------------------------*/
36/* Private define ------------------------------------------------------------*/
37
38#define QUEUE_SIZE (uint32_t) 10
39#define READ_CPLT_MSG (uint32_t) 1
40#define WRITE_CPLT_MSG (uint32_t) 2
41/*
42==================================================================
43enable the defines below to send custom rtos messages
44when an error or an abort occurs.
45Notice: depending on the HAL/SD driver the HAL_SD_ErrorCallback()
46may not be available.
47See BSP_SD_ErrorCallback() and BSP_SD_AbortCallback() below
48==================================================================
49
50#define RW_ERROR_MSG (uint32_t) 3
51#define RW_ABORT_MSG (uint32_t) 4
52*/
53/*
54 * the following Timeout is useful to give the control back to the applications
55 * in case of errors in either BSP_SD_ReadCpltCallback() or BSP_SD_WriteCpltCallback()
56 * the value by default is as defined in the BSP platform driver otherwise 30 secs
57 */
58#define SD_TIMEOUT 30 * 1000
59
60#define SD_DEFAULT_BLOCK_SIZE 512
61
62/*
63 * Depending on the use case, the SD card initialization could be done at the
64 * application level: if it is the case define the flag below to disable
65 * the BSP_SD_Init() call in the SD_Initialize() and add a call to
66 * BSP_SD_Init() elsewhere in the application.
67 */
68/* USER CODE BEGIN disableSDInit */
69/* #define DISABLE_SD_INIT */
70/* USER CODE END disableSDInit */
71
72/*
73 * when using cachable memory region, it may be needed to maintain the cache
74 * validity. Enable the define below to activate a cache maintenance at each
75 * read and write operation.
76 * Notice: This is applicable only for cortex M7 based platform.
77 */
78/* USER CODE BEGIN enableSDDmaCacheMaintenance */
79/* #define ENABLE_SD_DMA_CACHE_MAINTENANCE 1 */
80/* USER CODE BEGIN enableSDDmaCacheMaintenance */
81
82/*
83* Some DMA requires 4-Byte aligned address buffer to correctly read/wite data,
84* in FatFs some accesses aren't thus we need a 4-byte aligned scratch buffer to correctly
85* transfer data
86*/
87/* USER CODE BEGIN enableScratchBuffer */
88/* #define ENABLE_SCRATCH_BUFFER */
89/* USER CODE END enableScratchBuffer */
90
91/* Private variables ---------------------------------------------------------*/
92#if defined(ENABLE_SCRATCH_BUFFER)
93#if defined (ENABLE_SD_DMA_CACHE_MAINTENANCE)
94ALIGN_32BYTES(static uint8_t scratch[BLOCKSIZE]); // 32-Byte aligned for cache maintenance
95#else
96__ALIGN_BEGIN static uint8_t scratch[BLOCKSIZE] __ALIGN_END;
97#endif
98#endif
99/* Disk status */
100static volatile DSTATUS Stat = STA_NOINIT;
101
102#if (osCMSIS <= 0x20000U)
103static osMessageQId SDQueueID = NULL;
104#else
105static osMessageQueueId_t SDQueueID = NULL;
106#endif
107/* Private function prototypes -----------------------------------------------*/
108static DSTATUS SD_CheckStatus(BYTE lun);
109DSTATUS SD_initialize (BYTE);
110DSTATUS SD_status (BYTE);
111DRESULT SD_read (BYTE, BYTE*, DWORD, UINT);
112#if _USE_WRITE == 1
113DRESULT SD_write (BYTE, const BYTE*, DWORD, UINT);
114#endif /* _USE_WRITE == 1 */
115#if _USE_IOCTL == 1
116DRESULT SD_ioctl (BYTE, BYTE, void*);
117#endif /* _USE_IOCTL == 1 */
118
119const Diskio_drvTypeDef SD_Driver =
120{
122 SD_status,
123 SD_read,
124#if _USE_WRITE == 1
125 SD_write,
126#endif /* _USE_WRITE == 1 */
127
128#if _USE_IOCTL == 1
129 SD_ioctl,
130#endif /* _USE_IOCTL == 1 */
131};
132
133/* USER CODE BEGIN beforeFunctionSection */
134/* can be used to modify / undefine following code or add new code */
135/* USER CODE END beforeFunctionSection */
136
137/* Private functions ---------------------------------------------------------*/
138
139static int SD_CheckStatusWithTimeout(uint32_t timeout)
140{
141 uint32_t timer;
142 /* block until SDIO peripherial is ready again or a timeout occur */
143#if (osCMSIS <= 0x20000U)
144 timer = osKernelSysTick();
145 while( osKernelSysTick() - timer < timeout)
146#else
147 timer = osKernelGetTickCount();
148 while( osKernelGetTickCount() - timer < timeout)
149#endif
150 {
152 {
153 return 0;
154 }
155 }
156
157 return -1;
158}
159
160static DSTATUS SD_CheckStatus(BYTE lun)
161{
162 Stat = STA_NOINIT;
163
165 {
166 Stat &= ~STA_NOINIT;
167 }
168
169 return Stat;
170}
171
177DSTATUS SD_initialize(BYTE lun)
178{
179Stat = STA_NOINIT;
180
181 /*
182 * check that the kernel has been started before continuing
183 * as the osMessage API will fail otherwise
184 */
185#if (osCMSIS <= 0x20000U)
186 if(osKernelRunning())
187#else
188 if(osKernelGetState() == osKernelRunning)
189#endif
190 {
191#if !defined(DISABLE_SD_INIT)
192
193 if(BSP_SD_Init() == MSD_OK)
194 {
195 Stat = SD_CheckStatus(lun);
196 }
197
198#else
199 Stat = SD_CheckStatus(lun);
200#endif
201
202 /*
203 * if the SD is correctly initialized, create the operation queue
204 * if not already created
205 */
206
207 if (Stat != STA_NOINIT)
208 {
209 if (SDQueueID == NULL)
210 {
211 #if (osCMSIS <= 0x20000U)
212 osMessageQDef(SD_Queue, QUEUE_SIZE, uint16_t);
213 SDQueueID = osMessageCreate (osMessageQ(SD_Queue), NULL);
214#else
215 SDQueueID = osMessageQueueNew(QUEUE_SIZE, 2, NULL);
216#endif
217 }
218
219 if (SDQueueID == NULL)
220 {
221 Stat |= STA_NOINIT;
222 }
223 }
224 }
225
226 return Stat;
227}
228
234DSTATUS SD_status(BYTE lun)
235{
236 return SD_CheckStatus(lun);
237}
238
239/* USER CODE BEGIN beforeReadSection */
240/* can be used to modify previous code / undefine following code / add new code */
241/* USER CODE END beforeReadSection */
251DRESULT SD_read(BYTE lun, BYTE *buff, DWORD sector, UINT count)
252{
253 DRESULT res = RES_ERROR;
254 uint32_t timer;
255#if (osCMSIS < 0x20000U)
256 osEvent event;
257#else
258 uint16_t event;
259 osStatus_t status;
260#endif
261#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1)
262 uint32_t alignedAddr;
263#endif
264 /*
265 * ensure the SDCard is ready for a new operation
266 */
267
268 if (SD_CheckStatusWithTimeout(SD_TIMEOUT) < 0)
269 {
270 return res;
271 }
272
273#if defined(ENABLE_SCRATCH_BUFFER)
274 if (!((uint32_t)buff & 0x3))
275 {
276#endif
277 /* Fast path cause destination buffer is correctly aligned */
278 uint8_t ret = BSP_SD_ReadBlocks_DMA((uint32_t*)buff, (uint32_t)(sector), count);
279
280 if (ret == MSD_OK) {
281#if (osCMSIS < 0x20000U)
282 /* wait for a message from the queue or a timeout */
283 event = osMessageGet(SDQueueID, SD_TIMEOUT);
284
285 if (event.status == osEventMessage)
286 {
287 if (event.value.v == READ_CPLT_MSG)
288 {
289 timer = osKernelSysTick();
290 /* block until SDIO IP is ready or a timeout occur */
291 while(osKernelSysTick() - timer <SD_TIMEOUT)
292#else
293 status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT);
294 if ((status == osOK) && (event == READ_CPLT_MSG))
295 {
296 timer = osKernelGetTickCount();
297 /* block until SDIO IP is ready or a timeout occur */
298 while(osKernelGetTickCount() - timer <SD_TIMEOUT)
299#endif
300 {
302 {
303 res = RES_OK;
304#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1)
305 /*
306 the SCB_InvalidateDCache_by_Addr() requires a 32-Byte aligned address,
307 adjust the address and the D-Cache size to invalidate accordingly.
308 */
309 alignedAddr = (uint32_t)buff & ~0x1F;
310 SCB_InvalidateDCache_by_Addr((uint32_t*)alignedAddr, count*BLOCKSIZE + ((uint32_t)buff - alignedAddr));
311#endif
312 break;
313 }
314 }
315#if (osCMSIS < 0x20000U)
316 }
317 }
318#else
319 }
320#endif
321 }
322
323#if defined(ENABLE_SCRATCH_BUFFER)
324 }
325 else
326 {
327 /* Slow path, fetch each sector a part and memcpy to destination buffer */
328 int i;
329
330 for (i = 0; i < count; i++)
331 {
332 ret = BSP_SD_ReadBlocks_DMA((uint32_t*)scratch, (uint32_t)sector++, 1);
333 if (ret == MSD_OK )
334 {
335 /* wait until the read is successful or a timeout occurs */
336#if (osCMSIS < 0x20000U)
337 /* wait for a message from the queue or a timeout */
338 event = osMessageGet(SDQueueID, SD_TIMEOUT);
339
340 if (event.status == osEventMessage)
341 {
342 if (event.value.v == READ_CPLT_MSG)
343 {
344 timer = osKernelSysTick();
345 /* block until SDIO IP is ready or a timeout occur */
346 while(osKernelSysTick() - timer <SD_TIMEOUT)
347#else
348 status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT);
349 if ((status == osOK) && (event == READ_CPLT_MSG))
350 {
351 timer = osKernelGetTickCount();
352 /* block until SDIO IP is ready or a timeout occur */
353 ret = MSD_ERROR;
354 while(osKernelGetTickCount() - timer < SD_TIMEOUT)
355#endif
356 {
357 ret = BSP_SD_GetCardState();
358
359 if (ret == MSD_OK)
360 {
361 break;
362 }
363 }
364
365 if (ret != MSD_OK)
366 {
367 break;
368 }
369#if (osCMSIS < 0x20000U)
370 }
371 }
372#else
373 }
374#endif
375#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1)
376 /*
377 *
378 * invalidate the scratch buffer before the next read to get the actual data instead of the cached one
379 */
380 SCB_InvalidateDCache_by_Addr((uint32_t*)scratch, BLOCKSIZE);
381#endif
382 memcpy(buff, scratch, BLOCKSIZE);
383 buff += BLOCKSIZE;
384 }
385 else
386 {
387 break;
388 }
389 }
390
391 if ((i == count) && (ret == MSD_OK ))
392 res = RES_OK;
393 }
394#endif
395 return res;
396}
397
398
399/* USER CODE BEGIN beforeWriteSection */
400/* can be used to modify previous code / undefine following code / add new code */
401/* USER CODE END beforeWriteSection */
410#if _USE_WRITE == 1
411
412DRESULT SD_write(BYTE lun, const BYTE *buff, DWORD sector, UINT count)
413{
414 DRESULT res = RES_ERROR;
415 uint32_t timer;
416
417#if (osCMSIS < 0x20000U)
418 osEvent event;
419#else
420 uint16_t event;
421 osStatus_t status;
422#endif
423
424#if defined(ENABLE_SCRATCH_BUFFER)
425 int32_t ret;
426#endif
427
428 /*
429 * ensure the SDCard is ready for a new operation
430 */
431
432 if (SD_CheckStatusWithTimeout(SD_TIMEOUT) < 0)
433 {
434 return res;
435 }
436
437#if defined(ENABLE_SCRATCH_BUFFER)
438 if (!((uint32_t)buff & 0x3))
439 {
440#endif
441#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1)
442 uint32_t alignedAddr;
443 /*
444 the SCB_CleanDCache_by_Addr() requires a 32-Byte aligned address
445 adjust the address and the D-Cache size to clean accordingly.
446 */
447 alignedAddr = (uint32_t)buff & ~0x1F;
448 SCB_CleanDCache_by_Addr((uint32_t*)alignedAddr, count*BLOCKSIZE + ((uint32_t)buff - alignedAddr));
449#endif
450
451 if(BSP_SD_WriteBlocks_DMA((uint32_t*)buff,
452 (uint32_t) (sector),
453 count) == MSD_OK)
454 {
455#if (osCMSIS < 0x20000U)
456 /* Get the message from the queue */
457 event = osMessageGet(SDQueueID, SD_TIMEOUT);
458
459 if (event.status == osEventMessage)
460 {
461 if (event.value.v == WRITE_CPLT_MSG)
462 {
463#else
464 status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT);
465 if ((status == osOK) && (event == WRITE_CPLT_MSG))
466 {
467#endif
468 #if (osCMSIS < 0x20000U)
469 timer = osKernelSysTick();
470 /* block until SDIO IP is ready or a timeout occur */
471 while(osKernelSysTick() - timer < SD_TIMEOUT)
472#else
473 timer = osKernelGetTickCount();
474 /* block until SDIO IP is ready or a timeout occur */
475 while(osKernelGetTickCount() - timer < SD_TIMEOUT)
476#endif
477 {
479 {
480 res = RES_OK;
481 break;
482 }
483 }
484#if (osCMSIS < 0x20000U)
485 }
486 }
487#else
488 }
489#endif
490 }
491#if defined(ENABLE_SCRATCH_BUFFER)
492 else {
493 /* Slow path, fetch each sector a part and memcpy to destination buffer */
494 int i;
495
496#if (ENABLE_SD_DMA_CACHE_MAINTENANCE == 1)
497 /*
498 * invalidate the scratch buffer before the next write to get the actual data instead of the cached one
499 */
500 SCB_InvalidateDCache_by_Addr((uint32_t*)scratch, BLOCKSIZE);
501#endif
502 for (i = 0; i < count; i++)
503 {
504 memcpy((void *)scratch, buff, BLOCKSIZE);
505 buff += BLOCKSIZE;
506
507 ret = BSP_SD_WriteBlocks_DMA((uint32_t*)scratch, (uint32_t)sector++, 1);
508 if (ret == MSD_OK )
509 {
510 /* wait until the read is successful or a timeout occurs */
511#if (osCMSIS < 0x20000U)
512 /* wait for a message from the queue or a timeout */
513 event = osMessageGet(SDQueueID, SD_TIMEOUT);
514
515 if (event.status == osEventMessage)
516 {
517 if (event.value.v == READ_CPLT_MSG)
518 {
519 timer = osKernelSysTick();
520 /* block until SDIO IP is ready or a timeout occur */
521 while(osKernelSysTick() - timer <SD_TIMEOUT)
522#else
523 status = osMessageQueueGet(SDQueueID, (void *)&event, NULL, SD_TIMEOUT);
524 if ((status == osOK) && (event == READ_CPLT_MSG))
525 {
526 timer = osKernelGetTickCount();
527 /* block until SDIO IP is ready or a timeout occur */
528 ret = MSD_ERROR;
529 while(osKernelGetTickCount() - timer < SD_TIMEOUT)
530#endif
531 {
532 ret = BSP_SD_GetCardState();
533
534 if (ret == MSD_OK)
535 {
536 break;
537 }
538 }
539
540 if (ret != MSD_OK)
541 {
542 break;
543 }
544#if (osCMSIS < 0x20000U)
545 }
546 }
547#else
548 }
549#endif
550 }
551 else
552 {
553 break;
554 }
555 }
556
557 if ((i == count) && (ret == MSD_OK ))
558 res = RES_OK;
559 }
560
561 }
562#endif
563
564 return res;
565}
566 #endif /* _USE_WRITE == 1 */
567
568/* USER CODE BEGIN beforeIoctlSection */
569/* can be used to modify previous code / undefine following code / add new code */
570/* USER CODE END beforeIoctlSection */
578#if _USE_IOCTL == 1
579DRESULT SD_ioctl(BYTE lun, BYTE cmd, void *buff)
580{
581 DRESULT res = RES_ERROR;
582 BSP_SD_CardInfo CardInfo;
583
584 if (Stat & STA_NOINIT) return RES_NOTRDY;
585
586 switch (cmd)
587 {
588 /* Make sure that no pending write process */
589 case CTRL_SYNC :
590 res = RES_OK;
591 break;
592
593 /* Get number of sectors on the disk (DWORD) */
594 case GET_SECTOR_COUNT :
595 BSP_SD_GetCardInfo(&CardInfo);
596 *(DWORD*)buff = CardInfo.LogBlockNbr;
597 res = RES_OK;
598 break;
599
600 /* Get R/W sector size (WORD) */
601 case GET_SECTOR_SIZE :
602 BSP_SD_GetCardInfo(&CardInfo);
603 *(WORD*)buff = CardInfo.LogBlockSize;
604 res = RES_OK;
605 break;
606
607 /* Get erase block size in unit of sector (DWORD) */
608 case GET_BLOCK_SIZE :
609 BSP_SD_GetCardInfo(&CardInfo);
610 *(DWORD*)buff = CardInfo.LogBlockSize / SD_DEFAULT_BLOCK_SIZE;
611 res = RES_OK;
612 break;
613
614 default:
615 res = RES_PARERR;
616 }
617
618 return res;
619}
620#endif /* _USE_IOCTL == 1 */
621
622/* USER CODE BEGIN afterIoctlSection */
623/* can be used to modify previous code / undefine following code / add new code */
624/* USER CODE END afterIoctlSection */
625
626/* USER CODE BEGIN callbackSection */
627/* can be used to modify / following code or add new code */
628/* USER CODE END callbackSection */
635{
636
637 /*
638 * No need to add an "osKernelRunning()" check here, as the SD_initialize()
639 * is always called before any SD_Read()/SD_Write() call
640 */
641#if (osCMSIS < 0x20000U)
642 osMessagePut(SDQueueID, WRITE_CPLT_MSG, 0);
643#else
644 const uint16_t msg = WRITE_CPLT_MSG;
645 osMessageQueuePut(SDQueueID, (const void *)&msg, 0, 0);
646#endif
647}
648
655{
656 /*
657 * No need to add an "osKernelRunning()" check here, as the SD_initialize()
658 * is always called before any SD_Read()/SD_Write() call
659 */
660#if (osCMSIS < 0x20000U)
661 osMessagePut(SDQueueID, READ_CPLT_MSG, 0);
662#else
663 const uint16_t msg = READ_CPLT_MSG;
664 osMessageQueuePut(SDQueueID, (const void *)&msg, 0, 0);
665#endif
666}
667
668/* USER CODE BEGIN ErrorAbortCallbacks */
669/*
670void BSP_SD_AbortCallback(void)
671{
672#if (osCMSIS < 0x20000U)
673 osMessagePut(SDQueueID, RW_ABORT_MSG, 0);
674#else
675 const uint16_t msg = RW_ABORT_MSG;
676 osMessageQueuePut(SDQueueID, (const void *)&msg, NULL, 0);
677#endif
678}
679*/
680/* USER CODE END ErrorAbortCallbacks */
681
682/* USER CODE BEGIN lastSection */
683/* can be used to modify / undefine previous code or add new code */
684/* USER CODE END lastSection */
685
686/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
687
#define MSD_OK
SD status structure definition
Definition: bsp_driver_sd.h:42
uint8_t BSP_SD_WriteBlocks_DMA(uint32_t *pData, uint32_t WriteAddr, uint32_t NumOfBlocks)
Writes block(s) to a specified address in an SD card, in DMA mode.
uint8_t BSP_SD_Init(void)
Initializes the SD card device.
Definition: bsp_driver_sd.c:47
#define BSP_SD_CardInfo
SD Card information structure.
Definition: bsp_driver_sd.h:36
uint8_t BSP_SD_GetCardState(void)
Gets the current SD card data status.
#define SD_TRANSFER_OK
SD transfer state definition
Definition: bsp_driver_sd.h:48
uint8_t BSP_SD_ReadBlocks_DMA(uint32_t *pData, uint32_t ReadAddr, uint32_t NumOfBlocks)
Reads block(s) from a specified address in an SD card, in DMA mode.
void BSP_SD_GetCardInfo(HAL_SD_CardInfoTypeDef *CardInfo)
Get SD information about specific SD card.
Header for ff_gen_drv.c module.
DRESULT SD_read(BYTE, BYTE *, DWORD, UINT)
Reads Sector(s)
Definition: sd_diskio.c:251
DSTATUS SD_status(BYTE)
Gets Disk Status.
Definition: sd_diskio.c:234
void BSP_SD_ReadCpltCallback(void)
Rx Transfer completed callbacks.
Definition: sd_diskio.c:654
DSTATUS SD_initialize(BYTE)
Initializes a Drive.
Definition: sd_diskio.c:177
void BSP_SD_WriteCpltCallback(void)
Writes Sector(s)
Definition: sd_diskio.c:634
Header for sd_diskio.c module.
Disk IO Driver structure definition.
Definition: ff_gen_drv.h:41