stm32DAC

STM32 的 DAC 模块(数字/模拟转换模块)是 12 位数字输入,电压输出型的 DAC。 DAC可以配置为 8 位或 12 位模式,也可以与 DMA 控制器配合使用。 DAC 工作在 12 位模式时,数据可以设置成左对齐或右对齐。 DAC 模块有 2 个输出通道,每个通道都有单独的转换器。在双 DAC 模式下, 2 个通道可以独立地进行转换,也可以同时进行转换并同步地更新 2 个通道的输出。

本节实验,我们将利用按键(或 USMART) 控制 STM32 内部 DAC1来输出电压,通过 ADC1的通道1 采集 DAC的输出电压,在 LCD 模块上面显示 ADC 获取到的电压值以及 DAC 的设定输出电压值等信息。 

STM32 的 DAC 模块主要特点有:
① 2 个 DAC 转换器:每个转换器对应 1 个输出通道
② 8 位或者 12 位单调输出
③ 12 位模式下数据左对齐或者右对齐
④ 同步更新功能
⑤ 噪声波形生成
⑥ 三角波形生成
⑦ 双 DAC 通道同时或者分别转换
⑧ 每个通道都有 DMA 功能


当 DAC 的参考电压为 Vref+的时候(对 STM32F103RC 来说就是 3.3V), DAC 的输出电压是线性的从 0~Vref+, 12 位模式下 DAC 输出电压与 Vref+以及 DORx 的计算公式如下:
                  DACx 输出电压=Vref*( DORx/4095)

我的ADC和DAC的理解是:

ADC可以采集电压把电压转换为数字,DAC可以把数字转换为电压的形式并输出


配置步骤:

1)开启 PA 口时钟,设置 PA4 为模拟输入。

STM32F103RCT6 的 DAC 通道 1 在 PA4 上,所以,我们先要使能 PORTA 的时钟,然后设置 PA4 为模拟输入。 DAC 本身是输出,但是为什么端口要设置为模拟输入模式呢?因为一但使能 DACx 通道之后,相应的 GPIO 引脚( PA4 或者 PA5)会自动与 DAC 的模拟输出相连,设置为输入,是为了避免额外的干扰。
2)使能 DAC1 时钟。
3) 初始化 DAC,设置 DAC 的工作模式。
4) 使能 DAC 转换通道
5) 设置 DAC 的输出值
-----------------------------------------------------------------------------
typedef struct { 
                    u32 DAC_Trigger;  //触发选择   
                    u32 DAC_WaveGeneration;    //波形发生   
                    u32 DAC_LFSRUnmask_TriangleAmplitude; //幅值选择 
                    u32 DAC_OutputBuffer;  //输出缓存控制 
                 }DAC_InitTypeDef;
-----------------------------------------------------------------------------------
dac.c
[html]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. #include "dac.h"  
  2. void Dac1_Init(void)  
  3. {  
  4.     GPIO_InitTypeDef GPIO_ist;  
  5.     DAC_InitTypeDef DAC_ist;  
  6.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );//使能 PA 时钟  
  7.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE ); //使能 DAC 时钟  
  8.     GPIO_ist.GPIO_Pin=GPIO_Pin_4;  
  9.     GPIO_ist.GPIO_Mode=GPIO_Mode_AIN;//模拟输入  
  10.     GPIO_ist.GPIO_Speed=GPIO_Speed_50MHz;  
  11.     GPIO_Init(GPIOA,&GPIO_ist);  
  12.     GPIO_SetBits(GPIOA,GPIO_Pin_4);//PA.4 输出高  
  13.       
  14.     DAC_ist.DAC_Trigger=DAC_Trigger_None; //不使用触发功能  
  15.     DAC_ist.DAC_WaveGeneration=DAC_WaveGeneration_None;//不使用波形发生  
  16.     DAC_ist.DAC_LFSRUnmask_TriangleAmplitude=DAC_LFSRUnmask_Bit0;  
  17.     DAC_ist.DAC_OutputBuffer=DAC_OutputBuffer_Disable ;//关DAC1输出缓存  
  18.     DAC_Init(DAC_Channel_1,&DAC_ist);  //初始化 DAC 通道 1  
  19.       
  20.     DAC_Cmd(DAC_Channel_1,ENABLE);//使能 DAC1  
  21.     DAC_SetChannel1Data(DAC_Align_12b_R,0);//12位右对齐,设置DAC初始值  
  22. }  
  23.   
  24. //设置通道 1 输出电压  
  25. //vol:0~3300,代表 0~3.3V  
  26. void Dac1_Set_Vol(u16 vol)  
  27. {  
  28.     float temp=vol;  
  29.     temp/=1000;  
  30.     temp=temp*4096/3.3;  
  31.     DAC_SetChannel1Data(DAC_Align_12b_R,temp);//12位右对齐设置DAC值  
  32. }  
  33. u16 Dac1_Get_Vol(void)  
  34. {  
  35.     return DAC_GetDataOutputValue(DAC_Channel_1);  
  36. }  


dac.h
[html]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. #ifndef _DAC_H  
  2. #define _DAC_H  
  3. #include "sys.h"  
  4. void Dac1_Init(void);  
  5. u16 Dac1_Get_Vol(void);  
  6. void Dac1_Set_Vol(u16 vol);  
  7. #endif  

主函数。。通过按键来调节DAC输出电压的强度,WKUP增强,KEY0减弱(也可以通过USMART组件设置)
[html]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. #include "led.h"  
  2. #include "delay.h"  
  3. #include "sys.h"  
  4. #include "usart.h"  
  5. #include "lcd.h"  
  6. #include "adc.h"  
  7. #include "dac.h"  
  8. #include "key.h"  
  9. void init()  
  10. {  
  11.     delay_init();            //延时函数初始化      
  12.     uart_init(9600);        //串口初始化为9600  
  13.     LED_Init();             //初始化与LED连接的硬件接口  
  14.     LCD_Init();  
  15.     KEY_Init();  
  16.     Adc_Init();  
  17.     Dac1_Init();  
  18.       
  19.     POINT_COLOR=RED;//设置字体为红色  
  20.     LCD_ShowString(60,40,200,24,24,"DAC Test ^-^");  
  21.     LCD_ShowString(60,70,200,16,16,"Bigggg difficulty");  
  22.     LCD_ShowString(60,90,200,16,16,"2015/1/25");  
  23.     LCD_ShowString(60,110,200,16,16,"By--Mr yh");  
  24.     LCD_ShowString(60,130,200,16,16,"WK_UP:+ KEY0:-");  
  25.     //显示提示信息  
  26.     POINT_COLOR=BLUE;//设置字体为蓝色  
  27.     LCD_ShowString(60,150,200,16,16,"DAC VAL:");  
  28.     LCD_ShowString(60,170,200,16,16,"DAC VOL:0.000V");   
  29.     LCD_ShowString(60,190,200,16,16,"ADC VOL:0.000V");  
  30.     Dac1_Set_Vol(330);  
  31. }  
  32. int main(void)  
  33. {  
  34.       
  35.     u16  adcnum=0;  
  36.     u16 dacnum=0;  
  37.     float tem;  
  38.     u8 key,t=0;  
  39.     init();  
  40.     while(1)  
  41.     {  
  42.         t++;  
  43.         key=KEY_Scan(0);  
  44.         if(key==WK_UP_PRES)  
  45.         {  
  46.             if(dacnum<4000)dacnum+=200;  
  47.             DAC_SetChannel1Data(DAC_Align_12b_R,dacnum);  
  48.         }  
  49.         else if(key==KEY0_PRES)  
  50.         {  
  51.             if(dacnum>200)dacnum-=200;  
  52.             else dacnum=0;  
  53.             DAC_SetChannel1Data(DAC_Align_12b_R,dacnum);  
  54.         }  
  55.           
  56.         if(t==10||key==WK_UP_PRES||key==KEY0_PRES)  
  57.         {  
  58.             adcnum=Dac1_Get_Vol();dacnum=adcnum;  
  59.             LCD_ShowxNum(124,150,adcnum,4,16,0);//显示DAC寄存器值  
  60.             tem=(float)adcnum*(3.3/4096); //得到DAC电压值  
  61.             adcnum=tem;  
  62.             LCD_ShowxNum(124,170,tem,1,16,0);//显示电压值整数部分  
  63.             tem-=adcnum;tem*=1000;  
  64.             LCD_ShowxNum(140,170,tem,3,16,0X80);//显示电压值的小数部分  
  65.             adcnum=Get_Adc_Average(ADC_Channel_1,10);  
  66.             tem=(float)adcnum*(3.3/4096);  
  67.             adcnum=tem;  
  68.             LCD_ShowxNum(124,190,tem,1,16,0);  
  69.             tem-=adcnum;tem*=1000;  
  70.             LCD_ShowxNum(140,190,tem,3,16,0X80);  
  71.             t=0;  
  72.             LED0=!LED0;  
  73.         }  
  74.         delay_ms(10);  
  75.     }  
  76. }  



最后在通过杜邦线将PA1和PA4连起来就好了。。至于为什么要连这两个口 看图


ADC1 和DAC1 分别挂在了 PA1和PA4上,所以我们想要通过ADC1采集DAC1输出的电压,就要将这两点连起来


STM32_DAC学习(固定值_三角波) 百度搜索

==================================================================================================

(STM32)使用DAC输出WAVE音频波形
 
    

笔记:

本想使用PWM输出音频的,但无论怎么样调试,PWM的音频的频率总不对。后来,改用DAC了。

配置:

芯片:STM32F103VET

DAC:DAC通道2(8位右对齐)、定时器TIM7中断更改DAC值

WAVE数据:以const形式放于芯片上(8kHz采样、8bit、单声道)

弯路:

(1)使用TIM7控制DAC输出,用TRIG方式,通过DMA2通道传送数据到DAC写寄存器 => 没有任何波形输出

(2)以为不需要使用与 DAC通道 一致的定时器作驱动,故使用了 TIM3 。使TIM3工作于8K的工作频率,并允许IT_UPDATE,在TIM3定时器中断中更改DAC的值,不使用DMA2通道 => 有波形信号输出,但频率太高,1秒的音频在约50ms内输出完毕

(3)不使用TIM3,使用与DAC通道2相应的TIM7。将TIM7配置为8K工作频率,允许IT_UPDATE,在TIM7定时器中断中更改DAC的值 ,可正常驱动DAC。估计弯路(2)是由于TIM7的工作频率没有配置,导致DAC频率不正常所致,但未证实

(4)使用步骤(3)的定时器,改用DMA传送数据 => DAC无法正常输出音频

例程:

void DACInit(void)
{
	DAC_InitTypeDef   DAC_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	 
	// 1. 配置TIM7
	TIM_DeInit(TIM7);
	/* Time base configuration */
	TIM_TimeBaseStructure.TIM_Period = GetARRValue(8000);        // 重置周期
	TIM_TimeBaseStructure.TIM_Prescaler = 0;                // 分频
	TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;        // 时钟分割
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Down;        // 计数模式
	TIM_TimeBaseInit(TIM7, &TIM_TimeBaseStructure);
	// 使能TIM7更新中断 
	TIM_ITConfig(TIM7, TIM_IT_Update, ENABLE);
	 
	// 2. 配置DAC
	DAC_DeInit();
	/* DAC channel1 Configuration */
	DAC_InitStructure.DAC_Trigger = DAC_Trigger_None;        // 
	DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None;
	DAC_InitStructure.DAC_LFSRUnmask_TriangleAmplitude = DAC_LFSRUnmask_Bits8_0;
	DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Enable;
	// DA通道2初始化
	DAC_Init(DAC_Channel_2, &DAC_InitStructure);
	/* Enable DAC Channel1: Once the DAC channel1 is enabled, PA.04 is 
	 automatically connected to the DAC converter. */
	DAC_Cmd(DAC_Channel_2, ENABLE);
	 
	// 3. 启动TIM7
	TIM_Cmd(TIM7, ENABLE);
}
 
 
void TIM7_IRQHandler(void)
{
	INT16U tmpCap;
	 
	if (TIM_GetITStatus(TIM7, TIM_IT_Update) != RESET)
	{
		if( wavecount < wave0Length )
		{
			tmpCap = wave0[wavecount];
			wavecount++;
			/* Set DAC Channel1 DHR register */
			DAC_SetChannel2Data(DAC_Align_8b_R,tmpCap);                        
		}
		else
		{
			// 完成传输,关闭中断
			wavecount = 0;
			TIM_ITConfig(TIM7, TIM_IT_Update, DISABLE);
			TIM_Cmd(TIM7, DISABLE);
			 // 需要关闭DAC,不然在没有声音的时候会有杂音
			DAC_Cmd(DAC_Channel_2, DISABLE);
		}
	}
 
	/* Clear TIM6 update interrupt */
	TIM_ClearITPendingBit(TIM7, TIM_IT_Update);
}
  
// 根据采样率获得定时器自动
// 摘自waveplayer.c
INT16U GetARRValue(INT16U sample)
{
	INT16U arrValue;
	/* 更新OCA值以符合.WAV文件采样率 */
	switch (sample)
	{
	case SAMPLE_RATE_8000 :
	    arrValue = (uint16_t)(/8000);
	    break; /* 8KHz = 2x36MHz / 9000 */
	case SAMPLE_RATE_11025:
	    arrValue = (uint16_t)(/11025);
	    break; /* 11.025KHz = 2x36MHz / 6531 */
	case SAMPLE_RATE_16000:
	     arrValue = (uint16_t)(/16000);
	    break; /* 16KHz = 2x36MHz / 4500 */
	case SAMPLE_RATE_22050:
	    arrValue = (uint16_t)(/22050);
	    break; /* 22.05KHz = 2x36MHz / 2365 */
	case SAMPLE_RATE_44100:
	    arrValue = (uint16_t)(/44100);
	    break; /* 44.1KHz = 2x36MHz / 1633 */
	case SAMPLE_RATE_48000:
	    arrValue = (uint16_t)(/48000);
	    break; /* 48KHz = 2x36MHz / 1500 */
	default:
	    arrValue = 0;
	    break;
	}
	return arrValue;
}


  • 1
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值