STM32 双ADC DMA的使用 调了几天了没进展 请论坛各位高手帮忙看一下 小弟谢过了

2019-08-14 04:29发布

//主函数
int main(void)
{
// u32 ADC1_Buf[32];
    u32 ADC1_Buf[1];
 u16 adcx1; 
 u16 adcx2;
 u8  arry1[6];
 u8  arry2[6];
 float temp;
 SystemInit();
// Stm32_Clock_Init(9);  //
 delay_init(72);      //
 KEY_Init();
 LED_Init();
 ADC1_Init();
 //ADC2_Init();
//显示初始化  LCD12864_InitPort(); //
 LCD12864_Init();  //
    LCD12864_Pos(3,5);
 LCD12684_Wdat(0x35);
// adcx1=0;
// adcx2=0;
//调用DMA   
   // MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)ADC1_Buf,32);
    MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)ADC1_Buf,1);
 MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输

 while(1)  
 {    
  if(DMA1->ISR&(1<<1)); //DMA1传输完    
      
   adcx1=ADC1_Buf&0xffff; // 将底16位分出来
   adcx2=ADC1_Buf>>16;   //分出高16位
   //DMA1->IFCR|=1<<1;
   MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输   //显示
     temp = (float)adcx2*(3.3/4096);
     adcx2 = temp;
     arry2[0]=adcx2+0X30;
     arry2[1] = '.';
     adcx2 = (temp - adcx2)*1000;
     arry2[2]=adcx2/100+0X30;
     arry2[3]=adcx2%100/10+0X30;
     arry2[4]=adcx2%10+0X30;
     arry2[5]='';  */         //adcx1=Get_ADC1(ADC_CH11);
  //显示
     temp = (float)adcx1*(3.3/4096);
     adcx1 = temp;
     arry1[0]=adcx1+0X30;
     arry1[1] = '.';
     adcx1 = (temp - adcx1)*1000;
     arry1[2]=adcx1/100+0X30;
     arry1[3]=adcx1%100/10+0X30;
     arry1[4]=adcx1%10+0X30;
     arry1[5]='';
  
  LCD_ShowString(1,0,arry1);
     LCD_ShowString(4,4,arry2);  
     LCD_ShowString(3,0,"电子信息");
     //LCD_ShowString(4,0,arry3);
  //LCD_ShowString(4,4,arry4);   delay_ms(10);
   
 }   
}
//双ADC初始化
#include "stm32f10x.h"
#include "adc.h"
#include "dma.h" //u32 ADC1_Buf;
    
//初始化ADC1
//这里我们仅以规则通道为例
//我们默认将开启通道10 11                   
void  ADC1_Init(void)
{   
 //先初始化IO口
  RCC->APB2ENR|=1<<4;    //使能PORTA口时钟
 GPIOC->CRL&=0XFFFF0000;//PA0 1 2 3 anolog输入//PC0 1 2 3 anolog输入
 //通道10/11设置   
 RCC->APB2ENR|=1<<9;    //ADC1时钟使能  
 RCC->APB2RSTR|=1<<9;   //ADC1复位
 RCC->APB2RSTR&=~(1<<9);//复位结束    
 RCC->CFGR&=~(3<<14);   //分频因子清零 
 //SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
 //否则将导致ADC准确度下降!
 RCC->CFGR|=2<<14;         ADC1->CR1&=0XF0FFFF;   //工作模式清零
 ADC1->CR1|=0<<16;      //独立工作模式
  //ADC1->CR1|=6<<16;//规则同步模式
 ADC1->CR1&=~(1<<8);    //非扫描模式
 //ADC1->CR1|=(1<<8);    //扫描模式  
 ADC1->CR2&=~(1<<1);    //单次转换模式
 //ADC1->CR2|=(1<<1);    //连续转换模式
 ADC1->CR2&=~(7<<17);   
 ADC1->CR2|=7<<17;    //软件控制转换 
 ADC1->CR2|=1<<20;      //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
 ADC1->CR2&=~(1<<11);   //右对齐 
 ADC1->SQR1&=~(0XF<<20);
 ADC1->SQR1&=0<<20;     //1个转换在规则序列中 也就是只转换规则序列1      
 //设置通道10 11的采样时间
// ADC1->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
// ADC1->SQR3|=11;    //通道ch=11
    ADC1->SMPR1&=0XFFFFFFC7;//通道11采样时间清空
 ADC1->SMPR1|=7<<3;      //通道11  239.5周期,提高采样时间可以提高精确度  ADC1->CR2|=1<<8;//开启ADC DMA转换
 
 ADC1->CR2|=1<<0;     //开启AD转换器   
// ADC1->CR2|=1<<22;       //启动规则转换通道
// while(!(ADC1->SR&1<<1));//等待转换结束  
 ADC1->CR2|=1<<3;        //使能复位校准 
 while(ADC1->CR2&1<<3);  //等待复位校准结束    
    //该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。   
 ADC1->CR2|=1<<2;        //开启AD校准   
 while(ADC1->CR2&1<<2);  //等待校准结束
 //该位由软件设置以开始校准,并在校准结束时由硬件清除
 
  // MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)&ADC1_Buf,1);//配置DMA1_CHx
    MYDMA_Enable(DMA1_Channel1);//使能DMA1_CHx
// MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输
// ADC1->CR2|=1<<0;//开启ADC1转换  
}     
//获得ADC值
//ch:通道值 10 11
/*u16 Get_ADC1(u8 ch)  
{
 //设置转换序列     
 ADC1->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
 ADC1->SQR3|=ch;          
 ADC1->CR2|=1<<22;       //启动规则转换通道
 while(!(ADC1->SR&1<<1));//等待转换结束     
 return ADC1->DR;  //返回adc值 
}*/ //初始化ADC2
void ADC2_Init(void)
{
    RCC->APB2ENR|=1<<10;  //ADC2时钟使能 
    RCC->APB2RSTR|=1<<10;   //ADC2复位
    RCC->APB2RSTR&=~(1<<10);//复位结束
    RCC->CFGR&=~(3<<14);   //分频因子清零 
 //SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
 //否则将导致ADC准确度下降!
    RCC->CFGR|=2<<14;   //ADC2->CR1|=1<<11;  //ADC2->CR1 11不能开,不然就不能转换
    ADC2->CR1&=0XF0FFFF;   //工作模式清零
    ADC2->CR1|=0X0<<16;      //独立工作模式
 //ADC1->CR1|=6<<16;  // 规则同步模式
    //ADC2->CR1&=~(1<<8);    //非扫描模式 
 ADC2->CR1|=(1<<8);    //扫描模式 
    //ADC2->CR2&=~(1<<1);    //单次转换模式
    ADC2->CR2|=1<<1;       //连续转换模式
// ADC2->CR2&=~(7<<17);   
 ADC2->CR2|=7<<17;    //软件控制转换 
 ADC2->CR2|=1<<20;      //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
 ADC2->CR2&=~(1<<11);   //右对齐 
 ADC2->SQR1&=~(0XF<<20);
 ADC2->SQR1&=0<<20;     //1个转换在规则序列中 也就是只转换规则序列1 V
 ADC2->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
 ADC2->SQR3|=10;  //  通道 ch=10   //设置通道10的采样时间   
 
    ADC2->SMPR1&=0XFFFFFFF8;//通道10采样时间清空
 ADC2->SMPR1|=7<<0;      //通道10  239.5周期,提高采样时间可以提高精确度     ADC2->CR2|=1<<0;     //开启AD转换器 // ADC2->CR2|=1<<22;       //启动规则转换通道
// while(!(ADC2->SR&1<<1));//等待转换结束
   
 ADC2->CR2|=1<<3;        //使能复位校准 
    while(ADC2->CR2&1<<3); //等待复位校准结束  
    //该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。  
    ADC2->CR2|=1<<2;       //开启AD校准   
    while(ADC2->CR2&1<<2); //等待校准结束
    //该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
     
} //获得ADC2值
//ch:通道值 10 11
/*u16 Get_ADC2(u8 ch)  
{
 //设置转换序列     
 ADC2->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
 ADC2->SQR3|=ch;          
 ADC2->CR2|=1<<22;       //启动规则转换通道
 while(!(ADC2->SR&1<<1));//等待转换结束     
 return ADC2->DR;  //返回adc值 
} */
//DMA初始化
#include "dma.h"
#include "delay.h"
u16 DMA1_MEM_LEN;//保存DMA每次数据传送的长度      
//DMA1的各通道配置
//这里的传输形式是固定的,这点要根据不同的情况来修改
//从存储器->外设模式/8位数据宽度/存储器增量模式
//DMA_CHxMA通道CHx
//cpar:外设地址
//cmar:存储器地址
//cndtr:数据传输量 
void MYDMA_Config(DMA_Channel_TypeDef*DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
  RCC->AHBENR|=1<<0;  //开启DMA1时钟
  delay_ms(1);   //等待DMA时钟稳定
  DMA_CHx->CPAR=cpar;   //DMA1 外设地址
 DMA_CHx->CMAR=(u32)cmar;//DMA1,存储器地址
 DMA1_MEM_LEN=cndtr;     //保存DMA传输数据量
 DMA_CHx->CNDTR=cndtr;   //DMA1,传输数据量
 DMA_CHx->CCR=0X00000000;//复位
 DMA_CHx->CCR|=1<<4;  //从存储器读
 DMA_CHx->CCR|=1<<5;  //连续模式
 DMA_CHx->CCR|=0<<6;  //外设地址非增量模式
 DMA_CHx->CCR|=1<<7;  //存储器增量模式
 DMA_CHx->CCR|=2<<8;  //外设数据宽度为32位
 DMA_CHx->CCR|=2<<10; //存储器数据宽度32位
 DMA_CHx->CCR|=1<<12; //中等优先级
 DMA_CHx->CCR|=0<<14; //非存储器到存储器模式 
 DMA_CHx->CCR|=1<<0;    
} void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR|=1<<0;  //开启DMA传输
}  //开启一次DMA传输
void MYDMA_dmago(DMA_Channel_TypeDef*DMA_CHx)
{
 DMA_CHx->CCR&=~(1<<0);       //关闭DMA传输
 DMA_CHx->CNDTR=DMA1_MEM_LEN; //DMA1,传输数据量
 DMA_CHx->CCR|=1<<0;          //开启DMA传输
}


//ADC和DMA的初始化参考了原子哥的程序,主函数中if(DMA1->ISR&(1<<1)); //DMA1传输完  可以进入,能分出高16位和低16位的值,但是分出的值从来都没变过,ADC1和ADC2我都接的3V的电压,采集回来的去是ADC1=0.556  ADC2=1.654,且一直都没变过。我头大了,搞了很久,麻烦各位前辈指点,感激不尽。
                  
   
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
41条回答
冯丑丑爱睡觉
1楼-- · 2019-08-15 08:40
 精彩回答 2  元偷偷看……
cacao
2楼-- · 2019-08-15 10:17
回复【7楼】冯丑丑爱睡觉:
---------------------------------
连续转换模式我屏蔽掉了,使用单词转换模式,规则序列?我屏蔽了,还是不行。原子哥说没用到DMA,我不清楚,能帮我看一下吗,谢谢了。
cacao
3楼-- · 2019-08-15 10:29
回复【8楼】冯丑丑爱睡觉:
---------------------------------回复【8楼】冯丑丑爱睡觉:
这是我前几天写的,可能不完善,希望对你有帮助
---------------------------------
谢谢,我研究一下。
cacao
4楼-- · 2019-08-15 15:48
u8 i;
u32 ADC_Buf[10];
int main(void)
{
        u16 t;
u32 adcx;
u16 adcx1;
// u16 adcx2; 
u8  arry1[6];
// u8  arry2[6];
float temp;
SystemInit();
delay_init(72);      //延时初始化
ADC1_Init();
//ADC2_Init();
LCD12864_InitPort(); //端口初始化
LCD12864_Init();  //液晶初始化
        LCD12864_Pos(3,5);
LCD12684_Wdat(0x35);
while(1)
 
{
   /*****利用DMA由ADC外设向储存器传送数据********/
   MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)ADC_Buf,10);  
   ADC1->CR2|=1<<8;    //开启ADC的DMA转换
//    ADC1->CR2|=1<<0;    //开启AD转换
   MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输
    /*****等待DMA传送完成*******/
   while(!DMA1_Channel1->CNDTR)
    {  
     if(DMA1->ISR&(1<<1)); //DMA1传输完    
     {   
    DMA1->IFCR|=1<<1;
    break;
 }   
    }
 /***数据处理****/   
    for(t=0;t<10;t++)
{
adcx = adcx + ADC_Buf[t];

adcx = adcx /10; // 取10次平均值
adcx1=adcx&0xffff; //取低16位
//delay_ms(1);
    temp = (float)adcx1*(3.3/4096);
    adcx1 = temp;
    arry1[0]=adcx1+0X30;
    arry1[1] = '.';
    adcx1 = (temp - adcx1)*1000;
    arry1[2]=adcx1/100+0X30;
    arry1[3]=adcx1%100/10+0X30;
    arry1[4]=adcx1%10+0X30;
    arry1[5]='';
    LCD_ShowString(1,0,arry1);
    LCD_ShowString(3,0,"电子信息工程学院");
   delay_ms(100);
   
}  
}

void  ADC1_Init(void)
{    
//先初始化IO口
  RCC->APB2ENR|=1<<4;    //使能PORTA口时钟 
GPIOC->CRL&=0XFFFF0000;//PC0 1 2 3 anolog输入
//通道10/11设置  
RCC->APB2ENR|=1<<9;    //ADC1时钟使能   
RCC->APB2RSTR|=1<<9;   //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束     
RCC->CFGR&=~(3<<14);   //分频因子清零
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降! 
RCC->CFGR|=2<<14;        

ADC1->CR1&=0XF0FFFF;   //工作模式清零
ADC1->CR1|=0<<16;      //独立工作模式 
ADC1->CR1|=(1<<8);    //扫描模式   
ADC1->CR2|=(1<<1);    //连续转换模式
ADC1->CR2&=~(7<<17);    
ADC1->CR2|=7<<17;    //软件控制转换  
ADC1->CR2|=1<<20;      //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC1->CR2&=~(1<<11);   //右对齐  
ADC1->SQR1&=~(0XF<<20);
   
//设置通道10 11的采样时间

ADC1->SQR3&=0XFFFFFFE0;   //规则序列1=通道0 
ADC1->SQR3|=11;


            ADC1->SMPR1&=0XFFFFFFC7;//通道11采样时间清空
ADC1->SMPR1|=7<<3;      //通道11  239.5周期,提高采样时间可以提高精确度

ADC1->CR2|=1<<8;//开启ADC DMA转换
 
ADC1->CR2|=1<<0;     //开启AD转换器
 
ADC1->CR2|=1<<3;        //使能复位校准  
while(ADC1->CR2&1<<3);  //等待复位校准结束   
    //该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。   
ADC1->CR2|=1<<2;        //开启AD校准    
while(ADC1->CR2&1<<2);  //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除 
}   
//获得ADC值
u16 Get_ADC1(u8 ch)   
{
//设置转换序列     
ADC1->SQR3&=0XFFFFFFE0;//
ADC1->SQR3|=ch; //      
ADC1->CR2|=1<<22;       //启动规则转换通道 
while(!(ADC1->SR&1<<1));//等待转换结束      
return ADC1->DR; //返回adc值
}
cacao
5楼-- · 2019-08-15 16:31
---------------------------------
回复【8楼】冯丑丑爱睡觉:
---------------------------------
我参考了你的程序,也是只用一个ADC,即ADC1,DMA有数据了,但是跳动变化直到0停止,不是想要的。还是没有调出来。急啊!!!
冯丑丑爱睡觉
6楼-- · 2019-08-15 18:55
 精彩回答 2  元偷偷看……

一周热门 更多>