stm32adc多通道采集数据,dma的内存数据如何按照通道来保存...

2019-12-27 18:57发布

参考  http://www.ourdev.cn/forum.php?m ... A%E9%80%9A%E9%81%93

代码如下:
/内部电压,内部温度
//ADC的电压采集
#define ChannelNumber 3
#define CollectNumber 12

#define ADC1_DR_Address    ((uint32_t)0x4001244C)
__IO uint16_t ADCConvertedValue[CollectNumber][ChannelNumber];

u16 vI = 0;
u32 vV = 0;

void SIM_ADC_Init(void)
{
     GPIO_InitTypeDef GPIO_InitStructure;
     ADC_InitTypeDef ADC_InitStructure;
     DMA_InitTypeDef DMA_InitStructure;
     
     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);
     RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
   
     GPIO_InitStructure.GPIO_Pin  =GPIO_Pin_5;
     GPIO_InitStructure.GPIO_Mode =GPIO_Mode_AIN;
     GPIO_Init(GPIOA,&GPIO_InitStructure);
     
     
      /* DMA channel1 configuration ----------------------------------------------*/
    DMA_DeInit(DMA1_Channel1);   //将DMA的通道1寄存器重设为缺省值
    DMA_InitStructure.DMA_PeripheralBaseAddr =  (u32)&ADC1->DR;  //DMA外设ADC基地址
    DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADCConvertedValue;  //DMA内存基地址
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  //内存作为数据传输的目的地
    DMA_InitStructure.DMA_BufferSize = ChannelNumber * CollectNumber;  //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的通道
   
      /* Enable DMA channel1 */
      DMA_Cmd(DMA1_Channel1, ENABLE);
     
  /* ADC1 configuration ------------------------------------------------------*/
  ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
  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 = 3;
  ADC_Init(ADC1, &ADC_InitStructure);

  /* ADC1 regular channel13 configuration */
  ADC_RegularChannelConfig(ADC1, ADC_Channel_5,  1, ADC_SampleTime_239Cycles5);
  ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 2, ADC_SampleTime_239Cycles5);   
  ADC_RegularChannelConfig(ADC1, ADC_Channel_17, 3, ADC_SampleTime_239Cycles5);
  
  /* Enable ADC1 DMA */
  ADC_DMACmd(ADC1, ENABLE);
  
  /* Enable ADC1 */
  ADC_Cmd(ADC1, ENABLE);

  ADC_TempSensorVrefintCmd(ENABLE);
  
  /* Enable ADC1 reset calibaration register */   
  ADC_ResetCalibration(ADC1);
  /* Check the end of ADC1 reset calibration register */
  while(ADC_GetResetCalibrationStatus(ADC1));
  /* Start ADC1 calibaration */
  ADC_StartCalibration(ADC1);
  /* Check the end of ADC1 calibration */
  while(ADC_GetCalibrationStatus(ADC1));
     
  /* Start ADC1 Software Conversion */
  ADC_SoftwareStartConvCmd(ADC1, ENABLE);
  
  //AD_value=((float)ADC_GetConversionValue(ADC1)/4095.0)*3.3;   
}

数据是否按照通道号讲采集的数据保存在数组里,:有点晕了,

2.png (50.92 KB, 下载次数: 0) 下载附件 2012-5-29 02:25 上传

在这个watch中,如何这样看:
ADCConvertedValue[0] ,列表为:12个采集数据
ADCConvertedValue[1] ,列表为:12个采集数据
ADCConvertedValue[2] ,列表为:12个采集数据
这样更加清爽点呢

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
17条回答
tiancaigao7
1楼-- · 2019-12-27 21:29
貌似不行,向dsPIC的DMA有分散模式,这样可以设定每个数组的长度以及数组的数量,然后DMA每次传输会在不同的数组之间跳转。但是STM32貌似没有这个功能
shellgo
2楼-- · 2019-12-27 22:44
tiancaigao7 发表于 2012-5-29 08:00
貌似不行,向dsPIC的DMA有分散模式,这样可以设定每个数组的长度以及数组的数量,然后DMA每次传输会在不同 ...

规则组
此模式通过设置ADC_CR1寄存器上的DISCEN位激活。它可以用来执行一个短序列的n次转换
(n<=8),此转换是ADC_SQRx寄存器所选择的转换序列的一部分。数值n由ADC_CR1寄存器的
DISCNUM[2:0]位给出。
一个外部触发信号可以启动ADC_SQRx寄存器中描述的下一轮n次转换,直到此序列所有的转
换完成为止。总的序列长度由ADC_SQR1寄存器的L[3:0]定义。
举例:
n=3,被转换的通道 = 0、1、2、3、6、7、9、10
第一次触发:转换的序列为 0、1、2
第二次触发:转换的序列为 3、6、7
第三次触发:转换的序列为 9、10,并产生EOC事件
第四次触发:转换的序列 0、1、2
注意:
当以间断模式转换一个规则组时,转换序列结束后不自动从头开始。
当所有子组被转换完成,下一次触发启动第一个子组的转换。在上面的例子中,第四次触发重
新转换第一子组的通道
0

1

2

呵呵,看结果,就是这样的
shellgo
3楼-- · 2019-12-28 03:03
现在增加了均值计算,__IO uint16_t ADCValue[ChannelNumber];
void DMA1_Channel1_IRQHandler(void)
{
    if(DMA_GetITStatus(DMA1_IT_TC1) != RESET)
    {
      Filter();
      DMA_ClearITPendingBit(DMA1_IT_TC1);
    }
}


void Sim_ADC_Filter(void)
{
    u32  sum = 0;
    u8 count,i;
    for(i=0;i<ChannelNumber;i++)
    {
        sum=0;
        for ( count=0;count<ChannelNumber;count++)
        {
            sum += ADCConvertedValue[count][i];
        }
        ADValue[i]=sum/ChannelNumber;
    }
}

为什么在别的函数中访问ADValue[1]就特别容易死掉啊??
shellgo
4楼-- · 2019-12-28 04:35
 精彩回答 2  元偷偷看……
gbhgbh
5楼-- · 2019-12-28 04:49
gbhgbh
6楼-- · 2019-12-28 09:26

一周热门 更多>