2 ******************************************************************************
3 * @file stm32f4xx_hal_dma.c
4 * @author MCD Application Team
7 * @brief DMA HAL module driver.
9 * This file provides firmware functions to manage the following
10 * functionalities of the Direct Memory Access (DMA) peripheral:
11 * + Initialization and de-initialization functions
12 * + IO operation functions
13 * + Peripheral State and errors functions
15 ==============================================================================
16 ##### How to use this driver #####
17 ==============================================================================
19 (#) Enable and configure the peripheral to be connected to the DMA Stream
20 (except for internal SRAM/FLASH memories: no initialization is
21 necessary) please refer to Reference manual for connection between peripherals
24 (#) For a given Stream, program the required configuration through the following parameters:
25 Transfer Direction, Source and Destination data formats,
26 Circular, Normal or peripheral flow control mode, Stream Priority level,
27 Source and Destination Increment mode, FIFO mode and its Threshold (if needed),
28 Burst mode for Source and/or Destination (if needed) using HAL_DMA_Init() function.
30 -@- Prior to HAL_DMA_Init() the clock must be enabled for DMA through the following macros:
31 __HAL_RCC_DMA1_CLK_ENABLE() or __HAL_RCC_DMA2_CLK_ENABLE().
33 *** Polling mode IO operation ***
34 =================================
36 (+) Use HAL_DMA_Start() to start DMA transfer after the configuration of Source
37 address and destination address and the Length of data to be transferred.
38 (+) Use HAL_DMA_PollForTransfer() to poll for the end of current transfer, in this
39 case a fixed Timeout can be configured by User depending from his application.
40 (+) Use HAL_DMA_Abort() function to abort the current transfer.
42 *** Interrupt mode IO operation ***
43 ===================================
45 (+) Configure the DMA interrupt priority using HAL_NVIC_SetPriority()
46 (+) Enable the DMA IRQ handler using HAL_NVIC_EnableIRQ()
47 (+) Use HAL_DMA_Start_IT() to start DMA transfer after the configuration of
48 Source address and destination address and the Length of data to be transferred. In this
49 case the DMA interrupt is configured
50 (+) Use HAL_DMA_IRQHandler() called under DMA_IRQHandler() Interrupt subroutine
51 (+) At the end of data transfer HAL_DMA_IRQHandler() function is executed and user can
52 add his own function by customization of function pointer XferCpltCallback and
53 XferErrorCallback (i.e a member of DMA handle structure).
55 (#) Use HAL_DMA_GetState() function to return the DMA state and HAL_DMA_GetError() in case of error
58 (#) Use HAL_DMA_Abort_IT() function to abort the current transfer
60 -@- In Memory-to-Memory transfer mode, Circular mode is not allowed.
62 -@- The FIFO is used mainly to reduce bus usage and to allow data packing/unpacking: it is
63 possible to set different Data Sizes for the Peripheral and the Memory (ie. you can set
64 Half-Word data size for the peripheral to access its data register and set Word data size
65 for the Memory to gain in access time. Each two half words will be packed and written in
66 a single access to a Word in the Memory).
68 -@- When FIFO is disabled, it is not allowed to configure different Data Sizes for Source
69 and Destination. In this case the Peripheral Data Size will be applied to both Source
72 *** DMA HAL driver macros list ***
73 =============================================
75 Below the list of most used macros in DMA HAL driver.
77 (+) __HAL_DMA_ENABLE: Enable the specified DMA Stream.
78 (+) __HAL_DMA_DISABLE: Disable the specified DMA Stream.
79 (+) __HAL_DMA_GET_IT_SOURCE: Check whether the specified DMA Stream interrupt has occurred or not.
82 (@) You can refer to the DMA HAL driver header file for more useful macros
85 ******************************************************************************
88 * <h2><center>© COPYRIGHT(c) 2017 STMicroelectronics</center></h2>
90 * Redistribution and use in source and binary forms, with or without modification,
91 * are permitted provided that the following conditions are met:
92 * 1. Redistributions of source code must retain the above copyright notice,
93 * this list of conditions and the following disclaimer.
94 * 2. Redistributions in binary form must reproduce the above copyright notice,
95 * this list of conditions and the following disclaimer in the documentation
96 * and/or other materials provided with the distribution.
97 * 3. Neither the name of STMicroelectronics nor the names of its contributors
98 * may be used to endorse or promote products derived from this software
99 * without specific prior written permission.
101 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
102 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
103 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
104 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
105 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
106 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
107 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
108 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
109 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
110 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
112 ******************************************************************************
115 /* Includes ------------------------------------------------------------------*/
116 #include "stm32f4xx_hal.h"
118 /** @addtogroup STM32F4xx_HAL_Driver
122 /** @defgroup DMA DMA
123 * @brief DMA HAL module driver
127 #ifdef HAL_DMA_MODULE_ENABLED
129 /* Private types -------------------------------------------------------------*/
132 __IO
uint32_t ISR
; /*!< DMA interrupt status register */
133 __IO
uint32_t Reserved0
;
134 __IO
uint32_t IFCR
; /*!< DMA interrupt flag clear register */
135 } DMA_Base_Registers
;
137 /* Private variables ---------------------------------------------------------*/
138 /* Private constants ---------------------------------------------------------*/
139 /** @addtogroup DMA_Private_Constants
142 #define HAL_TIMEOUT_DMA_ABORT 5U /* 5 ms */
146 /* Private macros ------------------------------------------------------------*/
147 /* Private functions ---------------------------------------------------------*/
148 /** @addtogroup DMA_Private_Functions
151 static void DMA_SetConfig(DMA_HandleTypeDef
*hdma
, uint32_t SrcAddress
, uint32_t DstAddress
, uint32_t DataLength
);
152 static uint32_t DMA_CalcBaseAndBitshift(DMA_HandleTypeDef
*hdma
);
153 static HAL_StatusTypeDef
DMA_CheckFifoParam(DMA_HandleTypeDef
*hdma
);
159 /* Exported functions ---------------------------------------------------------*/
160 /** @addtogroup DMA_Exported_Functions
164 /** @addtogroup DMA_Exported_Functions_Group1
167 ===============================================================================
168 ##### Initialization and de-initialization functions #####
169 ===============================================================================
171 This section provides functions allowing to initialize the DMA Stream source
172 and destination addresses, incrementation and data sizes, transfer direction,
173 circular/normal mode selection, memory-to-memory mode selection and Stream priority value.
175 The HAL_DMA_Init() function follows the DMA configuration procedures as described in
183 * @brief Initialize the DMA according to the specified
184 * parameters in the DMA_InitTypeDef and create the associated handle.
185 * @param hdma: Pointer to a DMA_HandleTypeDef structure that contains
186 * the configuration information for the specified DMA Stream.
189 HAL_StatusTypeDef
HAL_DMA_Init(DMA_HandleTypeDef
*hdma
)
192 uint32_t tickstart
= HAL_GetTick();
193 DMA_Base_Registers
*regs
;
195 /* Check the DMA peripheral state */
201 /* Check the parameters */
202 assert_param(IS_DMA_STREAM_ALL_INSTANCE(hdma
->Instance
));
203 assert_param(IS_DMA_CHANNEL(hdma
->Init
.Channel
));
204 assert_param(IS_DMA_DIRECTION(hdma
->Init
.Direction
));
205 assert_param(IS_DMA_PERIPHERAL_INC_STATE(hdma
->Init
.PeriphInc
));
206 assert_param(IS_DMA_MEMORY_INC_STATE(hdma
->Init
.MemInc
));
207 assert_param(IS_DMA_PERIPHERAL_DATA_SIZE(hdma
->Init
.PeriphDataAlignment
));
208 assert_param(IS_DMA_MEMORY_DATA_SIZE(hdma
->Init
.MemDataAlignment
));
209 assert_param(IS_DMA_MODE(hdma
->Init
.Mode
));
210 assert_param(IS_DMA_PRIORITY(hdma
->Init
.Priority
));
211 assert_param(IS_DMA_FIFO_MODE_STATE(hdma
->Init
.FIFOMode
));
212 /* Check the memory burst, peripheral burst and FIFO threshold parameters only
213 when FIFO mode is enabled */
214 if(hdma
->Init
.FIFOMode
!= DMA_FIFOMODE_DISABLE
)
216 assert_param(IS_DMA_FIFO_THRESHOLD(hdma
->Init
.FIFOThreshold
));
217 assert_param(IS_DMA_MEMORY_BURST(hdma
->Init
.MemBurst
));
218 assert_param(IS_DMA_PERIPHERAL_BURST(hdma
->Init
.PeriphBurst
));
221 /* Allocate lock resource */
224 /* Change DMA peripheral state */
225 hdma
->State
= HAL_DMA_STATE_BUSY
;
227 /* Disable the peripheral */
228 __HAL_DMA_DISABLE(hdma
);
230 /* Check if the DMA Stream is effectively disabled */
231 while((hdma
->Instance
->CR
& DMA_SxCR_EN
) != RESET
)
233 /* Check for the Timeout */
234 if((HAL_GetTick() - tickstart
) > HAL_TIMEOUT_DMA_ABORT
)
236 /* Update error code */
237 hdma
->ErrorCode
= HAL_DMA_ERROR_TIMEOUT
;
239 /* Change the DMA state */
240 hdma
->State
= HAL_DMA_STATE_TIMEOUT
;
246 /* Get the CR register value */
247 tmp
= hdma
->Instance
->CR
;
249 /* Clear CHSEL, MBURST, PBURST, PL, MSIZE, PSIZE, MINC, PINC, CIRC, DIR, CT and DBM bits */
250 tmp
&= ((uint32_t)~(DMA_SxCR_CHSEL
| DMA_SxCR_MBURST
| DMA_SxCR_PBURST
| \
251 DMA_SxCR_PL
| DMA_SxCR_MSIZE
| DMA_SxCR_PSIZE
| \
252 DMA_SxCR_MINC
| DMA_SxCR_PINC
| DMA_SxCR_CIRC
| \
253 DMA_SxCR_DIR
| DMA_SxCR_CT
| DMA_SxCR_DBM
));
255 /* Prepare the DMA Stream configuration */
256 tmp
|= hdma
->Init
.Channel
| hdma
->Init
.Direction
|
257 hdma
->Init
.PeriphInc
| hdma
->Init
.MemInc
|
258 hdma
->Init
.PeriphDataAlignment
| hdma
->Init
.MemDataAlignment
|
259 hdma
->Init
.Mode
| hdma
->Init
.Priority
;
261 /* the Memory burst and peripheral burst are not used when the FIFO is disabled */
262 if(hdma
->Init
.FIFOMode
== DMA_FIFOMODE_ENABLE
)
264 /* Get memory burst and peripheral burst */
265 tmp
|= hdma
->Init
.MemBurst
| hdma
->Init
.PeriphBurst
;
268 /* Write to DMA Stream CR register */
269 hdma
->Instance
->CR
= tmp
;
271 /* Get the FCR register value */
272 tmp
= hdma
->Instance
->FCR
;
274 /* Clear Direct mode and FIFO threshold bits */
275 tmp
&= (uint32_t)~(DMA_SxFCR_DMDIS
| DMA_SxFCR_FTH
);
277 /* Prepare the DMA Stream FIFO configuration */
278 tmp
|= hdma
->Init
.FIFOMode
;
280 /* The FIFO threshold is not used when the FIFO mode is disabled */
281 if(hdma
->Init
.FIFOMode
== DMA_FIFOMODE_ENABLE
)
283 /* Get the FIFO threshold */
284 tmp
|= hdma
->Init
.FIFOThreshold
;
286 /* Check compatibility between FIFO threshold level and size of the memory burst */
287 /* for INCR4, INCR8, INCR16 bursts */
288 if (hdma
->Init
.MemBurst
!= DMA_MBURST_SINGLE
)
290 if (DMA_CheckFifoParam(hdma
) != HAL_OK
)
292 /* Update error code */
293 hdma
->ErrorCode
= HAL_DMA_ERROR_PARAM
;
295 /* Change the DMA state */
296 hdma
->State
= HAL_DMA_STATE_READY
;
303 /* Write to DMA Stream FCR */
304 hdma
->Instance
->FCR
= tmp
;
306 /* Initialize StreamBaseAddress and StreamIndex parameters to be used to calculate
307 DMA steam Base Address needed by HAL_DMA_IRQHandler() and HAL_DMA_PollForTransfer() */
308 regs
= (DMA_Base_Registers
*)DMA_CalcBaseAndBitshift(hdma
);
310 /* Clear all interrupt flags */
311 regs
->IFCR
= 0x3FU
<< hdma
->StreamIndex
;
313 /* Initialize the error code */
314 hdma
->ErrorCode
= HAL_DMA_ERROR_NONE
;
316 /* Initialize the DMA state */
317 hdma
->State
= HAL_DMA_STATE_READY
;
323 * @brief DeInitializes the DMA peripheral
324 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
325 * the configuration information for the specified DMA Stream.
328 HAL_StatusTypeDef
HAL_DMA_DeInit(DMA_HandleTypeDef
*hdma
)
330 DMA_Base_Registers
*regs
;
332 /* Check the DMA peripheral state */
338 /* Check the DMA peripheral state */
339 if(hdma
->State
== HAL_DMA_STATE_BUSY
)
341 /* Return error status */
345 /* Check the parameters */
346 assert_param(IS_DMA_STREAM_ALL_INSTANCE(hdma
->Instance
));
348 /* Disable the selected DMA Streamx */
349 __HAL_DMA_DISABLE(hdma
);
351 /* Reset DMA Streamx control register */
352 hdma
->Instance
->CR
= 0U;
354 /* Reset DMA Streamx number of data to transfer register */
355 hdma
->Instance
->NDTR
= 0U;
357 /* Reset DMA Streamx peripheral address register */
358 hdma
->Instance
->PAR
= 0U;
360 /* Reset DMA Streamx memory 0 address register */
361 hdma
->Instance
->M0AR
= 0U;
363 /* Reset DMA Streamx memory 1 address register */
364 hdma
->Instance
->M1AR
= 0U;
366 /* Reset DMA Streamx FIFO control register */
367 hdma
->Instance
->FCR
= 0x00000021U
;
369 /* Get DMA steam Base Address */
370 regs
= (DMA_Base_Registers
*)DMA_CalcBaseAndBitshift(hdma
);
372 /* Clear all interrupt flags at correct offset within the register */
373 regs
->IFCR
= 0x3FU
<< hdma
->StreamIndex
;
375 /* Initialize the error code */
376 hdma
->ErrorCode
= HAL_DMA_ERROR_NONE
;
378 /* Initialize the DMA state */
379 hdma
->State
= HAL_DMA_STATE_RESET
;
391 /** @addtogroup DMA_Exported_Functions_Group2
394 ===============================================================================
395 ##### IO operation functions #####
396 ===============================================================================
397 [..] This section provides functions allowing to:
398 (+) Configure the source, destination address and data length and Start DMA transfer
399 (+) Configure the source, destination address and data length and
400 Start DMA transfer with interrupt
401 (+) Abort DMA transfer
402 (+) Poll for transfer complete
403 (+) Handle DMA interrupt request
410 * @brief Starts the DMA Transfer.
411 * @param hdma : pointer to a DMA_HandleTypeDef structure that contains
412 * the configuration information for the specified DMA Stream.
413 * @param SrcAddress: The source memory Buffer address
414 * @param DstAddress: The destination memory Buffer address
415 * @param DataLength: The length of data to be transferred from source to destination
418 HAL_StatusTypeDef
HAL_DMA_Start(DMA_HandleTypeDef
*hdma
, uint32_t SrcAddress
, uint32_t DstAddress
, uint32_t DataLength
)
420 HAL_StatusTypeDef status
= HAL_OK
;
422 /* Check the parameters */
423 assert_param(IS_DMA_BUFFER_SIZE(DataLength
));
428 if(HAL_DMA_STATE_READY
== hdma
->State
)
430 /* Change DMA peripheral state */
431 hdma
->State
= HAL_DMA_STATE_BUSY
;
433 /* Initialize the error code */
434 hdma
->ErrorCode
= HAL_DMA_ERROR_NONE
;
436 /* Configure the source, destination address and the data length */
437 DMA_SetConfig(hdma
, SrcAddress
, DstAddress
, DataLength
);
439 /* Enable the Peripheral */
440 __HAL_DMA_ENABLE(hdma
);
444 /* Process unlocked */
447 /* Return error status */
454 * @brief Start the DMA Transfer with interrupt enabled.
455 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
456 * the configuration information for the specified DMA Stream.
457 * @param SrcAddress: The source memory Buffer address
458 * @param DstAddress: The destination memory Buffer address
459 * @param DataLength: The length of data to be transferred from source to destination
462 HAL_StatusTypeDef
HAL_DMA_Start_IT(DMA_HandleTypeDef
*hdma
, uint32_t SrcAddress
, uint32_t DstAddress
, uint32_t DataLength
)
464 HAL_StatusTypeDef status
= HAL_OK
;
466 /* calculate DMA base and stream number */
467 DMA_Base_Registers
*regs
= (DMA_Base_Registers
*)hdma
->StreamBaseAddress
;
469 /* Check the parameters */
470 assert_param(IS_DMA_BUFFER_SIZE(DataLength
));
475 if(HAL_DMA_STATE_READY
== hdma
->State
)
477 /* Change DMA peripheral state */
478 hdma
->State
= HAL_DMA_STATE_BUSY
;
480 /* Initialize the error code */
481 hdma
->ErrorCode
= HAL_DMA_ERROR_NONE
;
483 /* Configure the source, destination address and the data length */
484 DMA_SetConfig(hdma
, SrcAddress
, DstAddress
, DataLength
);
486 /* Clear all interrupt flags at correct offset within the register */
487 regs
->IFCR
= 0x3FU
<< hdma
->StreamIndex
;
489 /* Enable Common interrupts*/
490 hdma
->Instance
->CR
|= DMA_IT_TC
| DMA_IT_TE
| DMA_IT_DME
;
491 hdma
->Instance
->FCR
|= DMA_IT_FE
;
493 if(hdma
->XferHalfCpltCallback
!= NULL
)
495 hdma
->Instance
->CR
|= DMA_IT_HT
;
498 /* Enable the Peripheral */
499 __HAL_DMA_ENABLE(hdma
);
503 /* Process unlocked */
506 /* Return error status */
514 * @brief Aborts the DMA Transfer.
515 * @param hdma : pointer to a DMA_HandleTypeDef structure that contains
516 * the configuration information for the specified DMA Stream.
518 * @note After disabling a DMA Stream, a check for wait until the DMA Stream is
519 * effectively disabled is added. If a Stream is disabled
520 * while a data transfer is ongoing, the current data will be transferred
521 * and the Stream will be effectively disabled only after the transfer of
522 * this single data is finished.
525 HAL_StatusTypeDef
HAL_DMA_Abort(DMA_HandleTypeDef
*hdma
)
527 /* calculate DMA base and stream number */
528 DMA_Base_Registers
*regs
= (DMA_Base_Registers
*)hdma
->StreamBaseAddress
;
530 uint32_t tickstart
= HAL_GetTick();
532 if(hdma
->State
!= HAL_DMA_STATE_BUSY
)
534 hdma
->ErrorCode
= HAL_DMA_ERROR_NO_XFER
;
536 /* Process Unlocked */
543 /* Disable all the transfer interrupts */
544 hdma
->Instance
->CR
&= ~(DMA_IT_TC
| DMA_IT_TE
| DMA_IT_DME
);
545 hdma
->Instance
->FCR
&= ~(DMA_IT_FE
);
547 if((hdma
->XferHalfCpltCallback
!= NULL
) || (hdma
->XferM1HalfCpltCallback
!= NULL
))
549 hdma
->Instance
->CR
&= ~(DMA_IT_HT
);
552 /* Disable the stream */
553 __HAL_DMA_DISABLE(hdma
);
555 /* Check if the DMA Stream is effectively disabled */
556 while((hdma
->Instance
->CR
& DMA_SxCR_EN
) != RESET
)
558 /* Check for the Timeout */
559 if((HAL_GetTick() - tickstart
) > HAL_TIMEOUT_DMA_ABORT
)
561 /* Update error code */
562 hdma
->ErrorCode
= HAL_DMA_ERROR_TIMEOUT
;
564 /* Process Unlocked */
567 /* Change the DMA state */
568 hdma
->State
= HAL_DMA_STATE_TIMEOUT
;
574 /* Clear all interrupt flags at correct offset within the register */
575 regs
->IFCR
= 0x3FU
<< hdma
->StreamIndex
;
577 /* Process Unlocked */
580 /* Change the DMA state*/
581 hdma
->State
= HAL_DMA_STATE_READY
;
587 * @brief Aborts the DMA Transfer in Interrupt mode.
588 * @param hdma : pointer to a DMA_HandleTypeDef structure that contains
589 * the configuration information for the specified DMA Stream.
592 HAL_StatusTypeDef
HAL_DMA_Abort_IT(DMA_HandleTypeDef
*hdma
)
594 if(hdma
->State
!= HAL_DMA_STATE_BUSY
)
596 hdma
->ErrorCode
= HAL_DMA_ERROR_NO_XFER
;
601 /* Set Abort State */
602 hdma
->State
= HAL_DMA_STATE_ABORT
;
604 /* Disable the stream */
605 __HAL_DMA_DISABLE(hdma
);
612 * @brief Polling for transfer complete.
613 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
614 * the configuration information for the specified DMA Stream.
615 * @param CompleteLevel: Specifies the DMA level complete.
616 * @note The polling mode is kept in this version for legacy. it is recommanded to use the IT model instead.
617 * This model could be used for debug purpose.
618 * @note The HAL_DMA_PollForTransfer API cannot be used in circular and double buffering mode (automatic circular mode).
619 * @param Timeout: Timeout duration.
622 HAL_StatusTypeDef
HAL_DMA_PollForTransfer(DMA_HandleTypeDef
*hdma
, HAL_DMA_LevelCompleteTypeDef CompleteLevel
, uint32_t Timeout
)
624 HAL_StatusTypeDef status
= HAL_OK
;
625 uint32_t mask_cpltlevel
;
626 uint32_t tickstart
= HAL_GetTick();
629 /* calculate DMA base and stream number */
630 DMA_Base_Registers
*regs
;
632 if(HAL_DMA_STATE_BUSY
!= hdma
->State
)
634 /* No transfer ongoing */
635 hdma
->ErrorCode
= HAL_DMA_ERROR_NO_XFER
;
640 /* Polling mode not supported in circular mode and double buffering mode */
641 if ((hdma
->Instance
->CR
& DMA_SxCR_CIRC
) != RESET
)
643 hdma
->ErrorCode
= HAL_DMA_ERROR_NOT_SUPPORTED
;
647 /* Get the level transfer complete flag */
648 if(CompleteLevel
== HAL_DMA_FULL_TRANSFER
)
650 /* Transfer Complete flag */
651 mask_cpltlevel
= DMA_FLAG_TCIF0_4
<< hdma
->StreamIndex
;
655 /* Half Transfer Complete flag */
656 mask_cpltlevel
= DMA_FLAG_HTIF0_4
<< hdma
->StreamIndex
;
659 regs
= (DMA_Base_Registers
*)hdma
->StreamBaseAddress
;
662 while(((tmpisr
& mask_cpltlevel
) == RESET
) && ((hdma
->ErrorCode
& HAL_DMA_ERROR_TE
) == RESET
))
664 /* Check for the Timeout (Not applicable in circular mode)*/
665 if(Timeout
!= HAL_MAX_DELAY
)
667 if((Timeout
== 0U)||((HAL_GetTick() - tickstart
) > Timeout
))
669 /* Update error code */
670 hdma
->ErrorCode
= HAL_DMA_ERROR_TIMEOUT
;
672 /* Process Unlocked */
675 /* Change the DMA state */
676 hdma
->State
= HAL_DMA_STATE_READY
;
682 /* Get the ISR register value */
685 if((tmpisr
& (DMA_FLAG_TEIF0_4
<< hdma
->StreamIndex
)) != RESET
)
687 /* Update error code */
688 hdma
->ErrorCode
|= HAL_DMA_ERROR_TE
;
690 /* Clear the transfer error flag */
691 regs
->IFCR
= DMA_FLAG_TEIF0_4
<< hdma
->StreamIndex
;
694 if((tmpisr
& (DMA_FLAG_FEIF0_4
<< hdma
->StreamIndex
)) != RESET
)
696 /* Update error code */
697 hdma
->ErrorCode
|= HAL_DMA_ERROR_FE
;
699 /* Clear the FIFO error flag */
700 regs
->IFCR
= DMA_FLAG_FEIF0_4
<< hdma
->StreamIndex
;
703 if((tmpisr
& (DMA_FLAG_DMEIF0_4
<< hdma
->StreamIndex
)) != RESET
)
705 /* Update error code */
706 hdma
->ErrorCode
|= HAL_DMA_ERROR_DME
;
708 /* Clear the Direct Mode error flag */
709 regs
->IFCR
= DMA_FLAG_DMEIF0_4
<< hdma
->StreamIndex
;
713 if(hdma
->ErrorCode
!= HAL_DMA_ERROR_NONE
)
715 if((hdma
->ErrorCode
& HAL_DMA_ERROR_TE
) != RESET
)
719 /* Clear the half transfer and transfer complete flags */
720 regs
->IFCR
= (DMA_FLAG_HTIF0_4
| DMA_FLAG_TCIF0_4
) << hdma
->StreamIndex
;
722 /* Process Unlocked */
725 /* Change the DMA state */
726 hdma
->State
= HAL_DMA_STATE_READY
;
732 /* Get the level transfer complete flag */
733 if(CompleteLevel
== HAL_DMA_FULL_TRANSFER
)
735 /* Clear the half transfer and transfer complete flags */
736 regs
->IFCR
= (DMA_FLAG_HTIF0_4
| DMA_FLAG_TCIF0_4
) << hdma
->StreamIndex
;
738 /* Process Unlocked */
741 hdma
->State
= HAL_DMA_STATE_READY
;
745 /* Clear the half transfer and transfer complete flags */
746 regs
->IFCR
= (DMA_FLAG_HTIF0_4
) << hdma
->StreamIndex
;
753 * @brief Handles DMA interrupt request.
754 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
755 * the configuration information for the specified DMA Stream.
758 void HAL_DMA_IRQHandler(DMA_HandleTypeDef
*hdma
)
761 __IO
uint32_t count
= 0U;
762 uint32_t timeout
= SystemCoreClock
/ 9600U;
764 /* calculate DMA base and stream number */
765 DMA_Base_Registers
*regs
= (DMA_Base_Registers
*)hdma
->StreamBaseAddress
;
769 /* Transfer Error Interrupt management ***************************************/
770 if ((tmpisr
& (DMA_FLAG_TEIF0_4
<< hdma
->StreamIndex
)) != RESET
)
772 if(__HAL_DMA_GET_IT_SOURCE(hdma
, DMA_IT_TE
) != RESET
)
774 /* Disable the transfer error interrupt */
775 hdma
->Instance
->CR
&= ~(DMA_IT_TE
);
777 /* Clear the transfer error flag */
778 regs
->IFCR
= DMA_FLAG_TEIF0_4
<< hdma
->StreamIndex
;
780 /* Update error code */
781 hdma
->ErrorCode
|= HAL_DMA_ERROR_TE
;
784 /* FIFO Error Interrupt management ******************************************/
785 if ((tmpisr
& (DMA_FLAG_FEIF0_4
<< hdma
->StreamIndex
)) != RESET
)
787 if(__HAL_DMA_GET_IT_SOURCE(hdma
, DMA_IT_FE
) != RESET
)
789 /* Clear the FIFO error flag */
790 regs
->IFCR
= DMA_FLAG_FEIF0_4
<< hdma
->StreamIndex
;
792 /* Update error code */
793 hdma
->ErrorCode
|= HAL_DMA_ERROR_FE
;
796 /* Direct Mode Error Interrupt management ***********************************/
797 if ((tmpisr
& (DMA_FLAG_DMEIF0_4
<< hdma
->StreamIndex
)) != RESET
)
799 if(__HAL_DMA_GET_IT_SOURCE(hdma
, DMA_IT_DME
) != RESET
)
801 /* Clear the direct mode error flag */
802 regs
->IFCR
= DMA_FLAG_DMEIF0_4
<< hdma
->StreamIndex
;
804 /* Update error code */
805 hdma
->ErrorCode
|= HAL_DMA_ERROR_DME
;
808 /* Half Transfer Complete Interrupt management ******************************/
809 if ((tmpisr
& (DMA_FLAG_HTIF0_4
<< hdma
->StreamIndex
)) != RESET
)
811 if(__HAL_DMA_GET_IT_SOURCE(hdma
, DMA_IT_HT
) != RESET
)
813 /* Clear the half transfer complete flag */
814 regs
->IFCR
= DMA_FLAG_HTIF0_4
<< hdma
->StreamIndex
;
816 /* Multi_Buffering mode enabled */
817 if(((hdma
->Instance
->CR
) & (uint32_t)(DMA_SxCR_DBM
)) != RESET
)
819 /* Current memory buffer used is Memory 0 */
820 if((hdma
->Instance
->CR
& DMA_SxCR_CT
) == RESET
)
822 if(hdma
->XferHalfCpltCallback
!= NULL
)
824 /* Half transfer callback */
825 hdma
->XferHalfCpltCallback(hdma
);
828 /* Current memory buffer used is Memory 1 */
831 if(hdma
->XferM1HalfCpltCallback
!= NULL
)
833 /* Half transfer callback */
834 hdma
->XferM1HalfCpltCallback(hdma
);
840 /* Disable the half transfer interrupt if the DMA mode is not CIRCULAR */
841 if((hdma
->Instance
->CR
& DMA_SxCR_CIRC
) == RESET
)
843 /* Disable the half transfer interrupt */
844 hdma
->Instance
->CR
&= ~(DMA_IT_HT
);
847 if(hdma
->XferHalfCpltCallback
!= NULL
)
849 /* Half transfer callback */
850 hdma
->XferHalfCpltCallback(hdma
);
855 /* Transfer Complete Interrupt management ***********************************/
856 if ((tmpisr
& (DMA_FLAG_TCIF0_4
<< hdma
->StreamIndex
)) != RESET
)
858 if(__HAL_DMA_GET_IT_SOURCE(hdma
, DMA_IT_TC
) != RESET
)
860 /* Clear the transfer complete flag */
861 regs
->IFCR
= DMA_FLAG_TCIF0_4
<< hdma
->StreamIndex
;
863 if(HAL_DMA_STATE_ABORT
== hdma
->State
)
865 /* Disable all the transfer interrupts */
866 hdma
->Instance
->CR
&= ~(DMA_IT_TC
| DMA_IT_TE
| DMA_IT_DME
);
867 hdma
->Instance
->FCR
&= ~(DMA_IT_FE
);
869 if((hdma
->XferHalfCpltCallback
!= NULL
) || (hdma
->XferM1HalfCpltCallback
!= NULL
))
871 hdma
->Instance
->CR
&= ~(DMA_IT_HT
);
874 /* Clear all interrupt flags at correct offset within the register */
875 regs
->IFCR
= 0x3FU
<< hdma
->StreamIndex
;
877 /* Process Unlocked */
880 /* Change the DMA state */
881 hdma
->State
= HAL_DMA_STATE_READY
;
883 if(hdma
->XferAbortCallback
!= NULL
)
885 hdma
->XferAbortCallback(hdma
);
890 if(((hdma
->Instance
->CR
) & (uint32_t)(DMA_SxCR_DBM
)) != RESET
)
892 /* Current memory buffer used is Memory 0 */
893 if((hdma
->Instance
->CR
& DMA_SxCR_CT
) == RESET
)
895 if(hdma
->XferM1CpltCallback
!= NULL
)
897 /* Transfer complete Callback for memory1 */
898 hdma
->XferM1CpltCallback(hdma
);
901 /* Current memory buffer used is Memory 1 */
904 if(hdma
->XferCpltCallback
!= NULL
)
906 /* Transfer complete Callback for memory0 */
907 hdma
->XferCpltCallback(hdma
);
911 /* Disable the transfer complete interrupt if the DMA mode is not CIRCULAR */
914 if((hdma
->Instance
->CR
& DMA_SxCR_CIRC
) == RESET
)
916 /* Disable the transfer complete interrupt */
917 hdma
->Instance
->CR
&= ~(DMA_IT_TC
);
919 /* Process Unlocked */
922 /* Change the DMA state */
923 hdma
->State
= HAL_DMA_STATE_READY
;
926 if(hdma
->XferCpltCallback
!= NULL
)
928 /* Transfer complete callback */
929 hdma
->XferCpltCallback(hdma
);
935 /* manage error case */
936 if(hdma
->ErrorCode
!= HAL_DMA_ERROR_NONE
)
938 if((hdma
->ErrorCode
& HAL_DMA_ERROR_TE
) != RESET
)
940 hdma
->State
= HAL_DMA_STATE_ABORT
;
942 /* Disable the stream */
943 __HAL_DMA_DISABLE(hdma
);
947 if (++count
> timeout
)
952 while((hdma
->Instance
->CR
& DMA_SxCR_EN
) != RESET
);
954 /* Process Unlocked */
957 /* Change the DMA state */
958 hdma
->State
= HAL_DMA_STATE_READY
;
961 if(hdma
->XferErrorCallback
!= NULL
)
963 /* Transfer error callback */
964 hdma
->XferErrorCallback(hdma
);
970 * @brief Register callbacks
971 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
972 * the configuration information for the specified DMA Stream.
973 * @param CallbackID: User Callback identifer
974 * a DMA_HandleTypeDef structure as parameter.
975 * @param pCallback: pointer to private callbacsk function which has pointer to
976 * a DMA_HandleTypeDef structure as parameter.
979 HAL_StatusTypeDef
HAL_DMA_RegisterCallback(DMA_HandleTypeDef
*hdma
, HAL_DMA_CallbackIDTypeDef CallbackID
, void (* pCallback
)(DMA_HandleTypeDef
*_hdma
))
982 HAL_StatusTypeDef status
= HAL_OK
;
987 if(HAL_DMA_STATE_READY
== hdma
->State
)
991 case HAL_DMA_XFER_CPLT_CB_ID
:
992 hdma
->XferCpltCallback
= pCallback
;
995 case HAL_DMA_XFER_HALFCPLT_CB_ID
:
996 hdma
->XferHalfCpltCallback
= pCallback
;
999 case HAL_DMA_XFER_M1CPLT_CB_ID
:
1000 hdma
->XferM1CpltCallback
= pCallback
;
1003 case HAL_DMA_XFER_M1HALFCPLT_CB_ID
:
1004 hdma
->XferM1HalfCpltCallback
= pCallback
;
1007 case HAL_DMA_XFER_ERROR_CB_ID
:
1008 hdma
->XferErrorCallback
= pCallback
;
1011 case HAL_DMA_XFER_ABORT_CB_ID
:
1012 hdma
->XferAbortCallback
= pCallback
;
1021 /* Return error status */
1032 * @brief UnRegister callbacks
1033 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
1034 * the configuration information for the specified DMA Stream.
1035 * @param CallbackID: User Callback identifer
1036 * a HAL_DMA_CallbackIDTypeDef ENUM as parameter.
1037 * @retval HAL status
1039 HAL_StatusTypeDef
HAL_DMA_UnRegisterCallback(DMA_HandleTypeDef
*hdma
, HAL_DMA_CallbackIDTypeDef CallbackID
)
1041 HAL_StatusTypeDef status
= HAL_OK
;
1043 /* Process locked */
1046 if(HAL_DMA_STATE_READY
== hdma
->State
)
1050 case HAL_DMA_XFER_CPLT_CB_ID
:
1051 hdma
->XferCpltCallback
= NULL
;
1054 case HAL_DMA_XFER_HALFCPLT_CB_ID
:
1055 hdma
->XferHalfCpltCallback
= NULL
;
1058 case HAL_DMA_XFER_M1CPLT_CB_ID
:
1059 hdma
->XferM1CpltCallback
= NULL
;
1062 case HAL_DMA_XFER_M1HALFCPLT_CB_ID
:
1063 hdma
->XferM1HalfCpltCallback
= NULL
;
1066 case HAL_DMA_XFER_ERROR_CB_ID
:
1067 hdma
->XferErrorCallback
= NULL
;
1070 case HAL_DMA_XFER_ABORT_CB_ID
:
1071 hdma
->XferAbortCallback
= NULL
;
1074 case HAL_DMA_XFER_ALL_CB_ID
:
1075 hdma
->XferCpltCallback
= NULL
;
1076 hdma
->XferHalfCpltCallback
= NULL
;
1077 hdma
->XferM1CpltCallback
= NULL
;
1078 hdma
->XferM1HalfCpltCallback
= NULL
;
1079 hdma
->XferErrorCallback
= NULL
;
1080 hdma
->XferAbortCallback
= NULL
;
1103 /** @addtogroup DMA_Exported_Functions_Group3
1106 ===============================================================================
1107 ##### State and Errors functions #####
1108 ===============================================================================
1110 This subsection provides functions allowing to
1111 (+) Check the DMA state
1119 * @brief Returns the DMA state.
1120 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
1121 * the configuration information for the specified DMA Stream.
1124 HAL_DMA_StateTypeDef
HAL_DMA_GetState(DMA_HandleTypeDef
*hdma
)
1130 * @brief Return the DMA error code
1131 * @param hdma : pointer to a DMA_HandleTypeDef structure that contains
1132 * the configuration information for the specified DMA Stream.
1133 * @retval DMA Error Code
1135 uint32_t HAL_DMA_GetError(DMA_HandleTypeDef
*hdma
)
1137 return hdma
->ErrorCode
;
1148 /** @addtogroup DMA_Private_Functions
1153 * @brief Sets the DMA Transfer parameter.
1154 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
1155 * the configuration information for the specified DMA Stream.
1156 * @param SrcAddress: The source memory Buffer address
1157 * @param DstAddress: The destination memory Buffer address
1158 * @param DataLength: The length of data to be transferred from source to destination
1159 * @retval HAL status
1161 static void DMA_SetConfig(DMA_HandleTypeDef
*hdma
, uint32_t SrcAddress
, uint32_t DstAddress
, uint32_t DataLength
)
1164 hdma
->Instance
->CR
&= (uint32_t)(~DMA_SxCR_DBM
);
1166 /* Configure DMA Stream data length */
1167 hdma
->Instance
->NDTR
= DataLength
;
1169 /* Peripheral to Memory */
1170 if((hdma
->Init
.Direction
) == DMA_MEMORY_TO_PERIPH
)
1172 /* Configure DMA Stream destination address */
1173 hdma
->Instance
->PAR
= DstAddress
;
1175 /* Configure DMA Stream source address */
1176 hdma
->Instance
->M0AR
= SrcAddress
;
1178 /* Memory to Peripheral */
1181 /* Configure DMA Stream source address */
1182 hdma
->Instance
->PAR
= SrcAddress
;
1184 /* Configure DMA Stream destination address */
1185 hdma
->Instance
->M0AR
= DstAddress
;
1190 * @brief Returns the DMA Stream base address depending on stream number
1191 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
1192 * the configuration information for the specified DMA Stream.
1193 * @retval Stream base address
1195 static uint32_t DMA_CalcBaseAndBitshift(DMA_HandleTypeDef
*hdma
)
1197 uint32_t stream_number
= (((uint32_t)hdma
->Instance
& 0xFFU
) - 16U) / 24U;
1199 /* lookup table for necessary bitshift of flags within status registers */
1200 static const uint8_t flagBitshiftOffset
[8U] = {0U, 6U, 16U, 22U, 0U, 6U, 16U, 22U};
1201 hdma
->StreamIndex
= flagBitshiftOffset
[stream_number
];
1203 if (stream_number
> 3U)
1205 /* return pointer to HISR and HIFCR */
1206 hdma
->StreamBaseAddress
= (((uint32_t)hdma
->Instance
& (uint32_t)(~0x3FFU
)) + 4U);
1210 /* return pointer to LISR and LIFCR */
1211 hdma
->StreamBaseAddress
= ((uint32_t)hdma
->Instance
& (uint32_t)(~0x3FFU
));
1214 return hdma
->StreamBaseAddress
;
1218 * @brief Check compatibility between FIFO threshold level and size of the memory burst
1219 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
1220 * the configuration information for the specified DMA Stream.
1221 * @retval HAL status
1223 static HAL_StatusTypeDef
DMA_CheckFifoParam(DMA_HandleTypeDef
*hdma
)
1225 HAL_StatusTypeDef status
= HAL_OK
;
1226 uint32_t tmp
= hdma
->Init
.FIFOThreshold
;
1228 /* Memory Data size equal to Byte */
1229 if(hdma
->Init
.MemDataAlignment
== DMA_MDATAALIGN_BYTE
)
1233 case DMA_FIFO_THRESHOLD_1QUARTERFULL
:
1234 case DMA_FIFO_THRESHOLD_3QUARTERSFULL
:
1235 if ((hdma
->Init
.MemBurst
& DMA_SxCR_MBURST_1
) == DMA_SxCR_MBURST_1
)
1240 case DMA_FIFO_THRESHOLD_HALFFULL
:
1241 if (hdma
->Init
.MemBurst
== DMA_MBURST_INC16
)
1246 case DMA_FIFO_THRESHOLD_FULL
:
1253 /* Memory Data size equal to Half-Word */
1254 else if (hdma
->Init
.MemDataAlignment
== DMA_MDATAALIGN_HALFWORD
)
1258 case DMA_FIFO_THRESHOLD_1QUARTERFULL
:
1259 case DMA_FIFO_THRESHOLD_3QUARTERSFULL
:
1262 case DMA_FIFO_THRESHOLD_HALFFULL
:
1263 if ((hdma
->Init
.MemBurst
& DMA_SxCR_MBURST_1
) == DMA_SxCR_MBURST_1
)
1268 case DMA_FIFO_THRESHOLD_FULL
:
1269 if (hdma
->Init
.MemBurst
== DMA_MBURST_INC16
)
1279 /* Memory Data size equal to Word */
1284 case DMA_FIFO_THRESHOLD_1QUARTERFULL
:
1285 case DMA_FIFO_THRESHOLD_HALFFULL
:
1286 case DMA_FIFO_THRESHOLD_3QUARTERSFULL
:
1289 case DMA_FIFO_THRESHOLD_FULL
:
1290 if ((hdma
->Init
.MemBurst
& DMA_SxCR_MBURST_1
) == DMA_SxCR_MBURST_1
)
1307 #endif /* HAL_DMA_MODULE_ENABLED */
1316 /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/