| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179 | /**  ******************************************************************************  * @file    stm32l4xx_hal_dma.c  * @author  MCD Application Team  * @brief   DMA HAL module driver.  *         This file provides firmware functions to manage the following  *         functionalities of the Direct Memory Access (DMA) peripheral:  *           + Initialization and de-initialization functions  *           + IO operation functions  *           + Peripheral State and errors functions  @verbatim  ==============================================================================                        ##### How to use this driver #####  ==============================================================================  [..]   (#) Enable and configure the peripheral to be connected to the DMA Channel       (except for internal SRAM / FLASH memories: no initialization is       necessary). Please refer to the Reference manual for connection between peripherals       and DMA requests.   (#) For a given Channel, program the required configuration through the following parameters:       Channel request, Transfer Direction, Source and Destination data formats,       Circular or Normal mode, Channel Priority level, Source and Destination Increment mode       using HAL_DMA_Init() function.       Prior to HAL_DMA_Init the peripheral clock shall be enabled for both DMA & DMAMUX       thanks to:      (##) DMA1 or DMA2: __HAL_RCC_DMA1_CLK_ENABLE() or  __HAL_RCC_DMA2_CLK_ENABLE() ;      (##) DMAMUX1:      __HAL_RCC_DMAMUX1_CLK_ENABLE();   (#) Use HAL_DMA_GetState() function to return the DMA state and HAL_DMA_GetError() in case of error       detection.   (#) Use HAL_DMA_Abort() function to abort the current transfer     -@-   In Memory-to-Memory transfer mode, Circular mode is not allowed.     *** Polling mode IO operation ***     =================================    [..]          (+) Use HAL_DMA_Start() to start DMA transfer after the configuration of Source              address and destination address and the Length of data to be transferred          (+) Use HAL_DMA_PollForTransfer() to poll for the end of current transfer, in this              case a fixed Timeout can be configured by User depending from his application.     *** Interrupt mode IO operation ***     ===================================    [..]          (+) Configure the DMA interrupt priority using HAL_NVIC_SetPriority()          (+) Enable the DMA IRQ handler using HAL_NVIC_EnableIRQ()          (+) Use HAL_DMA_Start_IT() to start DMA transfer after the configuration of              Source address and destination address and the Length of data to be transferred.              In this case the DMA interrupt is configured          (+) Use HAL_DMA_IRQHandler() called under DMA_IRQHandler() Interrupt subroutine          (+) At the end of data transfer HAL_DMA_IRQHandler() function is executed and user can              add his own function to register callbacks with HAL_DMA_RegisterCallback().     *** DMA HAL driver macros list ***     =============================================      [..]       Below the list of macros in DMA HAL driver.       (+) __HAL_DMA_ENABLE: Enable the specified DMA Channel.       (+) __HAL_DMA_DISABLE: Disable the specified DMA Channel.       (+) __HAL_DMA_GET_FLAG: Get the DMA Channel pending flags.       (+) __HAL_DMA_CLEAR_FLAG: Clear the DMA Channel pending flags.       (+) __HAL_DMA_ENABLE_IT: Enable the specified DMA Channel interrupts.       (+) __HAL_DMA_DISABLE_IT: Disable the specified DMA Channel interrupts.       (+) __HAL_DMA_GET_IT_SOURCE: Check whether the specified DMA Channel interrupt is enabled or not.     [..]      (@) You can refer to the DMA HAL driver header file for more useful macros  @endverbatim  ******************************************************************************  * @attention  *  * <h2><center>© COPYRIGHT(c) 2017 STMicroelectronics</center></h2>  *  * Redistribution and use in source and binary forms, with or without modification,  * are permitted provided that the following conditions are met:  *   1. Redistributions of source code must retain the above copyright notice,  *      this list of conditions and the following disclaimer.  *   2. Redistributions in binary form must reproduce the above copyright notice,  *      this list of conditions and the following disclaimer in the documentation  *      and/or other materials provided with the distribution.  *   3. Neither the name of STMicroelectronics nor the names of its contributors  *      may be used to endorse or promote products derived from this software  *      without specific prior written permission.  *  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.  *  ******************************************************************************  *//* Includes ------------------------------------------------------------------*/#include "stm32l4xx_hal.h"/** @addtogroup STM32L4xx_HAL_Driver  * @{  *//** @defgroup DMA DMA  * @brief DMA HAL module driver  * @{  */#ifdef HAL_DMA_MODULE_ENABLED/* Private typedef -----------------------------------------------------------*//* Private define ------------------------------------------------------------*//* Private macro -------------------------------------------------------------*//* Private variables ---------------------------------------------------------*//* Private function prototypes -----------------------------------------------*//** @defgroup DMA_Private_Functions DMA Private Functions  * @{  */static void DMA_SetConfig(DMA_HandleTypeDef *hdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t DataLength);#if defined(DMAMUX1)static void DMA_CalcDMAMUXChannelBaseAndMask(DMA_HandleTypeDef *hdma);static void DMA_CalcDMAMUXRequestGenBaseAndMask(DMA_HandleTypeDef *hdma);#endif /* DMAMUX1 *//**  * @}  *//* Exported functions ---------------------------------------------------------*//** @defgroup DMA_Exported_Functions DMA Exported Functions  * @{  *//** @defgroup DMA_Exported_Functions_Group1 Initialization and de-initialization functions *  @brief   Initialization and de-initialization functions *@verbatim ===============================================================================             ##### Initialization and de-initialization functions  ##### ===============================================================================    [..]    This section provides functions allowing to initialize the DMA Channel source    and destination addresses, incrementation and data sizes, transfer direction,    circular/normal mode selection, memory-to-memory mode selection and Channel priority value.    [..]    The HAL_DMA_Init() function follows the DMA configuration procedures as described in    reference manual.@endverbatim  * @{  *//**  * @brief  Initialize the DMA according to the specified  *         parameters in the DMA_InitTypeDef and initialize the associated handle.  * @param  hdma: Pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_Init(DMA_HandleTypeDef *hdma){  uint32_t tmp = 0;  /* Check the DMA handle allocation */  if(hdma == NULL)  {    return HAL_ERROR;  }  /* Check the parameters */  assert_param(IS_DMA_ALL_INSTANCE(hdma->Instance));  assert_param(IS_DMA_DIRECTION(hdma->Init.Direction));  assert_param(IS_DMA_PERIPHERAL_INC_STATE(hdma->Init.PeriphInc));  assert_param(IS_DMA_MEMORY_INC_STATE(hdma->Init.MemInc));  assert_param(IS_DMA_PERIPHERAL_DATA_SIZE(hdma->Init.PeriphDataAlignment));  assert_param(IS_DMA_MEMORY_DATA_SIZE(hdma->Init.MemDataAlignment));  assert_param(IS_DMA_MODE(hdma->Init.Mode));  assert_param(IS_DMA_PRIORITY(hdma->Init.Priority));  assert_param(IS_DMA_ALL_REQUEST(hdma->Init.Request));  /* Compute the channel index */  if ((uint32_t)(hdma->Instance) < (uint32_t)(DMA2_Channel1))  {    /* DMA1 */    hdma->ChannelIndex = (((uint32_t)hdma->Instance - (uint32_t)DMA1_Channel1) / ((uint32_t)DMA1_Channel2 - (uint32_t)DMA1_Channel1)) << 2;    hdma->DmaBaseAddress = DMA1;  }  else  {    /* DMA2 */    hdma->ChannelIndex = (((uint32_t)hdma->Instance - (uint32_t)DMA2_Channel1) / ((uint32_t)DMA2_Channel2 - (uint32_t)DMA2_Channel1)) << 2;    hdma->DmaBaseAddress = DMA2;  }  /* Change DMA peripheral state */  hdma->State = HAL_DMA_STATE_BUSY;  /* Get the CR register value */  tmp = hdma->Instance->CCR;  /* Clear PL, MSIZE, PSIZE, MINC, PINC, CIRC, DIR and MEM2MEM bits */  tmp &= ((uint32_t)~(DMA_CCR_PL    | DMA_CCR_MSIZE  | DMA_CCR_PSIZE  |                      DMA_CCR_MINC  | DMA_CCR_PINC   | DMA_CCR_CIRC   |                      DMA_CCR_DIR   | DMA_CCR_MEM2MEM));  /* Prepare the DMA Channel configuration */  tmp |=  hdma->Init.Direction        |          hdma->Init.PeriphInc           | hdma->Init.MemInc           |          hdma->Init.PeriphDataAlignment | hdma->Init.MemDataAlignment |          hdma->Init.Mode                | hdma->Init.Priority;  /* Write to DMA Channel CR register */  hdma->Instance->CCR = tmp;#if defined(DMAMUX1)  /* Initialize parameters for DMAMUX channel :     DMAmuxChannel, DMAmuxChannelStatus and DMAmuxChannelStatusMask  */  DMA_CalcDMAMUXChannelBaseAndMask(hdma);  if(hdma->Init.Direction == DMA_MEMORY_TO_MEMORY)  {    /* if memory to memory force the request to 0*/    hdma->Init.Request = DMA_REQUEST_MEM2MEM;  }  /* Set peripheral request  to DMAMUX channel */  hdma->DMAmuxChannel->CCR = (hdma->Init.Request & DMAMUX_CxCR_DMAREQ_ID);  /* Clear the DMAMUX synchro overrun flag */  hdma->DMAmuxChannelStatus->CFR = hdma->DMAmuxChannelStatusMask;  if(((hdma->Init.Request >  0) && (hdma->Init.Request <= DMA_REQUEST_GENERATOR3)))  {    /* Initialize parameters for DMAMUX request generator :       DMAmuxRequestGen, DMAmuxRequestGenStatus and DMAmuxRequestGenStatusMask    */    DMA_CalcDMAMUXRequestGenBaseAndMask(hdma);    /* Reset the DMAMUX request generator register*/    hdma->DMAmuxRequestGen->RGCR = 0U;    /* Clear the DMAMUX request generator overrun flag */    hdma->DMAmuxRequestGenStatus->RGCFR = hdma->DMAmuxRequestGenStatusMask;  }  else  {    hdma->DMAmuxRequestGen = 0U;    hdma->DMAmuxRequestGenStatus = 0U;    hdma->DMAmuxRequestGenStatusMask = 0U;  }#endif /* DMAMUX1 */#if !defined (DMAMUX1)  /* Set request selection */  if(hdma->Init.Direction != DMA_MEMORY_TO_MEMORY)  {    /* Write to DMA channel selection register */    if (DMA1 == hdma->DmaBaseAddress)    {      /* Reset request selection for DMA1 Channelx */      DMA1_CSELR->CSELR &= ~(DMA_CSELR_C1S << hdma->ChannelIndex);      /* Configure request selection for DMA1 Channelx */      DMA1_CSELR->CSELR |= (uint32_t) (hdma->Init.Request << (hdma->ChannelIndex));    }    else /* DMA2 */    {      /* Reset request selection for DMA2 Channelx */      DMA2_CSELR->CSELR &= ~(DMA_CSELR_C1S << hdma->ChannelIndex);      /* Configure request selection for DMA2 Channelx */      DMA2_CSELR->CSELR |= (uint32_t) (hdma->Init.Request << (hdma->ChannelIndex));    }  }#endif /* STM32L431xx || STM32L432xx || STM32L433xx || STM32L442xx || STM32L443xx */       /* STM32L471xx || STM32L475xx || STM32L476xx || STM32L442xx || STM32L486xx */       /* STM32L496xx || STM32L4A6xx                                              */  /* Clean callbacks */  hdma->XferCpltCallback = NULL;  hdma->XferHalfCpltCallback = NULL;  hdma->XferErrorCallback = NULL;  hdma->XferAbortCallback = NULL;  /* Initialise the error code */  hdma->ErrorCode = HAL_DMA_ERROR_NONE;  /* Initialize the DMA state*/  hdma->State  = HAL_DMA_STATE_READY;  /* Allocate lock resource and initialize it */  hdma->Lock = HAL_UNLOCKED;  return HAL_OK;}/**  * @brief  DeInitialize the DMA peripheral.  * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_DeInit(DMA_HandleTypeDef *hdma){  /* Check the DMA handle allocation */  if (NULL == hdma )  {    return HAL_ERROR;  }  /* Check the parameters */  assert_param(IS_DMA_ALL_INSTANCE(hdma->Instance));  /* Disable the selected DMA Channelx */  __HAL_DMA_DISABLE(hdma);  /* Compute the channel index */  if ((uint32_t)(hdma->Instance) < (uint32_t)(DMA2_Channel1))  {    /* DMA1 */    hdma->ChannelIndex = (((uint32_t)hdma->Instance - (uint32_t)DMA1_Channel1) / ((uint32_t)DMA1_Channel2 - (uint32_t)DMA1_Channel1)) << 2;    hdma->DmaBaseAddress = DMA1;  }  else  {    /* DMA2 */    hdma->ChannelIndex = (((uint32_t)hdma->Instance - (uint32_t)DMA2_Channel1) / ((uint32_t)DMA2_Channel2 - (uint32_t)DMA2_Channel1)) << 2;    hdma->DmaBaseAddress = DMA2;  }  /* Reset DMA Channel control register */  hdma->Instance->CCR  = 0;  /* Clear all flags */  hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << (hdma->ChannelIndex));#if !defined (DMAMUX1)  /* Reset DMA channel selection register */  if (DMA1 == hdma->DmaBaseAddress)  {    /* DMA1 */    DMA1_CSELR->CSELR &= ~(DMA_CSELR_C1S << (hdma->ChannelIndex));  }  else  {    /* DMA2 */    DMA2_CSELR->CSELR &= ~(DMA_CSELR_C1S << (hdma->ChannelIndex));  }#endif /* STM32L431xx || STM32L432xx || STM32L433xx || STM32L442xx || STM32L443xx */       /* STM32L471xx || STM32L475xx || STM32L476xx || STM32L442xx || STM32L486xx */       /* STM32L496xx || STM32L4A6xx                                              */#if defined(DMAMUX1)  /* Initialize parameters for DMAMUX channel :     DMAmuxChannel, DMAmuxChannelStatus and DMAmuxChannelStatusMask */  DMA_CalcDMAMUXChannelBaseAndMask(hdma);  /* Reset the DMAMUX channel that corresponds to the DMA channel */  hdma->DMAmuxChannel->CCR = 0;  /* Clear the DMAMUX synchro overrun flag */  hdma->DMAmuxChannelStatus->CFR = hdma->DMAmuxChannelStatusMask;  /* Reset Request generator parameters if any */  if(((hdma->Init.Request >  0) && (hdma->Init.Request <= DMA_REQUEST_GENERATOR3)))  {    /* Initialize parameters for DMAMUX request generator :       DMAmuxRequestGen, DMAmuxRequestGenStatus and DMAmuxRequestGenStatusMask    */    DMA_CalcDMAMUXRequestGenBaseAndMask(hdma);    /* Reset the DMAMUX request generator register*/    hdma->DMAmuxRequestGen->RGCR = 0U;    /* Clear the DMAMUX request generator overrun flag */    hdma->DMAmuxRequestGenStatus->RGCFR = hdma->DMAmuxRequestGenStatusMask;  }    hdma->DMAmuxRequestGen = 0U;  hdma->DMAmuxRequestGenStatus = 0U;  hdma->DMAmuxRequestGenStatusMask = 0U;#endif /* DMAMUX1 */  /* Initialise the error code */  hdma->ErrorCode = HAL_DMA_ERROR_NONE;  /* Initialize the DMA state */  hdma->State = HAL_DMA_STATE_RESET;  /* Release Lock */  __HAL_UNLOCK(hdma);  return HAL_OK;}/**  * @}  *//** @defgroup DMA_Exported_Functions_Group2 Input and Output operation functions *  @brief   Input and Output operation functions *@verbatim ===============================================================================                      #####  IO operation functions  ##### ===============================================================================    [..]  This section provides functions allowing to:      (+) Configure the source, destination address and data length and Start DMA transfer      (+) Configure the source, destination address and data length and          Start DMA transfer with interrupt      (+) Abort DMA transfer      (+) Poll for transfer complete      (+) Handle DMA interrupt request@endverbatim  * @{  *//**  * @brief  Start the DMA Transfer.  * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.  * @param  SrcAddress: The source memory Buffer address  * @param  DstAddress: The destination memory Buffer address  * @param  DataLength: The length of data to be transferred from source to destination  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_Start(DMA_HandleTypeDef *hdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t DataLength){  HAL_StatusTypeDef status = HAL_OK;  /* Check the parameters */  assert_param(IS_DMA_BUFFER_SIZE(DataLength));  /* Process locked */  __HAL_LOCK(hdma);  if(HAL_DMA_STATE_READY == hdma->State)  {    /* Change DMA peripheral state */    hdma->State = HAL_DMA_STATE_BUSY;    hdma->ErrorCode = HAL_DMA_ERROR_NONE;    /* Disable the peripheral */    __HAL_DMA_DISABLE(hdma);    /* Configure the source, destination address and the data length & clear flags*/    DMA_SetConfig(hdma, SrcAddress, DstAddress, DataLength);    /* Enable the Peripheral */    __HAL_DMA_ENABLE(hdma);  }  else  {    /* Process Unlocked */    __HAL_UNLOCK(hdma);    status = HAL_BUSY;  }  return status;}/**  * @brief  Start the DMA Transfer with interrupt enabled.  * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.  * @param  SrcAddress: The source memory Buffer address  * @param  DstAddress: The destination memory Buffer address  * @param  DataLength: The length of data to be transferred from source to destination  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_Start_IT(DMA_HandleTypeDef *hdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t DataLength){  HAL_StatusTypeDef status = HAL_OK;  /* Check the parameters */  assert_param(IS_DMA_BUFFER_SIZE(DataLength));  /* Process locked */  __HAL_LOCK(hdma);  if(HAL_DMA_STATE_READY == hdma->State)  {    /* Change DMA peripheral state */    hdma->State = HAL_DMA_STATE_BUSY;    hdma->ErrorCode = HAL_DMA_ERROR_NONE;    /* Disable the peripheral */    __HAL_DMA_DISABLE(hdma);    /* Configure the source, destination address and the data length & clear flags*/    DMA_SetConfig(hdma, SrcAddress, DstAddress, DataLength);    /* Enable the transfer complete interrupt */    /* Enable the transfer Error interrupt */    if(NULL != hdma->XferHalfCpltCallback )    {      /* Enable the Half transfer complete interrupt as well */      __HAL_DMA_ENABLE_IT(hdma, (DMA_IT_TC | DMA_IT_HT | DMA_IT_TE));    }    else    {      __HAL_DMA_DISABLE_IT(hdma, DMA_IT_HT);      __HAL_DMA_ENABLE_IT(hdma, (DMA_IT_TC | DMA_IT_TE));    }#ifdef DMAMUX1    /* Check if DMAMUX Synchronization is enabled*/    if((hdma->DMAmuxChannel->CCR & DMAMUX_CxCR_SE) != 0U)    {      /* Enable DMAMUX sync overrun IT*/      hdma->DMAmuxChannel->CCR |= DMAMUX_CxCR_SOIE;    }    if(hdma->DMAmuxRequestGen != 0U)    {      /* if using DMAMUX request generator, enable the DMAMUX request generator overrun IT*/      /* enable the request gen overrun IT*/      hdma->DMAmuxRequestGen->RGCR |= DMAMUX_RGxCR_OIE;    }#endif /* DMAMUX1 */    /* Enable the Peripheral */    __HAL_DMA_ENABLE(hdma);  }  else  {    /* Process Unlocked */    __HAL_UNLOCK(hdma);    /* Remain BUSY */    status = HAL_BUSY;  }  return status;}/**  * @brief  Abort the DMA Transfer.  * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.    * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_Abort(DMA_HandleTypeDef *hdma){  HAL_StatusTypeDef status = HAL_OK;  /* Check the DMA peripheral handle */  if(NULL == hdma)  {    return HAL_ERROR;  }  /* Disable DMA IT */  __HAL_DMA_DISABLE_IT(hdma, (DMA_IT_TC | DMA_IT_HT | DMA_IT_TE));#if defined(DMAMUX1)  /* disable the DMAMUX sync overrun IT*/  hdma->DMAmuxChannel->CCR &= ~DMAMUX_CxCR_SOIE;#endif /* DMAMUX1 */  /* Disable the channel */  __HAL_DMA_DISABLE(hdma);  /* Clear all flags */  hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << hdma->ChannelIndex);#if defined(DMAMUX1)  /* Clear the DMAMUX synchro overrun flag */  hdma->DMAmuxChannelStatus->CFR = hdma->DMAmuxChannelStatusMask;  if(hdma->DMAmuxRequestGen != 0U)  {    /* if using DMAMUX request generator, disable the DMAMUX request generator overrun IT*/    /* disable the request gen overrun IT*/    hdma->DMAmuxRequestGen->RGCR &= ~DMAMUX_RGxCR_OIE;    /* Clear the DMAMUX request generator overrun flag */    hdma->DMAmuxRequestGenStatus->RGCFR = hdma->DMAmuxRequestGenStatusMask;  }#endif /* DMAMUX1 */  /* Change the DMA state */  hdma->State = HAL_DMA_STATE_READY;  /* Process Unlocked */  __HAL_UNLOCK(hdma);  return status;}/**  * @brief  Aborts the DMA Transfer in Interrupt mode.  * @param  hdma  : pointer to a DMA_HandleTypeDef structure that contains  *                 the configuration information for the specified DMA Channel.  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_Abort_IT(DMA_HandleTypeDef *hdma){  HAL_StatusTypeDef status = HAL_OK;  if(HAL_DMA_STATE_BUSY != hdma->State)  {    /* no transfer ongoing */    hdma->ErrorCode = HAL_DMA_ERROR_NO_XFER;    status = HAL_ERROR;  }  else  {    /* Disable DMA IT */    __HAL_DMA_DISABLE_IT(hdma, (DMA_IT_TC | DMA_IT_HT | DMA_IT_TE));    /* Disable the channel */    __HAL_DMA_DISABLE(hdma);#if defined(DMAMUX1)    /* disable the DMAMUX sync overrun IT*/    hdma->DMAmuxChannel->CCR &= ~DMAMUX_CxCR_SOIE;    /* Clear all flags */    hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << hdma->ChannelIndex);    /* Clear the DMAMUX synchro overrun flag */    hdma->DMAmuxChannelStatus->CFR = hdma->DMAmuxChannelStatusMask;    if(hdma->DMAmuxRequestGen != 0U)    {      /* if using DMAMUX request generator, disable the DMAMUX request generator overrun IT*/      /* disable the request gen overrun IT*/      hdma->DMAmuxRequestGen->RGCR &= ~DMAMUX_RGxCR_OIE;      /* Clear the DMAMUX request generator overrun flag */      hdma->DMAmuxRequestGenStatus->RGCFR = hdma->DMAmuxRequestGenStatusMask;    }#else    /* Clear all flags */    hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << hdma->ChannelIndex);#endif /* DMAMUX1 */    /* Change the DMA state */    hdma->State = HAL_DMA_STATE_READY;    /* Process Unlocked */    __HAL_UNLOCK(hdma);    /* Call User Abort callback */    if(hdma->XferAbortCallback != NULL)    {      hdma->XferAbortCallback(hdma);    }  }  return status;}/**  * @brief  Polling for transfer complete.  * @param  hdma:    pointer to a DMA_HandleTypeDef structure that contains  *                  the configuration information for the specified DMA Channel.  * @param  CompleteLevel: Specifies the DMA level complete.  * @param  Timeout:       Timeout duration.  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_PollForTransfer(DMA_HandleTypeDef *hdma, HAL_DMA_LevelCompleteTypeDef CompleteLevel, uint32_t Timeout){  uint32_t temp;  uint32_t tickstart = 0;  if(HAL_DMA_STATE_BUSY != hdma->State)  {    /* no transfer ongoing */    hdma->ErrorCode = HAL_DMA_ERROR_NO_XFER;    __HAL_UNLOCK(hdma);    return HAL_ERROR;  }  /* Polling mode not supported in circular mode */  if (RESET != (hdma->Instance->CCR & DMA_CCR_CIRC))  {    hdma->ErrorCode = HAL_DMA_ERROR_NOT_SUPPORTED;    return HAL_ERROR;  }  /* Get the level transfer complete flag */  if (HAL_DMA_FULL_TRANSFER == CompleteLevel)  {    /* Transfer Complete flag */    temp = DMA_FLAG_TC1 << hdma->ChannelIndex;  }  else  {    /* Half Transfer Complete flag */    temp = DMA_FLAG_HT1 << hdma->ChannelIndex;  }  /* Get tick */  tickstart = HAL_GetTick();  while(RESET == (hdma->DmaBaseAddress->ISR & temp))  {    if((RESET != (hdma->DmaBaseAddress->ISR & (DMA_FLAG_TE1 << hdma->ChannelIndex))))    {      /* When a DMA transfer error occurs */      /* A hardware clear of its EN bits is performed */      /* Clear all flags */      hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << hdma->ChannelIndex);      /* Update error code */      hdma->ErrorCode = HAL_DMA_ERROR_TE;      /* Change the DMA state */      hdma->State= HAL_DMA_STATE_READY;      /* Process Unlocked */      __HAL_UNLOCK(hdma);      return HAL_ERROR;    }    /* Check for the Timeout */    if(Timeout != HAL_MAX_DELAY)    {      if((Timeout == 0) || ((HAL_GetTick() - tickstart) > Timeout))      {        /* Update error code */        hdma->ErrorCode = HAL_DMA_ERROR_TIMEOUT;        /* Change the DMA state */        hdma->State = HAL_DMA_STATE_READY;        /* Process Unlocked */        __HAL_UNLOCK(hdma);        return HAL_ERROR;      }    }  }#if defined(DMAMUX1)  /*Check for DMAMUX Request generator (if used) overrun status */  if(hdma->DMAmuxRequestGen != 0U)  {    /* if using DMAMUX request generator Check for DMAMUX request generator overrun */    if((hdma->DMAmuxRequestGenStatus->RGSR & hdma->DMAmuxRequestGenStatusMask) != 0U)    {      /* Disable the request gen overrun interrupt */      hdma->DMAmuxRequestGen->RGCR |= DMAMUX_RGxCR_OIE;      /* Clear the DMAMUX request generator overrun flag */      hdma->DMAmuxRequestGenStatus->RGCFR = hdma->DMAmuxRequestGenStatusMask;      /* Update error code */      hdma->ErrorCode |= HAL_DMA_ERROR_REQGEN;    }  }  /* Check for DMAMUX Synchronization overrun */  if((hdma->DMAmuxChannelStatus->CSR & hdma->DMAmuxChannelStatusMask) != 0U)  {    /* Clear the DMAMUX synchro overrun flag */    hdma->DMAmuxChannelStatus->CFR = hdma->DMAmuxChannelStatusMask;    /* Update error code */    hdma->ErrorCode |= HAL_DMA_ERROR_SYNC;  }#endif /* DMAMUX1 */  if(HAL_DMA_FULL_TRANSFER == CompleteLevel)  {    /* Clear the transfer complete flag */    hdma->DmaBaseAddress->IFCR = (DMA_FLAG_TC1 << hdma->ChannelIndex);    /* The selected Channelx EN bit is cleared (DMA is disabled and    all transfers are complete) */    hdma->State = HAL_DMA_STATE_READY;  }  else  {    /* Clear the half transfer complete flag */    hdma->DmaBaseAddress->IFCR = (DMA_FLAG_HT1 << hdma->ChannelIndex);  }  /* Process unlocked */  __HAL_UNLOCK(hdma);  return HAL_OK;}/**  * @brief  Handle DMA interrupt request.  * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.  * @retval None  */void HAL_DMA_IRQHandler(DMA_HandleTypeDef *hdma){  uint32_t flag_it = hdma->DmaBaseAddress->ISR;  uint32_t source_it = hdma->Instance->CCR;  /* Half Transfer Complete Interrupt management ******************************/  if ((RESET != (flag_it & (DMA_FLAG_HT1 << hdma->ChannelIndex))) && (RESET != (source_it & DMA_IT_HT)))  {      /* Disable the half transfer interrupt if the DMA mode is not CIRCULAR */      if((hdma->Instance->CCR & DMA_CCR_CIRC) == 0)      {        /* Disable the half transfer interrupt */        __HAL_DMA_DISABLE_IT(hdma, DMA_IT_HT);      }      /* Clear the half transfer complete flag */      hdma->DmaBaseAddress->IFCR = (DMA_ISR_HTIF1 << hdma->ChannelIndex);      /* DMA peripheral state is not updated in Half Transfer */      /* but in Transfer Complete case */     if(hdma->XferHalfCpltCallback != NULL)      {        /* Half transfer callback */        hdma->XferHalfCpltCallback(hdma);      }  }  /* Transfer Complete Interrupt management ***********************************/  else if ((RESET != (flag_it & (DMA_FLAG_TC1 << hdma->ChannelIndex))) && (RESET != (source_it & DMA_IT_TC)))  {    if((hdma->Instance->CCR & DMA_CCR_CIRC) == 0)    {      /* Disable the transfer complete and error interrupt */      __HAL_DMA_DISABLE_IT(hdma, DMA_IT_TE | DMA_IT_TC);      /* Change the DMA state */      hdma->State = HAL_DMA_STATE_READY;    }    /* Clear the transfer complete flag */    hdma->DmaBaseAddress->IFCR = (DMA_ISR_TCIF1 << hdma->ChannelIndex);    /* Process Unlocked */    __HAL_UNLOCK(hdma);    if(hdma->XferCpltCallback != NULL)    {      /* Transfer complete callback */      hdma->XferCpltCallback(hdma);    }  }  /* Transfer Error Interrupt management **************************************/  else if (( RESET != (flag_it & (DMA_FLAG_TE1 << hdma->ChannelIndex))) && (RESET != (source_it & DMA_IT_TE)))  {    /* When a DMA transfer error occurs */    /* A hardware clear of its EN bits is performed */    /* Disable ALL DMA IT */    __HAL_DMA_DISABLE_IT(hdma, (DMA_IT_TC | DMA_IT_HT | DMA_IT_TE));    /* Clear all flags */    hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << hdma->ChannelIndex);    /* Update error code */    hdma->ErrorCode = HAL_DMA_ERROR_TE;    /* Change the DMA state */    hdma->State = HAL_DMA_STATE_READY;    /* Process Unlocked */    __HAL_UNLOCK(hdma);    if (hdma->XferErrorCallback != NULL)    {      /* Transfer error callback */      hdma->XferErrorCallback(hdma);    }  }  return;}/**  * @brief  Register callbacks  * @param  hdma:                 pointer to a DMA_HandleTypeDef structure that contains  *                               the configuration information for the specified DMA Channel.  * @param  CallbackID:           User Callback identifer  *                               a HAL_DMA_CallbackIDTypeDef ENUM as parameter.  * @param  pCallback:            pointer to private callbacsk function which has pointer to  *                               a DMA_HandleTypeDef structure as parameter.  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_RegisterCallback(DMA_HandleTypeDef *hdma, HAL_DMA_CallbackIDTypeDef CallbackID, void (* pCallback)( DMA_HandleTypeDef * _hdma)){  HAL_StatusTypeDef status = HAL_OK;  /* Process locked */  __HAL_LOCK(hdma);  if(HAL_DMA_STATE_READY == hdma->State)  {    switch (CallbackID)    {     case  HAL_DMA_XFER_CPLT_CB_ID:           hdma->XferCpltCallback = pCallback;           break;     case  HAL_DMA_XFER_HALFCPLT_CB_ID:           hdma->XferHalfCpltCallback = pCallback;           break;     case  HAL_DMA_XFER_ERROR_CB_ID:           hdma->XferErrorCallback = pCallback;           break;     case  HAL_DMA_XFER_ABORT_CB_ID:           hdma->XferAbortCallback = pCallback;           break;     default:           status = HAL_ERROR;           break;    }  }  else  {    status = HAL_ERROR;  }  /* Release Lock */  __HAL_UNLOCK(hdma);  return status;}/**  * @brief  UnRegister callbacks  * @param  hdma:                 pointer to a DMA_HandleTypeDef structure that contains  *                               the configuration information for the specified DMA Channel.  * @param  CallbackID:           User Callback identifer  *                               a HAL_DMA_CallbackIDTypeDef ENUM as parameter.  * @retval HAL status  */HAL_StatusTypeDef HAL_DMA_UnRegisterCallback(DMA_HandleTypeDef *hdma, HAL_DMA_CallbackIDTypeDef CallbackID){  HAL_StatusTypeDef status = HAL_OK;    /* Process locked */  __HAL_LOCK(hdma);  if(HAL_DMA_STATE_READY == hdma->State)  {    switch (CallbackID)    {     case  HAL_DMA_XFER_CPLT_CB_ID:           hdma->XferCpltCallback = NULL;           break;     case  HAL_DMA_XFER_HALFCPLT_CB_ID:           hdma->XferHalfCpltCallback = NULL;           break;     case  HAL_DMA_XFER_ERROR_CB_ID:           hdma->XferErrorCallback = NULL;           break;     case  HAL_DMA_XFER_ABORT_CB_ID:           hdma->XferAbortCallback = NULL;           break;    case   HAL_DMA_XFER_ALL_CB_ID:           hdma->XferCpltCallback = NULL;           hdma->XferHalfCpltCallback = NULL;           hdma->XferErrorCallback = NULL;           hdma->XferAbortCallback = NULL;           break;    default:           status = HAL_ERROR;           break;    }  }  else  {    status = HAL_ERROR;  }  /* Release Lock */  __HAL_UNLOCK(hdma);  return status;}/**  * @}  *//** @defgroup DMA_Exported_Functions_Group3 Peripheral State and Errors functions *  @brief    Peripheral State and Errors functions *@verbatim ===============================================================================            ##### Peripheral State and Errors functions ##### ===============================================================================    [..]    This subsection provides functions allowing to      (+) Check the DMA state      (+) Get error code@endverbatim  * @{  *//**  * @brief  Return the DMA hande state.  * @param  hdma: pointer to a DMA_HandleTypeDef structure that contains  *               the configuration information for the specified DMA Channel.  * @retval HAL state  */HAL_DMA_StateTypeDef HAL_DMA_GetState(DMA_HandleTypeDef *hdma){  /* Return DMA handle state */  return hdma->State;}/**  * @brief  Return the DMA error code.  * @param  hdma : pointer to a DMA_HandleTypeDef structure that contains  *              the configuration information for the specified DMA Channel.  * @retval DMA Error Code  */uint32_t HAL_DMA_GetError(DMA_HandleTypeDef *hdma){  return hdma->ErrorCode;}/**  * @}  *//**  * @}  *//** @addtogroup DMA_Private_Functions  * @{  *//**  * @brief  Sets the DMA Transfer parameter.  * @param  hdma:       pointer to a DMA_HandleTypeDef structure that contains  *                     the configuration information for the specified DMA Channel.  * @param  SrcAddress: The source memory Buffer address  * @param  DstAddress: The destination memory Buffer address  * @param  DataLength: The length of data to be transferred from source to destination  * @retval HAL status  */static void DMA_SetConfig(DMA_HandleTypeDef *hdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t DataLength){#if defined(DMAMUX1)  /* Clear the DMAMUX synchro overrun flag */  hdma->DMAmuxChannelStatus->CFR = hdma->DMAmuxChannelStatusMask;  if(hdma->DMAmuxRequestGen != 0U)  {    /* Clear the DMAMUX request generator overrun flag */    hdma->DMAmuxRequestGenStatus->RGCFR = hdma->DMAmuxRequestGenStatusMask;  }#endif  /* Clear all flags */  hdma->DmaBaseAddress->IFCR = (DMA_ISR_GIF1 << hdma->ChannelIndex);  /* Configure DMA Channel data length */  hdma->Instance->CNDTR = DataLength;  /* Peripheral to Memory */  if((hdma->Init.Direction) == DMA_MEMORY_TO_PERIPH)  {    /* Configure DMA Channel destination address */    hdma->Instance->CPAR = DstAddress;    /* Configure DMA Channel source address */    hdma->Instance->CMAR = SrcAddress;  }  /* Memory to Peripheral */  else  {    /* Configure DMA Channel source address */    hdma->Instance->CPAR = SrcAddress;    /* Configure DMA Channel destination address */    hdma->Instance->CMAR = DstAddress;  }}#if defined(DMAMUX1)/**  * @brief  Updates the DMA handle with the DMAMUX  channel and status mask depending on stream number  * @param  hdma:       pointer to a DMA_HandleTypeDef structure that contains  *                     the configuration information for the specified DMA Stream.  * @retval None  */static void DMA_CalcDMAMUXChannelBaseAndMask(DMA_HandleTypeDef *hdma){  uint32_t channel_number = 0;  DMAMUX_Channel_TypeDef *DMAMUX1_ChannelBase;  /* check if instance is not outside the DMA channel range */  if ((uint32_t)hdma->Instance < (uint32_t)DMA2_Channel1)  {    /* DMA1 */    DMAMUX1_ChannelBase = DMAMUX1_Channel0;  }  else  {    /* DMA2 */    DMAMUX1_ChannelBase = DMAMUX1_Channel7;  }  channel_number = (((uint32_t)hdma->Instance & 0xFF) - 8) / 20;  hdma->DMAmuxChannel = (DMAMUX_Channel_TypeDef *)(uint32_t)((uint32_t)DMAMUX1_ChannelBase + (hdma->ChannelIndex >> 2) * ((uint32_t)DMAMUX1_Channel1 - (uint32_t)DMAMUX1_Channel0));  hdma->DMAmuxChannelStatus = DMAMUX1_ChannelStatus;  hdma->DMAmuxChannelStatusMask = 1U << channel_number;}/**  * @brief  Updates the DMA handle with the DMAMUX  request generator params  * @param  hdma:       pointer to a DMA_HandleTypeDef structure that contains  *                     the configuration information for the specified DMA Stream.  * @retval None  */static void DMA_CalcDMAMUXRequestGenBaseAndMask(DMA_HandleTypeDef *hdma){  uint32_t request =  hdma->Init.Request & DMAMUX_CxCR_DMAREQ_ID;  /* DMA Channels are connected to DMAMUX1 request generator blocks*/  hdma->DMAmuxRequestGen = (DMAMUX_RequestGen_TypeDef *)((uint32_t)(((uint32_t)DMAMUX1_RequestGenerator0) + ((request - 1U) * 4U)));  hdma->DMAmuxRequestGenStatus = DMAMUX1_RequestGenStatus;  hdma->DMAmuxRequestGenStatusMask = 1U << (request - 1U);}#endif /* DMAMUX1 *//**  * @}  *//**  * @}  */#endif /* HAL_DMA_MODULE_ENABLED *//**  * @}  *//**  * @}  *//************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
 |