Skip to content

Commit

Permalink
stmhal: Add dma support for sdcard.
Browse files Browse the repository at this point in the history
This started out using IgorLektorovEpam work in PR #1389
and reworked it.
  • Loading branch information
dhylands committed Nov 16, 2015
1 parent a1ebf69 commit 445f4bd
Showing 1 changed file with 76 additions and 69 deletions.
145 changes: 76 additions & 69 deletions stmhal/sdcard.c
Expand Up @@ -24,8 +24,6 @@
* THE SOFTWARE.
*/

// TODO make it work with DMA

#include STM32_HAL_H

#include "py/nlr.h"
Expand All @@ -34,6 +32,8 @@
#include "pin.h"
#include "genhdr/pins.h"
#include "bufhelper.h"
#include "dma.h"
#include "irq.h"

#if MICROPY_HW_HAS_SDCARD

Expand Down Expand Up @@ -64,8 +64,12 @@

#endif


// TODO: I think that as an optimization, we can allocate these dynamically
// if an sd card is detected. This will save approx 260 bytes of RAM
// when no sdcard was being used.
static SD_HandleTypeDef sd_handle;
static DMA_HandleTypeDef sd_rx_dma, sd_tx_dma;
static DMA_InitTypeDef sd_rx_dma_init, sd_tx_dma_init;

void sdcard_init(void) {
GPIO_InitTypeDef GPIO_Init_Structure;
Expand Down Expand Up @@ -98,13 +102,45 @@ void HAL_SD_MspInit(SD_HandleTypeDef *hsd) {
// enable SDIO clock
__SDIO_CLK_ENABLE();

// GPIO have already been initialised by sdcard_init
// NVIC configuration for SDIO interrupts
HAL_NVIC_SetPriority(SDIO_IRQn, IRQ_PRI_SDIO, IRQ_SUBPRI_SDIO);
HAL_NVIC_EnableIRQ(SDIO_IRQn);

// TODO: Since SDIO is fundamentally half-duplex, we really only need to
// tie up one DMA channel. However, the HAL DMA API doesn't
// seem to provide a convenient way to change the direction. I believe that
// its as simple as changing the CR register and the Init.Direction field
// and make DMA_SetConfig public.

// Configure DMA Rx parameters
sd_rx_dma_init.PeriphInc = DMA_PINC_DISABLE;
sd_rx_dma_init.MemInc = DMA_MINC_ENABLE;
sd_rx_dma_init.PeriphDataAlignment = DMA_PDATAALIGN_WORD;
sd_rx_dma_init.MemDataAlignment = DMA_MDATAALIGN_WORD;
sd_rx_dma_init.Mode = DMA_PFCTRL;
sd_rx_dma_init.Priority = DMA_PRIORITY_VERY_HIGH;
sd_rx_dma_init.FIFOMode = DMA_FIFOMODE_ENABLE;
sd_rx_dma_init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
sd_rx_dma_init.MemBurst = DMA_MBURST_INC4;
sd_rx_dma_init.PeriphBurst = DMA_PBURST_INC4;

// Configure DMA Tx parameters
sd_tx_dma_init.PeriphInc = DMA_PINC_DISABLE;
sd_tx_dma_init.MemInc = DMA_MINC_ENABLE;
sd_tx_dma_init.PeriphDataAlignment = DMA_PDATAALIGN_WORD;
sd_tx_dma_init.MemDataAlignment = DMA_MDATAALIGN_WORD;
sd_tx_dma_init.Mode = DMA_PFCTRL;
sd_tx_dma_init.Priority = DMA_PRIORITY_VERY_HIGH;
sd_tx_dma_init.FIFOMode = DMA_FIFOMODE_ENABLE;
sd_tx_dma_init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
sd_tx_dma_init.MemBurst = DMA_MBURST_INC4;
sd_tx_dma_init.PeriphBurst = DMA_PBURST_INC4;

// interrupts are not used at the moment
// they are needed only for DMA transfer (I think...)
// GPIO have already been initialised by sdcard_init
}

void HAL_SD_MspDeInit(SD_HandleTypeDef *hsd) {
HAL_NVIC_DisableIRQ(SDIO_IRQn);
__SDIO_CLK_DISABLE();
}

Expand Down Expand Up @@ -168,6 +204,10 @@ uint64_t sdcard_get_capacity_in_bytes(void) {
return cardinfo.CardCapacity;
}

void SDIO_IRQHandler(void) {
HAL_SD_IRQHandler(&sd_handle);
}

mp_uint_t sdcard_read_blocks(uint8_t *dest, uint32_t block_num, uint32_t num_blocks) {
// check that dest pointer is aligned on a 4-byte boundary
if (((uint32_t)dest & 3) != 0) {
Expand All @@ -179,90 +219,57 @@ mp_uint_t sdcard_read_blocks(uint8_t *dest, uint32_t block_num, uint32_t num_blo
return SD_ERROR;
}

// We must disable IRQs because the SDIO peripheral has a small FIFO
// buffer and we can't let it fill up in the middle of a read.
// This will not be needed when SD uses DMA for transfer.
mp_uint_t atomic_state = MICROPY_BEGIN_ATOMIC_SECTION();
HAL_SD_ErrorTypedef err = HAL_SD_ReadBlocks_BlockNumber(&sd_handle, (uint32_t*)dest, block_num, SDCARD_BLOCK_SIZE, num_blocks);
MICROPY_END_ATOMIC_SECTION(atomic_state);
HAL_SD_ErrorTypedef err = SD_OK;

return err;
}
if (query_irq() == IRQ_STATE_ENABLED) {
dma_init(&sd_rx_dma, DMA_STREAM_SDIO_RX, &sd_rx_dma_init, DMA_CHANNEL_SDIO_RX, DMA_PERIPH_TO_MEMORY, &sd_handle);
sd_handle.hdmarx = &sd_rx_dma;

mp_uint_t sdcard_write_blocks(const uint8_t *src, uint32_t block_num, uint32_t num_blocks) {
// check that src pointer is aligned on a 4-byte boundary
if (((uint32_t)src & 3) != 0) {
return SD_ERROR;
}
err = HAL_SD_ReadBlocks_BlockNumber_DMA(&sd_handle, (uint32_t*)dest, block_num, SDCARD_BLOCK_SIZE, num_blocks);
if (err == SD_OK) {
// wait for DMA transfer to finish, with a large timeout
err = HAL_SD_CheckReadOperation(&sd_handle, 100000000);
}

// check that SD card is initialised
if (sd_handle.Instance == NULL) {
return SD_ERROR;
dma_deinit(sd_handle.hdmarx);
sd_handle.hdmarx = NULL;
} else {
err = HAL_SD_ReadBlocks_BlockNumber(&sd_handle, (uint32_t*)dest, block_num, SDCARD_BLOCK_SIZE, num_blocks);
}

// We must disable IRQs because the SDIO peripheral has a small FIFO
// buffer and we can't let it drain to empty in the middle of a write.
// This will not be needed when SD uses DMA for transfer.
mp_uint_t atomic_state = MICROPY_BEGIN_ATOMIC_SECTION();
HAL_SD_ErrorTypedef err = HAL_SD_WriteBlocks_BlockNumber(&sd_handle, (uint32_t*)src, block_num, SDCARD_BLOCK_SIZE, num_blocks);
MICROPY_END_ATOMIC_SECTION(atomic_state);

return err;
}

#if 0
DMA not implemented
bool sdcard_read_blocks_dma(uint8_t *dest, uint32_t block_num, uint32_t num_blocks) {
// check that dest pointer is aligned on a 4-byte boundary
if (((uint32_t)dest & 3) != 0) {
return false;
}

// check that SD card is initialised
if (sd_handle.Instance == NULL) {
return false;
}

// do the read
if (HAL_SD_ReadBlocks_BlockNumber_DMA(&sd_handle, (uint32_t*)dest, block_num, SDCARD_BLOCK_SIZE) != SD_OK) {
return false;
}

// wait for DMA transfer to finish, with a large timeout
if (HAL_SD_CheckReadOperation(&sd_handle, 100000000) != SD_OK) {
return false;
}

return true;
}

bool sdcard_write_blocks_dma(const uint8_t *src, uint32_t block_num, uint32_t num_blocks) {
mp_uint_t sdcard_write_blocks(const uint8_t *src, uint32_t block_num, uint32_t num_blocks) {
// check that src pointer is aligned on a 4-byte boundary
if (((uint32_t)src & 3) != 0) {
return false;
return SD_ERROR;
}

// check that SD card is initialised
if (sd_handle.Instance == NULL) {
return false;
return SD_ERROR;
}

SD_Error status;
HAL_SD_ErrorTypedef err = SD_OK;

status = HAL_SD_WriteBlocks_BlockNumber_DMA(&sd_handle, (uint32_t*)src, block_num, SDCARD_BLOCK_SIZE, num_blocks);
if (status != SD_OK) {
return false;
}
if (query_irq() == IRQ_STATE_ENABLED) {
dma_init(&sd_tx_dma, DMA_STREAM_SDIO_TX, &sd_tx_dma_init, DMA_CHANNEL_SDIO_TX, DMA_MEMORY_TO_PERIPH, &sd_handle);
sd_handle.hdmatx = &sd_tx_dma;

// wait for DMA transfer to finish, with a large timeout
status = HAL_SD_CheckWriteOperation(&sd_handle, 100000000);
if (status != SD_OK) {
return false;
err = HAL_SD_WriteBlocks_BlockNumber_DMA(&sd_handle, (uint32_t*)src, block_num, SDCARD_BLOCK_SIZE, num_blocks);
if (err == SD_OK) {
// wait for DMA transfer to finish, with a large timeout
err = HAL_SD_CheckWriteOperation(&sd_handle, 100000000);
}
dma_deinit(sd_handle.hdmatx);
sd_handle.hdmatx = NULL;
} else {
err = HAL_SD_WriteBlocks_BlockNumber(&sd_handle, (uint32_t*)src, block_num, SDCARD_BLOCK_SIZE, num_blocks);
}

return true;
return err;
}
#endif

/******************************************************************************/
// Micro Python bindings
Expand Down

0 comments on commit 445f4bd

Please sign in to comment.