OpenEdv-开源电子网

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

找了好多关于F4的多通道采集代码,实在找不出我的代码问题在哪,希望不是硬件问题,原子哥求帮忙

[复制链接]

1

主题

6

帖子

0

精华

新手上路

积分
41
金钱
41
注册时间
2014-5-13
在线时间
1 小时
发表于 2015-4-19 17:04:11 | 显示全部楼层 |阅读模式
5金钱
#include "adc.h"
#include "lcd.h"
#include "delay.h"
#include "sys.h"
__IO uint16_t adc_value_final[ADC_CH_NUM] ={0};
__IO uint16_t adc_value[ADC_CH_NUM * ADC_CONV_COUNT] = {0};

/*多通道独立DMA采集模式*/
 void ADC1_CH10to15_DMA_Config(void)
{
  ADC_InitTypeDef       ADC_InitStructure;
  ADC_CommonInitTypeDef ADC_CommonInitStructure;
  DMA_InitTypeDef       DMA_InitStructure;
  GPIO_InitTypeDef      GPIO_InitStructure;

  /* Enable ADC1, DMA2 and GPIO clocks ****************************************/
  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2 | RCC_AHB1Periph_GPIOC, ENABLE);
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);

  /* DMA2 Stream0 channel0 configuration **************************************/
  DMA_InitStructure.DMA_Channel = DMA_Channel_0;  
  DMA_InitStructure.DMA_PeripheralBaseAddr = ((uint32_t)&ADC1->DR);
  DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)adc_value;
  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
  DMA_InitStructure.DMA_BufferSize = ADC_CONV_COUNT*ADC_CH_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_FIFOMode = DMA_FIFOMode_Disable;         
  DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
  DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
  DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
  DMA_Init(DMA2_Stream0, &DMA_InitStructure);
  DMA_Cmd(DMA2_Stream0, ENABLE);

  /* Configure ADC1 Channel10to15 pin as analog input ******************************/
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;
  GPIO_Init(GPIOC, &GPIO_InitStructure);

  RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,ENABLE);
  RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,DISABLE);

  /* ADC Common Init **********************************************************/
  ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;
  ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;
  ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_1;
  ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_20Cycles;
  ADC_CommonInit(&ADC_CommonInitStructure);

  /* ADC1 Init ****************************************************************/
  ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
  ADC_InitStructure.ADC_ScanConvMode = ENABLE;
  ADC_InitStructure.ADC_ContinuousConvMode =  ENABLE;
  ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;
  ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
  ADC_InitStructure.ADC_NbrOfConversion =ADC_CH_NUM;
  ADC_Init(ADC1, &ADC_InitStructure);

  /* ADC1 regular channel10to15 configuration *************************************/
  ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_480Cycles);
  ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTime_480Cycles);
  ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 3, ADC_SampleTime_480Cycles);
  ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 4, ADC_SampleTime_480Cycles);

 /* Enable DMA request after last transfer (Single-ADC mode) */
  ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE);

  /* Enable ADC1 DMA */
  ADC_DMACmd(ADC1, ENABLE);

  /* Enable ADC1 */
  ADC_Cmd(ADC1, ENABLE);
}
/*
进行一次AD数据采集
*/
void ADC_Sampling(void)
{
  u8 i,j;
  uint16_t sum;
  ADC_SoftwareStartConv(ADC1);/* Start ADC Software Conversion */ 
  while(RESET == DMA_GetFlagStatus(DMA2_Stream0,DMA_FLAG_TCIF0));
  for(i=0; i<ADC_CH_NUM; i++) 
{
sum=0;
for(j=0; j<ADC_CONV_COUNT; j++)
sum+=adc_value[j*ADC_CONV_COUNT+i];
adc_value_final=sum/ADC_CONV_COUNT;
  }
}


/****
获取指定通道的值
*/
int ADC_GetChannelValue(u8 ch)
{
  if(ch<=0 || ch >ADC_CH_NUM) return -1;
  else 
    return adc_value_final[ch-1];
}/*ADC值显示*/
void Display_ADC1Voltage(void)
{
u16 adcx;
float temp;
  u16 i;
      for(i=0; i<ADC_CH_NUM; i++)
        {
            adcx=adc_value_final;
            switch(i)
              {
               case 0:
                LCD_ShowxNum(134,130,adcx,4,16,0);    //显示ADCC采样后的原始值
             temp=(float)adcx*(3.3/4096);          //获取计算后的带小数的实际电压值,比如3.1111
              adcx=temp;                            //赋值整数部分给adcx变量,因为adcx为u16整形
             LCD_ShowxNum(134,150,adcx,1,16,0);    //显示电压值的整数部分,3.1111的话,这里就是显示3
             temp-=adcx;                           //把已经显示的整数部分去掉,留下小数部分,比如3.1111-3=0.1111
             temp*=1000;                           //小数部分乘以1000,例如:0.1111就转换为111.1,相当于保留三位小数。
             LCD_ShowxNum(150,150,temp,3,16,0X80); //显示小数部分(前面转换为了整形显示),这里显示的就是111.
   break;
case 1:
 LCD_ShowxNum(134,170,adcx,4,16,0);    //显示ADCC采样后的原始值
             temp=(float)adcx*(3.3/4096);          //获取计算后的带小数的实际电压值,比如3.1111
              adcx=temp;                            //赋值整数部分给adcx变量,因为adcx为u16整形
             LCD_ShowxNum(134,190,adcx,1,16,0);    //显示电压值的整数部分,3.1111的话,这里就是显示3
             temp-=adcx;                           //把已经显示的整数部分去掉,留下小数部分,比如3.1111-3=0.1111
             temp*=1000;                           //小数部分乘以1000,例如:0.1111就转换为111.1,相当于保留三位小数。
             LCD_ShowxNum(150,190,temp,3,16,0X80); //显示小数部分(前面转换为了整形显示),这里显示的就是111.
   break; 
case 2:
LCD_ShowxNum(134,210,adcx,4,16,0);    //显示ADCC采样后的原始值
temp=(float)adcx*(3.3/4096);          //获取计算后的带小数的实际电压值,比如3.1111
adcx=temp;                            //赋值整数部分给adcx变量,因为adcx为u16整形
LCD_ShowxNum(134,230,adcx,1,16,0);    //显示电压值的整数部分,3.1111的话,这里就是显示3
temp-=adcx;                           //把已经显示的整数部分去掉,留下小数部分,比如3.1111-3=0.1111
temp*=1000;                           //小数部分乘以1000,例如:0.1111就转换为111.1,相当于保留三位小数。
LCD_ShowxNum(150,230,temp,3,16,0X80); //显示小数部分(前面转换为了整形显示),这里显示的就是111.
   break;
case 3:
LCD_ShowxNum(134,250,adcx,4,16,0);    //显示ADCC采样后的原始值
temp=(float)adcx*(3.3/4096);          //获取计算后的带小数的实际电压值,比如3.1111
adcx=temp;                            //赋值整数部分给adcx变量,因为adcx为u16整形
LCD_ShowxNum(134,270,adcx,1,16,0);    //显示电压值的整数部分,3.1111的话,这里就是显示3
temp-=adcx;                           //把已经显示的整数部分去掉,留下小数部分,比如3.1111-3=0.1111
temp*=1000;                           //小数部分乘以1000,例如:0.1111就转换为111.1,相当于保留三位小数。
LCD_ShowxNum(150,270,temp,3,16,0X80); //显示小数部分(前面转换为了整形显示),这里显示的就是111.
       break;
default:
     break;
}
}
  
}

最佳答案

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

我写的ADC 2通道+DMA
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

6

主题

93

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
300
金钱
300
注册时间
2014-6-12
在线时间
54 小时
发表于 2015-4-19 17:04:12 | 显示全部楼层
我写的ADC 2通道+DMA

ADC1通道4和5通过DMA得到旋钮当前值.rar

8.58 MB, 下载次数: 126

回复

使用道具 举报

39

主题

2026

帖子

1

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
2020
金钱
2020
注册时间
2013-5-1
在线时间
87 小时
发表于 2015-4-19 18:12:41 | 显示全部楼层
你的问题是什么.....不要只看液晶,仿真下看看你的数组里有没有数据,再去检查配置
博观而约取,厚积而薄发。
回复

使用道具 举报

0

主题

10

帖子

0

精华

初级会员

Rank: 2

积分
99
金钱
99
注册时间
2014-8-21
在线时间
15 小时
发表于 2015-4-20 22:31:42 | 显示全部楼层
DMA操作学习了。正在头疼中
回复

使用道具 举报

1

主题

6

帖子

0

精华

新手上路

积分
41
金钱
41
注册时间
2014-5-13
在线时间
1 小时
 楼主| 发表于 2015-4-22 12:07:32 | 显示全部楼层
回复【3楼】xmetoo:
---------------------------------
我调了一周了,还没出结果正看手册着
回复

使用道具 举报

4

主题

22

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
238
金钱
238
注册时间
2017-4-17
在线时间
60 小时
发表于 2017-7-9 11:39:43 | 显示全部楼层
楼主想问一下  __IO uint16_t adc_value_final[ADC_CH_NUM] ={0};
__IO uint16_t adc_value[ADC_CH_NUM * ADC_CONV_COUNT] = {0};

这两行代码的定义,,,谢谢
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2024-12-18 12:42

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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