cancel
Showing results for 
Search instead for 
Did you mean: 

Is it a bug? STM32F0xx HAL USART Driver

avinash_elec
Associate III

 

Hello All,

I am using USART peripheral to transfer a block of bytes. I am trying to do it using DMA. Because I want all bytes of USART frame should be dispatched without any gaps between them and my system has few interrupts that could bring gaps between the byte if I transfer using polling mode.

So I have configured the DMA channel for this. But my problem is that DMA Transfer complete Callback is not getting invoked by HAL Driver. I am using Normal DMA mode and NOT the circular mode.

void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart)

when I debugged the code I found that the HAL driver file stm32f0xx_hal_uart.c file line no 3533

static void UART_DMATransmitCplt(DMA_HandleTypeDef *hdma)

is only invoking the HAL_UART_TxCpltCallback() when the DMA type is Circular (see the screenshot below) so is it a bug? Because HAL documentation doesn't state that the callback will be called only for CIRCULAR mode and NOT normal more. Nor does this code is making any sense.

dma_circular.png

When I removed the "else" keyword, my callback is getting called. I have also referred to a previous version of the HAL driver, there also the logic is same and also in STM32F4's HAL driver.

1 ACCEPTED SOLUTION

Accepted Solutions
TDK
Guru

Fairly sure this works.

At end of normal DMA, TCIE gets set, which eventually triggers the IRQ, which calls UART_EndTransmit_IT which calls HAL_UART_TxCpltCallback. TC gets set when no characters are being sent (i.e. transmission is actually complete).

So why does it do this? Because when the DMA TC flag gets set, the last character is still being sent out. TC only indicates that the DMA is done, not that the transmission is done. If it called HAL_UART_TxCpltCallback then, and you disabled the UART, it would cause issues.

If you feel a post has answered your question, please click "Accept as Solution".

View solution in original post

5 REPLIES 5

Yes, that does look unhelpful. Not sure it's by design.

@STTwo-32 

Tips, Buy me a coffee, or three.. PayPal Venmo
Up vote any posts that you find helpful, it shows what's working..
TDK
Guru

Fairly sure this works.

At end of normal DMA, TCIE gets set, which eventually triggers the IRQ, which calls UART_EndTransmit_IT which calls HAL_UART_TxCpltCallback. TC gets set when no characters are being sent (i.e. transmission is actually complete).

So why does it do this? Because when the DMA TC flag gets set, the last character is still being sent out. TC only indicates that the DMA is done, not that the transmission is done. If it called HAL_UART_TxCpltCallback then, and you disabled the UART, it would cause issues.

If you feel a post has answered your question, please click "Accept as Solution".

Test on a STM32F429 new CubeMX project. It does work as intended as described above. However, you need to enable the global UART IRQ interrupt, which is not enabled by default.

If you feel a post has answered your question, please click "Accept as Solution".

Thank you for taking time to look into my issue! I now understood my fault. I have gone through the reference manual also and found this.

dma_complete.png

dma_complete_2.png

Yes now I have enabled the USART global interrupt, written the ISR and in ISR used the HAL handler for USART interrupt. Now everything is working as expected.