打印

【求助】STM32F051-ADC-DMA方式采集2路数据时出现问题

[复制链接]
5070|10
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
各位大侠,本人在用STM32F051R8T6 DMA方式实现采样ADC数据时,结果发现,从串口输出来的数值不对,一路数据很大,一路数据为零,弄了半天,也没有解决这个问题,只好请各位大侠们帮忙给看看,非常感谢!
【说明】:利用DMA中断方式,进行一路数据采集时是正常的。
相关源码如下:
   /* ADC1数据寄存器地址 */
#define ADC1_DR_Address        0x40012440
/* 宏定义        -------------------------------------------------------------*/
/* 变量定义          ---------------------------------------------------------*/
extern float  AD_CH_1 = 0;
extern float  AD_CH_2 = 0 ;
extern unsigned int   ADC_Value[2]={0,0};//内存数组
/*******************************************************************************
* 函数名称: ADC_GPIO_Init();
* 功能描述: ADC--GPIO输入引脚配置---在此可以设置16路外部输入通道
* 输入参数: void
* 返回参数: 无
********************************************************************************/
void ADC_GPIO_Init(void)
{
     GPIO_InitTypeDef  GPIO_InitStructure;
     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOB, ENABLE);   
     GPIO_InitStructure.GPIO_Pin   = GPIO_Pin_0 |GPIO_Pin_1;
     GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AN;
     GPIO_InitStructure.GPIO_PuPd  = GPIO_PuPd_NOPULL;
     GPIO_Init(GPIOB, &GPIO_InitStructure);
}
/*******************************************************************************
* 函数名称:  ADC_DMA_Reset()
* 功能描述:  ADC模块初始化
* 输入参数:  void
* 返回参数:  无
********************************************************************************/
void ADC_DMA_Reset(void)
{   
     ADC_InitTypeDef  ADC_InitStructure;
     DMA_InitTypeDef  DMA_InitStructure;

     ADC_DeInit(ADC1);  //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值
     RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);  // 使能ADC1通道时钟   
     RCC_AHBPeriphClockCmd( RCC_AHBPeriph_DMA1 , ENABLE);   // 使能DMA通道时钟
     ADC_GPIO_Init(); //ADC-GPIO引脚初始化
//////////////////////////////////////////////////////////////////////////////////
     /* DMA1 通道1配置设置*/
     DMA_DeInit(DMA1_Channel1);
     DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)ADC1_DR_Address; //DMA对应的外设基地址
     DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_Value; //内存存储基地址,定义的一个数组
     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //DMA转换模式为SRC模式,由外设搬移到内存
     DMA_InitStructure.DMA_BufferSize = 2;  // DMA缓存大小,2个(设置DMA在传输时缓冲区的长度)
     DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//接收一次数据后,设备地址禁止后移(设置DMA的外设递增模式)
     DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //关闭接收一次数据后,目标内存地址后移(设置DMA的内存递增模式)
     DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//定义外设数据长度--16位
     DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
     DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;  // 循环模式开启,Buf写满后,自动回到初始地址开始传输
     DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh; //DMA_Priority_High;
     DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
     DMA_Init(DMA1_Channel1, &DMA_InitStructure);
   
     DMA_ITConfig(DMA1_Channel1, DMA_IT_TC, ENABLE);
     DMA_Cmd(DMA1_Channel1, ENABLE);     // DMA1 Channel1 使能
     ADC_DMARequestModeConfig(ADC1, ADC_DMAMode_Circular); // ADC DMA 配置在循环模式下面   
////////////////////////////////////////////////////////////////////////////////////
     ADC_StructInit(&ADC_InitStructure);
     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_Backward; //后前:0--18通道
     ADC_Init(ADC1, &ADC_InitStructure);
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////
     // ADC_ChannelConfig(ADC1, ADC_Channel_TempSensor, ADC_SampleTime_239_5Cycles );
     // ADC_TempSensorCmd(ENABLE); //使能内部温度传感器
     // ADC_ChannelConfig(ADC1, ADC_Channel_Vrefint,ADC_SampleTime_239_5Cycles);
      //ADC_VrefintCmd(ENABLE);
       ADC_ChannelConfig(ADC1, ADC_Channel_8 , ADC_SampleTime_71_5Cycles );
       ADC_ChannelConfig(ADC1, ADC_Channel_9 , ADC_SampleTime_71_5Cycles );
////////////////////////////////////////////////////////////////////////////////////////      
       ADC_GetCalibrationFactor(ADC1);        // 开始ADC校准
       ADC_Cmd(ADC1, ENABLE);                //  使能指定的ADC1      
       while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_ADEN));   /* 等待ADC准备好 */
   
       ADC_DMACmd(ADC1, ENABLE);               // 使能 ADC_DMA
   
       ADC_StartOfConversion(ADC1); //开始温度转换;
}

DMA中断函数:
void DMA1_Channel1_IRQHandler(void)
{
   if(DMA_GetFlagStatus(DMA1_FLAG_TC1) != RESET ) //检测DMA1TC1标志
  {         
          Delay(50000);
                 AD_CH_1  =  (float)ADC_Value[0];
          AD_CH_1  =  AD_CH_1 /4096*3.3;
          AD_CH_2  =  (float)ADC_Value[1];
                 AD_CH_2  =  AD_CH_2/4096*3.3;
          DMA_ClearFlag(DMA1_FLAG_TC1);//清除DMA TC1标志   
    }
}

20121110103113953.jpg (27.46 KB )

串口显示的电压值--一路数据不对,一路为零

串口显示的电压值--一路数据不对,一路为零
沙发
acgean| | 2012-11-19 18:27 | 只看该作者
把 float 数值计算放在中断里, 呵呵, 够折腾了.
还是希望中断里尽量简单一些.

使用特权

评论回复
板凳
吾要单片机| | 2012-11-19 19:09 | 只看该作者
怎么会用到浮点数呢?

使用特权

评论回复
地板
lut1lut| | 2012-11-20 17:30 | 只看该作者
ADC_Value[2]这个数组定义的是32位宽度的,而DMA的源和目的传输宽度是16位宽度的。那么第一次ADC转换值被DMA放到了ADC_Value[0]的低16位;第二次ADC转换值被DMA放到了ADC_Value[0]的高16位;然后触发EOC中断,进行浮点计算处理。那么此时ADC_Value[1]还是0呢。

使用特权

评论回复
5
fengye5340|  楼主 | 2012-11-20 20:59 | 只看该作者
感谢楼主们的解答,问题已经解决了

使用特权

评论回复
6
lut1lut| | 2012-11-21 13:53 | 只看该作者
LZ应该把问题的原因和如何解决的公布一下吧。

也不枉大家帮你分析半天哦?

使用特权

评论回复
7
fengye5340|  楼主 | 2012-11-21 18:11 | 只看该作者
原来是 这样定义的 extern unsigned int   ADC_Value[2]={0,0};//内存数组
改成  extern __IO uint16_t    ADC_Value[2]={0,0};//内存数组
后问题解决,非常感谢大家的帮忙!昨天的网络不好,没有及时给出原因,抱歉!

使用特权

评论回复
8
zxm19820916| | 2013-2-27 15:57 | 只看该作者
采样数组随时有可能变化,是应该使用volatile修饰。

使用特权

评论回复
9
jcsasm| | 2014-2-12 16:14 | 只看该作者
我这怎么2路以上就有问题呢

使用特权

评论回复
10
fu_yifan| | 2014-6-21 23:30 | 只看该作者

请都各位高手,今天开了四个通道采样,为什么采样时间都设为7.5周,程序就走不下去了,只改变了第四通道的采样时间,改为13.5就可以了, 全都设为1.5不行,  只改最后一个改为13.5又可以了,单次采样一个序列

使用特权

评论回复
11
yyxlmq1314| | 2014-9-27 11:26 | 只看该作者
我也遇到这个问题,刚刚解决

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

99

主题

454

帖子

11

粉丝