2025-09-09 5:34 AM
Hello,
I would like to transfer 3 linked list blocks over UART DMA. When I started HAL_MDMA_START_IT, it transmits only 16 bytes. How can I send big files back to back? Configurations are as follows:
MCU : STM32H7 series
Firmware : Last release
2025-09-09 7:33 AM - edited 2025-09-09 7:36 AM
Hello @farukisiker
Please refer to the article below to configure your linked list dma transfer.
How to configure the linked list mode in STM32Cube... - STMicroelectronics Community
also see the workshop below:
STM32H5 workshop - 05 More advanced GPDMA - Linked Lists (intermediate)
2025-09-09 7:48 AM
Hello @farukisiker
Please refer to this example as starting point.
2025-09-17 6:11 AM
Hello @farukisiker
Any update on this topic please ?
2025-09-17 7:37 AM
I haven't tried that solution yet. Should I close that topic till trying?
2025-09-22 5:34 AM
In NUCLEO-U575 example includes "DMA_QListTypeDef" structure at stm32u5xx_hal_dma_ex.h. However, it is not defined in H7 library. And, queue is created around this structure.
2025-09-25 3:24 AM
Hello @farukisiker
To configure your linked list transfer, please refer to the section How to use this driver in the stm32h7xx_hal_mdma.c
==============================================================================
##### How to use this driver #####
==============================================================================
[..]
(#) Enable and configure the peripheral to be connected to the MDMA Channel
(except for internal SRAM/FLASH memories: no initialization is
necessary) please refer to Reference manual for connection between peripherals
and MDMA requests.
(#)
For a given Channel use HAL_MDMA_Init function to program the required configuration through the following parameters:
transfer request , channel priority, data endianness, Source increment, destination increment ,
source data size, destination data size, data alignment, source Burst, destination Burst ,
buffer Transfer Length, Transfer Trigger Mode (buffer transfer, block transfer, repeated block transfer
or full transfer) source and destination block address offset, mask address and data.
If using the MDMA in linked list mode then use function HAL_MDMA_LinkedList_CreateNode to fill a transfer node.
Note that parameters given to the function HAL_MDMA_Init corresponds always to the node zero.
Use function HAL_MDMA_LinkedList_AddNode to connect the created node to the linked list at a given position.
User can make a linked list circular using function HAL_MDMA_LinkedList_EnableCircularMode , this function will automatically connect the
last node of the list to the first one in order to make the list circular.
In this case the linked list will loop on node 1 : first node connected after the initial transfer defined by the HAL_MDMA_Init
-@- The initial transfer itself (node 0 corresponding to the Init).
User can disable the circular mode using function HAL_MDMA_LinkedList_DisableCircularMode, this function will then remove
the connection between last node and first one.
Function HAL_MDMA_LinkedList_RemoveNode can be used to remove (disconnect) a node from the transfer linked list.
When a linked list is circular (last node connected to first one), if removing node1 (node where the linked list loops),
the linked list remains circular and node 2 becomes the first one.
Note that if the linked list is made circular the transfer will loop infinitely (or until aborted by the user).
[..]
(+) User can select the transfer trigger mode (parameter TransferTriggerMode) to define the amount of data to be
transfer upon a request :
(++) MDMA_BUFFER_TRANSFER : each request triggers a transfer of BufferTransferLength data
with BufferTransferLength defined within the HAL_MDMA_Init.
(++) MDMA_BLOCK_TRANSFER : each request triggers a transfer of a block
with block size defined within the function HAL_MDMA_Start/HAL_MDMA_Start_IT
or within the current linked list node parameters.
(++) MDMA_REPEAT_BLOCK_TRANSFER : each request triggers a transfer of a number of blocks
with block size and number of blocks defined within the function HAL_MDMA_Start/HAL_MDMA_Start_IT
or within the current linked list node parameters.
(++) MDMA_FULL_TRANSFER : each request triggers a full transfer
all blocks and all nodes(if a linked list has been created using HAL_MDMA_LinkedList_CreateNode \ HAL_MDMA_LinkedList_AddNode).
*** Polling mode IO operation ***
=================================
[..]
(+) Use HAL_MDMA_Start() to start MDMA transfer after the configuration of Source
address and destination address and the Length of data to be transferred.
(+) Use HAL_MDMA_PollForTransfer() to poll for the end of current transfer or a transfer level
In this case a fixed Timeout can be configured by User depending from his application.
(+) Use HAL_MDMA_Abort() function to abort the current transfer : blocking method this API returns
when the abort ends or timeout (should not be called from an interrupt service routine).
*** Interrupt mode IO operation ***
===================================
[..]
(+) Configure the MDMA interrupt priority using HAL_NVIC_SetPriority()
(+) Enable the MDMA IRQ handler using HAL_NVIC_EnableIRQ()
(+) Use HAL_MDMA_Start_IT() to start MDMA transfer after the configuration of
Source address and destination address and the Length of data to be transferred. In this
case the MDMA interrupt is configured.
(+) Use HAL_MDMA_IRQHandler() called under MDMA_IRQHandler() Interrupt subroutine
(+) At the end of data transfer HAL_MDMA_IRQHandler() function is executed and user can
add his own function by customization of function pointer XferCpltCallback and
XferErrorCallback (i.e a member of MDMA handle structure).
(+) Use HAL_MDMA_Abort_IT() function to abort the current transfer : non-blocking method. This API will finish the execution immediately
then the callback XferAbortCallback (if specified by the user) is asserted once the MDMA channel has effectively aborted.
(could be called from an interrupt service routine).
(+) Use functions HAL_MDMA_RegisterCallback and HAL_MDMA_UnRegisterCallback respectevely to register unregister user callbacks
from the following list :
(++) XferCpltCallback : transfer complete callback.
(++) XferBufferCpltCallback : buffer transfer complete callback.
(++) XferBlockCpltCallback : block transfer complete callback.
(++) XferRepeatBlockCpltCallback : repeated block transfer complete callback.
(++) XferErrorCallback : transfer error callback.
(++) XferAbortCallback : transfer abort complete callback.
[..]
(+) If the transfer Request corresponds to SW request (MDMA_REQUEST_SW) User can use function HAL_MDMA_GenerateSWRequest to
trigger requests manually. Function HAL_MDMA_GenerateSWRequest must be used with the following precautions:
(++) This function returns an error if used while the Transfer has ended or not started.
(++) If used while the current request has not been served yet (current request transfer on going)
this function returns an error and the new request is ignored.
Generally this function should be used in conjunctions with the MDMA callbacks:
(++) example 1:
(+++) Configure a transfer with request set to MDMA_REQUEST_SW and trigger mode set to MDMA_BUFFER_TRANSFER
(+++) Register a callback for buffer transfer complete (using callback ID set to HAL_MDMA_XFER_BUFFERCPLT_CB_ID)
(+++) After calling HAL_MDMA_Start_IT the MDMA will issue the transfer of a first BufferTransferLength data.
(+++) When the buffer transfer complete callback is asserted first buffer has been transferred and user can ask for a new buffer transfer
request using HAL_MDMA_GenerateSWRequest.
(++) example 2:
(+++) Configure a transfer with request set to MDMA_REQUEST_SW and trigger mode set to MDMA_BLOCK_TRANSFER
(+++) Register a callback for block transfer complete (using callback ID HAL_MDMA_XFER_BLOCKCPLT_CB_ID)
(+++) After calling HAL_MDMA_Start_IT the MDMA will issue the transfer of a first block of data.
(+++) When the block transfer complete callback is asserted the first block has been transferred and user can ask
for a new block transfer request using HAL_MDMA_GenerateSWRequest.
[..] Use HAL_MDMA_GetState() function to return the MDMA state and HAL_MDMA_GetError() in case of error detection.
*** MDMA HAL driver macros list ***
=============================================
[..]
Below the list of most used macros in MDMA HAL driver.
(+) __HAL_MDMA_ENABLE: Enable the specified MDMA Channel.
(+) __HAL_MDMA_DISABLE: Disable the specified MDMA Channel.
(+) __HAL_MDMA_GET_FLAG: Get the MDMA Channel pending flags.
(+) __HAL_MDMA_CLEAR_FLAG: Clear the MDMA Channel pending flags.
(+) __HAL_MDMA_ENABLE_IT: Enable the specified MDMA Channel interrupts.
(+) __HAL_MDMA_DISABLE_IT: Disable the specified MDMA Channel interrupts.
(+) __HAL_MDMA_GET_IT_SOURCE: Check whether the specified MDMA Channel interrupt has occurred or not.
[..]
(@) You can refer to the header file of the MDMA HAL driver for more useful macros.
[..]
2025-09-25 5:35 AM
Hello @farukisiker
You can refer to the example below also:
2025-09-28 11:37 PM - last edited on 2025-09-29 1:39 AM by Saket_Om
In this example, we do not forward all buffers to an interface like UART tx buffer. I already can transfer datas between memory by using MDMA. My problem is with peripheral interface. I want to forward 3 buffers to uart tx buffer back to back.
//-------------------------------------------------
mdmaLinkNodeConfig.SrcAddress = (uint32_t)SRC_Const_Buffer1;
mdmaLinkNodeConfig.DstAddress = (uint32_t)DESTBuffer1_SRAM1;
mdmaLinkNodeConfig.BlockDataLength = (BUFFER_SIZE1*4);
mdmaLinkNodeConfig.BlockCount = 1;
//-------------------------------------------------
mdmaLinkNodeConfig.SrcAddress = (uint32_t)SRC_Const_Buffer2;
mdmaLinkNodeConfig.DstAddress = (uint32_t)DESTBuffer2_AHBSRAM;
mdmaLinkNodeConfig.BlockDataLength = (BUFFER_SIZE2*4);
mdmaLinkNodeConfig.BlockCount = 1;
//-------------------------------------------------
if (HAL_MDMA_Start_IT(&MDMA_Handle, (uint32_t)&SRC_Const_Buffer0, (uint32_t)&DESTBuffer0_AXISRAM, (BUFFER_SIZE0 * 4), 1) != HAL_OK)
//-------------------------------------------------
2025-09-29 1:35 AM
Hello @farukisiker
Could you please share your code for UART MDMA transfer? The code you shared is for memory-to-memory transfer.
Please use </> button to paste your code. Read this post for more details: How to insert source code