OpenEdv-开源电子网

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

ADC转换结果很乱》》》》

[复制链接]

145

主题

445

帖子

0

精华

高级会员

Rank: 4

积分
823
金钱
823
注册时间
2014-1-15
在线时间
350 小时
发表于 2014-12-9 11:31:06 | 显示全部楼层 |阅读模式
5金钱
[mw_shl_code=c,true]volatile u8 ADCfinishsign = 0; u16 sensor[3]={1,1,1}; u16 ADCval[adccount][adcchcount]; /*********************************************************************************** 单通道,连续转换,得到10组数据后停止,启动下一通道 ADC_Channel_12,ADC_Channel_13,ADC_Channel_14 ***********************************************************************************/ void Adc_Init(void) { ADC_InitTypeDef ADC_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOC|RCC_APB2Periph_ADC1, ENABLE ); //使能ADC1通道时钟 RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M //PA1 作为模拟通道输入引脚 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚 GPIO_Init(GPIOC, &GPIO_InitStructure); ADC_DeInit(ADC1); //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值 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 = 3; //顺序进行规则转换的ADC通道的数目 ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器 ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 1, ADC_SampleTime_71Cycles5 ); ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 2, ADC_SampleTime_71Cycles5 ); ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 3, ADC_SampleTime_71Cycles5 ); ADC_DMACmd(ADC1, ENABLE); ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1 ADC_ResetCalibration(ADC1); //使能复位校准 while(ADC_GetResetCalibrationStatus(ADC1)) ; //等待复位校准结束 ADC_ResetCalibration(ADC1); //使能复位校准 while(ADC_GetResetCalibrationStatus(ADC1)) ; //等待复位校准结束 } /*********************************************************************************** ***********************************************************************************/ void ADC_Start(void) { DMA_Cmd( DMA1_Channel1, DISABLE ); ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 DMA1_Channel1->CNDTR = adccount*adcchcount; DMA_Cmd(DMA1_Channel1, ENABLE); } /*********************************************************************************** ***********************************************************************************/[/mw_shl_code] [mw_shl_code=c,true]#define adcchcount 3[/mw_shl_code] [mw_shl_code=c,true] #define adccount   10[/mw_shl_code] [mw_shl_code=c,true]void GetADCAverage(void)  {   u8  i,j;   u16 temmin=0,temmax=0,temsum=0;   if(ADCfinishsign)   {     ADCfinishsign=0;     for( j=0; j<adcchcount; j++ )[/mw_shl_code] [mw_shl_code=c,true]    {       temmin = ADCval[0][j];       temmax = ADCval[0][j];       temsum=0;       for( i=0; i<adccount; i++ )       {         temsum += ADCval[j];                  if( ADCval[j]<temmin )         {           temmin = ADCval[j];                 }         else           if( ADCval[j]>temmax )            {             temmax = ADCval[j];                   }       }       temsum   -= (temmin+temmax);       sensor[j] = temsum/(adccount-2);     }       } } [/mw_shl_code]

[mw_shl_code=c,true]void DMA1_Channel1Config(void) { DMA_InitTypeDef DMA_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输 DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1->DR; DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADCval; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize = adccount*adcchcount; 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_Normal; //工作在正常缓存模式 DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级 DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(DMA1_Channel1, &DMA_InitStructure); DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,ENABLE); //发送完成中断 NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } /*********************************************************************************** ***********************************************************************************/ void DMA1_Channel1_IRQHandler(void) { if(DMA1->ISR&(1<<1)) { ADC_SoftwareStartConvCmd(ADC1, DISABLE); DMA1->IFCR|=(1<<1); ADCfinishsign=1; } } [/mw_shl_code]

我是这样设计的:ADC扫描模式,PC2、3、4。DMA只传输一次(非循环模式),启动ADC后设置DMA传输的数量,DMA传输完成中断关闭ADC。并将标志置位。
目的是:每次打开ADC后转换10次然后关闭。
结果:14通道接到3.3V,其他接地。可以看到14通道的结果乱跑。



最佳答案

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

竟然没人??? 解决办法:加上ADC_SoftwareStartConvCmd(ADC1, DISABLE); ADC_Cmd(ADC1, DISABLE);
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

145

主题

445

帖子

0

精华

高级会员

Rank: 4

积分
823
金钱
823
注册时间
2014-1-15
在线时间
350 小时
 楼主| 发表于 2014-12-9 11:31:07 | 显示全部楼层
竟然没人???
解决办法:加上ADC_SoftwareStartConvCmd(ADC1, DISABLE);
ADC_Cmd(ADC1, DISABLE);
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2014-12-9 22:19:29 | 显示全部楼层
回复【2楼】Di诺:
---------------------------------
恭喜解决了。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-6-28 05:00

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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