stm32中DMA基本使用
直接存儲(chǔ)器存取用來(lái)提供在外設(shè)和存儲(chǔ)器之間或者存儲(chǔ)器和存儲(chǔ)器之間的高速數(shù)據(jù)傳輸。無(wú)須CPU的干預(yù),通過(guò)DMA數(shù)據(jù)可以快速地移動(dòng)。這就節(jié)省了CPU的資源來(lái)做其他操作。
本文引用地址:http://www.ex-cimer.com/article/201611/318277.htm有多少個(gè)DMA資源?
有兩個(gè)DMA控制器,DMA1有7個(gè)通道,DMA2有5個(gè)通道。
數(shù)據(jù)從什么地方送到什么地方?
外設(shè)到SRAM(I2C/UART等獲取數(shù)據(jù)并送入SRAM);
SRAM的兩個(gè)區(qū)域之間;
外設(shè)到外設(shè)(ADC讀取數(shù)據(jù)后送到TIM1控制其產(chǎn)生不同的PWM占空比);
SRAM到外設(shè)(SRAM中預(yù)先保存的數(shù)據(jù)送入DAC產(chǎn)生各種波形);
……還有一些目前還搞不清楚的。
DMA可以傳遞多少數(shù)據(jù)?
傳統(tǒng)的DMA的概念是用于大批量數(shù)據(jù)的傳輸,但是我理解,在STM32中,它的概念被擴(kuò)展了,也許更多的時(shí)候快速是其應(yīng)用的重點(diǎn)。數(shù)據(jù)可以從1~65535個(gè)。
直接存儲(chǔ)器存取(Direct Memory Access,DMA)是計(jì)算機(jī)科學(xué)中的一種內(nèi)存訪問(wèn)技術(shù)。它允許某些電腦內(nèi)部的硬體子系統(tǒng)(電腦外設(shè)),可以獨(dú)立地直接讀寫(xiě)系統(tǒng)存儲(chǔ)器,而不需繞道 CPU。在同等程度的CPU負(fù)擔(dān)下,DMA是一種快速的數(shù)據(jù)傳送方式。它允許不同速度的硬件裝置來(lái)溝通,而不需要依于 CPU的大量中斷請(qǐng)求。
現(xiàn)在越來(lái)越多的單片機(jī)采用DMA技術(shù),提供外設(shè)和存儲(chǔ)器之間或者存儲(chǔ)器之間的高速數(shù)據(jù)傳輸。當(dāng) CPU 初始化這個(gè)傳輸動(dòng)作,傳輸動(dòng)作本身是由DMA 控制器來(lái)實(shí)行和完成。STM32就有一個(gè)DMA控制器,它有7個(gè)通道,每個(gè)通道專門用來(lái)管理一個(gè)或多個(gè)外設(shè)對(duì)存儲(chǔ)器訪問(wèn)的請(qǐng)求,還有一個(gè)仲裁器來(lái)協(xié)調(diào)各個(gè)DMA請(qǐng)求的優(yōu)先權(quán)。
DMA 控制器和Cortex-M3核共享系統(tǒng)數(shù)據(jù)總線執(zhí)行直接存儲(chǔ)器數(shù)據(jù)傳輸。當(dāng)CPU和DMA同時(shí)訪問(wèn)相同的目標(biāo)(RAM或外設(shè))時(shí),DMA請(qǐng)求可能會(huì)停止 CPU訪問(wèn)系統(tǒng)總線達(dá)若干個(gè)周期,總線仲裁器執(zhí)行循環(huán)調(diào)度,以保證CPU至少可以得到一半的系統(tǒng)總線(存儲(chǔ)器或外設(shè))帶寬。
在發(fā)生一個(gè)事件后,外設(shè)發(fā)送一個(gè)請(qǐng)求信號(hào)到DMA控制器。DMA控制器根據(jù)通道的優(yōu)先權(quán)處理請(qǐng)求。當(dāng)DMA控制器開(kāi)始訪問(wèn)外設(shè)的時(shí)候,DMA控制器立即發(fā)送給外設(shè)一個(gè)應(yīng)答信號(hào)。當(dāng)從DMA控制器得到應(yīng)答信號(hào)時(shí),外設(shè)立即釋放它的請(qǐng)求。一旦外設(shè)釋放了這個(gè)請(qǐng)求,DMA控制器同時(shí)撤銷應(yīng)答信號(hào)。如果發(fā)生更多的請(qǐng)求時(shí),外設(shè)可以啟動(dòng)下次處理。
總之,每個(gè)DMA傳送由3個(gè)操作組成:
1. 從外設(shè)數(shù)據(jù)寄存器或者從DMA_CMARx寄存器指定地址的存儲(chǔ)器單元執(zhí)行加載操作。
2. 存數(shù)據(jù)到外設(shè)數(shù)據(jù)寄存器或者存數(shù)據(jù)到DMA_CMARx寄存器指定地址的存儲(chǔ)器單元。
3. 執(zhí)行一次DMA_CNDTRx寄存器的遞減操作。該寄存器包含未完成的操作數(shù)目。
仲裁器根據(jù)通道請(qǐng)求的優(yōu)先級(jí)來(lái)啟動(dòng)外設(shè)/存儲(chǔ)器的訪問(wèn)。優(yōu)先級(jí)分為兩個(gè)等級(jí):軟件(4個(gè)等級(jí):最高、高、中等、低)、硬件(有較低編號(hào)的通道比擁有較高編號(hào)的通道有較高的優(yōu)先權(quán))。
可以在DMA傳輸過(guò)半、傳輸完成和傳輸錯(cuò)誤時(shí)產(chǎn)生中斷。
STM32中DMA的不同中斷(傳輸完成、半傳輸、傳輸完成)通過(guò)“線或”方式連接至NVIC,需要在中斷例程中進(jìn)行判斷。
進(jìn)行DMA配置前,不要忘了在RCC設(shè)置中使能DMA時(shí)鐘。STM32的DMA控制器掛在AHB總線上。
DMA總共有7個(gè)通道,各個(gè)通道的DMA映射關(guān)系如下:
外設(shè)的事件連接至相應(yīng)DMA通道,每個(gè)通道均可以通過(guò)軟件觸發(fā)實(shí)現(xiàn)存儲(chǔ)器內(nèi)部的DMA數(shù)據(jù)傳輸(M2M模式)
Tips:庫(kù)2.0中函數(shù)RCC_AHBPeriphClockCmd的參數(shù)由“RCC_AHBPeriph_DMA”改成“RCC_AHBPeriph_DMA1”(如果是DMA1控制器的話)。
DMA的傳輸標(biāo)志位(CHTIFx、CTCIFx、CGIFx)由硬件設(shè)置為“1”,但需要軟件清零,在中斷服務(wù)程序中清除。當(dāng)CGIFx(全局中斷標(biāo)志位)清零后,CHTIFx 和 CTCIFx均清零。
過(guò)程:怎樣啟用DMA?首先,眾所周知的是初始化,任何設(shè)備啟用前都要對(duì)其進(jìn)行初始化,要對(duì)模塊初始化,還要先了解該模塊相應(yīng)的結(jié)構(gòu)及其函數(shù),以便正確的設(shè)置;由于DMA較為復(fù)雜,我就只談?wù)凞MA的基本結(jié)構(gòu)和和常用函數(shù),這些都是ST公司提供在庫(kù)函數(shù)中的。
1、 下面代碼是一個(gè)標(biāo)準(zhǔn)DMA設(shè)置,當(dāng)然實(shí)際應(yīng)用中可根據(jù)實(shí)際情況進(jìn)行裁減:
DMA_DeInit(DMA_Channel1);
上面這句是給DMA配置通道,根據(jù)ST提供的資料,STM3210Fx中DMA包含7個(gè)通道(CH1~CH7),也就是說(shuō)可以為外設(shè)或memory提供7座“橋梁”(請(qǐng)?jiān)试S我使用橋梁一詞,我覺(jué)得更容易理解,,別“拍磚”呀?。?/p>
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
上面語(yǔ)句中的DMA_InitStructure是一個(gè)DMA結(jié)構(gòu)體,在庫(kù)中有聲明了,當(dāng)然使用時(shí)就要先定義了;DMA_PeripheralBaseAddr是該結(jié)構(gòu)體中一個(gè)數(shù)據(jù)成員,給DMA一個(gè)起始地址,好比是一個(gè)buffer起始地址,數(shù)據(jù)流程是:外設(shè)寄存器à DMA_PeripheralBaseAddàmemory中變量空間(或flash中數(shù)據(jù)空間等),ADC1_DR_Address是我定義的一個(gè)地址變量;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_ConvertedValue;
上面這句很顯然是DMA要連接在Memory中變量的地址,ADC_ConvertedValue是我自己在memory中定義的一個(gè)變量;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
上面的這句是設(shè)置DMA的傳輸方向,就如前面我所說(shuō)的,DMA可以雙向傳輸,也可以單向傳輸,這里設(shè)置的是單向傳輸,如果需要雙向傳輸:把DMA_DIR_PeripheralSRC改成DMA_DIR_PeripheralDST即可。
DMA_InitStructure.DMA_BufferSize = 2;
上面的這句是設(shè)置DMA在傳輸時(shí)緩沖區(qū)的長(zhǎng)度,前面有定義過(guò)了buffer的起始地址:ADC1_DR_Address ,為了安全性和可靠性,一般需要給buffer定義一個(gè)儲(chǔ)存片區(qū),這個(gè)參數(shù)的單位有三種類型:Byte、HalfWord、word,我設(shè)置的2個(gè)half-word(見(jiàn)下面的設(shè)置);32位的MCU中1個(gè)half-word占16 bits。
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
上面的這句是設(shè)置DMA的外設(shè)遞增模式,如果DMA選用的通道(CHx)有多個(gè)外設(shè)連接,需要使用外設(shè)遞增模式:DMA_PeripheralInc_Enable;我的例子里DMA只與ADC1建立了聯(lián)系,所以選用DMA_PeripheralInc_Disable
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
上面的這句是設(shè)置DMA的內(nèi)存遞增模式,DMA訪問(wèn)多個(gè)內(nèi)存參數(shù)時(shí),需要使用DMA_MemoryInc_Enable,當(dāng)DMA只訪問(wèn)一個(gè)內(nèi)存參數(shù)時(shí),可設(shè)置成:DMA_MemoryInc_Disable。
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
上面的這句是設(shè)置DMA在訪問(wèn)時(shí)每次操作的數(shù)據(jù)長(zhǎng)度。有三種數(shù)據(jù)長(zhǎng)度類型,前面已經(jīng)講過(guò)了,這里不在敘述。
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
與上面雷同。在此不再說(shuō)明。
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
上面的這句是設(shè)置DMA的傳輸模式:連續(xù)不斷的循環(huán)模式,若只想訪問(wèn)一次后就不要訪問(wèn)了(或按指令操作來(lái)反問(wèn),也就是想要它訪問(wèn)的時(shí)候就訪問(wèn),不要它訪問(wèn)的時(shí)候就停止),可以設(shè)置成通用模式:DMA_Mode_Normal
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
上面的這句是設(shè)置DMA的優(yōu)先級(jí)別:可以分為4級(jí):VeryHigh,High,Medium,Low.
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
上面的這句是設(shè)置DMA的2個(gè)memory中的變量互相訪問(wèn)的
DMA_Init(DMA_Channel1,&DMA_InitStructure);
前面那些都是對(duì)DMA結(jié)構(gòu)體成員的設(shè)置,在次再統(tǒng)一對(duì)DMA整個(gè)模塊做一次初始化,使得DMA各成員與上面的參數(shù)一致。
/*DMA Enable*/
DMA_Cmd(DMA_Channel1,ENABLE);
{
DMA_InitTypeDef DMA_InitStructure;
//DMA設(shè)置:
//設(shè)置DMA源:內(nèi)存地址&串口數(shù)據(jù)寄存器地址
//方向:內(nèi)存-->外設(shè)
//每次傳輸位:8bit
//傳輸大小DMA_BufferSize=SENDBUFF_SIZE
//地址自增模式:外設(shè)地址不增,內(nèi)存地址自增1
//DMA模式:一次傳輸,非循環(huán)
//優(yōu)先級(jí):中
DMA_DeInit(DMA1_Channel4);//串口1的DMA傳輸通道是通道4
DMA_InitStructure.DMA_PeripheralBaseAddr = USART1_DR_Base;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)SendBuff;
DMA_InitStructure.DMA_DIR =DMA_DIR_PeripheralDST;//外設(shè)作為DMA的目的端
DMA_InitStructure.DMA_BufferSize = SENDBUFF_SIZE;//傳輸大小
DMA_InitStructure.DMA_PeripheralInc =DMA_PeripheralInc_Disable;//外設(shè)地址不增加
DMA_InitStructure.DMA_MemoryInc =DMA_MemoryInc_Enable;//內(nèi)存地址自增1
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStructure.DMA_Mode =DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;//(DMA傳送優(yōu)先級(jí)為中等)
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel4, &DMA_InitStructure);
}
vu8 SendBuff[SENDBUFF_SIZE];
SendBuff[i] = i%10+0;
}
4、開(kāi)始DMA傳輸(使能對(duì)應(yīng)的DMA通道)
DMA_Cmd(DMA1_Channel4, ENABLE);
{
LED_1_REV; //LED翻轉(zhuǎn)
Delay(); //浪費(fèi)時(shí)間
}
相關(guān)推薦
技術(shù)專區(qū)
- FPGA
- DSP
- MCU
- 示波器
- 步進(jìn)電機(jī)
- Zigbee
- LabVIEW
- Arduino
- RFID
- NFC
- STM32
- Protel
- GPS
- MSP430
- Multisim
- 濾波器
- CAN總線
- 開(kāi)關(guān)電源
- 單片機(jī)
- PCB
- USB
- ARM
- CPLD
- 連接器
- MEMS
- CMOS
- MIPS
- EMC
- EDA
- ROM
- 陀螺儀
- VHDL
- 比較器
- Verilog
- 穩(wěn)壓電源
- RAM
- AVR
- 傳感器
- 可控硅
- IGBT
- 嵌入式開(kāi)發(fā)
- 逆變器
- Quartus
- RS-232
- Cyclone
- 電位器
- 電機(jī)控制
- 藍(lán)牙
- PLC
- PWM
- 汽車電子
- 轉(zhuǎn)換器
- 電源管理
- 信號(hào)放大器
評(píng)論