STM32F072-ADC_DMA ,多通道电压DMA采集

adc.c

#include "stm32f0xx.h"

#include "adc.h"

uint16_t ADC_Buf[16];

void ADC_DMA_Init(void)
{
    ADC_InitTypeDef ADC_InitStructure;
    DMA_InitTypeDef DMA_InitStructure;
    
    /* 使能时钟 */
    RCC_AHBPeriphClockCmd( RCC_AHBPeriph_DMA1, ENABLE );
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
    /* 在此配置ADC通道所用到的GPIO,I/O口进行初始化配置,请参考数据手册
例如:
    GPIO_InitTypeDef GPIO_InitStructure;
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA,ENABLE);
    
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
*/ 
    /* 复位DMA1通道1,可做可不做 */
    DMA_DeInit(DMA1_Channel1);
    /* 外设地址,即ADC1的数据寄存器,数据长度为16位 */
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)0x40012440;//(&(ADC->DR));//ADC1_BASE+0X04;
    /* 内存地址,自定义数组并取首地址,注意要用16位,与ADC数据对称 */
    DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_Buf;
    /* 外设作为传输源,即ADC->DMA */
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
    /* 一轮传输的数据数量,最好和内存数组长度对应 */
    DMA_InitStructure.DMA_BufferSize = 16;
    /* 外设地址在传输过程中不自增,因为ADC规则通道的数据寄存器只有一个 */
    DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
    /* 内存地址传输过程自增 */
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
    /* 传输的数据都是16位的,即半字(字,不是字节) */
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
    /* Normal模式单次传输,Circular模式可以自动重装循环传输 */
    DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
    /* DMA任务优先级 */
    DMA_InitStructure.DMA_Priority = DMA_Priority_High;
    /* 非内存->内存模式 */
    DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
    
    DMA_Init(DMA1_Channel1, &DMA_InitStructure);
    
    DMA_Cmd(DMA1_Channel1, ENABLE);//使能DMA通道1
    
    DMA_ITConfig(DMA1_Channel1, DMA_IT_TC, ENABLE);//使能传输完毕中断,这样才能接收到该中断
    
    
    ADC_DeInit(ADC1);//ADC恢复默认设置
    ADC_StructInit(&ADC_InitStructure);//初始化ADC结构
    
    ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//12位精度
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //规定模式装换工作在连续模式
    ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//数据对其为右对齐
    ADC_InitStructure.ADC_ScanDirection = ADC_ScanDirection_Upward; //ADC的扫描方向
    
    ADC_Init(ADC1,&ADC_InitStructure);
    
    /* ADC通道配置,包括通道、采样顺序和采样周期 */
    ADC_ChannelConfig(ADC1, ADC_Channel_0 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_1 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_4 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_5 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_6 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_7 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_8 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_9 , ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_10, ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_11, ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_12, ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_13, ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_14, ADC_SampleTime_239_5Cycles);
    ADC_ChannelConfig(ADC1, ADC_Channel_15, ADC_SampleTime_239_5Cycles);
    
    ADC_ChannelConfig(ADC1, ADC_Channel_Vrefint, ADC_SampleTime_239_5Cycles);//17
    
    ADC_GetCalibrationFactor(ADC1);//ADC校准
    ADC_VrefintCmd(ENABLE);//打开内部参照电压
    ADC_DMARequestModeConfig(ADC1, ADC_DMAMode_Circular);//循环模式下的 ADC DMA 请求
    ADC_DMACmd(ADC1, ENABLE);//使能 ADC_DMA
    
    ADC_OverrunModeCmd(ADC1, ENABLE);
    ADC_Cmd(ADC1, ENABLE);//使能 ADC
    while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_ADEN));//等待 ADCEN 标志
    ADC_StartOfConversion(ADC1);//ADC 常规软件启动转换
}

/* 取ADC平均值 */
uint32_t Read_ADC_DMA_number(uint16_t buf,uint8_t number_count)
{
    uint16_t  MinAD = 0x0fff,MaxAd = 0;
    uint16_t  num =0;
    uint32_t   number = 0;
    uint8_t t;
    for(t = 0; t < number_count; t++) 
    {
        num = buf;
        MaxAd = ADC_MAX(MaxAd,num);
        MinAD = ADC_MIN(MinAD,num);
        number +=num;
    }
    number = (number-MaxAd-MinAD)/(number_count-2);
    return number;
}

/* 取ADC最大值 */
uint32_t Read_ADC_DMA_Max(uint16_t buf,uint32_t number_count)
{
    uint32_t MaxAd = 0;
    uint32_t num =0;
    uint32_t number = 0;
    uint32_t t;
    for(t = 0; t < number_count; t++) 
    {
        num = buf;
        MaxAd = ADC_MAX(MaxAd,num);
    }
    number = MaxAd;
    return number;
}

adc.h

#ifndef __ADC_H__
#define __ADC_H__

#include "stm32f0xx.h"


void ADC_DMA_Init(void);
uint32_t Read_ADC_DMA_number(uint16_t buf,uint8_t number_count);
uint32_t Read_ADC_DMA_Max(uint16_t buf,uint32_t number_count);

#define ADC_MAX(x, y)   ((x) > (y) ? (x) : (y))
#define ADC_MIN(x, y)   ((x) < (y) ? (x) : (y))


#endif 

  • 16
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值