做嵌入式開發(fā)時,大家是不是都有過這種崩潰場景:
- 采集一個傳感器數(shù)據(jù),本來以為只是簡單的 ADC → 內(nèi)存,結(jié)果發(fā)現(xiàn) CPU 忙得要死;
- 想收點串口數(shù)據(jù),CPU 每次被中斷打斷,延遲累積到系統(tǒng)卡頓;
- 數(shù)據(jù)傳輸量一上來,系統(tǒng)直接掉幀甚至死機。
很多初學(xué)者遇到這種情況,第一反應(yīng)是“是不是代碼寫得不夠高效”。但事實是:你再怎么優(yōu)化循環(huán),CPU 親自搬數(shù)據(jù)就是效率低。
解決方案其實很明確——用 DMA(Direct Memory Access,直接存儲器訪問)。如果你會合理使用 DMA,很多“看似不可避免的性能瓶頸”都能迎刃而解。今天我們就來聊聊 DMA 的工作原理、常見應(yīng)用和實戰(zhàn)技巧。
一、DMA 究竟是什么?
DMA 的核心思路很簡單:把數(shù)據(jù)搬運工作交給硬件去做。
在沒有 DMA 的情況下,數(shù)據(jù)傳輸?shù)牧鞒檀蟾攀牵?/strong>
- 外設(shè)產(chǎn)生數(shù)據(jù);
- CPU 中斷響應(yīng);
- CPU 把數(shù)據(jù)讀出來放到內(nèi)存。
而有了 DMA:
- 外設(shè)和 DMA 控制器直接“勾搭”;
- 數(shù)據(jù)繞過 CPU,直接搬到內(nèi)存。
這樣一來,CPU 不用再做“快遞員”,可以專心處理邏輯,系統(tǒng)響應(yīng)速度和并發(fā)能力都能上一個臺階。
二、DMA 的常見應(yīng)用場景
1. 串口數(shù)據(jù)接收(UART DMA)
如果你做過大數(shù)據(jù)量的串口通信,就會發(fā)現(xiàn)中斷方式很容易丟數(shù)據(jù)。
- DMA 可以配置成循環(huán)緩沖區(qū),數(shù)據(jù)一來就自動寫入 RAM;
- CPU 只需要在合適的時候檢查緩沖區(qū),不用每個字節(jié)都處理中斷。
2. ADC 連續(xù)采樣
- 普通方式下,CPU 每次采樣要響應(yīng) ADC 中斷,很快就“累趴”;
- 用 DMA,可以把采樣結(jié)果自動存到數(shù)組里,形成數(shù)據(jù)流,CPU 后續(xù)再批量處理。
3. 內(nèi)存到外設(shè)數(shù)據(jù)傳輸
比如 SPI 發(fā)送、DAC 輸出波形:
- 傳統(tǒng)方式要一個字節(jié)一個字節(jié)寫寄存器;
- DMA 可以直接把內(nèi)存中的一段緩沖區(qū)刷到外設(shè)寄存器,效率極高。
4. 內(nèi)存到內(nèi)存?zhèn)鬏?/strong>
一些芯片的 DMA 支持內(nèi)存塊搬運,可以用來快速清零數(shù)組、搬運數(shù)據(jù)結(jié)構(gòu),CPU 不用一個循環(huán)一個循環(huán)寫。
三、DMA 配置的關(guān)鍵要點
很多同學(xué)第一次用 DMA,發(fā)現(xiàn)配置比想象中復(fù)雜。其實總結(jié)起來,主要是以下幾個步驟:
- 確定通道/流
- DMA 控制器通常有多個通道,對應(yīng)不同的外設(shè)。
- 要查手冊,看你的外設(shè)掛在哪個 DMA 通道上。
- 配置源地址和目的地址
- 源地址可以是外設(shè)寄存器,比如 ADC_DR。
- 目的地址一般是內(nèi)存數(shù)組。
- 有時候是反過來,比如內(nèi)存 → SPI。
- 配置傳輸方向
- 外設(shè)到內(nèi)存(ADC、UART RX);
- 內(nèi)存到外設(shè)(SPI TX、DAC);
- 內(nèi)存到內(nèi)存。
- 配置數(shù)據(jù)長度和傳輸模式
- 單次搬運幾個字節(jié)?總共搬多少?
- 是循環(huán)模式(buffer 自動回繞)還是普通模式?
- 啟動 DMA
- 記得在外設(shè)使能之前配置好 DMA;
- 啟動順序有講究,比如 UART DMA 要先開 DMA 再開 UART 接收。
四、實戰(zhàn)技巧:避免常見坑
技巧 1:循環(huán)緩沖 vs 普通模式
- 如果數(shù)據(jù)源是持續(xù)的(比如串口、ADC),用循環(huán)緩沖更穩(wěn)。
- 如果只是一次性發(fā)送一段數(shù)據(jù)(比如 SPI 發(fā)命令),普通模式即可。
技巧 2:善用中斷回調(diào)
DMA 雖然能自動搬數(shù)據(jù),但你還是得知道“什么時候搬完”。
- 可以開傳輸完成中斷,在回調(diào)函數(shù)里處理數(shù)據(jù)。
- 對于循環(huán)模式,可以用半傳輸中斷,做到“邊采集邊處理”。
技巧 3:緩存對齊問題
有些 MCU 的 DMA 對地址有對齊要求,比如 4 字節(jié)對齊,否則性能下降甚至報錯。寫代碼前要看手冊。
技巧 4:注意總線帶寬
DMA 不是“無限快”,它也占用內(nèi)存總線。
- 多個 DMA 通道同時工作時,可能會互相搶占。
- 解決方法:錯峰啟動,或者降低非關(guān)鍵任務(wù)的優(yōu)先級。
技巧 5:與 CPU 配合
DMA 搬數(shù)據(jù)的同時,CPU 可能要訪問同一片內(nèi)存。
- 要小心數(shù)據(jù)一致性問題,可以用“雙緩沖”或者加鎖機制。
五、案例分享
案例 1:ADC + DMA 實現(xiàn)波形采集
某項目需要 10kHz 的 ADC 采樣,用中斷方式 CPU 占用率高達 70%。
→ 換成 DMA 循環(huán)搬運到數(shù)組,CPU 占用率直接降到 5% 以下,系統(tǒng)反應(yīng)流暢。
案例 2:UART 接收不丟包
串口調(diào)試工具長時間發(fā)送數(shù)據(jù),CPU 用中斷處理時經(jīng)常丟字節(jié)。
→ 改用 DMA + 環(huán)形緩沖 + IDLE 中斷檢測幀間隔,接收穩(wěn)定,再也沒掉過包。
案例 3:SPI 高速傳輸
某 OLED 屏幕更新一幀圖像需要傳 8KB 數(shù)據(jù),用循環(huán)寫寄存器方式刷新率只有 20fps。
→ DMA 一次傳輸緩沖區(qū),刷新率提高到 60fps,畫面流暢無比。
六、總結(jié)
DMA 對嵌入式開發(fā)來說,是“必學(xué)技能”之一。很多人初學(xué)時嫌它復(fù)雜,繼續(xù)讓 CPU 自己“跑腿”,結(jié)果系統(tǒng)卡頓、性能不達標。其實只要掌握幾個要點:
- 了解外設(shè)和 DMA 通道的映射關(guān)系;
- 正確配置源地址、目的地址和傳輸模式;
- 結(jié)合中斷回調(diào)實現(xiàn)數(shù)據(jù)處理;
- 注意帶寬和緩存一致性問題。
你就能真正實現(xiàn)“零 CPU 占用的高速傳輸”,讓系統(tǒng)既高效又穩(wěn)定。
-
cpu
+關(guān)注
關(guān)注
68文章
11275瀏覽量
224911 -
數(shù)據(jù)傳輸
+關(guān)注
關(guān)注
9文章
2198瀏覽量
67574 -
dma
+關(guān)注
關(guān)注
3文章
581瀏覽量
105912
發(fā)布評論請先 登錄
CW32L052單片機支持DMA實現(xiàn)高速數(shù)據(jù)傳輸
ADXRS453Z的芯片支持使用STM32單片機的DMA數(shù)據(jù)傳輸功能嗎?
基于FPGA的高速LVDS數(shù)據(jù)傳輸
Serial RapidIO接口DMA數(shù)據(jù)傳輸
請問ADXRS453Z的芯片支持使用STM32單片機的DMA數(shù)據(jù)傳輸功能嗎?
DMA的數(shù)據(jù)傳輸速率是多少?
怎么實現(xiàn)基于FPGA的具有流量控制機制的高速串行數(shù)據(jù)傳輸系統(tǒng)設(shè)計?
stm32數(shù)據(jù)傳輸的相關(guān)資料分享
DMA進行數(shù)據(jù)傳輸和CPU進行數(shù)據(jù)傳輸的疑問
基于FPDP的高速數(shù)據(jù)傳輸系統(tǒng)設(shè)計
基于Zynq-7000的SRIO高速數(shù)據(jù)傳輸設(shè)計與實現(xiàn)
DMA數(shù)據(jù)傳輸(源代碼分享)
DMA數(shù)據(jù)傳輸在SPEAR300實現(xiàn)高速串口驅(qū)動設(shè)計
stm32數(shù)據(jù)傳輸
DMA 實戰(zhàn)指南:零 CPU 占用玩轉(zhuǎn)高速數(shù)據(jù)傳輸
評論