请问STM32的ADC,对于指定通道如何实现多次采样后再将AD值通过DMA传入内存

-网上有很多ADC与DMA结合的例程,但是发现都是每次ADC转换完成就通过DMA传输给内存。-我现在希望达到这样的效果,为了让AD转换值更稳定,需要其连续转换10次后,再... -网上有很多ADC与DMA结合的例程,但是发现都是每次ADC转换完成就通过DMA传输给内存。
-我现在希望达到这样的效果,为了让AD转换值更稳定,需要其连续转换10次后,再通过DMA传输给内存,但是不知道怎么设置这个10次这个值。
-之前的方法一直是在ADC终端里,通过软件判断转换次数,达到10次后再软件赋给指定内存,不知道这次能不能用DMA实现这个方法。
展开
 我来答
百度网友cc21aee
推荐于2017-11-26 · TA获得超过2044个赞
知道小有建树答主
回答量:1273
采纳率:100%
帮助的人:1216万
展开全部

多次采集取平均值即可,给你个我写的四通道转换,想更平稳的话可以在条件允许的情况下简单的用下递推平均滤波

/*********************************************************************************
 * 文件名  :ADC.c
 * 描述    :ADC模块
 * 库版本  :ST3.5.0   
 * 编写时间:2013年12月3日
 * 编写人  :LiuHui
 *
 *
 * 修改时间:2013年7月8
 * 修改内容:
 *          @1更改ADC通道为PC0~3
 *          @2加入过采样,提高精度减小波动 
 *  @!!!!
 *  程序在开发板和核心板上运行完全正常,放到SD板上ADC_ConvertedValue[1]总是等于ADC_ConvertedValue[0]
 *  原因未查明,初步判断为DMA问题,使用SD卡型时勿用ADC_ConvertedValue[1]
 *  原因现已查明为SD板问题  
 *  @!!!!
*/
#include "system.h"
#include "ADC.h"
vu16 ADC_ConvertedValue[Sample_Num][Channel_Num];
void ADC_DMA_Config(void)
{
 DMA_InitTypeDef DMA_InitStructure;
             
 RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
 DMA_DeInit(DMA1_Channel1);
 DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
 DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;
 DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
 DMA_InitStructure.DMA_BufferSize = Sample_Num*Channel_Num;
 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);
 DMA_Cmd(DMA1_Channel1,ENABLE);
}
void ADC1_Config(void)
{
 GPIO_InitTypeDef GPIO_InitStructure;
 ADC_InitTypeDef ADC_InitStructure;
 
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
 
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
 GPIO_Init(GPIOC, &GPIO_InitStructure);
 ADC_DMA_Config();
 
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
// ADC_DeInit(ADC1);
 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_InitStructure.ADC_NbrOfChannel = Channel_Num;//通道数目
 ADC_Init(ADC1, &ADC_InitStructure); 
// ADC_TempSensorVrefintCmd(ENABLE);//使能片内温度传感器
 
 RCC_ADCCLKConfig(RCC_PCLK2_Div6); //PCLK 6分频
 ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_55Cycles5);//通道,转换次序,转换时间
 ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTime_55Cycles5);
 ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 3, ADC_SampleTime_55Cycles5);
 ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 4, ADC_SampleTime_55Cycles5);
// ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 6, ADC_SampleTime_239Cycles5);
 
 ADC_DMACmd(ADC1, ENABLE);
 
 ADC_Cmd(ADC1, ENABLE);
 ADC_ResetCalibration(ADC1);
 while(ADC_GetResetCalibrationStatus(ADC1));
 ADC_StartCalibration(ADC1);//开始校准
 while(ADC_GetCalibrationStatus(ADC1));
 ADC_SoftwareStartConvCmd(ADC1, ENABLE);//使能ADC的软件转换启动功能
}
uint16_t ReadADCAverageValue(uint16_t Channel)
{
 uint8_t i;
 uint32_t sum = 0;
 for(i=0; i<Sample_Num; i++)
 {
  sum+=ADC_ConvertedValue[i][Channel];
 }
 return (sum/Sample_Num);
}


头文件
#ifndef __ADC_H
#define __ADC_H
/*使用示例:
*
 printf("%d\r\n",ADC_ConvertedValue[0]);
 printf("%d\r\n",ADC_ConvertedValue[1]);
 printf("%d\r\n",ADC_ConvertedValue[2]);
 printf("о?:%f\r\n",(1.43-(ADC_ConvertedValue[3]*3.3/4096))/0.0043+25);
*
 sprintf(ch,"AD0 Value Is:%5d.",ADC_ConvertedValue[0]);
 printf("%s",ch);
 LCD_ShowString(20,20,(uint8_t*)ch,Color[i]);
 sprintf(ch,"AD1 Value Is:%5d.",ADC_ConvertedValue[1]);
 printf("%s",ch);
 LCD_ShowString(20,40,(uint8_t*)ch,Color[i]);
 sprintf(ch,"AD2 Value Is:%5d.",ADC_ConvertedValue[2]);
 printf("%s",ch);
 LCD_ShowString(20,60,(uint8_t*)ch,Color[i]);
 sprintf(ch,"AD3 Value Is:%5d.",ADC_ConvertedValue[3]);
 printf("%s",ch);
 LCD_ShowString(20,80,(uint8_t*)ch,Color[i]);
*/
#define Channel_Num  4
#define Sample_Num  10
extern vu16 ADC_ConvertedValue[Sample_Num][Channel_Num];
void ADC1_Config(void);
uint16_t ReadADCAverageValue(uint16_t Channel);
#endif
追问
请问是谁来调用的ReadADCAverageValue,是在DMA的中断,还是在还是在需要调用ADC值之前的代码中进行?
追答
主程序需要使用ADC转换的数据的时候,比如读取通道10直接调用ReadADCAverageValue(0x00);
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式