<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > STM32 八路AD轉(zhuǎn)換用DMA傳輸調(diào)試成功,DMA傳輸不錯位

          STM32 八路AD轉(zhuǎn)換用DMA傳輸調(diào)試成功,DMA傳輸不錯位

          作者: 時間:2016-11-23 來源:網(wǎng)絡(luò) 收藏
          //


          //
          #include"stm32f10x_conf.h" //把DMA.h和ADC.h 的注釋去掉
          #include "stm32f10x.h"
          #include "stm32_eval.h"

          本文引用地址:http://www.ex-cimer.com/article/201611/320088.htm

          #include

          #define N 50 //每通道采50次
          #define M 8 //為8個通道


          #define ADC1_DR_Address ((u32)0x4001244C)

          vu16 After_filter[M]; //用來存放求平均值之后的結(jié)果
          vu16 AD_Value[N][M]; //用來存放ADC轉(zhuǎn)換結(jié)果,也是DMA的目標地址
          //這個變量存放ADC轉(zhuǎn)換后的值,在后面DMA設(shè)置的時候取了這個變了的地址,將轉(zhuǎn)

          //換結(jié)果直接傳輸?shù)竭@個地址。當需要多路AD轉(zhuǎn)換時,定義此為一個數(shù)組后面

          //DMA設(shè)置時取數(shù)組的基地址,依次傳輸

          void IO_cfg(void);
          void EXTI_cfg(void);
          void NVIC_cfg(void);
          void USART_cfg(void);
          void ADC_cfg(void);
          void DMA_cfg(void);

          u16 GetVolt(u16 advalue);
          void filter(void);

          void SerialPutChar(uint8_t c);
          void SerialReceivechar(uint8_t c);

          void Delay(vu32 nCount)
          {
          for(; nCount != 0; nCount--);
          }


          void RCC_cfg()
          {
          //打開PA端口時鐘,并且打開復(fù)用時鐘
          RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC | RCC_APB2Periph_AFIO, ENABLE);

          RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA時鐘
          RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //使能ADC

          }


          ////////////////////////////幾乎沒用,為了系統(tǒng)兼容/////////////////////////////////////
          #ifdef __GNUC__

          #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
          #else
          #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
          #endif
          //////////////////////////////////////////////////////////////////////////////////////////////////

          int main()
          {
          int i; //用于串口輸出

          u16 value[M]; //存放求完平均值之后再將結(jié)果擴大100倍

          RCC_cfg();
          EXTI_cfg();
          IO_cfg();
          NVIC_cfg();
          USART_cfg();

          DMA_cfg();
          ADC_cfg();

          while(1)
          {
          while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);//等待傳輸完成否則第一位數(shù)據(jù)容易失
          filter();
          for(i=0;i
          {
          value[i]= GetVolt(After_filter[i]);

          printf("value[%d]:t%d.%dvn",i,value[i]/100,value[i]0) ;
          Delay(0x80000);
          }

          }

          }

          //
          void IO_cfg()
          {
          //八路AD輸入

          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOB, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOB, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOC, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOC, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOC, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOC, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOC, &GPIO_InitStructure);


          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
          GPIO_Init(GPIOC, &GPIO_InitStructure);

          }

          // 配置串口參數(shù)

          void USART_cfg()
          {
          USART_InitTypeDef USART_InitStructure;

          USART_InitStructure.USART_BaudRate = 115200;
          USART_InitStructure.USART_WordLength = USART_WordLength_8b;
          USART_InitStructure.USART_StopBits = USART_StopBits_1;
          USART_InitStructure.USART_Parity = USART_Parity_No;
          USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
          USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

          STM_EVAL_COMInit(COM1, &USART_InitStructure);

          }


          void SerialPutChar(uint8_t c)
          {
          USART_SendData(EVAL_COM1, c);
          while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TXE) == RESET)
          {
          }
          }

          void SerialReceivechar(uint8_t c)
          {

          USART_ReceiveData(EVAL_COM1);

          while( USART_GetFlagStatus(EVAL_COM1, USART_FLAG_RXNE) != RESET)
          {
          }
          }

          void ADC_cfg(void)
          {
          ADC_InitTypeDef ADC_InitStructure;

          ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; // ADC1和ADC2工作在獨立模式 ;
          ADC_InitStructure.ADC_ScanConvMode = ENABLE; // 模數(shù)轉(zhuǎn)換工作在掃描模式(多通道)模式 ;
          ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; // 模數(shù)轉(zhuǎn)換工作在連續(xù)模式 ;
          ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;// 轉(zhuǎn)換由軟件而不是外部觸發(fā)

          //啟動 ;
          ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // ADC數(shù)據(jù)右對齊 ;
          //八個通道轉(zhuǎn)換時下面賦值為8(相當于開啟通道的數(shù)目)
          ADC_InitStructure.ADC_NbrOfChannel = M; // 順序進行規(guī)則轉(zhuǎn)換的ADC通道的數(shù)目 8 ;
          ADC_Init(ADC1, &ADC_InitStructure);

          //下面設(shè)置指定的ADC規(guī)則通道,一共需要設(shè)置8個

          // 設(shè)置指定ADC的規(guī)則組通道8-15,設(shè)置它們的轉(zhuǎn)化順序和采樣時間為55.5周期;
          ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 2, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 3, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 4, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 5, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 6, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 7, ADC_SampleTime_55Cycles5);
          ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 8, ADC_SampleTime_55Cycles5);

          ADC_DMACmd(ADC1, ENABLE); // 使能指定的ADC的DMA請求 ;
          ADC_Cmd(ADC1, ENABLE); // 使能指定的ADC ;

          ADC_ResetCalibration(ADC1);// 重置指定的ADC的校準寄存器 ;
          while(ADC_GetResetCalibrationStatus(ADC1)); // 獲取ADC重置校準寄存器的狀態(tài) ;

          ADC_StartCalibration(ADC1); // 開始指定ADC的校準程序 ;
          while(ADC_GetCalibrationStatus(ADC1)); // 獲取指定ADC的校準狀態(tài) ;
          ADC_SoftwareStartConvCmd(ADC1, ENABLE);// 使能指定的ADC的軟件轉(zhuǎn)換啟動功能 ;
          }

          void DMA_cfg(void)
          {
          DMA_InitTypeDef DMA_InitStructure;

          DMA_DeInit(DMA1_Channel1);
          DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;


          DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&AD_Value; //上面這句很顯然是DMA要連接在Memory

          //中變量的地址,
          //AD_Value是我自己在memory中定義的一個變量數(shù)組;

          DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //這里設(shè)置的是單向傳輸,如果需要雙向傳輸:
          //把DMA_DIR_PeripheralSRC改成

          //DMA_DIR_PeripheralDST即可。

          //轉(zhuǎn)換通道為M,每組轉(zhuǎn)換N次
          DMA_InitStructure.DMA_BufferSize = M*N;//上面的這句是設(shè)置DMA在傳輸時緩沖區(qū)的長度,前面有定義過了

          //buffer的起始地址:ADC1_DR_Address ,為了安全性和可靠性,一

          //般需要給buffer定義
          //一個儲存片區(qū),這個參數(shù)的單位有三種類型:Byte、HalfWord、word,
          //我設(shè)置的2個half-word(見下面的設(shè)置);32位的MCU中1個half-word占16 bits。

          DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

          //上面的這句是設(shè)置DMA的外設(shè)遞增模式,如果DMA
          //選用的通道(CHx)有多個外設(shè)連接,需要使用外設(shè)遞增模式:
          //DMA_PeripheralInc_Enable;我的例子里DMA只與ADC1建立了聯(lián)系,
          //所以選用DMA_PeripheralInc_Disable
          DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//上面的這句是設(shè)置DMA的內(nèi)存遞增模式,DMA訪

          //問多個內(nèi)存參數(shù)時,
          //需要使用DMA_MemoryInc_Enable,當DMA只訪問一個內(nèi)存參數(shù)時,
          //可設(shè)置成:DMA_MemoryInc_Disable。

          DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;

          //定義外設(shè)數(shù)據(jù)寬度為16位

          //上面的這句是設(shè)置DMA在訪問時每次操作的數(shù)據(jù)長度。

          //有三種數(shù)據(jù)長度類型,前面已經(jīng)講過了,這里不在敘述。
          DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//與上面雷同。在此不再說明。

          DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;

          //上面的這句是設(shè)置DMA的傳輸模式:連續(xù)不斷的循環(huán)式,
          //若只想訪問一次后就不要訪問了(或按指令操作來反問,也就是想要它訪問
          //的時候就訪問,不要它訪問的時候就停止),可以設(shè)置成通用模式:DMA_Mode_Normal

          DMA_InitStructure.DMA_Priority = DMA_Priority_High;//上面的這句是設(shè)置DMA的優(yōu)先級別:可以分為4級:

          //VeryHigh,High,Medium,Low.

          DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//上面的這句是設(shè)置DMA的2個memory中的變量互相訪問的

          DMA_Init(DMA1_Channel1,&DMA_InitStructure);//前面那些都是對DMA結(jié)構(gòu)體成員的設(shè)置,在次再統(tǒng)一對DMA整

          //個模塊做一次初始化,
          //使得DMA各成員與上面的參數(shù)一致。

          DMA_Cmd(DMA1_Channel1,ENABLE);

          }

          ///////////////////////////////////
          u16 GetVolt(u16 advalue)
          {
          return (u16)(advalue * 330 / 4096); //求的結(jié)果擴大了100倍,方便下面求出小數(shù)
          }

          //求AD轉(zhuǎn)換結(jié)果的平均值函數(shù)
          void filter(void)
          {
          int i;
          int sum = 0;
          u8 count;
          for(i=0;i<12;i++)
          {
          for ( count=0;count
          {
          sum += AD_Value[count][i];
          }
          After_filter[i]=sum/N;
          sum=0;
          }
          }


          PUTCHAR_PROTOTYPE
          {


          USART_SendData(EVAL_COM1, (uint8_t) ch);


          while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TC) == RESET)
          {
          }

          return ch;
          }

          #ifdef USE_FULL_ASSERT


          void assert_failed(uint8_t* file, uint32_t line)
          {


          while (1)
          {
          }
          }

          #endif



          評論


          技術(shù)專區(qū)

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();