diff options
| author | Damien George | 2015-06-10 13:06:48 +0100 |
|---|---|---|
| committer | Damien George | 2015-06-10 14:01:44 +0100 |
| commit | 3d30d605f5286774edb7669f3958628e5c656048 (patch) | |
| tree | 29a1e5b6a8a1bb21c09cb8a6d7fa2b3147da03d3 /stmhal/dma.c | |
| parent | 7ed58cb66379f4d87e3e8fbb68baada19048ac18 (diff) | |
stmhal: Factor out DMA initialisation code from spi.c.
This is so that the DMA can be shared by multiple peripherals.
Diffstat (limited to 'stmhal/dma.c')
| -rw-r--r-- | stmhal/dma.c | 148 |
1 files changed, 148 insertions, 0 deletions
diff --git a/stmhal/dma.c b/stmhal/dma.c new file mode 100644 index 000000000..93c35d61d --- /dev/null +++ b/stmhal/dma.c @@ -0,0 +1,148 @@ +/* + * This file is part of the Micro Python project, http://micropython.org/ + * + * The MIT License (MIT) + * + * Copyright (c) 2015 Damien P. George + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +#include <stdio.h> +#include <string.h> +#include <stdint.h> +#include <stm32f4xx_hal.h> + +#include "dma.h" + +#define NSTREAM (16) + +static const uint8_t dma_irqn[NSTREAM] = { + DMA1_Stream0_IRQn, + DMA1_Stream1_IRQn, + DMA1_Stream2_IRQn, + DMA1_Stream3_IRQn, + DMA1_Stream4_IRQn, + DMA1_Stream5_IRQn, + DMA1_Stream6_IRQn, + DMA1_Stream7_IRQn, + DMA2_Stream0_IRQn, + DMA2_Stream1_IRQn, + DMA2_Stream2_IRQn, + DMA2_Stream3_IRQn, + DMA2_Stream4_IRQn, + DMA2_Stream5_IRQn, + DMA2_Stream6_IRQn, + DMA2_Stream7_IRQn, +}; + +static DMA_HandleTypeDef *dma_handle[NSTREAM] = {NULL}; +static uint32_t dma_last_channel[NSTREAM]; + +void DMA1_Stream0_IRQHandler(void) { if (dma_handle[0] != NULL) { HAL_DMA_IRQHandler(dma_handle[0]); } } +void DMA1_Stream1_IRQHandler(void) { if (dma_handle[1] != NULL) { HAL_DMA_IRQHandler(dma_handle[1]); } } +void DMA1_Stream2_IRQHandler(void) { if (dma_handle[2] != NULL) { HAL_DMA_IRQHandler(dma_handle[2]); } } +void DMA1_Stream3_IRQHandler(void) { if (dma_handle[3] != NULL) { HAL_DMA_IRQHandler(dma_handle[3]); } } +void DMA1_Stream4_IRQHandler(void) { if (dma_handle[4] != NULL) { HAL_DMA_IRQHandler(dma_handle[4]); } } +void DMA1_Stream5_IRQHandler(void) { if (dma_handle[5] != NULL) { HAL_DMA_IRQHandler(dma_handle[5]); } } +void DMA1_Stream6_IRQHandler(void) { if (dma_handle[6] != NULL) { HAL_DMA_IRQHandler(dma_handle[6]); } } +void DMA1_Stream7_IRQHandler(void) { if (dma_handle[7] != NULL) { HAL_DMA_IRQHandler(dma_handle[7]); } } +void DMA2_Stream0_IRQHandler(void) { if (dma_handle[8] != NULL) { HAL_DMA_IRQHandler(dma_handle[8]); } } +void DMA2_Stream1_IRQHandler(void) { if (dma_handle[9] != NULL) { HAL_DMA_IRQHandler(dma_handle[9]); } } +void DMA2_Stream2_IRQHandler(void) { if (dma_handle[10] != NULL) { HAL_DMA_IRQHandler(dma_handle[10]); } } +void DMA2_Stream3_IRQHandler(void) { if (dma_handle[11] != NULL) { HAL_DMA_IRQHandler(dma_handle[11]); } } +void DMA2_Stream4_IRQHandler(void) { if (dma_handle[12] != NULL) { HAL_DMA_IRQHandler(dma_handle[12]); } } +void DMA2_Stream5_IRQHandler(void) { if (dma_handle[13] != NULL) { HAL_DMA_IRQHandler(dma_handle[13]); } } +void DMA2_Stream6_IRQHandler(void) { if (dma_handle[14] != NULL) { HAL_DMA_IRQHandler(dma_handle[14]); } } +void DMA2_Stream7_IRQHandler(void) { if (dma_handle[15] != NULL) { HAL_DMA_IRQHandler(dma_handle[15]); } } + +static int get_dma_id(DMA_Stream_TypeDef *dma_stream) { + if ((uint32_t)dma_stream < DMA2_BASE) { + return ((uint32_t)dma_stream - DMA1_Stream0_BASE) / 0x18; + } else { + return (NSTREAM / 2) + ((uint32_t)dma_stream - DMA2_Stream0_BASE) / 0x18; + } +} + +void dma_init(DMA_HandleTypeDef *dma, DMA_Stream_TypeDef *dma_stream, uint32_t dma_channel, uint32_t direction, void *data) { + int dma_id = get_dma_id(dma_stream); + //printf("dma_init(%p, %p(%d), 0x%x, 0x%x, %p)\n", dma, dma_stream, dma_id, (uint)dma_channel, (uint)direction, data); + + // TODO possibly don't need to clear the entire structure + memset(dma, 0, sizeof(*dma)); + + // set global pointer for IRQ handler + dma_handle[dma_id] = dma; + + // initialise critical parameters + dma->Instance = dma_stream; + dma->Init.Direction = direction; + + // half of __HAL_LINKDMA(data, xxx, *dma) + // caller must implement other half by doing: data->xxx = dma + dma->Parent = data; + + // if this stream was previously configured for this channel then we + // can skip most of the initialisation + if (dma_last_channel[dma_id] == dma_channel) { + goto same_channel; + } + dma_last_channel[dma_id] = dma_channel; + + // set DMA parameters (these are only used by HAL_DMA_Init) + dma->Init.Channel = dma_channel; + dma->Init.PeriphInc = DMA_PINC_DISABLE; + dma->Init.MemInc = DMA_MINC_ENABLE; + dma->Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE; + dma->Init.MemDataAlignment = DMA_MDATAALIGN_BYTE; + dma->Init.Mode = DMA_NORMAL; + dma->Init.Priority = DMA_PRIORITY_LOW; + dma->Init.FIFOMode = DMA_FIFOMODE_DISABLE; + dma->Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL; + dma->Init.MemBurst = DMA_MBURST_INC4; + dma->Init.PeriphBurst = DMA_PBURST_INC4; + + // enable clock for needed DMA peripheral + if (dma_id <= 7) { + __DMA1_CLK_ENABLE(); + } else { + __DMA2_CLK_ENABLE(); + } + + // reset and configure DMA peripheral + HAL_DMA_DeInit(dma); + HAL_DMA_Init(dma); + HAL_NVIC_SetPriority(dma_irqn[dma_id], 6, 0); + +same_channel: + HAL_NVIC_EnableIRQ(dma_irqn[dma_id]); +} + +void dma_deinit(DMA_HandleTypeDef *dma) { + int dma_id = get_dma_id(dma->Instance); + HAL_NVIC_DisableIRQ(dma_irqn[dma_id]); + dma_handle[dma_id] = NULL; +} + +void dma_invalidate_channel(DMA_Stream_TypeDef *dma_stream, uint32_t dma_channel) { + int dma_id = get_dma_id(dma_stream); + if (dma_last_channel[dma_id] == dma_channel) { + dma_last_channel[dma_id] = 0xffffffff; + } +} |
