关于DAC实验问题

2019-10-14 22:32发布

本帖最后由 arao和xiaomi 于 2017-2-25 11:47 编辑

在学习原子DAC实验的时候出现一个现象,不知道是什么原因,求解答。现象描述:在最开始的时候初始值都为0,然后通过按键使DAC的值变大,每次增大200,在DAC的VAL 为1800以下时,DAC和ADC的值都能同步,就是DAC的值增大,ADC的值也跟着增大,但是当DAC的VAL 为1800以上时,再按键的时候DAC的VAL 的值继续增大,但是ADC的值却没有同步变化,只是在很小的增幅变化。疑问通过了杜邦线将DAC和ADC的IO口连起来了,为什么在电压值比较大(大于1.5v)的时候,ADC和DAC的值没有同步变化呢?



后面楼主自己再尝试了一下,直接给DAC写入3000,对应的DAC vol值为2.416v,对应转换的ADC vol值却只有1.703v,这个是什么原因呀?感觉不是很理解,DAC出来的模拟量直接接到ADC上面去了,为什么ADC得到的值却不一样呢?

然后当我直接给DAC写入1600,对应的DAC vol值为1.289v,对应转换的ADC vol值却只有1.289v,这个时候它们之间转换的结果就比较准确,然后我就在想,是不是ADC配置的时钟的原因,导致当模拟量的值比较大的时候,就把那个误差放大了,而产生了比较大的误差呀?
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
15条回答
arao和xiaomi
1楼-- · 2019-10-15 11:17
 精彩回答 2  元偷偷看……
电子之旅
2楼-- · 2019-10-15 13:30
arao和xiaomi 发表于 2017-2-26 13:09
@正点原子   原子哥帮我解答一下呗

把你的ADC、DAC代码配置都贴出来看看
arao和xiaomi
3楼-- · 2019-10-15 15:44
电子之旅 发表于 2017-2-27 14:11
把你的ADC、DAC代码配置都贴出来看看

[mw_shl_code=c,true]adc.c

#include "adc.h"
#include "delay.h"
//初始化ADC
//这里我们仅以规则通道为例
//我们默认将开启通道0~3                                                                                                                                          
void  Adc_Init(void)
{        
        ADC_InitTypeDef ADC_InitStructure;
        GPIO_InitTypeDef GPIO_InitStructure;

        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1        , ENABLE );          //使能ADC1通道时钟


        RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

        //PA1 作为模拟通道输入引脚                        
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;                //模拟输入引脚
        GPIO_Init(GPIOA, &GPIO_InitStructure);       

        ADC_DeInit(ADC1);  //复位ADC1

        ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;        //ADC工作模式:ADC1和ADC2工作在独立模式
        ADC_InitStructure.ADC_ScanConvMode = DISABLE;        //模数转换工作在单通道模式
        ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;        //模数转换工作在单次转换模式
        ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;        //转换由软件而不是外部触发启动
        ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;        //ADC数据右对齐
        ADC_InitStructure.ADC_NbrOfChannel = 1;        //顺序进行规则转换的ADC通道的数目
        ADC_Init(ADC1, &ADC_InitStructure);        //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   

  
        ADC_Cmd(ADC1, ENABLE);        //使能指定的ADC1
       
        ADC_ResetCalibration(ADC1);        //使能复位校准  
         
        while(ADC_GetResetCalibrationStatus(ADC1));        //等待复位校准结束
       
        ADC_StartCalibration(ADC1);         //开启AD校准

        while(ADC_GetCalibrationStatus(ADC1));         //等待校准结束

//        ADC_SoftwareStartConvCmd(ADC1, ENABLE);                //使能指定的ADC1的软件转换启动功能

}                                  
//获得ADC值
//ch:通道值 0~3
u16 Get_Adc(u8 ch)   
{
          //设置指定ADC的规则组通道,一个序列,采样时间
        ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );        //ADC1,ADC通道,采样时间为239.5周期                                      
  
        ADC_SoftwareStartConvCmd(ADC1, ENABLE);                //使能指定的ADC1的软件转换启动功能       
         
        while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

        return ADC_GetConversionValue(ADC1);        //返回最近一次ADC1规则组的转换结果
}

u16 Get_Adc_Average(u8 ch,u8 times)
{
        u32 temp_val=0;
        u8 t;
        for(t=0;t<times;t++)
        {
                temp_val+=Get_Adc(ch);
                delay_ms(5);
        }
        return temp_val/times;
}          

dac.c

#include "dac.h"
void Dac1_Init(void)
{
  
        GPIO_InitTypeDef GPIO_InitStructure;
        DAC_InitTypeDef DAC_InitType;

        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );          //使能PORTA通道时钟
           RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE );          //使能DAC通道时钟

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;                                 // 端口配置
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;                  //模拟输入
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_Init(GPIOA, &GPIO_InitStructure);
        GPIO_SetBits(GPIOA,GPIO_Pin_4)        ;//PA.4 输出高
                                       
        DAC_InitType.DAC_Trigger=DAC_Trigger_None;        //不使用触发功能 TEN1=0
        DAC_InitType.DAC_WaveGeneration=DAC_WaveGeneration_None;//不使用波形发生
        DAC_InitType.DAC_LFSRUnmask_TriangleAmplitude=DAC_LFSRUnmask_Bit0;//屏蔽、幅值设置
        DAC_InitType.DAC_OutputBuffer=DAC_OutputBuffer_Disable ;        //DAC1输出缓存关闭 BOFF1=1
    DAC_Init(DAC_Channel_1,&DAC_InitType);         //初始化DAC通道1

        DAC_Cmd(DAC_Channel_1, ENABLE);  //使能DAC1
  
    DAC_SetChannel1Data(DAC_Align_12b_R, 0);  //12位右对齐数据格式,设置DAC值

}

//设置通道1输出电压
//vol:0~3300,代表0~3.3V
void Dac1_Set_Vol(u16 vol)
{
        float temp=vol;
        temp/=1000;
        temp=temp*4096/3.3;
        DAC_SetChannel1Data(DAC_Align_12b_R,temp);//12位右对齐数据格式设置DAC值
}

主函数

int main(void)
{         
       
         u16 adcx;
         float temp;
         u8 t=0;
         u16 dacval=0;
        u8 key;
        u8 lcd_id[12];                        //存放LCD ID字符串
        delay_init();                     //延时函数初始化          
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);         //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
        uart_init(115200);                 //串口初始化为115200
        LED_Init();                             //LED端口初始化
    LCD_Init();
        Adc_Init();                                  //ADC初始化
        Dac1_Init();                                //DAC初始化
         
       
        POINT_COLOR=RED;
        sprintf((char*)lcd_id,"LCD ID:%04X",lcddev.id);//将LCD ID打印到lcd_id数组。               
    LCD_ShowString(30,40,210,24,24,"STM32 ^_^");
        LCD_ShowString(30,70,200,16,16,"DAC TEST");
       
        LCD_ShowString(30,110,200,16,16,lcd_id);                //显示LCD ID
        LCD_ShowString(30,130,200,12,12,"2017/2/23");             
        LCD_ShowString(60,150,200,16,16,"WK_UP:+  KEY1:-");
       
         
        //显示提示信息
        POINT_COLOR=BLUE;//设置字体为蓝 {MOD}
        LCD_ShowString(60,170,200,16,16,"DAC VAL:");             
        LCD_ShowString(60,190,200,16,16,"DAC VOL:0.000V");             
        LCD_ShowString(60,210,200,16,16,"ADC VOL:0.000V");
       
        DAC_SetChannel1Data(DAC_Align_12b_R, 0);//初始值为0
       
          while(1)
        {                 
               
                t++;
                dacval=3000;
                 DAC_SetChannel1Data(DAC_Align_12b_R, dacval);//设置DAC值       
               
         if(t==10||key==KEY1_PRES||key==WKUP_PRES) //WKUP/KEY1按下了,或者定时时间到了
                {          
                        adcx=DAC_GetDataOutputValue(DAC_Channel_1);//读取前面设置DAC的值
                        LCD_ShowxNum(124,170,adcx,4,16,0);             //显示DAC寄存器值
                        temp=(float)adcx*(3.3/4096);                        //得到DAC电压值
                        adcx=temp;
                        LCD_ShowxNum(124,190,temp,1,16,0);             //显示电压值整数部分
                        temp-=adcx;
                        temp*=1000;
                        LCD_ShowxNum(140,190,temp,3,16,0X80);         //显示电压值的小数部分
                        adcx=Get_Adc_Average(ADC_Channel_1,10);                //得到ADC转换值          
                        temp=(float)adcx*(3.3/4096);                        //得到ADC电压值
                        adcx=temp;
                        LCD_ShowxNum(124,210,temp,1,16,0);             //显示电压值整数部分
                        temp-=adcx;
                        temp*=1000;
                        LCD_ShowxNum(140,210,temp,3,16,0X80);         //显示电压值的小数部分
                        LED0=!LED0;          
                        t=0;
                }            
                delay_ms(10);
               
               

        }
}
[/mw_shl_code]
arao和xiaomi
4楼-- · 2019-10-15 18:25
电子之旅 发表于 2017-2-27 14:11
把你的ADC、DAC代码配置都贴出来看看

代码在下面,就是原子哥的里面例程,但是为什么会出现那个情况呢?
电子之旅
5楼-- · 2019-10-15 19:49
arao和xiaomi 发表于 2017-2-27 14:38
代码在下面,就是原子哥的里面例程,但是为什么会出现那个情况呢?

你的代码没什么问题,有可能是你的单片机电路有问题。
正点原子
6楼-- · 2019-10-16 01:47
 精彩回答 2  元偷偷看……

一周热门 更多>