STM32F407 ADC的DMA传输

2019-07-20 02:42发布

不过有个问题,从第二次启动ADC转换,第一个传输的数据是不对的(数据是上一次的数据,现在的做法是去掉最大和最小值),这个一直找不到问题所在。下面附上代码。

#define Adc_len 7
u16 Adc_Buf[Adc_len];
//初始化ADC                                                                                                                          
void  Adc_Init(void)
{
  GPIO_InitTypeDef  GPIO_InitStructure;
        ADC_CommonInitTypeDef ADC_CommonInitStructure;
        ADC_InitTypeDef       ADC_InitStructure;
        DMA_InitTypeDef  DMA_InitStructure;
       
  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能GPIOA时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //使能ADC1时钟

  //先初始化ADC1通道0 IO口
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;//PA0 通道0
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;//模拟输入
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;//不带上下拉
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化

        // Enable DMA Clock.
        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2,  ENABLE);
        // USART1 发送数据流.
        DMA_DeInit(DMA2_Stream0);
        while (DMA_GetCmdStatus(DMA2_Stream0) != DISABLE){}//等待DMA可配置

        // USART1 DMA Cell Init.
        DMA_StructInit(&DMA_InitStructure);
        DMA_InitStructure.DMA_Channel             =  DMA_Channel_0;                                                // 所在 DMA通道
        DMA_InitStructure.DMA_DIR                 =  DMA_DIR_PeripheralToMemory;                // 外设到存储器

        DMA_InitStructure.DMA_Memory0BaseAddr     =  (u32)(& Adc_Buf[0]);                // 存储器地址
        DMA_InitStructure.DMA_MemoryInc           =  DMA_MemoryInc_Enable;                                // 存储器地址递增
        DMA_InitStructure.DMA_MemoryDataSize      =  DMA_PeripheralDataSize_HalfWord;                        // 半字

        DMA_InitStructure.DMA_PeripheralBaseAddr  =  (u32)(& ADC1->DR);                                // 外设地址
        DMA_InitStructure.DMA_PeripheralInc       =  DMA_PeripheralInc_Disable;                        // 外设地址不变
        DMA_InitStructure.DMA_PeripheralDataSize  =  DMA_PeripheralDataSize_HalfWord;                // 半字

        DMA_InitStructure.DMA_BufferSize          =  Adc_len;                                // 存储器长度
        DMA_InitStructure.DMA_Mode                =  DMA_Mode_Normal;                                        // 不使用循环模式( NDTR计数到0时停止接收 ) DMA_Mode_Circular
        DMA_InitStructure.DMA_Priority            =  DMA_Priority_VeryHigh;                                // 优先级

        DMA_InitStructure.DMA_FIFOMode            =  DMA_FIFOMode_Disable;                                // 不使用FIFO
        DMA_InitStructure.DMA_FIFOThreshold       =  DMA_FIFOThreshold_1QuarterFull;        // 阈值级别为1/4
        DMA_InitStructure.DMA_MemoryBurst         =  DMA_MemoryBurst_Single;                        // 存储器单次传输
        DMA_InitStructure.DMA_PeripheralBurst     =  DMA_PeripheralBurst_Single;                // 外设单次传输

        DMA_Init(DMA2_Stream0,  &DMA_InitStructure);

//        DMA2_Stream0->CR &= ~(1<<5);
        // 不使用双缓冲模式.
        DMA_DoubleBufferModeCmd(DMA2_Stream0,     DISABLE);
        DMA_DoubleBufferModeConfig(DMA2_Stream0,  (u32)(& Adc_Buf[0]),  DMA_Memory_0);                // 使用第一缓冲区
       

        RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,ENABLE);          //ADC1复位
        RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,DISABLE);        //复位结束

  ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式
  ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;//两个采样阶段之间的延迟5个时钟
  ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //DMA失能
  ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;//预分频4分频。ADCCLK=PCLK2/4=84/4=21Mhz,ADC时钟最好不要超过36Mhz
  ADC_CommonInit(&ADC_CommonInitStructure);//初始化
       
  ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//12位模式
  ADC_InitStructure.ADC_ScanConvMode = DISABLE;//非扫描模式       
  ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//关闭连续转换
  ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;//禁止触发检测,使用软件触发
  ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐
  ADC_InitStructure.ADC_NbrOfConversion = 1;//1个转换在规则序列中 也就是只转换规则序列1
  ADC_Init(ADC1, &ADC_InitStructure);//ADC初始化
        ADC1->CR2 |= 3<<8;//开启AD_DMA
       
        ADC_Cmd(ADC1, ENABLE);//开启AD转换器
}

u16 Get_Adc(u8 ch)
{
        u32 sum=0,i;
        u16 min = 0XFFFF, max = 0;
        DMA2_Stream0->CR &= ~(uint32_t)DMA_SxCR_EN;// 关闭DMA.
        while (DMA_GetCmdStatus(DMA2_Stream0) != DISABLE){}        //确保DMA可以被设置
        // 重置接收长度和设定接收缓冲区.
        DMA2_Stream0->NDTR  =  Adc_len;
        DMA2_Stream0->M0AR  = (uint32_t)Adc_Buf;
        DMA2_Stream0->CR |= (uint32_t)DMA_SxCR_EN;// 重启DMA.
        //设置指定ADC的规则组通道,一个序列,采样时间
        ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_480Cycles );        //ADC1,ADC通道,480个周期,提高采样时间可以提高精确度                            
//        ADC1->SQR3 &= ~(0X1F);//规则通道1序列清0
//        ADC1->SQR3 |= 0;//设置规则通道
        ADC1->CR2    |= (1<<1);//开启连续转换模式
        ADC_SoftwareStartConv(ADC1);                //使能指定的ADC1的软件转换启动功能
        //等待转换完成
        while(DMA_GetCurrDataCounter(DMA2_Stream0))
        {
        }
       
        DMA2_Stream0->CR &= ~(uint32_t)DMA_SxCR_EN;// 关闭DMA.
        ADC1->CR2  &= ~(1<<1);//关闭连续转换模式
        //清除DMA和ADC传输标记 不可少
        ADC1->SR = 0;
        DMA2->LIFCR = 0X7D;
        for(i=0;i<Adc_len;i++)
        {
                if(min > Adc_Buf[i])
                        min = Adc_Buf[i];
                if(max < Adc_Buf[i])
                        max = Adc_Buf[i];
                sum += Adc_Buf[i];
//                printf("Buf%d=%d ",i,Adc_Buf[i]);
        }
        sum -= (min + max);
        return sum/(Adc_len - 2)*3300/4095;
}

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
3条回答
功夫孬嫑昆
1楼-- · 2019-07-20 08:13
怀疑是DMA传输完成后,但ADC是连续转换模式,在还没关闭连续转换之前ADC又转换了一次,所以转换成功后还会发生DMA请求,但此时DMA传输已经完成,这个值也就存在那里,当下次再次启动DMA后,这个值就存起来了。
念尘lovingu
2楼-- · 2019-07-20 13:15
谢谢分享
zuan
3楼-- · 2019-07-20 13:17
 精彩回答 2  元偷偷看……

一周热门 更多>