久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

STM32 DMA使用詳解

[復制鏈接]
跳轉到指定樓層
樓主
ID:80436 發表于 2015-5-21 00:20 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
STM32 DMA使用詳解
DMA部分我用到的相對簡單,當然,可能這是新東西,我暫時還用不到它的復雜功能吧。下面用問答的形式表達我的思路。
DMA有什么用?
       直接存儲器存取用來提供在外設和存儲器之間或者存儲器和存儲器之間的高速數據傳輸。無須CPU的干預,通過DMA數據可以快速地移動。這就節省了CPU的資源來做其他操作。
有多少個DMA資源?
       有兩個DMA控制器,DMA1有7個通道,DMA2有5個通道。
數據從什么地方送到什么地方?
       外設到SRAM(I2C/UART等獲取數據并送入SRAM);
       SRAM的兩個區域之間;
       外設到外設(ADC讀取數據后送到TIM1控制其產生不同的PWM占空比);
       SRAM到外設(SRAM中預先保存的數據送入DAC產生各種波形);
       ……還有一些目前還搞不清楚的。
DMA可以傳遞多少數據?
       傳統的DMA的概念是用于大批量數據的傳輸,但是我理解,在STM32中,它的概念被擴展了,也許更多的時候快速是其應用的重點。數據可以從1~65535個。
直接存儲器存取(Direct Memory Access,DMA)是計算機科學中的一種內存訪問技術。它允許某些電腦內部的硬體子系統(電腦外設),可以獨立地直接讀寫系統存儲器,而不需繞道 CPU。在同等程度的CPU負擔下,DMA是一種快速的數據傳送方式。它允許不同速度的硬件裝置來溝通,而不需要依于 CPU的大量中斷請求。【摘自Wikipedia】
現在越來越多的單片機采用DMA技術,提供外設和存儲器之間或者存儲器之間的高速數據傳輸。當 CPU 初始化這個傳輸動作,傳輸動作本身是由 DMA 控制器 來實行和完成。STM32就有一個DMA控制器,它有7個通道,每個通道專門用來管理一個或多個外設對存儲器訪問的請求,還有一個仲裁器來協調各個DMA請求的優先權。
DMA 控制器和Cortex-M3核共享系統數據總線執行直接存儲器數據傳輸。當CPU和DMA同時訪問相同的目標(RAM或外設)時,DMA請求可能會停止 CPU訪問系統總線達若干個周期,總線仲裁器執行循環調度,以保證CPU至少可以得到一半的系統總線(存儲器或外設)帶寬。
在發生一個事件后,外設發送一個請求信號到DMA控制器。DMA控制器根據通道的優先權處理請求。當DMA控制器開始訪問外設的時候,DMA控制器立即發送給外設一個應答信號。當從DMA控制器得到應答信號時,外設立即釋放它的請求。一旦外設釋放了這個請求,DMA控制器同時撤銷應答信號。如果發生更多的請求時,外設可以啟動下次處理。
總之,每個DMA傳送由3個操作組成:
1. 從外設數據寄存器或者從DMA_CMARx寄存器指定地址的存儲器單元執行加載操作。
2. 存數據到外設數據寄存器或者存數據到DMA_CMARx寄存器指定地址的存儲器單元。
3. 執行一次DMA_CNDTRx寄存器的遞減操作。該寄存器包含未完成的操作數目。
仲裁器根據通道請求的優先級來啟動外設/存儲器的訪問。優先級分為兩個等級:軟件(4個等級:最高、高、中等、低)、硬件(有較低編號的通道比擁有較高編號的通道有較高的優先權)。
可以在DMA傳輸過半、傳輸完成和傳輸錯誤時產生中斷。
STM32中DMA的不同中斷(傳輸完成、半傳輸、傳輸完成)通過“線或”方式連接至NVIC,需要在中斷例程中進行判斷。
進行DMA配置前,不要忘了在RCC設置中使能DMA時鐘。STM32的DMA控制器掛在AHB總線上。
DMA總共有7個通道,各個通道的DMA映射關系如下:
外設的事件連接至相應DMA通道,每個通道均可以通過軟件觸發實現存儲器內部的DMA數據傳輸(M2M模式)
Tips:庫2.0中函數RCC_AHBPeriphClockCmd的參數由“RCC_AHBPeriph_DMA”改成“RCC_AHBPeriph_DMA1”(如果是DMA1控制器的話)。
DMA的傳輸標志位(CHTIFx、CTCIFx、CGIFx)由硬件設置為“1”,但需要軟件清零,在中斷服務程序中清除。當CGIFx(全局中斷標志位)清零后,CHTIFx 和 CTCIFx均清零。

過程:怎樣啟用DMA?首先,眾所周知的是初始化,任何設備啟用前都要對其進行初始化,要對模塊初始化,還要先了解該模塊相應的結構及其函數,以便正確的設置;由于DMA較為復雜,我就只談談DMA的基本結構和和常用函數,這些都是ST公司提供在庫函數中的。
1、 下面代碼是一個標準DMA設置,當然實際應用中可根據實際情況進行裁減:
DMA_DeInit(DMA_Channel1);
上面這句是給DMA配置通道,根據ST提供的資料,STM3210Fx中DMA包含7個通道(CH1~CH7),也就是說可以為外設或memory提供7座“橋梁”(請允許我使用橋梁一詞,我覺得更容易理解,哈哈,別“拍磚”呀!);
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
上面語句中的DMA_InitStructure是一個DMA結構體,在庫中有聲明了,當然使用時就要先定義了;DMA_PeripheralBaseAddr是該結構體中一個數據成員,給DMA一個起始地址,好比是一個buffer起始地址,數據流程是:外設寄存器à DMA_PeripheralBaseAddàmemory中變量空間(或flash中數據空間等),ADC1_DR_Address是我定義的一個地址變量;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_ConvertedValue;
上面這句很顯然是DMA要連接在Memory中變量的地址,ADC_ConvertedValue是我自己在memory中定義的一個變量;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
上面的這句是設置DMA的傳輸方向,就如前面我所說的,DMA可以雙向傳輸,也可以單向傳輸,這里設置的是單向傳輸,如果需要雙向傳輸:把DMA_DIR_PeripheralSRC改成DMA_DIR_PeripheralDST即可。
DMA_InitStructure.DMA_BufferSize = 2;
上面的這句是設置DMA在傳輸時緩沖區的長度,前面有定義過了buffer的起始地址:ADC1_DR_Address ,為了安全性和可靠性,一般需要給buffer定義一個儲存片區,這個參數的單位有三種類型:Byte、HalfWord、word,我設置的2個half-word(見下面的設置);32位的MCU中1個half-word占16 bits。
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
上面的這句是設置DMA的外設遞增模式,如果DMA選用的通道(CHx)有多個外設連接,需要使用外設遞增模式:DMA_PeripheralInc_Enable;我的例子里DMA只與ADC1建立了聯系,所以選用DMA_PeripheralInc_Disable
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
上面的這句是設置DMA的內存遞增模式,DMA訪問多個內存參數時,需要使用DMA_MemoryInc_Enable,當DMA只訪問一個內存參數時,可設置成:DMA_MemoryInc_Disable。
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
上面的這句是設置DMA在訪問時每次操作的數據長度。有三種數據長度類型,前面已經講過了,這里不在敘述。
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
與上面雷同。在此不再說明。
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
上面的這句是設置DMA的傳輸模式:連續不斷的循環模式,若只想訪問一次后就不要訪問了(或按指令操作來反問,也就是想要它訪問的時候就訪問,不要它訪問的時候就停止),可以設置成通用模式:DMA_Mode_Normal
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
上面的這句是設置DMA的優先級別:可以分為4級:VeryHigh,High,Medium,Low.
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
上面的這句是設置DMA的2個memory中的變量互相訪問的
DMA_Init(DMA_Channel1,&DMA_InitStructure);
前面那些都是對DMA結構體成員的設置,在次再統一對DMA整個模塊做一次初始化,使得DMA各成員與上面的參數一致。
/*DMA Enable*/
DMA_Cmd(DMA_Channel1,ENABLE);
哈哈哈!這一句我想我就不羅嗦了,大家一看就明白。
至此,整個DMA總算設置好了,但是,DMA通道又是怎樣與外設聯系在一起的呢?哈哈,這也是我當初最想知道的一個事情,別急!容我想喝口茶~~~~~~哈哈哈!
要使DMA與外設建立有效連接,這不是DMA自身的事情,是各個外設的事情,每個外設都有 一個xxx_DMACmd(XXXx,Enable )函數,如果使DMA與ADC建立有效聯系,就使用ADC_DMACmd(ADC1,Enable); (這里我啟用了ADC中的ADC1模塊)。

一個簡單的例子 transfer  a word data buffer from FLASH memory to embedded SRAM memory.
在V3.1.2庫的位置
STM32F10x_StdPeriph_Lib_V3.1.2\Project\STM32F10x_StdPeriph_Examples\DMA\FLASH_RAM

/* DMA1 channel6 configuration */
DMA_DeInit(DMA1_Channel6);
  //peripheral base address
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)SRC_Const_Buffer;
  //memory base address   
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)DST_Buffer;
  //數據傳輸方向    Peripheral is source               
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
//緩沖區大小 Number of data to be transferred (0 up to 65535).數據傳輸數目     
DMA_InitStructure.DMA_BufferSize = BufferSize;
   // the Peripheral address register is incremented      
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;
  //the memory address register is incremented
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
//the Peripheral data width      
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
//the DMAy Channelx will be used in memory-to-memory transfer
//DMA通道的操作可以在沒有外設請求的情況下進行,這種操作就是存儲器到存儲器模式。
DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;   
DMA_Init(DMA1_Channel6, &DMA_InitStructure);

/* Enable DMA1 Channel6 Transfer Complete interrupt */
DMA_ITConfig(DMA1_Channel6, DMA_IT_TC, ENABLE);


/* Enable DMA1 Channel6 transfer */
DMA_Cmd(DMA1_Channel6, ENABLE);
=======================================================================

外設的DMA請求映像


要使DMA與外設建立有效連接,這不是DMA自身的事情,是各個外設的事情,每個外設都有 一個

xxx_DMACmd(XXXx,Enable )函數,如果使DMA與ADC建立有效聯系,就使用 ADC_DMACmd

(ADC1,Enable); (這里我啟用了ADC中的ADC1模塊)。

/* DMA1 channel1 configuration ----------------------------------------------*/
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&AD_Value;   
//u16  AD_Value[2];   不加&應該也可以  數組名 代表地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = 2;      //############## 改了
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //##############     改了
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);

/* Enable DMA1 channel 1 */
DMA_Cmd(DMA1_Channel1, ENABLE);

/* ADC1 configuration ------------------------------------------------------*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 2;      //##############     改了
ADC_Init(ADC1, &ADC_InitStructure);
//內部溫度傳感器  添加這一句
/* Enable the temperature sensor and vref internal channel */
ADC_TempSensorVrefintCmd(ENABLE);
//##############     改了

//################ Channel 10(電位器)
ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_13Cycles5);
//###### 內部溫度傳感器  Channel 16 ###################
ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 2, ADC_SampleTime_55Cycles5);

  /* Enable ADC1 DMA */使能ADC1的DMA請求映像
  ADC_DMACmd(ADC1, ENABLE);

/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);

/* Enable ADC1 reset calibaration register */   //使用之前一定要校準
ADC_ResetCalibration(ADC1);
/* Check the end of ADC1 reset calibration register */
while(ADC_GetResetCalibrationStatus(ADC1));

/* Start ADC1 calibaration */
ADC_StartCalibration(ADC1);
/* Check the end of ADC1 calibration */
while(ADC_GetCalibrationStatus(ADC1));

/* Start ADC1 Software Conversion */

ADC_SoftwareStartConvCmd(ADC1, ENABLE);



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

使用道具 舉報

沙發
ID:94120 發表于 2015-10-31 10:03 | 只看該作者
SD傳輸的時候一直卡在了DMA等待這里,  用野火的不行,原子的驅動我也看了,沒條理,都看不下去,安富萊的中規中矩的,不錯,換了安富萊的驅動就可以了
回復

使用道具 舉報

板凳
ID:95206 發表于 2015-11-9 20:53 | 只看該作者
謝謝分享,正在學習這方面的內容
回復

使用道具 舉報

地板
ID:221323 發表于 2017-8-16 21:29 | 只看該作者
給力分享 頂
回復

使用道具 舉報

5#
ID:238016 發表于 2017-10-10 08:54 | 只看該作者
謝謝分享,正在學習這方面的內容
回復

使用道具 舉報

6#
ID:311091 發表于 2018-4-18 18:12 | 只看該作者
好資料共分享!
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 成人欧美一区二区三区视频xxx | 老司机久久| 午夜三级视频 | 国产我和子的乱视频网站 | 国产一区二区精品在线观看 | 97国产精品视频人人做人人爱 | 欧美日韩国产精品 | 色婷婷综合在线观看 | 一区二区三区在线观看免费视频 | 免费人成激情视频在线观看冫 | 久久久国产一区二区 | 欧美久久久久久久久 | 国产精品高潮呻吟 | 亚洲日本欧美日韩高观看 | 日韩精品视频在线观看一区二区三区 | 国产丝袜一区二区三区免费视频 | 亚洲欧美自拍偷拍视频 | 国产一级一级毛片 | 人人干人人看 | 欧美日韩专区 | 一区二区三区成人 | 精品一区电影 | 91天堂网| 蜜桃臀av一区二区三区 | 国产精品视频久久 | 欧美多人在线 | 久草在线 | 国产成人99av超碰超爽 | 免费福利视频一区二区三区 | 91视频在线看 | 黄色毛片网站在线观看 | 日韩福利在线 | 国产精品久久久久久久久久久免费看 | www.久久国产精品 | 91免费在线看 | 一区二区精品视频 | 精品视频999 | 日韩美女在线看免费观看 | 久久精品国产亚洲 | 久久久www成人免费精品 | 国产精品成人一区二区三区 |