找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 12892|回復: 1
打印 上一主題 下一主題
收起左側

關于STM32 DMA重新使能的話題

[復制鏈接]
跳轉到指定樓層
樓主
ID:98618 發(fā)表于 2016-11-1 00:05 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
有人發(fā)現(xiàn)當DMA配置為normal模式,一輪傳輸完成后再使能同一DMA數(shù)據流時,即使有DMA請求產生,DMA根本不進行數(shù)據傳輸。這里不妨以STM32F4為例聊聊該話題。


其實,在非循環(huán)模式下配置的DMA數(shù)據流傳輸結束后(即要傳輸?shù)臄?shù)據數(shù)目達到零),除非軟件重新對數(shù)據流編程并使能該數(shù)據流(通過將 DMA_SxCR 寄存器中的 EN 位置 1),否則DMA 會停止傳輸(硬件會將 DMA_SxCR 寄存器中的EN 位清零)并不再響應任何 DMA 請求。也就是說,當一輪DMA傳輸完成后,簡單來一句 __HAL_DMA_ENABLE(hdma)并不能再次重啟DMA傳輸,必須先行對數(shù)據流重新進行初始化配置,然后使能該DMA數(shù)據流。


不過,要對STM32F4系列內部各DMA數(shù)據流進行配置,首先要保證 DMA_SxCR 寄存器中的 EN 位已被清零。一般來講,在進行DMA數(shù)據流配置前,先對該EN位進行寫0去禁用該DMA流,然后讀取此位以確認沒有正在進行的數(shù)據流傳輸操作。將此位寫 0 可能不會立即生效,因為只有當前所有傳輸都已完成時才會將其寫為 0。當讀取到 EN 位為 0 時,才可以配置DMA數(shù)據流。配置完成后,如果是再次配置的話,記得將先前的DMA 傳輸中在狀態(tài)寄存器(DMA_LISR 和 DMA_HISR)中置 1 的所有數(shù)據流專用位置0, 然后重新使能數(shù)據流,即將 DMA_SxCR 寄存器中的 EN位置 1 激活數(shù)據流。


OK,這里拿個實際案例看看以加深下印象。


有人用STM32F4芯片做產品,他是基于STM32CUBE庫做應用開發(fā)。用DMA做USART通信的數(shù)據傳輸,DMA配置為循環(huán)模式。

他發(fā)現(xiàn)HAL_UART_Transmit_DMA()這個函數(shù)在做了一次DMA配置后,第二次使用它更新配置時無法生效。比如按下面步驟操作:


1HAL_UART_Transmit_DMA(&huart1,DataBuff, 10);

2HAL_UART_DMAPause(&huart1);                           

3 HAL_UART_Transmit_DMA(&huart1,&DataBuff[5],5); 

4HAL_UART_DMAResume(&huart1);                           

             

開發(fā)者的目的是希望先將DataBuff[0]至DataBuff[9]10個數(shù)據傳到串口上去。中途調整DMA配置[即第【3】句],然后發(fā)送DataBuff[5]至DataBuff[9]的5個數(shù)據到串口助手上去。


結果發(fā)現(xiàn)經過2次配置后的結果沒變,發(fā)送的都是DataBuff[0]至DataBuff[9]的10個數(shù)據。為什么呢?不妨一起看看。

上面第【1】句做usart TX的DMA傳輸配置。循環(huán)從DataBuf處開始取10個數(shù)據送往串口;

接著第【2】句是在DMA完成中斷里進行。暫停USART TX傳輸?shù)腄MA請求,并延時2S;

然后第【3】句再做USART TX的DMA配置,循環(huán)從&DataBuf【5】處開始取5個數(shù)據送往串口;

最后第【4】句重新開啟USART TX的DMA傳輸;


按照上面幾步也貌似條理清晰,有根有據?山Y果為什么發(fā)現(xiàn)第【3】句的二次配置沒法生效呢?


從前面的描述我們得知,如果需要對某STREAM進行DMA配置的話,首先須DISABLE該STREAM,即先對DMA_SxCR寄存器的EN位清零。而且這個清零并不一定立即生效,必須保證該STREAM當前沒有在進行DMA傳輸。所以,做清零操作后,還要去讀該EN位,直至讀到該位為0后方能對該DMA STREAM 進行再次配置。

HAL_UART_Transmit_DMA()函數(shù)里有對DMA_SxCR的EN位清零的動作,不過沒有讀取確認的動作。復位后DMA_SxCR寄存器的EN位默認是0毫無疑問,所以復位后第【1】句用HAL_UART_Transmit_DMA()函數(shù)配置后生效自然沒問題。


上面的第2】句在DMA完成中斷里進行,暫停UART TX的DMA請求,還延時了2S。本意是想讓DMA停下來,為后面二次配置做準備。但他這里只是暫停了DMA請求,加上這里配置DMA為循環(huán)模式,也就是說在進入DMA完成中斷時,DMA又做了下一輪傳輸?shù)臏蕚,其中包括傳輸?shù)據項數(shù)目寄存器DMA_SxNDTR的重裝。此時盡管暫停了DMA請求,但該DMA流是處于傳輸未完成狀態(tài)。


那么緊跟著的第【3】句對USART TX的DMA二次配置,HAL_UART_Transmit_DMA()函數(shù)里雖然有對DMA_SxCR的EN位清零,但由于此時該STREAM處于傳輸未完成狀態(tài),所以無法實現(xiàn)最終清零。


既然無法對EN位清零,相關配置就無法寫入生效。加上該函數(shù)沒有對EN位是否為0的未做進一步確認判斷,自然而然地下行到第【4】句重新開啟USART TX的DMA傳輸。最后的結果當然還是按照老配置運行了!有心的話,可以去STM32F4參考手冊看看數(shù)據流 x 配置寄存器 DMA_SxCR,里面多個參數(shù)只有在EN位為0時方可進行寫操作

 

好了,就此打住。MCU開發(fā)中不少問題往往可能就卡在技術手冊看得不到位。在開發(fā)過程中遇到難解問題時,有時細心看看手冊的相關部分,或許會有踏破鐵鞋無覓處,得來全不費工夫的感覺。

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏1 分享淘帖 頂1 踩
回復

使用道具 舉報

沙發(fā)
ID:303001 發(fā)表于 2018-8-23 13:17 | 只看該作者
朋友,多謝你這篇帖子,我今天找了一上午DMA使能無效的問題,看過你這篇文章后,醍醐灌頂。栴}終于解決,多謝!這次問題提醒我,以后得多看看手冊,即便是用庫編程。
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表