stm32h7xx_hal_mdma.c 63 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899
  1. /**
  2. ******************************************************************************
  3. * @file stm32h7xx_hal_mdma.c
  4. * @author MCD Application Team
  5. * @brief This file provides firmware functions to manage the following
  6. * functionalities of the Master Direct Memory Access (MDMA) peripheral:
  7. * + Initialization/de-initialization functions
  8. * + I/O operation functions
  9. * + Peripheral State and errors functions
  10. ******************************************************************************
  11. * @attention
  12. *
  13. * Copyright (c) 2017 STMicroelectronics.
  14. * All rights reserved.
  15. *
  16. * This software is licensed under terms that can be found in the LICENSE file
  17. * in the root directory of this software component.
  18. * If no LICENSE file comes with this software, it is provided AS-IS.
  19. *
  20. ******************************************************************************
  21. @verbatim
  22. ==============================================================================
  23. ##### How to use this driver #####
  24. ==============================================================================
  25. [..]
  26. (#) Enable and configure the peripheral to be connected to the MDMA Channel
  27. (except for internal SRAM/FLASH memories: no initialization is
  28. necessary) please refer to Reference manual for connection between peripherals
  29. and MDMA requests.
  30. (#)
  31. For a given Channel use HAL_MDMA_Init function to program the required configuration through the following parameters:
  32. transfer request , channel priority, data endianness, Source increment, destination increment ,
  33. source data size, destination data size, data alignment, source Burst, destination Burst ,
  34. buffer Transfer Length, Transfer Trigger Mode (buffer transfer, block transfer, repeated block transfer
  35. or full transfer) source and destination block address offset, mask address and data.
  36. If using the MDMA in linked list mode then use function HAL_MDMA_LinkedList_CreateNode to fill a transfer node.
  37. Note that parameters given to the function HAL_MDMA_Init corresponds always to the node zero.
  38. Use function HAL_MDMA_LinkedList_AddNode to connect the created node to the linked list at a given position.
  39. User can make a linked list circular using function HAL_MDMA_LinkedList_EnableCircularMode , this function will automatically connect the
  40. last node of the list to the first one in order to make the list circular.
  41. In this case the linked list will loop on node 1 : first node connected after the initial transfer defined by the HAL_MDMA_Init
  42. -@- The initial transfer itself (node 0 corresponding to the Init).
  43. User can disable the circular mode using function HAL_MDMA_LinkedList_DisableCircularMode, this function will then remove
  44. the connection between last node and first one.
  45. Function HAL_MDMA_LinkedList_RemoveNode can be used to remove (disconnect) a node from the transfer linked list.
  46. When a linked list is circular (last node connected to first one), if removing node1 (node where the linked list loops),
  47. the linked list remains circular and node 2 becomes the first one.
  48. Note that if the linked list is made circular the transfer will loop infinitely (or until aborted by the user).
  49. [..]
  50. (+) User can select the transfer trigger mode (parameter TransferTriggerMode) to define the amount of data to be
  51. transfer upon a request :
  52. (++) MDMA_BUFFER_TRANSFER : each request triggers a transfer of BufferTransferLength data
  53. with BufferTransferLength defined within the HAL_MDMA_Init.
  54. (++) MDMA_BLOCK_TRANSFER : each request triggers a transfer of a block
  55. with block size defined within the function HAL_MDMA_Start/HAL_MDMA_Start_IT
  56. or within the current linked list node parameters.
  57. (++) MDMA_REPEAT_BLOCK_TRANSFER : each request triggers a transfer of a number of blocks
  58. with block size and number of blocks defined within the function HAL_MDMA_Start/HAL_MDMA_Start_IT
  59. or within the current linked list node parameters.
  60. (++) MDMA_FULL_TRANSFER : each request triggers a full transfer
  61. all blocks and all nodes(if a linked list has been created using HAL_MDMA_LinkedList_CreateNode \ HAL_MDMA_LinkedList_AddNode).
  62. *** Polling mode IO operation ***
  63. =================================
  64. [..]
  65. (+) Use HAL_MDMA_Start() to start MDMA transfer after the configuration of Source
  66. address and destination address and the Length of data to be transferred.
  67. (+) Use HAL_MDMA_PollForTransfer() to poll for the end of current transfer or a transfer level
  68. In this case a fixed Timeout can be configured by User depending from his application.
  69. (+) Use HAL_MDMA_Abort() function to abort the current transfer : blocking method this API returns
  70. when the abort ends or timeout (should not be called from an interrupt service routine).
  71. *** Interrupt mode IO operation ***
  72. ===================================
  73. [..]
  74. (+) Configure the MDMA interrupt priority using HAL_NVIC_SetPriority()
  75. (+) Enable the MDMA IRQ handler using HAL_NVIC_EnableIRQ()
  76. (+) Use HAL_MDMA_Start_IT() to start MDMA transfer after the configuration of
  77. Source address and destination address and the Length of data to be transferred. In this
  78. case the MDMA interrupt is configured.
  79. (+) Use HAL_MDMA_IRQHandler() called under MDMA_IRQHandler() Interrupt subroutine
  80. (+) At the end of data transfer HAL_MDMA_IRQHandler() function is executed and user can
  81. add his own function by customization of function pointer XferCpltCallback and
  82. XferErrorCallback (i.e a member of MDMA handle structure).
  83. (+) Use HAL_MDMA_Abort_IT() function to abort the current transfer : non-blocking method. This API will finish the execution immediately
  84. then the callback XferAbortCallback (if specified by the user) is asserted once the MDMA channel has effectively aborted.
  85. (could be called from an interrupt service routine).
  86. (+) Use functions HAL_MDMA_RegisterCallback and HAL_MDMA_UnRegisterCallback respectevely to register unregister user callbacks
  87. from the following list :
  88. (++) XferCpltCallback : transfer complete callback.
  89. (++) XferBufferCpltCallback : buffer transfer complete callback.
  90. (++) XferBlockCpltCallback : block transfer complete callback.
  91. (++) XferRepeatBlockCpltCallback : repeated block transfer complete callback.
  92. (++) XferErrorCallback : transfer error callback.
  93. (++) XferAbortCallback : transfer abort complete callback.
  94. [..]
  95. (+) If the transfer Request corresponds to SW request (MDMA_REQUEST_SW) User can use function HAL_MDMA_GenerateSWRequest to
  96. trigger requests manually. Function HAL_MDMA_GenerateSWRequest must be used with the following precautions:
  97. (++) This function returns an error if used while the Transfer has ended or not started.
  98. (++) If used while the current request has not been served yet (current request transfer on going)
  99. this function returns an error and the new request is ignored.
  100. Generally this function should be used in conjunctions with the MDMA callbacks:
  101. (++) example 1:
  102. (+++) Configure a transfer with request set to MDMA_REQUEST_SW and trigger mode set to MDMA_BUFFER_TRANSFER
  103. (+++) Register a callback for buffer transfer complete (using callback ID set to HAL_MDMA_XFER_BUFFERCPLT_CB_ID)
  104. (+++) After calling HAL_MDMA_Start_IT the MDMA will issue the transfer of a first BufferTransferLength data.
  105. (+++) When the buffer transfer complete callback is asserted first buffer has been transferred and user can ask for a new buffer transfer
  106. request using HAL_MDMA_GenerateSWRequest.
  107. (++) example 2:
  108. (+++) Configure a transfer with request set to MDMA_REQUEST_SW and trigger mode set to MDMA_BLOCK_TRANSFER
  109. (+++) Register a callback for block transfer complete (using callback ID HAL_MDMA_XFER_BLOCKCPLT_CB_ID)
  110. (+++) After calling HAL_MDMA_Start_IT the MDMA will issue the transfer of a first block of data.
  111. (+++) When the block transfer complete callback is asserted the first block has been transferred and user can ask
  112. for a new block transfer request using HAL_MDMA_GenerateSWRequest.
  113. [..] Use HAL_MDMA_GetState() function to return the MDMA state and HAL_MDMA_GetError() in case of error detection.
  114. *** MDMA HAL driver macros list ***
  115. =============================================
  116. [..]
  117. Below the list of most used macros in MDMA HAL driver.
  118. (+) __HAL_MDMA_ENABLE: Enable the specified MDMA Channel.
  119. (+) __HAL_MDMA_DISABLE: Disable the specified MDMA Channel.
  120. (+) __HAL_MDMA_GET_FLAG: Get the MDMA Channel pending flags.
  121. (+) __HAL_MDMA_CLEAR_FLAG: Clear the MDMA Channel pending flags.
  122. (+) __HAL_MDMA_ENABLE_IT: Enable the specified MDMA Channel interrupts.
  123. (+) __HAL_MDMA_DISABLE_IT: Disable the specified MDMA Channel interrupts.
  124. (+) __HAL_MDMA_GET_IT_SOURCE: Check whether the specified MDMA Channel interrupt has occurred or not.
  125. [..]
  126. (@) You can refer to the header file of the MDMA HAL driver for more useful macros.
  127. [..]
  128. @endverbatim
  129. */
  130. /* Includes ------------------------------------------------------------------*/
  131. #include "stm32h7xx_hal.h"
  132. /** @addtogroup STM32H7xx_HAL_Driver
  133. * @{
  134. */
  135. /** @defgroup MDMA MDMA
  136. * @brief MDMA HAL module driver
  137. * @{
  138. */
  139. #ifdef HAL_MDMA_MODULE_ENABLED
  140. /* Private typedef -----------------------------------------------------------*/
  141. /* Private constants ---------------------------------------------------------*/
  142. /** @addtogroup MDMA_Private_Constants
  143. * @{
  144. */
  145. #define HAL_TIMEOUT_MDMA_ABORT 5U /* 5 ms */
  146. #define HAL_MDMA_CHANNEL_SIZE 0x40U /* an MDMA instance channel size is 64 byte */
  147. /**
  148. * @}
  149. */
  150. /* Private macro -------------------------------------------------------------*/
  151. /* Private variables ---------------------------------------------------------*/
  152. /* Private function prototypes -----------------------------------------------*/
  153. /** @addtogroup MDMA_Private_Functions_Prototypes
  154. * @{
  155. */
  156. static void MDMA_SetConfig(MDMA_HandleTypeDef *hmdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t BlockDataLength, uint32_t BlockCount);
  157. static void MDMA_Init(MDMA_HandleTypeDef *hmdma);
  158. /**
  159. * @}
  160. */
  161. /** @addtogroup MDMA_Exported_Functions MDMA Exported Functions
  162. * @{
  163. */
  164. /** @addtogroup MDMA_Exported_Functions_Group1
  165. *
  166. @verbatim
  167. ===============================================================================
  168. ##### Initialization and de-initialization functions #####
  169. ===============================================================================
  170. [..]
  171. This section provides functions allowing to :
  172. Initialize and de-initialize the MDMA channel.
  173. Register and Unregister MDMA callbacks
  174. [..]
  175. The HAL_MDMA_Init() function follows the MDMA channel configuration procedures as described in
  176. reference manual.
  177. The HAL_MDMA_DeInit function allows to deinitialize the MDMA channel.
  178. HAL_MDMA_RegisterCallback and HAL_MDMA_UnRegisterCallback functions allows
  179. respectevely to register/unregister an MDMA callback function.
  180. @endverbatim
  181. * @{
  182. */
  183. /**
  184. * @brief Initializes the MDMA according to the specified
  185. * parameters in the MDMA_InitTypeDef and create the associated handle.
  186. * @param hmdma: Pointer to a MDMA_HandleTypeDef structure that contains
  187. * the configuration information for the specified MDMA Channel.
  188. * @retval HAL status
  189. */
  190. HAL_StatusTypeDef HAL_MDMA_Init(MDMA_HandleTypeDef *hmdma)
  191. {
  192. uint32_t tickstart = HAL_GetTick();
  193. /* Check the MDMA peripheral handle */
  194. if(hmdma == NULL)
  195. {
  196. return HAL_ERROR;
  197. }
  198. /* Check the parameters */
  199. assert_param(IS_MDMA_STREAM_ALL_INSTANCE(hmdma->Instance));
  200. assert_param(IS_MDMA_PRIORITY(hmdma->Init.Priority));
  201. assert_param(IS_MDMA_ENDIANNESS_MODE(hmdma->Init.Endianness));
  202. assert_param(IS_MDMA_REQUEST(hmdma->Init.Request));
  203. assert_param(IS_MDMA_SOURCE_INC(hmdma->Init.SourceInc));
  204. assert_param(IS_MDMA_DESTINATION_INC(hmdma->Init.DestinationInc));
  205. assert_param(IS_MDMA_SOURCE_DATASIZE(hmdma->Init.SourceDataSize));
  206. assert_param(IS_MDMA_DESTINATION_DATASIZE(hmdma->Init.DestDataSize));
  207. assert_param(IS_MDMA_DATA_ALIGNMENT(hmdma->Init.DataAlignment));
  208. assert_param(IS_MDMA_SOURCE_BURST(hmdma->Init.SourceBurst));
  209. assert_param(IS_MDMA_DESTINATION_BURST(hmdma->Init.DestBurst));
  210. assert_param(IS_MDMA_BUFFER_TRANSFER_LENGTH(hmdma->Init.BufferTransferLength));
  211. assert_param(IS_MDMA_TRANSFER_TRIGGER_MODE(hmdma->Init.TransferTriggerMode));
  212. assert_param(IS_MDMA_BLOCK_ADDR_OFFSET(hmdma->Init.SourceBlockAddressOffset));
  213. assert_param(IS_MDMA_BLOCK_ADDR_OFFSET(hmdma->Init.DestBlockAddressOffset));
  214. /* Allocate lock resource */
  215. __HAL_UNLOCK(hmdma);
  216. /* Change MDMA peripheral state */
  217. hmdma->State = HAL_MDMA_STATE_BUSY;
  218. /* Disable the MDMA channel */
  219. __HAL_MDMA_DISABLE(hmdma);
  220. /* Check if the MDMA channel is effectively disabled */
  221. while((hmdma->Instance->CCR & MDMA_CCR_EN) != 0U)
  222. {
  223. /* Check for the Timeout */
  224. if((HAL_GetTick() - tickstart ) > HAL_TIMEOUT_MDMA_ABORT)
  225. {
  226. /* Update error code */
  227. hmdma->ErrorCode = HAL_MDMA_ERROR_TIMEOUT;
  228. /* Change the MDMA state */
  229. hmdma->State = HAL_MDMA_STATE_ERROR;
  230. return HAL_ERROR;
  231. }
  232. }
  233. /* Initialize the MDMA channel registers */
  234. MDMA_Init(hmdma);
  235. /* Reset the MDMA first/last linkedlist node addresses and node counter */
  236. hmdma->FirstLinkedListNodeAddress = 0;
  237. hmdma->LastLinkedListNodeAddress = 0;
  238. hmdma->LinkedListNodeCounter = 0;
  239. /* Initialize the error code */
  240. hmdma->ErrorCode = HAL_MDMA_ERROR_NONE;
  241. /* Initialize the MDMA state */
  242. hmdma->State = HAL_MDMA_STATE_READY;
  243. return HAL_OK;
  244. }
  245. /**
  246. * @brief DeInitializes the MDMA peripheral
  247. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  248. * the configuration information for the specified MDMA Channel.
  249. * @retval HAL status
  250. */
  251. HAL_StatusTypeDef HAL_MDMA_DeInit(MDMA_HandleTypeDef *hmdma)
  252. {
  253. /* Check the MDMA peripheral handle */
  254. if(hmdma == NULL)
  255. {
  256. return HAL_ERROR;
  257. }
  258. /* Disable the selected MDMA Channelx */
  259. __HAL_MDMA_DISABLE(hmdma);
  260. /* Reset MDMA Channel control register */
  261. hmdma->Instance->CCR = 0;
  262. hmdma->Instance->CTCR = 0;
  263. hmdma->Instance->CBNDTR = 0;
  264. hmdma->Instance->CSAR = 0;
  265. hmdma->Instance->CDAR = 0;
  266. hmdma->Instance->CBRUR = 0;
  267. hmdma->Instance->CLAR = 0;
  268. hmdma->Instance->CTBR = 0;
  269. hmdma->Instance->CMAR = 0;
  270. hmdma->Instance->CMDR = 0;
  271. /* Clear all flags */
  272. __HAL_MDMA_CLEAR_FLAG(hmdma,(MDMA_FLAG_TE | MDMA_FLAG_CTC | MDMA_FLAG_BRT | MDMA_FLAG_BT | MDMA_FLAG_BFTC));
  273. /* Reset the MDMA first/last linkedlist node addresses and node counter */
  274. hmdma->FirstLinkedListNodeAddress = 0;
  275. hmdma->LastLinkedListNodeAddress = 0;
  276. hmdma->LinkedListNodeCounter = 0;
  277. /* Initialize the error code */
  278. hmdma->ErrorCode = HAL_MDMA_ERROR_NONE;
  279. /* Initialize the MDMA state */
  280. hmdma->State = HAL_MDMA_STATE_RESET;
  281. /* Release Lock */
  282. __HAL_UNLOCK(hmdma);
  283. return HAL_OK;
  284. }
  285. /**
  286. * @brief Config the Post request Mask address and Mask data
  287. * @param hmdma : pointer to a MDMA_HandleTypeDef structure that contains
  288. * the configuration information for the specified MDMA Channel.
  289. * @param MaskAddress: specifies the address to be updated (written) with MaskData after a request is served.
  290. * @param MaskData: specifies the value to be written to MaskAddress after a request is served.
  291. * MaskAddress and MaskData could be used to automatically clear a peripheral flag when the request is served.
  292. * @retval HAL status
  293. */
  294. HAL_StatusTypeDef HAL_MDMA_ConfigPostRequestMask(MDMA_HandleTypeDef *hmdma, uint32_t MaskAddress, uint32_t MaskData)
  295. {
  296. HAL_StatusTypeDef status = HAL_OK;
  297. /* Check the MDMA peripheral handle */
  298. if(hmdma == NULL)
  299. {
  300. return HAL_ERROR;
  301. }
  302. /* Process locked */
  303. __HAL_LOCK(hmdma);
  304. if(HAL_MDMA_STATE_READY == hmdma->State)
  305. {
  306. /* if HW request set Post Request MaskAddress and MaskData, */
  307. if((hmdma->Instance->CTCR & MDMA_CTCR_SWRM) == 0U)
  308. {
  309. /* Set the HW request clear Mask and Data */
  310. hmdma->Instance->CMAR = MaskAddress;
  311. hmdma->Instance->CMDR = MaskData;
  312. /*
  313. -If the request is done by SW : BWM could be set to 1 or 0.
  314. -If the request is done by a peripheral :
  315. If mask address not set (0) => BWM must be set to 0
  316. If mask address set (different than 0) => BWM could be set to 1 or 0
  317. */
  318. if(MaskAddress == 0U)
  319. {
  320. hmdma->Instance->CTCR &= ~MDMA_CTCR_BWM;
  321. }
  322. else
  323. {
  324. hmdma->Instance->CTCR |= MDMA_CTCR_BWM;
  325. }
  326. }
  327. else
  328. {
  329. /* Return error status */
  330. status = HAL_ERROR;
  331. }
  332. }
  333. else
  334. {
  335. /* Return error status */
  336. status = HAL_ERROR;
  337. }
  338. /* Release Lock */
  339. __HAL_UNLOCK(hmdma);
  340. return status;
  341. }
  342. /**
  343. * @brief Register callbacks
  344. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  345. * the configuration information for the specified MDMA Channel.
  346. * @param CallbackID: User Callback identifier
  347. * @param pCallback: pointer to callbacsk function.
  348. * @retval HAL status
  349. */
  350. HAL_StatusTypeDef HAL_MDMA_RegisterCallback(MDMA_HandleTypeDef *hmdma, HAL_MDMA_CallbackIDTypeDef CallbackID, void (* pCallback)(MDMA_HandleTypeDef *_hmdma))
  351. {
  352. HAL_StatusTypeDef status = HAL_OK;
  353. /* Check the MDMA peripheral handle */
  354. if(hmdma == NULL)
  355. {
  356. return HAL_ERROR;
  357. }
  358. /* Process locked */
  359. __HAL_LOCK(hmdma);
  360. if(HAL_MDMA_STATE_READY == hmdma->State)
  361. {
  362. switch (CallbackID)
  363. {
  364. case HAL_MDMA_XFER_CPLT_CB_ID:
  365. hmdma->XferCpltCallback = pCallback;
  366. break;
  367. case HAL_MDMA_XFER_BUFFERCPLT_CB_ID:
  368. hmdma->XferBufferCpltCallback = pCallback;
  369. break;
  370. case HAL_MDMA_XFER_BLOCKCPLT_CB_ID:
  371. hmdma->XferBlockCpltCallback = pCallback;
  372. break;
  373. case HAL_MDMA_XFER_REPBLOCKCPLT_CB_ID:
  374. hmdma->XferRepeatBlockCpltCallback = pCallback;
  375. break;
  376. case HAL_MDMA_XFER_ERROR_CB_ID:
  377. hmdma->XferErrorCallback = pCallback;
  378. break;
  379. case HAL_MDMA_XFER_ABORT_CB_ID:
  380. hmdma->XferAbortCallback = pCallback;
  381. break;
  382. default:
  383. break;
  384. }
  385. }
  386. else
  387. {
  388. /* Return error status */
  389. status = HAL_ERROR;
  390. }
  391. /* Release Lock */
  392. __HAL_UNLOCK(hmdma);
  393. return status;
  394. }
  395. /**
  396. * @brief UnRegister callbacks
  397. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  398. * the configuration information for the specified MDMA Channel.
  399. * @param CallbackID: User Callback identifier
  400. * a HAL_MDMA_CallbackIDTypeDef ENUM as parameter.
  401. * @retval HAL status
  402. */
  403. HAL_StatusTypeDef HAL_MDMA_UnRegisterCallback(MDMA_HandleTypeDef *hmdma, HAL_MDMA_CallbackIDTypeDef CallbackID)
  404. {
  405. HAL_StatusTypeDef status = HAL_OK;
  406. /* Check the MDMA peripheral handle */
  407. if(hmdma == NULL)
  408. {
  409. return HAL_ERROR;
  410. }
  411. /* Process locked */
  412. __HAL_LOCK(hmdma);
  413. if(HAL_MDMA_STATE_READY == hmdma->State)
  414. {
  415. switch (CallbackID)
  416. {
  417. case HAL_MDMA_XFER_CPLT_CB_ID:
  418. hmdma->XferCpltCallback = NULL;
  419. break;
  420. case HAL_MDMA_XFER_BUFFERCPLT_CB_ID:
  421. hmdma->XferBufferCpltCallback = NULL;
  422. break;
  423. case HAL_MDMA_XFER_BLOCKCPLT_CB_ID:
  424. hmdma->XferBlockCpltCallback = NULL;
  425. break;
  426. case HAL_MDMA_XFER_REPBLOCKCPLT_CB_ID:
  427. hmdma->XferRepeatBlockCpltCallback = NULL;
  428. break;
  429. case HAL_MDMA_XFER_ERROR_CB_ID:
  430. hmdma->XferErrorCallback = NULL;
  431. break;
  432. case HAL_MDMA_XFER_ABORT_CB_ID:
  433. hmdma->XferAbortCallback = NULL;
  434. break;
  435. case HAL_MDMA_XFER_ALL_CB_ID:
  436. hmdma->XferCpltCallback = NULL;
  437. hmdma->XferBufferCpltCallback = NULL;
  438. hmdma->XferBlockCpltCallback = NULL;
  439. hmdma->XferRepeatBlockCpltCallback = NULL;
  440. hmdma->XferErrorCallback = NULL;
  441. hmdma->XferAbortCallback = NULL;
  442. break;
  443. default:
  444. status = HAL_ERROR;
  445. break;
  446. }
  447. }
  448. else
  449. {
  450. status = HAL_ERROR;
  451. }
  452. /* Release Lock */
  453. __HAL_UNLOCK(hmdma);
  454. return status;
  455. }
  456. /**
  457. * @}
  458. */
  459. /** @addtogroup MDMA_Exported_Functions_Group2
  460. *
  461. @verbatim
  462. ===============================================================================
  463. ##### Linked list operation functions #####
  464. ===============================================================================
  465. [..] This section provides functions allowing to:
  466. (+) Create a linked list node
  467. (+) Add a node to the MDMA linked list
  468. (+) Remove a node from the MDMA linked list
  469. (+) Enable/Disable linked list circular mode
  470. @endverbatim
  471. * @{
  472. */
  473. /**
  474. * @brief Initializes an MDMA Link Node according to the specified
  475. * parameters in the pMDMA_LinkedListNodeConfig .
  476. * @param pNode: Pointer to a MDMA_LinkNodeTypeDef structure that contains Linked list node
  477. * registers configurations.
  478. * @param pNodeConfig: Pointer to a MDMA_LinkNodeConfTypeDef structure that contains
  479. * the configuration information for the specified MDMA Linked List Node.
  480. * @retval HAL status
  481. */
  482. HAL_StatusTypeDef HAL_MDMA_LinkedList_CreateNode(MDMA_LinkNodeTypeDef *pNode, MDMA_LinkNodeConfTypeDef *pNodeConfig)
  483. {
  484. uint32_t addressMask;
  485. uint32_t blockoffset;
  486. /* Check the MDMA peripheral state */
  487. if((pNode == NULL) || (pNodeConfig == NULL))
  488. {
  489. return HAL_ERROR;
  490. }
  491. /* Check the parameters */
  492. assert_param(IS_MDMA_PRIORITY(pNodeConfig->Init.Priority));
  493. assert_param(IS_MDMA_ENDIANNESS_MODE(pNodeConfig->Init.Endianness));
  494. assert_param(IS_MDMA_REQUEST(pNodeConfig->Init.Request));
  495. assert_param(IS_MDMA_SOURCE_INC(pNodeConfig->Init.SourceInc));
  496. assert_param(IS_MDMA_DESTINATION_INC(pNodeConfig->Init.DestinationInc));
  497. assert_param(IS_MDMA_SOURCE_DATASIZE(pNodeConfig->Init.SourceDataSize));
  498. assert_param(IS_MDMA_DESTINATION_DATASIZE(pNodeConfig->Init.DestDataSize));
  499. assert_param(IS_MDMA_DATA_ALIGNMENT(pNodeConfig->Init.DataAlignment));
  500. assert_param(IS_MDMA_SOURCE_BURST(pNodeConfig->Init.SourceBurst));
  501. assert_param(IS_MDMA_DESTINATION_BURST(pNodeConfig->Init.DestBurst));
  502. assert_param(IS_MDMA_BUFFER_TRANSFER_LENGTH(pNodeConfig->Init.BufferTransferLength));
  503. assert_param(IS_MDMA_TRANSFER_TRIGGER_MODE(pNodeConfig->Init.TransferTriggerMode));
  504. assert_param(IS_MDMA_BLOCK_ADDR_OFFSET(pNodeConfig->Init.SourceBlockAddressOffset));
  505. assert_param(IS_MDMA_BLOCK_ADDR_OFFSET(pNodeConfig->Init.DestBlockAddressOffset));
  506. assert_param(IS_MDMA_TRANSFER_LENGTH(pNodeConfig->BlockDataLength));
  507. assert_param(IS_MDMA_BLOCK_COUNT(pNodeConfig->BlockCount));
  508. /* Configure next Link node Address Register to zero */
  509. pNode->CLAR = 0;
  510. /* Configure the Link Node registers*/
  511. pNode->CTBR = 0;
  512. pNode->CMAR = 0;
  513. pNode->CMDR = 0;
  514. pNode->Reserved = 0;
  515. /* Write new CTCR Register value */
  516. pNode->CTCR = pNodeConfig->Init.SourceInc | pNodeConfig->Init.DestinationInc | \
  517. pNodeConfig->Init.SourceDataSize | pNodeConfig->Init.DestDataSize | \
  518. pNodeConfig->Init.DataAlignment| pNodeConfig->Init.SourceBurst | \
  519. pNodeConfig->Init.DestBurst | \
  520. ((pNodeConfig->Init.BufferTransferLength - 1U) << MDMA_CTCR_TLEN_Pos) | \
  521. pNodeConfig->Init.TransferTriggerMode;
  522. /* If SW request set the CTCR register to SW Request Mode*/
  523. if(pNodeConfig->Init.Request == MDMA_REQUEST_SW)
  524. {
  525. pNode->CTCR |= MDMA_CTCR_SWRM;
  526. }
  527. /*
  528. -If the request is done by SW : BWM could be set to 1 or 0.
  529. -If the request is done by a peripheral :
  530. If mask address not set (0) => BWM must be set to 0
  531. If mask address set (different than 0) => BWM could be set to 1 or 0
  532. */
  533. if((pNodeConfig->Init.Request == MDMA_REQUEST_SW) || (pNodeConfig->PostRequestMaskAddress != 0U))
  534. {
  535. pNode->CTCR |= MDMA_CTCR_BWM;
  536. }
  537. /* Set the new CBNDTR Register value */
  538. pNode->CBNDTR = ((pNodeConfig->BlockCount - 1U) << MDMA_CBNDTR_BRC_Pos) & MDMA_CBNDTR_BRC;
  539. /* if block source address offset is negative set the Block Repeat Source address Update Mode to decrement */
  540. if(pNodeConfig->Init.SourceBlockAddressOffset < 0)
  541. {
  542. pNode->CBNDTR |= MDMA_CBNDTR_BRSUM;
  543. /*write new CBRUR Register value : source repeat block offset */
  544. blockoffset = (uint32_t)(- pNodeConfig->Init.SourceBlockAddressOffset);
  545. pNode->CBRUR = blockoffset & 0x0000FFFFU;
  546. }
  547. else
  548. {
  549. /*write new CBRUR Register value : source repeat block offset */
  550. pNode->CBRUR = (((uint32_t) pNodeConfig->Init.SourceBlockAddressOffset) & 0x0000FFFFU);
  551. }
  552. /* if block destination address offset is negative set the Block Repeat destination address Update Mode to decrement */
  553. if(pNodeConfig->Init.DestBlockAddressOffset < 0)
  554. {
  555. pNode->CBNDTR |= MDMA_CBNDTR_BRDUM;
  556. /*write new CBRUR Register value : destination repeat block offset */
  557. blockoffset = (uint32_t)(- pNodeConfig->Init.DestBlockAddressOffset);
  558. pNode->CBRUR |= ((blockoffset & 0x0000FFFFU) << MDMA_CBRUR_DUV_Pos);
  559. }
  560. else
  561. {
  562. /*write new CBRUR Register value : destination repeat block offset */
  563. pNode->CBRUR |= ((((uint32_t)pNodeConfig->Init.DestBlockAddressOffset) & 0x0000FFFFU) << MDMA_CBRUR_DUV_Pos);
  564. }
  565. /* Configure MDMA Link Node data length */
  566. pNode->CBNDTR |= pNodeConfig->BlockDataLength;
  567. /* Configure MDMA Link Node destination address */
  568. pNode->CDAR = pNodeConfig->DstAddress;
  569. /* Configure MDMA Link Node Source address */
  570. pNode->CSAR = pNodeConfig->SrcAddress;
  571. /* if HW request set the HW request and the requet CleraMask and ClearData MaskData, */
  572. if(pNodeConfig->Init.Request != MDMA_REQUEST_SW)
  573. {
  574. /* Set the HW request in CTBR register */
  575. pNode->CTBR = pNodeConfig->Init.Request & MDMA_CTBR_TSEL;
  576. /* Set the HW request clear Mask and Data */
  577. pNode->CMAR = pNodeConfig->PostRequestMaskAddress;
  578. pNode->CMDR = pNodeConfig->PostRequestMaskData;
  579. }
  580. addressMask = pNodeConfig->SrcAddress & 0xFF000000U;
  581. if((addressMask == 0x20000000U) || (addressMask == 0x00000000U))
  582. {
  583. /*The AHBSbus is used as source (read operation) on channel x */
  584. pNode->CTBR |= MDMA_CTBR_SBUS;
  585. }
  586. addressMask = pNodeConfig->DstAddress & 0xFF000000U;
  587. if((addressMask == 0x20000000U) || (addressMask == 0x00000000U))
  588. {
  589. /*The AHB bus is used as destination (write operation) on channel x */
  590. pNode->CTBR |= MDMA_CTBR_DBUS;
  591. }
  592. return HAL_OK;
  593. }
  594. /**
  595. * @brief Connect a node to the linked list.
  596. * @param hmdma : Pointer to a MDMA_HandleTypeDef structure that contains
  597. * the configuration information for the specified MDMA Channel.
  598. * @param pNewNode : Pointer to a MDMA_LinkNodeTypeDef structure that contains Linked list node
  599. * to be add to the list.
  600. * @param pPrevNode : Pointer to the new node position in the linked list or zero to insert the new node
  601. * at the end of the list
  602. *
  603. * @retval HAL status
  604. */
  605. HAL_StatusTypeDef HAL_MDMA_LinkedList_AddNode(MDMA_HandleTypeDef *hmdma, MDMA_LinkNodeTypeDef *pNewNode, MDMA_LinkNodeTypeDef *pPrevNode)
  606. {
  607. MDMA_LinkNodeTypeDef *pNode;
  608. uint32_t counter = 0, nodeInserted = 0;
  609. HAL_StatusTypeDef hal_status = HAL_OK;
  610. /* Check the MDMA peripheral handle */
  611. if((hmdma == NULL) || (pNewNode == NULL))
  612. {
  613. return HAL_ERROR;
  614. }
  615. /* Process locked */
  616. __HAL_LOCK(hmdma);
  617. if(HAL_MDMA_STATE_READY == hmdma->State)
  618. {
  619. /* Change MDMA peripheral state */
  620. hmdma->State = HAL_MDMA_STATE_BUSY;
  621. /* Check if this is the first node (after the Inititlization node) */
  622. if((uint32_t)hmdma->FirstLinkedListNodeAddress == 0U)
  623. {
  624. if(pPrevNode == NULL)
  625. {
  626. /* if this is the first node after the initialization
  627. connect this node to the node 0 by updating
  628. the MDMA channel CLAR register to this node address */
  629. hmdma->Instance->CLAR = (uint32_t)pNewNode;
  630. /* Set the MDMA handle First linked List node*/
  631. hmdma->FirstLinkedListNodeAddress = pNewNode;
  632. /*reset New node link */
  633. pNewNode->CLAR = 0;
  634. /* Update the Handle last node address */
  635. hmdma->LastLinkedListNodeAddress = pNewNode;
  636. hmdma->LinkedListNodeCounter = 1;
  637. }
  638. else
  639. {
  640. hal_status = HAL_ERROR;
  641. }
  642. }
  643. else if(hmdma->FirstLinkedListNodeAddress != pNewNode)
  644. {
  645. /* Check if the node to insert already exists*/
  646. pNode = hmdma->FirstLinkedListNodeAddress;
  647. while((counter < hmdma->LinkedListNodeCounter) && (hal_status == HAL_OK))
  648. {
  649. if(pNode->CLAR == (uint32_t)pNewNode)
  650. {
  651. hal_status = HAL_ERROR; /* error this node already exist in the linked list and it is not first node */
  652. }
  653. pNode = (MDMA_LinkNodeTypeDef *)pNode->CLAR;
  654. counter++;
  655. }
  656. if(hal_status == HAL_OK)
  657. {
  658. /* Check if the previous node is the last one in the current list or zero */
  659. if((pPrevNode == hmdma->LastLinkedListNodeAddress) || (pPrevNode == NULL))
  660. {
  661. /* insert the new node at the end of the list */
  662. pNewNode->CLAR = hmdma->LastLinkedListNodeAddress->CLAR;
  663. hmdma->LastLinkedListNodeAddress->CLAR = (uint32_t)pNewNode;
  664. /* Update the Handle last node address */
  665. hmdma->LastLinkedListNodeAddress = pNewNode;
  666. /* Increment the linked list node counter */
  667. hmdma->LinkedListNodeCounter++;
  668. }
  669. else
  670. {
  671. /*insert the new node after the pPreviousNode node */
  672. pNode = hmdma->FirstLinkedListNodeAddress;
  673. counter = 0;
  674. while((counter < hmdma->LinkedListNodeCounter) && (nodeInserted == 0U))
  675. {
  676. counter++;
  677. if(pNode == pPrevNode)
  678. {
  679. /*Insert the new node after the previous one */
  680. pNewNode->CLAR = pNode->CLAR;
  681. pNode->CLAR = (uint32_t)pNewNode;
  682. /* Increment the linked list node counter */
  683. hmdma->LinkedListNodeCounter++;
  684. nodeInserted = 1;
  685. }
  686. else
  687. {
  688. pNode = (MDMA_LinkNodeTypeDef *)pNode->CLAR;
  689. }
  690. }
  691. if(nodeInserted == 0U)
  692. {
  693. hal_status = HAL_ERROR;
  694. }
  695. }
  696. }
  697. }
  698. else
  699. {
  700. hal_status = HAL_ERROR;
  701. }
  702. /* Process unlocked */
  703. __HAL_UNLOCK(hmdma);
  704. hmdma->State = HAL_MDMA_STATE_READY;
  705. return hal_status;
  706. }
  707. else
  708. {
  709. /* Process unlocked */
  710. __HAL_UNLOCK(hmdma);
  711. /* Return error status */
  712. return HAL_BUSY;
  713. }
  714. }
  715. /**
  716. * @brief Disconnect/Remove a node from the transfer linked list.
  717. * @param hmdma : Pointer to a MDMA_HandleTypeDef structure that contains
  718. * the configuration information for the specified MDMA Channel.
  719. * @param pNode : Pointer to a MDMA_LinkNodeTypeDef structure that contains Linked list node
  720. * to be removed from the list.
  721. *
  722. * @retval HAL status
  723. */
  724. HAL_StatusTypeDef HAL_MDMA_LinkedList_RemoveNode(MDMA_HandleTypeDef *hmdma, MDMA_LinkNodeTypeDef *pNode)
  725. {
  726. MDMA_LinkNodeTypeDef *ptmpNode;
  727. uint32_t counter = 0, nodeDeleted = 0;
  728. HAL_StatusTypeDef hal_status = HAL_OK;
  729. /* Check the MDMA peripheral handle */
  730. if((hmdma == NULL) || (pNode == NULL))
  731. {
  732. return HAL_ERROR;
  733. }
  734. /* Process locked */
  735. __HAL_LOCK(hmdma);
  736. if(HAL_MDMA_STATE_READY == hmdma->State)
  737. {
  738. /* Change MDMA peripheral state */
  739. hmdma->State = HAL_MDMA_STATE_BUSY;
  740. /* If first and last node are null (no nodes in the list) : return error*/
  741. if(((uint32_t)hmdma->FirstLinkedListNodeAddress == 0U) || ((uint32_t)hmdma->LastLinkedListNodeAddress == 0U) || (hmdma->LinkedListNodeCounter == 0U))
  742. {
  743. hal_status = HAL_ERROR;
  744. }
  745. else if(hmdma->FirstLinkedListNodeAddress == pNode) /* Deleting first node */
  746. {
  747. /* Delete 1st node */
  748. if(hmdma->LastLinkedListNodeAddress == pNode)
  749. {
  750. /*if the last node is at the same time the first one (1 single node after the init node 0)
  751. then update the last node too */
  752. hmdma->FirstLinkedListNodeAddress = 0;
  753. hmdma->LastLinkedListNodeAddress = 0;
  754. hmdma->LinkedListNodeCounter = 0;
  755. hmdma->Instance->CLAR = 0;
  756. }
  757. else
  758. {
  759. if((uint32_t)hmdma->FirstLinkedListNodeAddress == hmdma->LastLinkedListNodeAddress->CLAR)
  760. {
  761. /* if last node is looping to first (circular list) one update the last node connection */
  762. hmdma->LastLinkedListNodeAddress->CLAR = pNode->CLAR;
  763. }
  764. /* if deleting the first node after the initialization
  765. connect the next node to the node 0 by updating
  766. the MDMA channel CLAR register to this node address */
  767. hmdma->Instance->CLAR = pNode->CLAR;
  768. hmdma->FirstLinkedListNodeAddress = (MDMA_LinkNodeTypeDef *)hmdma->Instance->CLAR;
  769. /* Update the Handle node counter */
  770. hmdma->LinkedListNodeCounter--;
  771. }
  772. }
  773. else /* Deleting any other node */
  774. {
  775. /*Deleted node is not the first one : find it */
  776. ptmpNode = hmdma->FirstLinkedListNodeAddress;
  777. while((counter < hmdma->LinkedListNodeCounter) && (nodeDeleted == 0U))
  778. {
  779. counter++;
  780. if(ptmpNode->CLAR == ((uint32_t)pNode))
  781. {
  782. /* if deleting the last node */
  783. if(pNode == hmdma->LastLinkedListNodeAddress)
  784. {
  785. /*Update the linked list last node address in the handle*/
  786. hmdma->LastLinkedListNodeAddress = ptmpNode;
  787. }
  788. /* update the next node link after deleting pMDMA_LinkedListNode */
  789. ptmpNode->CLAR = pNode->CLAR;
  790. nodeDeleted = 1;
  791. /* Update the Handle node counter */
  792. hmdma->LinkedListNodeCounter--;
  793. }
  794. else
  795. {
  796. ptmpNode = (MDMA_LinkNodeTypeDef *)ptmpNode->CLAR;
  797. }
  798. }
  799. if(nodeDeleted == 0U)
  800. {
  801. /* last node reashed without finding the node to delete : return error */
  802. hal_status = HAL_ERROR;
  803. }
  804. }
  805. /* Process unlocked */
  806. __HAL_UNLOCK(hmdma);
  807. hmdma->State = HAL_MDMA_STATE_READY;
  808. return hal_status;
  809. }
  810. else
  811. {
  812. /* Process unlocked */
  813. __HAL_UNLOCK(hmdma);
  814. /* Return error status */
  815. return HAL_BUSY;
  816. }
  817. }
  818. /**
  819. * @brief Make the linked list circular by connecting the last node to the first.
  820. * @param hmdma : Pointer to a MDMA_HandleTypeDef structure that contains
  821. * the configuration information for the specified MDMA Channel.
  822. * @retval HAL status
  823. */
  824. HAL_StatusTypeDef HAL_MDMA_LinkedList_EnableCircularMode(MDMA_HandleTypeDef *hmdma)
  825. {
  826. HAL_StatusTypeDef hal_status = HAL_OK;
  827. /* Check the MDMA peripheral handle */
  828. if(hmdma == NULL)
  829. {
  830. return HAL_ERROR;
  831. }
  832. /* Process locked */
  833. __HAL_LOCK(hmdma);
  834. if(HAL_MDMA_STATE_READY == hmdma->State)
  835. {
  836. /* Change MDMA peripheral state */
  837. hmdma->State = HAL_MDMA_STATE_BUSY;
  838. /* If first and last node are null (no nodes in the list) : return error*/
  839. if(((uint32_t)hmdma->FirstLinkedListNodeAddress == 0U) || ((uint32_t)hmdma->LastLinkedListNodeAddress == 0U) || (hmdma->LinkedListNodeCounter == 0U))
  840. {
  841. hal_status = HAL_ERROR;
  842. }
  843. else
  844. {
  845. /* to enable circular mode Last Node should be connected to first node */
  846. hmdma->LastLinkedListNodeAddress->CLAR = (uint32_t)hmdma->FirstLinkedListNodeAddress;
  847. }
  848. }
  849. /* Process unlocked */
  850. __HAL_UNLOCK(hmdma);
  851. hmdma->State = HAL_MDMA_STATE_READY;
  852. return hal_status;
  853. }
  854. /**
  855. * @brief Disable the linked list circular mode by setting the last node connection to null
  856. * @param hmdma : Pointer to a MDMA_HandleTypeDef structure that contains
  857. * the configuration information for the specified MDMA Channel.
  858. * @retval HAL status
  859. */
  860. HAL_StatusTypeDef HAL_MDMA_LinkedList_DisableCircularMode(MDMA_HandleTypeDef *hmdma)
  861. {
  862. HAL_StatusTypeDef hal_status = HAL_OK;
  863. /* Check the MDMA peripheral handle */
  864. if(hmdma == NULL)
  865. {
  866. return HAL_ERROR;
  867. }
  868. /* Process locked */
  869. __HAL_LOCK(hmdma);
  870. if(HAL_MDMA_STATE_READY == hmdma->State)
  871. {
  872. /* Change MDMA peripheral state */
  873. hmdma->State = HAL_MDMA_STATE_BUSY;
  874. /* If first and last node are null (no nodes in the list) : return error*/
  875. if(((uint32_t)hmdma->FirstLinkedListNodeAddress == 0U) || ((uint32_t)hmdma->LastLinkedListNodeAddress == 0U) || (hmdma->LinkedListNodeCounter == 0U))
  876. {
  877. hal_status = HAL_ERROR;
  878. }
  879. else
  880. {
  881. /* to disable circular mode Last Node should be connected to NULL */
  882. hmdma->LastLinkedListNodeAddress->CLAR = 0;
  883. }
  884. }
  885. /* Process unlocked */
  886. __HAL_UNLOCK(hmdma);
  887. hmdma->State = HAL_MDMA_STATE_READY;
  888. return hal_status;
  889. }
  890. /**
  891. * @}
  892. */
  893. /** @addtogroup MDMA_Exported_Functions_Group3
  894. *
  895. @verbatim
  896. ===============================================================================
  897. ##### IO operation functions #####
  898. ===============================================================================
  899. [..] This section provides functions allowing to:
  900. (+) Configure the source, destination address and data length and Start MDMA transfer
  901. (+) Configure the source, destination address and data length and
  902. Start MDMA transfer with interrupt
  903. (+) Abort MDMA transfer
  904. (+) Poll for transfer complete
  905. (+) Generate a SW request (when Request is set to MDMA_REQUEST_SW)
  906. (+) Handle MDMA interrupt request
  907. @endverbatim
  908. * @{
  909. */
  910. /**
  911. * @brief Starts the MDMA Transfer.
  912. * @param hmdma : pointer to a MDMA_HandleTypeDef structure that contains
  913. * the configuration information for the specified MDMA Channel.
  914. * @param SrcAddress : The source memory Buffer address
  915. * @param DstAddress : The destination memory Buffer address
  916. * @param BlockDataLength : The length of a block transfer in bytes
  917. * @param BlockCount : The number of a blocks to be transfer
  918. * @retval HAL status
  919. */
  920. HAL_StatusTypeDef HAL_MDMA_Start(MDMA_HandleTypeDef *hmdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t BlockDataLength, uint32_t BlockCount)
  921. {
  922. /* Check the parameters */
  923. assert_param(IS_MDMA_TRANSFER_LENGTH(BlockDataLength));
  924. assert_param(IS_MDMA_BLOCK_COUNT(BlockCount));
  925. /* Check the MDMA peripheral handle */
  926. if(hmdma == NULL)
  927. {
  928. return HAL_ERROR;
  929. }
  930. /* Process locked */
  931. __HAL_LOCK(hmdma);
  932. if(HAL_MDMA_STATE_READY == hmdma->State)
  933. {
  934. /* Change MDMA peripheral state */
  935. hmdma->State = HAL_MDMA_STATE_BUSY;
  936. /* Initialize the error code */
  937. hmdma->ErrorCode = HAL_MDMA_ERROR_NONE;
  938. /* Disable the peripheral */
  939. __HAL_MDMA_DISABLE(hmdma);
  940. /* Configure the source, destination address and the data length */
  941. MDMA_SetConfig(hmdma, SrcAddress, DstAddress, BlockDataLength, BlockCount);
  942. /* Enable the Peripheral */
  943. __HAL_MDMA_ENABLE(hmdma);
  944. if(hmdma->Init.Request == MDMA_REQUEST_SW)
  945. {
  946. /* activate If SW request mode*/
  947. hmdma->Instance->CCR |= MDMA_CCR_SWRQ;
  948. }
  949. }
  950. else
  951. {
  952. /* Process unlocked */
  953. __HAL_UNLOCK(hmdma);
  954. /* Return error status */
  955. return HAL_BUSY;
  956. }
  957. return HAL_OK;
  958. }
  959. /**
  960. * @brief Starts the MDMA Transfer with interrupts enabled.
  961. * @param hmdma : pointer to a MDMA_HandleTypeDef structure that contains
  962. * the configuration information for the specified MDMA Channel.
  963. * @param SrcAddress : The source memory Buffer address
  964. * @param DstAddress : The destination memory Buffer address
  965. * @param BlockDataLength : The length of a block transfer in bytes
  966. * @param BlockCount : The number of a blocks to be transfer
  967. * @retval HAL status
  968. */
  969. HAL_StatusTypeDef HAL_MDMA_Start_IT(MDMA_HandleTypeDef *hmdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t BlockDataLength, uint32_t BlockCount)
  970. {
  971. /* Check the parameters */
  972. assert_param(IS_MDMA_TRANSFER_LENGTH(BlockDataLength));
  973. assert_param(IS_MDMA_BLOCK_COUNT(BlockCount));
  974. /* Check the MDMA peripheral handle */
  975. if(hmdma == NULL)
  976. {
  977. return HAL_ERROR;
  978. }
  979. /* Process locked */
  980. __HAL_LOCK(hmdma);
  981. if(HAL_MDMA_STATE_READY == hmdma->State)
  982. {
  983. /* Change MDMA peripheral state */
  984. hmdma->State = HAL_MDMA_STATE_BUSY;
  985. /* Initialize the error code */
  986. hmdma->ErrorCode = HAL_MDMA_ERROR_NONE;
  987. /* Disable the peripheral */
  988. __HAL_MDMA_DISABLE(hmdma);
  989. /* Configure the source, destination address and the data length */
  990. MDMA_SetConfig(hmdma, SrcAddress, DstAddress, BlockDataLength, BlockCount);
  991. /* Enable Common interrupts i.e Transfer Error IT and Channel Transfer Complete IT*/
  992. __HAL_MDMA_ENABLE_IT(hmdma, (MDMA_IT_TE | MDMA_IT_CTC));
  993. if(hmdma->XferBlockCpltCallback != NULL)
  994. {
  995. /* if Block transfer complete Callback is set enable the corresponding IT*/
  996. __HAL_MDMA_ENABLE_IT(hmdma, MDMA_IT_BT);
  997. }
  998. if(hmdma->XferRepeatBlockCpltCallback != NULL)
  999. {
  1000. /* if Repeated Block transfer complete Callback is set enable the corresponding IT*/
  1001. __HAL_MDMA_ENABLE_IT(hmdma, MDMA_IT_BRT);
  1002. }
  1003. if(hmdma->XferBufferCpltCallback != NULL)
  1004. {
  1005. /* if buffer transfer complete Callback is set enable the corresponding IT*/
  1006. __HAL_MDMA_ENABLE_IT(hmdma, MDMA_IT_BFTC);
  1007. }
  1008. /* Enable the Peripheral */
  1009. __HAL_MDMA_ENABLE(hmdma);
  1010. if(hmdma->Init.Request == MDMA_REQUEST_SW)
  1011. {
  1012. /* activate If SW request mode*/
  1013. hmdma->Instance->CCR |= MDMA_CCR_SWRQ;
  1014. }
  1015. }
  1016. else
  1017. {
  1018. /* Process unlocked */
  1019. __HAL_UNLOCK(hmdma);
  1020. /* Return error status */
  1021. return HAL_BUSY;
  1022. }
  1023. return HAL_OK;
  1024. }
  1025. /**
  1026. * @brief Aborts the MDMA Transfer.
  1027. * @param hmdma : pointer to a MDMA_HandleTypeDef structure that contains
  1028. * the configuration information for the specified MDMA Channel.
  1029. *
  1030. * @note After disabling a MDMA Channel, a check for wait until the MDMA Channel is
  1031. * effectively disabled is added. If a Channel is disabled
  1032. * while a data transfer is ongoing, the current data will be transferred
  1033. * and the Channel will be effectively disabled only after the transfer of
  1034. * this single data is finished.
  1035. * @retval HAL status
  1036. */
  1037. HAL_StatusTypeDef HAL_MDMA_Abort(MDMA_HandleTypeDef *hmdma)
  1038. {
  1039. uint32_t tickstart = HAL_GetTick();
  1040. /* Check the MDMA peripheral handle */
  1041. if(hmdma == NULL)
  1042. {
  1043. return HAL_ERROR;
  1044. }
  1045. if(HAL_MDMA_STATE_BUSY != hmdma->State)
  1046. {
  1047. hmdma->ErrorCode = HAL_MDMA_ERROR_NO_XFER;
  1048. /* Process Unlocked */
  1049. __HAL_UNLOCK(hmdma);
  1050. return HAL_ERROR;
  1051. }
  1052. else
  1053. {
  1054. /* Disable all the transfer interrupts */
  1055. __HAL_MDMA_DISABLE_IT(hmdma, (MDMA_IT_TE | MDMA_IT_CTC | MDMA_IT_BT | MDMA_IT_BRT | MDMA_IT_BFTC));
  1056. /* Disable the channel */
  1057. __HAL_MDMA_DISABLE(hmdma);
  1058. /* Check if the MDMA Channel is effectively disabled */
  1059. while((hmdma->Instance->CCR & MDMA_CCR_EN) != 0U)
  1060. {
  1061. /* Check for the Timeout */
  1062. if( (HAL_GetTick() - tickstart ) > HAL_TIMEOUT_MDMA_ABORT)
  1063. {
  1064. /* Update error code */
  1065. hmdma->ErrorCode |= HAL_MDMA_ERROR_TIMEOUT;
  1066. /* Process Unlocked */
  1067. __HAL_UNLOCK(hmdma);
  1068. /* Change the MDMA state */
  1069. hmdma->State = HAL_MDMA_STATE_ERROR;
  1070. return HAL_ERROR;
  1071. }
  1072. }
  1073. /* Clear all interrupt flags */
  1074. __HAL_MDMA_CLEAR_FLAG(hmdma, (MDMA_FLAG_TE | MDMA_FLAG_CTC | MDMA_FLAG_BT | MDMA_FLAG_BRT | MDMA_FLAG_BFTC));
  1075. /* Process Unlocked */
  1076. __HAL_UNLOCK(hmdma);
  1077. /* Change the MDMA state*/
  1078. hmdma->State = HAL_MDMA_STATE_READY;
  1079. }
  1080. return HAL_OK;
  1081. }
  1082. /**
  1083. * @brief Aborts the MDMA Transfer in Interrupt mode.
  1084. * @param hmdma : pointer to a MDMA_HandleTypeDef structure that contains
  1085. * the configuration information for the specified MDMA Channel.
  1086. * @retval HAL status
  1087. */
  1088. HAL_StatusTypeDef HAL_MDMA_Abort_IT(MDMA_HandleTypeDef *hmdma)
  1089. {
  1090. /* Check the MDMA peripheral handle */
  1091. if(hmdma == NULL)
  1092. {
  1093. return HAL_ERROR;
  1094. }
  1095. if(HAL_MDMA_STATE_BUSY != hmdma->State)
  1096. {
  1097. /* No transfer ongoing */
  1098. hmdma->ErrorCode = HAL_MDMA_ERROR_NO_XFER;
  1099. return HAL_ERROR;
  1100. }
  1101. else
  1102. {
  1103. /* Set Abort State */
  1104. hmdma->State = HAL_MDMA_STATE_ABORT;
  1105. /* Disable the stream */
  1106. __HAL_MDMA_DISABLE(hmdma);
  1107. }
  1108. return HAL_OK;
  1109. }
  1110. /**
  1111. * @brief Polling for transfer complete.
  1112. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  1113. * the configuration information for the specified MDMA Channel.
  1114. * @param CompleteLevel: Specifies the MDMA level complete.
  1115. * @param Timeout: Timeout duration.
  1116. * @retval HAL status
  1117. */
  1118. HAL_StatusTypeDef HAL_MDMA_PollForTransfer(MDMA_HandleTypeDef *hmdma, HAL_MDMA_LevelCompleteTypeDef CompleteLevel, uint32_t Timeout)
  1119. {
  1120. uint32_t levelFlag, errorFlag;
  1121. uint32_t tickstart;
  1122. /* Check the parameters */
  1123. assert_param(IS_MDMA_LEVEL_COMPLETE(CompleteLevel));
  1124. /* Check the MDMA peripheral handle */
  1125. if(hmdma == NULL)
  1126. {
  1127. return HAL_ERROR;
  1128. }
  1129. if(HAL_MDMA_STATE_BUSY != hmdma->State)
  1130. {
  1131. /* No transfer ongoing */
  1132. hmdma->ErrorCode = HAL_MDMA_ERROR_NO_XFER;
  1133. return HAL_ERROR;
  1134. }
  1135. /* Get the level transfer complete flag */
  1136. levelFlag = ((CompleteLevel == HAL_MDMA_FULL_TRANSFER) ? MDMA_FLAG_CTC : \
  1137. (CompleteLevel == HAL_MDMA_BUFFER_TRANSFER)? MDMA_FLAG_BFTC : \
  1138. (CompleteLevel == HAL_MDMA_BLOCK_TRANSFER) ? MDMA_FLAG_BT : \
  1139. MDMA_FLAG_BRT);
  1140. /* Get timeout */
  1141. tickstart = HAL_GetTick();
  1142. while(__HAL_MDMA_GET_FLAG(hmdma, levelFlag) == 0U)
  1143. {
  1144. if((__HAL_MDMA_GET_FLAG(hmdma, MDMA_FLAG_TE) != 0U))
  1145. {
  1146. /* Get the transfer error source flag */
  1147. errorFlag = hmdma->Instance->CESR;
  1148. if((errorFlag & MDMA_CESR_TED) == 0U)
  1149. {
  1150. /* Update error code : Read Transfer error */
  1151. hmdma->ErrorCode |= HAL_MDMA_ERROR_READ_XFER;
  1152. }
  1153. else
  1154. {
  1155. /* Update error code : Write Transfer error */
  1156. hmdma->ErrorCode |= HAL_MDMA_ERROR_WRITE_XFER;
  1157. }
  1158. if((errorFlag & MDMA_CESR_TEMD) != 0U)
  1159. {
  1160. /* Update error code : Error Mask Data */
  1161. hmdma->ErrorCode |= HAL_MDMA_ERROR_MASK_DATA;
  1162. }
  1163. if((errorFlag & MDMA_CESR_TELD) != 0U)
  1164. {
  1165. /* Update error code : Error Linked list */
  1166. hmdma->ErrorCode |= HAL_MDMA_ERROR_LINKED_LIST;
  1167. }
  1168. if((errorFlag & MDMA_CESR_ASE) != 0U)
  1169. {
  1170. /* Update error code : Address/Size alignment error */
  1171. hmdma->ErrorCode |= HAL_MDMA_ERROR_ALIGNMENT;
  1172. }
  1173. if((errorFlag & MDMA_CESR_BSE) != 0U)
  1174. {
  1175. /* Update error code : Block Size error */
  1176. hmdma->ErrorCode |= HAL_MDMA_ERROR_BLOCK_SIZE;
  1177. }
  1178. (void) HAL_MDMA_Abort(hmdma); /* if error then abort the current transfer */
  1179. /*
  1180. Note that the Abort function will
  1181. - Clear all transfer flags
  1182. - Unlock
  1183. - Set the State
  1184. */
  1185. return HAL_ERROR;
  1186. }
  1187. /* Check for the Timeout */
  1188. if(Timeout != HAL_MAX_DELAY)
  1189. {
  1190. if(((HAL_GetTick() - tickstart ) > Timeout) || (Timeout == 0U))
  1191. {
  1192. /* Update error code */
  1193. hmdma->ErrorCode |= HAL_MDMA_ERROR_TIMEOUT;
  1194. (void) HAL_MDMA_Abort(hmdma); /* if timeout then abort the current transfer */
  1195. /*
  1196. Note that the Abort function will
  1197. - Clear all transfer flags
  1198. - Unlock
  1199. - Set the State
  1200. */
  1201. return HAL_ERROR;
  1202. }
  1203. }
  1204. }
  1205. /* Clear the transfer level flag */
  1206. if(CompleteLevel == HAL_MDMA_BUFFER_TRANSFER)
  1207. {
  1208. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_BFTC);
  1209. }
  1210. else if(CompleteLevel == HAL_MDMA_BLOCK_TRANSFER)
  1211. {
  1212. __HAL_MDMA_CLEAR_FLAG(hmdma, (MDMA_FLAG_BFTC | MDMA_FLAG_BT));
  1213. }
  1214. else if(CompleteLevel == HAL_MDMA_REPEAT_BLOCK_TRANSFER)
  1215. {
  1216. __HAL_MDMA_CLEAR_FLAG(hmdma, (MDMA_FLAG_BFTC | MDMA_FLAG_BT | MDMA_FLAG_BRT));
  1217. }
  1218. else if(CompleteLevel == HAL_MDMA_FULL_TRANSFER)
  1219. {
  1220. __HAL_MDMA_CLEAR_FLAG(hmdma, (MDMA_FLAG_BRT | MDMA_FLAG_BT | MDMA_FLAG_BFTC | MDMA_FLAG_CTC));
  1221. /* Process unlocked */
  1222. __HAL_UNLOCK(hmdma);
  1223. hmdma->State = HAL_MDMA_STATE_READY;
  1224. }
  1225. else
  1226. {
  1227. return HAL_ERROR;
  1228. }
  1229. return HAL_OK;
  1230. }
  1231. /**
  1232. * @brief Generate an MDMA SW request trigger to activate the request on the given Channel.
  1233. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  1234. * the configuration information for the specified MDMA Stream.
  1235. * @retval HAL status
  1236. */
  1237. HAL_StatusTypeDef HAL_MDMA_GenerateSWRequest(MDMA_HandleTypeDef *hmdma)
  1238. {
  1239. uint32_t request_mode;
  1240. /* Check the MDMA peripheral handle */
  1241. if(hmdma == NULL)
  1242. {
  1243. return HAL_ERROR;
  1244. }
  1245. /* Get the softawre request mode */
  1246. request_mode = hmdma->Instance->CTCR & MDMA_CTCR_SWRM;
  1247. if((hmdma->Instance->CCR & MDMA_CCR_EN) == 0U)
  1248. {
  1249. /* if no Transfer on going (MDMA enable bit not set) return error */
  1250. hmdma->ErrorCode = HAL_MDMA_ERROR_NO_XFER;
  1251. return HAL_ERROR;
  1252. }
  1253. else if(((hmdma->Instance->CISR & MDMA_CISR_CRQA) != 0U) || (request_mode == 0U))
  1254. {
  1255. /* if an MDMA ongoing request has not yet end or if request mode is not SW request return error */
  1256. hmdma->ErrorCode = HAL_MDMA_ERROR_BUSY;
  1257. return HAL_ERROR;
  1258. }
  1259. else
  1260. {
  1261. /* Set the SW request bit to activate the request on the Channel */
  1262. hmdma->Instance->CCR |= MDMA_CCR_SWRQ;
  1263. return HAL_OK;
  1264. }
  1265. }
  1266. /**
  1267. * @brief Handles MDMA interrupt request.
  1268. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  1269. * the configuration information for the specified MDMA Channel.
  1270. * @retval None
  1271. */
  1272. void HAL_MDMA_IRQHandler(MDMA_HandleTypeDef *hmdma)
  1273. {
  1274. __IO uint32_t count = 0;
  1275. uint32_t timeout = SystemCoreClock / 9600U;
  1276. uint32_t generalIntFlag, errorFlag;
  1277. /* General Interrupt Flag management ****************************************/
  1278. generalIntFlag = 1UL << ((((uint32_t)hmdma->Instance - (uint32_t)(MDMA_Channel0))/HAL_MDMA_CHANNEL_SIZE) & 0x1FU);
  1279. if((MDMA->GISR0 & generalIntFlag) == 0U)
  1280. {
  1281. return; /* the General interrupt flag for the current channel is down , nothing to do */
  1282. }
  1283. /* Transfer Error Interrupt management ***************************************/
  1284. if((__HAL_MDMA_GET_FLAG(hmdma, MDMA_FLAG_TE) != 0U))
  1285. {
  1286. if(__HAL_MDMA_GET_IT_SOURCE(hmdma, MDMA_IT_TE) != 0U)
  1287. {
  1288. /* Disable the transfer error interrupt */
  1289. __HAL_MDMA_DISABLE_IT(hmdma, MDMA_IT_TE);
  1290. /* Get the transfer error source flag */
  1291. errorFlag = hmdma->Instance->CESR;
  1292. if((errorFlag & MDMA_CESR_TED) == 0U)
  1293. {
  1294. /* Update error code : Read Transfer error */
  1295. hmdma->ErrorCode |= HAL_MDMA_ERROR_READ_XFER;
  1296. }
  1297. else
  1298. {
  1299. /* Update error code : Write Transfer error */
  1300. hmdma->ErrorCode |= HAL_MDMA_ERROR_WRITE_XFER;
  1301. }
  1302. if((errorFlag & MDMA_CESR_TEMD) != 0U)
  1303. {
  1304. /* Update error code : Error Mask Data */
  1305. hmdma->ErrorCode |= HAL_MDMA_ERROR_MASK_DATA;
  1306. }
  1307. if((errorFlag & MDMA_CESR_TELD) != 0U)
  1308. {
  1309. /* Update error code : Error Linked list */
  1310. hmdma->ErrorCode |= HAL_MDMA_ERROR_LINKED_LIST;
  1311. }
  1312. if((errorFlag & MDMA_CESR_ASE) != 0U)
  1313. {
  1314. /* Update error code : Address/Size alignment error */
  1315. hmdma->ErrorCode |= HAL_MDMA_ERROR_ALIGNMENT;
  1316. }
  1317. if((errorFlag & MDMA_CESR_BSE) != 0U)
  1318. {
  1319. /* Update error code : Block Size error error */
  1320. hmdma->ErrorCode |= HAL_MDMA_ERROR_BLOCK_SIZE;
  1321. }
  1322. /* Clear the transfer error flags */
  1323. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_TE);
  1324. }
  1325. }
  1326. /* Buffer Transfer Complete Interrupt management ******************************/
  1327. if((__HAL_MDMA_GET_FLAG(hmdma, MDMA_FLAG_BFTC) != 0U))
  1328. {
  1329. if(__HAL_MDMA_GET_IT_SOURCE(hmdma, MDMA_IT_BFTC) != 0U)
  1330. {
  1331. /* Clear the buffer transfer complete flag */
  1332. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_BFTC);
  1333. if(hmdma->XferBufferCpltCallback != NULL)
  1334. {
  1335. /* Buffer transfer callback */
  1336. hmdma->XferBufferCpltCallback(hmdma);
  1337. }
  1338. }
  1339. }
  1340. /* Block Transfer Complete Interrupt management ******************************/
  1341. if((__HAL_MDMA_GET_FLAG(hmdma, MDMA_FLAG_BT) != 0U))
  1342. {
  1343. if(__HAL_MDMA_GET_IT_SOURCE(hmdma, MDMA_IT_BT) != 0U)
  1344. {
  1345. /* Clear the block transfer complete flag */
  1346. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_BT);
  1347. if(hmdma->XferBlockCpltCallback != NULL)
  1348. {
  1349. /* Block transfer callback */
  1350. hmdma->XferBlockCpltCallback(hmdma);
  1351. }
  1352. }
  1353. }
  1354. /* Repeated Block Transfer Complete Interrupt management ******************************/
  1355. if((__HAL_MDMA_GET_FLAG(hmdma, MDMA_FLAG_BRT) != 0U))
  1356. {
  1357. if(__HAL_MDMA_GET_IT_SOURCE(hmdma, MDMA_IT_BRT) != 0U)
  1358. {
  1359. /* Clear the repeat block transfer complete flag */
  1360. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_BRT);
  1361. if(hmdma->XferRepeatBlockCpltCallback != NULL)
  1362. {
  1363. /* Repeated Block transfer callback */
  1364. hmdma->XferRepeatBlockCpltCallback(hmdma);
  1365. }
  1366. }
  1367. }
  1368. /* Channel Transfer Complete Interrupt management ***********************************/
  1369. if((__HAL_MDMA_GET_FLAG(hmdma, MDMA_FLAG_CTC) != 0U))
  1370. {
  1371. if(__HAL_MDMA_GET_IT_SOURCE(hmdma, MDMA_IT_CTC) != 0U)
  1372. {
  1373. /* Disable all the transfer interrupts */
  1374. __HAL_MDMA_DISABLE_IT(hmdma, (MDMA_IT_TE | MDMA_IT_CTC | MDMA_IT_BT | MDMA_IT_BRT | MDMA_IT_BFTC));
  1375. if(HAL_MDMA_STATE_ABORT == hmdma->State)
  1376. {
  1377. /* Process Unlocked */
  1378. __HAL_UNLOCK(hmdma);
  1379. /* Change the DMA state */
  1380. hmdma->State = HAL_MDMA_STATE_READY;
  1381. if(hmdma->XferAbortCallback != NULL)
  1382. {
  1383. hmdma->XferAbortCallback(hmdma);
  1384. }
  1385. return;
  1386. }
  1387. /* Clear the Channel Transfer Complete flag */
  1388. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_CTC);
  1389. /* Process Unlocked */
  1390. __HAL_UNLOCK(hmdma);
  1391. /* Change MDMA peripheral state */
  1392. hmdma->State = HAL_MDMA_STATE_READY;
  1393. if(hmdma->XferCpltCallback != NULL)
  1394. {
  1395. /* Channel Transfer Complete callback */
  1396. hmdma->XferCpltCallback(hmdma);
  1397. }
  1398. }
  1399. }
  1400. /* manage error case */
  1401. if(hmdma->ErrorCode != HAL_MDMA_ERROR_NONE)
  1402. {
  1403. hmdma->State = HAL_MDMA_STATE_ABORT;
  1404. /* Disable the channel */
  1405. __HAL_MDMA_DISABLE(hmdma);
  1406. do
  1407. {
  1408. if (++count > timeout)
  1409. {
  1410. break;
  1411. }
  1412. }
  1413. while((hmdma->Instance->CCR & MDMA_CCR_EN) != 0U);
  1414. /* Process Unlocked */
  1415. __HAL_UNLOCK(hmdma);
  1416. if((hmdma->Instance->CCR & MDMA_CCR_EN) != 0U)
  1417. {
  1418. /* Change the MDMA state to error if MDMA disable fails */
  1419. hmdma->State = HAL_MDMA_STATE_ERROR;
  1420. }
  1421. else
  1422. {
  1423. /* Change the MDMA state to Ready if MDMA disable success */
  1424. hmdma->State = HAL_MDMA_STATE_READY;
  1425. }
  1426. if (hmdma->XferErrorCallback != NULL)
  1427. {
  1428. /* Transfer error callback */
  1429. hmdma->XferErrorCallback(hmdma);
  1430. }
  1431. }
  1432. }
  1433. /**
  1434. * @}
  1435. */
  1436. /** @addtogroup MDMA_Exported_Functions_Group4
  1437. *
  1438. @verbatim
  1439. ===============================================================================
  1440. ##### State and Errors functions #####
  1441. ===============================================================================
  1442. [..]
  1443. This subsection provides functions allowing to
  1444. (+) Check the MDMA state
  1445. (+) Get error code
  1446. @endverbatim
  1447. * @{
  1448. */
  1449. /**
  1450. * @brief Returns the MDMA state.
  1451. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  1452. * the configuration information for the specified MDMA Channel.
  1453. * @retval HAL state
  1454. */
  1455. HAL_MDMA_StateTypeDef HAL_MDMA_GetState(MDMA_HandleTypeDef *hmdma)
  1456. {
  1457. return hmdma->State;
  1458. }
  1459. /**
  1460. * @brief Return the MDMA error code
  1461. * @param hmdma : pointer to a MDMA_HandleTypeDef structure that contains
  1462. * the configuration information for the specified MDMA Channel.
  1463. * @retval MDMA Error Code
  1464. */
  1465. uint32_t HAL_MDMA_GetError(MDMA_HandleTypeDef *hmdma)
  1466. {
  1467. return hmdma->ErrorCode;
  1468. }
  1469. /**
  1470. * @}
  1471. */
  1472. /**
  1473. * @}
  1474. */
  1475. /** @addtogroup MDMA_Private_Functions
  1476. * @{
  1477. */
  1478. /**
  1479. * @brief Sets the MDMA Transfer parameter.
  1480. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  1481. * the configuration information for the specified MDMA Channel.
  1482. * @param SrcAddress: The source memory Buffer address
  1483. * @param DstAddress: The destination memory Buffer address
  1484. * @param BlockDataLength : The length of a block transfer in bytes
  1485. * @param BlockCount: The number of blocks to be transferred
  1486. * @retval HAL status
  1487. */
  1488. static void MDMA_SetConfig(MDMA_HandleTypeDef *hmdma, uint32_t SrcAddress, uint32_t DstAddress, uint32_t BlockDataLength, uint32_t BlockCount)
  1489. {
  1490. uint32_t addressMask;
  1491. /* Configure the MDMA Channel data length */
  1492. MODIFY_REG(hmdma->Instance->CBNDTR ,MDMA_CBNDTR_BNDT, (BlockDataLength & MDMA_CBNDTR_BNDT));
  1493. /* Configure the MDMA block repeat count */
  1494. MODIFY_REG(hmdma->Instance->CBNDTR , MDMA_CBNDTR_BRC , ((BlockCount - 1U) << MDMA_CBNDTR_BRC_Pos) & MDMA_CBNDTR_BRC);
  1495. /* Clear all interrupt flags */
  1496. __HAL_MDMA_CLEAR_FLAG(hmdma, MDMA_FLAG_TE | MDMA_FLAG_CTC | MDMA_CISR_BRTIF | MDMA_CISR_BTIF | MDMA_CISR_TCIF);
  1497. /* Configure MDMA Channel destination address */
  1498. hmdma->Instance->CDAR = DstAddress;
  1499. /* Configure MDMA Channel Source address */
  1500. hmdma->Instance->CSAR = SrcAddress;
  1501. addressMask = SrcAddress & 0xFF000000U;
  1502. if((addressMask == 0x20000000U) || (addressMask == 0x00000000U))
  1503. {
  1504. /*The AHBSbus is used as source (read operation) on channel x */
  1505. hmdma->Instance->CTBR |= MDMA_CTBR_SBUS;
  1506. }
  1507. else
  1508. {
  1509. /*The AXI bus is used as source (read operation) on channel x */
  1510. hmdma->Instance->CTBR &= (~MDMA_CTBR_SBUS);
  1511. }
  1512. addressMask = DstAddress & 0xFF000000U;
  1513. if((addressMask == 0x20000000U) || (addressMask == 0x00000000U))
  1514. {
  1515. /*The AHB bus is used as destination (write operation) on channel x */
  1516. hmdma->Instance->CTBR |= MDMA_CTBR_DBUS;
  1517. }
  1518. else
  1519. {
  1520. /*The AXI bus is used as destination (write operation) on channel x */
  1521. hmdma->Instance->CTBR &= (~MDMA_CTBR_DBUS);
  1522. }
  1523. /* Set the linked list register to the first node of the list */
  1524. hmdma->Instance->CLAR = (uint32_t)hmdma->FirstLinkedListNodeAddress;
  1525. }
  1526. /**
  1527. * @brief Initializes the MDMA handle according to the specified
  1528. * parameters in the MDMA_InitTypeDef
  1529. * @param hmdma: pointer to a MDMA_HandleTypeDef structure that contains
  1530. * the configuration information for the specified MDMA Channel.
  1531. * @retval None
  1532. */
  1533. static void MDMA_Init(MDMA_HandleTypeDef *hmdma)
  1534. {
  1535. uint32_t blockoffset;
  1536. /* Prepare the MDMA Channel configuration */
  1537. hmdma->Instance->CCR = hmdma->Init.Priority | hmdma->Init.Endianness;
  1538. /* Write new CTCR Register value */
  1539. hmdma->Instance->CTCR = hmdma->Init.SourceInc | hmdma->Init.DestinationInc | \
  1540. hmdma->Init.SourceDataSize | hmdma->Init.DestDataSize | \
  1541. hmdma->Init.DataAlignment | hmdma->Init.SourceBurst | \
  1542. hmdma->Init.DestBurst | \
  1543. ((hmdma->Init.BufferTransferLength - 1U) << MDMA_CTCR_TLEN_Pos) | \
  1544. hmdma->Init.TransferTriggerMode;
  1545. /* If SW request set the CTCR register to SW Request Mode */
  1546. if(hmdma->Init.Request == MDMA_REQUEST_SW)
  1547. {
  1548. /*
  1549. -If the request is done by SW : BWM could be set to 1 or 0.
  1550. -If the request is done by a peripheral :
  1551. If mask address not set (0) => BWM must be set to 0
  1552. If mask address set (different than 0) => BWM could be set to 1 or 0
  1553. */
  1554. hmdma->Instance->CTCR |= (MDMA_CTCR_SWRM | MDMA_CTCR_BWM);
  1555. }
  1556. /* Reset CBNDTR Register */
  1557. hmdma->Instance->CBNDTR = 0;
  1558. /* if block source address offset is negative set the Block Repeat Source address Update Mode to decrement */
  1559. if(hmdma->Init.SourceBlockAddressOffset < 0)
  1560. {
  1561. hmdma->Instance->CBNDTR |= MDMA_CBNDTR_BRSUM;
  1562. /* Write new CBRUR Register value : source repeat block offset */
  1563. blockoffset = (uint32_t)(- hmdma->Init.SourceBlockAddressOffset);
  1564. hmdma->Instance->CBRUR = (blockoffset & 0x0000FFFFU);
  1565. }
  1566. else
  1567. {
  1568. /* Write new CBRUR Register value : source repeat block offset */
  1569. hmdma->Instance->CBRUR = (((uint32_t)hmdma->Init.SourceBlockAddressOffset) & 0x0000FFFFU);
  1570. }
  1571. /* If block destination address offset is negative set the Block Repeat destination address Update Mode to decrement */
  1572. if(hmdma->Init.DestBlockAddressOffset < 0)
  1573. {
  1574. hmdma->Instance->CBNDTR |= MDMA_CBNDTR_BRDUM;
  1575. /* Write new CBRUR Register value : destination repeat block offset */
  1576. blockoffset = (uint32_t)(- hmdma->Init.DestBlockAddressOffset);
  1577. hmdma->Instance->CBRUR |= ((blockoffset & 0x0000FFFFU) << MDMA_CBRUR_DUV_Pos);
  1578. }
  1579. else
  1580. {
  1581. /*write new CBRUR Register value : destination repeat block offset */
  1582. hmdma->Instance->CBRUR |= ((((uint32_t)hmdma->Init.DestBlockAddressOffset) & 0x0000FFFFU) << MDMA_CBRUR_DUV_Pos);
  1583. }
  1584. /* if HW request set the HW request and the requet CleraMask and ClearData MaskData, */
  1585. if(hmdma->Init.Request != MDMA_REQUEST_SW)
  1586. {
  1587. /* Set the HW request in CTRB register */
  1588. hmdma->Instance->CTBR = hmdma->Init.Request & MDMA_CTBR_TSEL;
  1589. }
  1590. else /* SW request : reset the CTBR register */
  1591. {
  1592. hmdma->Instance->CTBR = 0;
  1593. }
  1594. /* Write Link Address Register */
  1595. hmdma->Instance->CLAR = 0;
  1596. }
  1597. /**
  1598. * @}
  1599. */
  1600. #endif /* HAL_MDMA_MODULE_ENABLED */
  1601. /**
  1602. * @}
  1603. */
  1604. /**
  1605. * @}
  1606. */