OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 10644|回复: 6

ADC多通道采集到的数据完全错误

[复制链接]

7

主题

15

帖子

0

精华

初级会员

Rank: 2

积分
112
金钱
112
注册时间
2014-9-19
在线时间
4 小时
发表于 2014-10-26 11:30:05 | 显示全部楼层 |阅读模式
5金钱

adc1文件
#include "adc1.h"
#include "stm32f10x.h"       //这个头文件包括STM32F10x所有外围寄存器、位、内存映射的定义
#include "stdio.h"

 /*STM32 ADC多通道转换
描述:用ADC连续采集6路模拟信号,并由DMA传输到内存。
ADC配置为扫描并且连续转换模式,ADC的时钟配置为12MHZ。
在每次转换结束后,由DMA循环将转换的数据传输到内存中。
ADC可以连续采集N次求平均值。*/
#define  N  50     //每通道采50次
#define  M  6     //为12个通道

vu16  AD_Value[N][M];   //用来存放ADC转换结果,也是DMA的目标地址
vu16  After_filter[M];    //用来存放求平均值之后的结果
int i;   

/*GPIO管脚的配置   选用ADC的通道0  1  2  3  8  9  10  11  12  13  14  15,
分别对应的管脚为PA0  A1  A2   PA3  B0  B1 */
void GPIO_Configuration(void)
{
GPIO_InitTypeDef  GPIO_InitStructure; 
  
    //PA0/1/2/3 作为模拟通道输入引脚            
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(GPIOA, &GPIO_InitStructure);

//PB0/1 作为模拟通道输入引脚                         
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOB, &GPIO_InitStructure);
}

/*配置ADC1*/
void ADC1_Configuration(void)
    {
ADC_InitTypeDef  ADC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE );  //使能ADC1通道时钟
/* Configure ADCCLK such as ADCCLK = PCLK2/6 */ 
RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //72M/6=12,ADC最大时间不能超过14M

ADC_DeInit(ADC1);  //将外设 ADC1 的全部寄存器重设为缺省值

/*-----------------ADC1 configuration --------------------------------------*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式: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数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = M; //顺序进行规则转换的ADC通道的数目
ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
 
/* ADC1 regular channel11 configuration */ 
//设置指定ADC的规则组通道,设置它们的转化顺序和采样时间
//ADC1,ADC通道x,规则采样顺序值为y,采样时间为239.5周期
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 5, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 6, ADC_SampleTime_239Cycles5 );

// 开启ADC的DMA支持(要实现DMA功能,还需独立配置DMA通道等参数)
    ADC_DMACmd(ADC1, ENABLE);

/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);   //使能指定的ADC1
/* Enable ADC1 reset calibaration register */   
ADC_ResetCalibration(ADC1);  //复位指定的ADC1的校准寄存器
/* Check the end of ADC1 reset calibration register */
while(ADC_GetResetCalibrationStatus(ADC1)); //获取ADC1复位校准寄存器的状态,设置状态则等待

/* Start ADC1 calibaration */
ADC_StartCalibration(ADC1); //开始指定ADC1的校准状态
/* Check the end of ADC1 calibration */
while(ADC_GetCalibrationStatus(ADC1)); //获取指定ADC1的校准程序,设置状态则等待


}
/*配置DMA*/
void DMA_Configuration(void)
{
/* ADC1  DMA1 Channel Config */
DMA_InitTypeDef DMA_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输

DMA_DeInit(DMA1_Channel1);   //将DMA的通道1寄存器重设为缺省值
DMA_InitStructure.DMA_PeripheralBaseAddr =  (u32)&ADC1->DR;  //DMA外设ADC基地址
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&AD_Value;  //DMA内存基地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  //内存作为数据传输的目的地
DMA_InitStructure.DMA_BufferSize = N*M;  //DMA通道的DMA缓存的大小
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器递增
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;  //数据宽度为16位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;  //工作在循环缓存模式
DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA通道 x拥有高优先级 
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
DMA_Init(DMA1_Channel1, &DMA_InitStructure);  //根据DMA_InitStruct中指定的参数初始化DMA的通道

DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,ENABLE); //配置DMA1通道1传输完成中断
DMA_Cmd(DMA1_Channel1,ENABLE); //使能DMA1
}  
//配置所有外设
void Init_All_Periph(void)
{

GPIO_Configuration();

ADC1_Configuration();

DMA_Configuration();
 
}
/*根据ADC结果计算电压*/
u16 GetVolt(u16 advalue)  
  {   
   return (float)(advalue * 330 / 4096);   //求的结果扩大了100倍,方便下面求出小数 
  }

 /*求平均值函数*/
void filter(void)
{
  u32  sum = 0;
u16  count;    
  for(i=0;i<M;i++)
      {
         for ( count=0;count<N;count++)
          {
           sum += AD_Value[count]
          }
          After_filter=sum/N;  
          sum=0;
      }


main.c文件
 int main(void)
{
vu16  AD_Value[N][M];   //用来存放ADC转换结果,也是DMA的目标地址
vu16  After_filter[M];    //用来存放求平均值之后的结果
int i=0;
vu16 value[M];
SystemInit();
NVIC_Configuration();
delay_init(72);
LCD_Init();
ADC1_Configuration();
LCD_Clear();  
Init_All_Periph();
 EXTIX_Init();
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
DMA_Cmd(DMA1_Channel1, ENABLE); //启动DMA通道
while(1)
{
filter();
for(i=0;i<6;i++)
{
value= GetVolt(After_filter);
switch(i)
{
case 0: LCD_Clear_XY(0,39,50,89);
LCD_Show2Num(50,210,value,4,18,1);break;
case 1: LCD_Clear_XY(0,39,90,129);
LCD_Show2Num(90,210,value,4,18,1);break;
case 2: LCD_Clear_XY(0,39,130,169);
LCD_Show2Num(130,210,value[2],4,18,1);break;
case 3: LCD_Clear_XY(0,39,170,209);
LCD_Show2Num(170,210,value[3],4,18,1);break;
case 4: LCD_Clear_XY(0,39,210,249);
LCD_Show2Num(210,210,value[4],4,18,1);break;
case 5: LCD_Clear_XY(0,39,250,300);
LCD_Show2Num(250,210,value[5],4,18,1);break; 
default: break;
}  
delay_ms(500);  

i=0;
}
}


最佳答案

查看完整内容[请看2#楼]

这样应该是读不到数据的吧,你在主程序里面一直处理"ADC的转换数据" 这样是不是会出现这种问题,ADC转换的数据都还没有通过DMA传送到RAM区,你就在对这个RAM区的数据进行处理,这样读出的结果是不是有问题呢? 而且在对DMA转换的结果进行处理的时候是不是应该把DMA关掉?要不然数据会被重新刷新的 还有就是既然你用了DMA中断进行对数据处理,怎么没有看到DMA中断服务函数?却要在主程序里面去处理DMA转换的结果? 应该是所有的数据 ...
做自己喜欢做的事
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

39

主题

156

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
417
金钱
417
注册时间
2012-12-23
在线时间
19 小时
发表于 2014-10-26 11:30:06 | 显示全部楼层
这样应该是读不到数据的吧,你在主程序里面一直处理"ADC的转换数据"
这样是不是会出现这种问题,ADC转换的数据都还没有通过DMA传送到RAM区,你就在对这个RAM区的数据进行处理,这样读出的结果是不是有问题呢?
而且在对DMA转换的结果进行处理的时候是不是应该把DMA关掉?要不然数据会被重新刷新的
还有就是既然你用了DMA中断进行对数据处理,怎么没有看到DMA中断服务函数?却要在主程序里面去处理DMA转换的结果?
应该是所有的数据都转换完成之后(即:触发了DMA传输完成中断),这时候要把ADC关掉,再把DMA传输通道关闭,
然后再去处理DMA的转换数据,而且在对数据进行处理的时候要对数据进行滤波再求平均值(因为ADC可能会读到杂信号,去掉几个最小值和几个最大值),你再试试看,希望能帮到你
天道酬勤
回复

使用道具 举报

7

主题

15

帖子

0

精华

初级会员

Rank: 2

积分
112
金钱
112
注册时间
2014-9-19
在线时间
4 小时
 楼主| 发表于 2014-10-26 11:35:42 | 显示全部楼层
问题描述:
这个程序是使用PA0,1,2,3和PB8,9共6个通道的ADC去采集数据,使用多通道扫面的方式,6个GPIO口处于悬空状态,没有接任何外设。可是LCD上最终显示6个通道的AD值分别为0,0,67,660,108,660.
并且改变GPIO口输入电压,显示值不变化,求各位大神指点,谢谢!!!!!!!!
做自己喜欢做的事
回复

使用道具 举报

81

主题

1002

帖子

0

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
1876
金钱
1876
注册时间
2014-9-10
在线时间
208 小时
发表于 2014-10-26 12:31:15 | 显示全部楼层
你为什么不在GPIO配置里面使能GPIOA时钟,却再ADC里面才使能。
小小蜗牛
回复

使用道具 举报

7

主题

15

帖子

0

精华

初级会员

Rank: 2

积分
112
金钱
112
注册时间
2014-9-19
在线时间
4 小时
 楼主| 发表于 2014-10-26 23:43:16 | 显示全部楼层
这个程序当时是在网上找的,忘了改这个地方了,但是我刚把程序改了一下,还是不对,麻烦您能帮我在检查一下,谢谢了
做自己喜欢做的事
回复

使用道具 举报

7

主题

15

帖子

0

精华

初级会员

Rank: 2

积分
112
金钱
112
注册时间
2014-9-19
在线时间
4 小时
 楼主| 发表于 2014-10-28 20:26:26 | 显示全部楼层
谢谢,你一句话直击我的要害,瞬间感觉自己压根就不会写程序,谢谢你
做自己喜欢做的事
回复

使用道具 举报

54

主题

105

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
341
金钱
341
注册时间
2014-10-30
在线时间
0 小时
发表于 2014-12-13 12:54:09 | 显示全部楼层
楼主,能把你改好的程序发给我吗?
如何等到ADC转换的数据通过DMA传送到RAM区,才对这个RAM区的数据进行处理? 
在对DMA转换的结果进行处理的时候如何把DMA关掉?
还有就是既然你用了DMA中断进行对数据处理,怎么没有看到DMA中断服务函数?却要在主程序里面去处理DMA转换的结果? 
应该是所有的数据都转换完成之后(即:触发了DMA传输完成中断),这时候要把ADC关掉,再把DMA传输通道关闭, 
然后再去处理DMA的转换数据,而且在对数据进行处理的时候要对数据进行滤波再求平均值(因为ADC可能会读到杂信号,去掉几个最小值和几个最大值)
4楼说的,该怎样做到啊?
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-27 23:33

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表