新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > STM32 八路AD轉換用DMA傳輸調試成功,DMA傳輸不錯位

        STM32 八路AD轉換用DMA傳輸調試成功,DMA傳輸不錯位

        作者: 時間:2016-11-23 來源:網絡 收藏
        //


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

        本文引用地址:http://www.104case.com/article/201611/320088.htm

        #include

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


        #define ADC1_DR_Address ((u32)0x4001244C)

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

        //換結果直接傳輸到這個地址。當需要多路AD轉換時,定義此為一個數組后面

        //DMA設置時取數組的基地址,依次傳輸

        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端口時鐘,并且打開復用時鐘
        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

        }


        ////////////////////////////幾乎沒用,為了系統兼容/////////////////////////////////////
        #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]; //存放求完平均值之后再將結果擴大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);//等待傳輸完成否則第一位數據容易失
        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);

        }

        // 配置串口參數

        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; // 模數轉換工作在掃描模式(多通道)模式 ;
        ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; // 模數轉換工作在連續模式 ;
        ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;// 轉換由軟件而不是外部觸發

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

        //下面設置指定的ADC規則通道,一共需要設置8個

        // 設置指定ADC的規則組通道8-15,設置它們的轉化順序和采樣時間為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重置校準寄存器的狀態 ;

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

        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中定義的一個變量數組;

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

        //DMA_DIR_PeripheralDST即可。

        //轉換通道為M,每組轉換N次
        DMA_InitStructure.DMA_BufferSize = M*N;//上面的這句是設置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;

        //定義外設數據寬度為16位

        //上面的這句是設置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(DMA1_Channel1,&DMA_InitStructure);//前面那些都是對DMA結構體成員的設置,在次再統一對DMA整

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

        DMA_Cmd(DMA1_Channel1,ENABLE);

        }

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

        //求AD轉換結果的平均值函數
        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



        評論


        技術專區

        關閉
        主站蜘蛛池模板: 应城市| 苍梧县| 睢宁县| 嵊州市| 平遥县| 塔河县| 田阳县| 元朗区| 房产| 吉安县| 托克逊县| 上高县| 滕州市| 西峡县| 东乡县| 金川县| 贺州市| 武义县| 新巴尔虎右旗| 镇安县| 特克斯县| 红河县| 黄梅县| 丰镇市| 车致| 广安市| 上犹县| 石柱| 平潭县| 扶风县| 论坛| 商河县| 翁源县| 临汾市| 临邑县| 新龙县| 辰溪县| 姚安县| 息烽县| 乡城县| 张家界市|