第二十五章 PWM DAC实验
上一章,我们介绍了STM32自带DAC模块的使用,但不是每个STM32都有DAC模块的,对于那些没有DAC模块的芯片,我们可以通过PWM+RC滤波来实一个PWM DAC。本章我们将向大家介绍如何使用STM32的PWM来设计一个DAC。我们将使用按键(或USMART)控制STM32的PWM输出,从而控制PWM DAC的输出电压,通过ADC1的通道1采集PWM DAC的输出电压,并在LCD模块上面显示ADC获取到的电压值以及PWM DAC的设定输出电压值等信息。本章将分为如下几个部分:
25.1 PWM DAC简介
25.2 硬件设计
25.3 软件设计
25.4 下载验证
25.1 PWM DAC简介 虽然大容量的STM32F103具有内部DAC,但是更多的型号是没有DAC的,不过STM32所有的芯片都有PWM输出,因此,我们可以用PWM+简单的RC滤波来实现DAC输出,从而节省成本。
PWM本质上其实就是是一种周期一定,而高低电平占空比可调的方波。实际电路的典型PWM波形,如图25.1.1所示:
图25.1.1 实际电路典型PWM波形
图25.1.1的PWM波形可以用分段函数表示为式①:
其中:T是单片机中计数脉冲的基本周期,也就是STM32定时器的计数频率的倒数。N是PWM波一个周期的计数脉冲个数,也就是STM32的ARR-1的值。n是PWM波一个周期中高电平的计数脉冲个数,也就是STM32的CCRx的值。VH和VL分别是PWM波的高低电平电压值,k为谐波次数,t为时间。我们将①式展开成傅里叶级数,得到公式②:
从②式可以看出,式中第1个方括弧为直流分量,第2项为1次谐波分量,第3项为大于1次的高次谐波分量。式②中的直流分量与n成线性关系,并随着n从0到N,直流分量从VL到VL+VH之间变化。这正是电压输出的DAC所需要的。因此,如果能把式②中除直流分量外的谐波过滤掉,则可以得到从PWM波到电压输出DAC的转换,即:PWM波可以通过一个低通滤波器进行解调。式②中的第2项的幅度和相角与n有关,频率为1/(NT),其实就是PWM的输出频率。该频率是设计低通滤波器的依据。如果能把1次谐波很好过滤
掉,则高次谐波就应该基本不存在了。
通过上面的了解,我们可以得到PWM DAC的分辨率,计算公式如下:
分辨率=log2(N)
这里假设n的最小变化为1,当N=256的时候,分辨率就是8位。而STM32的定时器都是16位的,可以很容易得到更高的分辨率,分辨率越高,速度就越慢。不过我们在本章要设计的DAC分辨率为8位。
在8位分辨条件下,我们一般要求1次谐波对输出电压的影响不要超过1个位的精度,也就是3.3/256=0.01289V。假设VH为3.3V,VL为0V,那么一次谐波的最大值是2*3.3/π=2.1V,这就要求我们的RC滤波电路提供至少-20lg(2.1/0.01289)=-44dB的衰减。
STM32的定时器最快的计数频率是72Mhz,8为分辨率的时候,PWM频率为72M/256=281.25Khz。如果是1阶RC滤波,则要求截止频率为1.77Khz,如果为2阶RC滤波,则要求截止频率为22.34Khz。
战舰STM32开发板的PWM DAC输出采用二阶RC滤波,该部分原理图如图25.1.2所示:
图25.1.2 PWM DAC二阶RC滤波原理图
二阶RC滤波截止频率计算公式为:
f=1/2πRC
以上公式要求R28*C37=R29*C38=RC。根据这个公式,我们计算出图25.1.2的截止频率为:33.8Khz超过了22.34Khz,这个和我们前面提到的要求有点出入,原因是该电路我们还需要用作PWM DAC音频输出,而音频信号带宽是22.05Khz,为了让音频信号能够通过该低通滤波,同时为了标准化参数选取,所以确定了这样的参数。实测精度在0.5LSB以内。
PWM DAC的原理部分,就为大家介绍到这里。
25.2硬件设计 本章用到的硬件资源有:
1) 指示灯DS0
2) WK_UP和KEY1按键
3) 串口
4) TFTLCD模块
5) ADC
6) PWM DAC
本章,我们使用STM32的TIM4_CH1(PB6)输出PWM,经过二阶RC滤波后,转换为直流输出,实现PWM DAC。同上一章一样,我们通过ADC1的通道1(PA1)读取PWM DAC的输出,并在LCD模块上显示相关数值,通过按键和USMART控制PWM DAC的输出值。我们需要用到ADC采集DAC的输出电压,所以需要在硬件上将PWM DAC和ADC短接起来,PWM DAC部分原理图如图25.2.1所示:
图25.2.1 PWM DAC原理图
在硬件上,我们还需要用跳线帽短接多功能端口的PDC和ADC,如图25.2.2所示:
图25.2.2 硬件连接示意图
25.3 软件设计 找到上一章的工程,然后打开USER文件夹下的工程,打开之前的timer.c文件,在最后添加一个新的函数:TIM4_PWM_Init,该函数代码如下:
//TIM4 CH1 PWM输出设置
//PWM输出初始化
//arr:自动重装值
//psc:时钟预分频数
void TIM4_PWM_Init(u16 arr,u16 psc)
{
RCC->APB1ENR|=1<<2; //TIM4时钟使能
RCC->APB2ENR|=1<<3; //使能PORTB时钟
GPIOB->CRL&=0X00FFFFFF; //PB6输出
GPIOB->CRL|=0X4B000000; //复用功能输出
TIM4->ARR=arr; //设定计数器自动重装值
TIM4->PSC=psc; //预分频器分频设置
TIM4->CCMR1|=7<<4; //CH1 PWM2模式
TIM4->CCMR1|=1<<3; //CH1 预装载使能
TIM4->CCER|=1<<1; //OC1 低电平有效
TIM4->CCER|=1<<0; //OC1输出使能
TIM4->CR1=0x0080; //ARPE使能
TIM4->CR1|=0x01; //使能定时器3
}
该函数用来初始化TIM4_CH1的PWM输出(PB6),其原理同之前介绍的PWM输出一模一样,只是换过一个定时器而已。这里就不细说了。
同时在timer.h里面,修改代码如下:
#ifndef __TIMER_H
#define __TIMER_H
#include "sys.h"
//通过改变TIM3->CCR2的值来改变占空比,从而控制LED0的亮度
#define LED0_PWM_VAL TIM3->CCR2
//TIM4 CH1作为PWM DAC的输出通道
#define PWM_DAC_VAL TIM4->CCR1
void TIM3_Int_Init(u16 arr,u16 psc);
void TIM3_PWM_Init(u16 arr,u16 psc);
void TIM5_Cap_Init(u16 arr,u16 psc);
void TIM4_PWM_Init(u16 arr,u16 psc);
#endif
该部分代码在原有基础上添加了TIM4_PWM_Init函数的声明,并添加PWM_DAC_VAL宏定义,用于改变TIM4_CH1的PWM占空比,从而控制PWM DAC的输出。
接下来我们在test.c里面,修改代码如下:
//设置输出电压
//vol:0~330,代表0~3.3V
void PWM_DAC_Set(u16 vol)
{
float temp=vol;
temp/=100;
temp=temp*256/3.3;
PWM_DAC_VAL=temp;
}
int main(void)
{
u16 adcx; u8 key;
float temp; u8 t=0; u16 pwmval=0;
Stm32_Clock_Init(9); //系统时钟设置
uart_init(72,9600); //串口初始化为9600
delay_init(72); //延时初始化
LED_Init(); //初始化与LED连接的硬件接口
LCD_Init(); //初始化LCD
usmart_dev.init(72); //初始化USMART
KEY_Init(); //按键初始化
Adc_Init(); //ADC初始化
TIM4_PWM_Init(255,0);//TIM4 PWM初始化, Fpwm=72M/256=281.25Khz.
POINT_COLOR=RED; //设置字体为红色
LCD_ShowString(60,50,200,16,16,"WarShip STM32");
LCD_ShowString(60,70,200,16,16,"PWM DAC TEST");
LCD_ShowString(60,90,200,16,16,"ATOM@ALIENTEK");
LCD_ShowString(60,110,200,16,16,"2012/9/8");
LCD_ShowString(60,130,200,16,16,"WK_UP:+ KEY1:-");
//显示提示信息
POINT_COLOR=BLUE;//设置字体为蓝色
LCD_ShowString(60,150,200,16,16,"PWM VAL:");
LCD_ShowString(60,170,200,16,16,"DAC VOL:0.000V");
LCD_ShowString(60,190,200,16,16,"ADC VOL:0.000V");
PWM_DAC_VAL=pwmval;//初始值为0
while(1)
{
t++;
key=KEY_Scan(0);
if(key==4)
{
if(pwmval<250)pwmval+=10;
PWM_DAC_VAL=pwmval; //输出
}else if(key==2)
{
if(pwmval>10)pwmval-=10;
else pwmval=0;
PWM_DAC_VAL=pwmval; //输出
}
if(t==10||key==2||key==4) //WK_UP/KEY1按下了,或者定时时间到了
{
adcx=PWM_DAC_VAL;
LCD_ShowxNum(124,150,adcx,3,16,0); //显示DAC寄存器值
temp=(float)adcx*(3.3/256); //得到DAC电压值
adcx=temp;
LCD_ShowxNum(124,170,temp,1,16,0); //显示电压值整数部分
temp-=adcx; temp*=1000;
LCD_ShowxNum(140,170,temp,3,16,0x80); //显示电压值的小数部分
adcx=Get_Adc_Average(ADC_CH1,20); //得到ADC转换值
temp=(float)adcx*(3.3/4096); //得到ADC电压值
adcx=temp;
LCD_ShowxNum(124,190,temp,1,16,0); //显示电压值整数部分
temp-=adcx; temp*=1000;
LCD_ShowxNum(140,190,temp,3,16,0x80); //显示电压值的小数部分
t=0; LED0=!LED0;
}
delay_ms(10);
}
}
此部分代码,同上一章的基本一样,先对需要用到的模块进行初始化,然后显示一些提示信息,本章我们通过WK_UP和KEY1(也就是上下键)来实现对PWM脉宽的控制,经过RC滤波,最终实现对DAC输出幅值的控制。按下WK_UP增加,按KEY1减小。同时在LCD上面显示TIM4_CCR1寄存器的值、PWM DAC设计输出电压以及ADC采集到的实际输出电压。同时DS0闪烁,提示程序运行状况。
不过此部分代码还有一个PWM_DAC_Set函数,用于USMART调用,从而通过串口控制PWM DAC的输出,所以还需要将PWM_DAC_Set函数加入USMART控制,方法前面已经有详细的介绍了,大家这里自行添加,或者直接查看我们光盘的源码。
25.4 下载验证 在代码编译成功之后,我们通过下载代码到ALIENTEK战舰STM32开发板上,可以看到LCD显示如图25.4.1所示:
图25.4.1 PWM DAC实验测试图
同时伴随DS0的不停闪烁,提示程序在运行。此时,我们通过按WK_UP按键,可以看到输出电压增大,按KEY1则变小。
大家可以试试在USMART调用PWM_DAC_Set函数,来设置PWM DAC的输出电压,如图25.4.2所示:
图25.4.2 通过usmart设置PWM DAC的电压输出