在做数字开关电源开发过程中使用最多的就是PWM功能和ADC采样功能。ADC采样时采样的时间点很重要,必须在PWM输出高电平的时候取采样,这样采样出来的数据才是最准确的。在STM8单片机中,可以通过定时器的TRG信号去触发ADC采样,这样就可以将PWM波形的输出和ADC采样结合在一起了。
下面就演示一下,如何在输出PWM波的过程中触发ADC采样。
首先看ADC的初始化代码。
#include "adc.h"
#include "main.h"
#include "led.h"
u16 DATAH = 0; //ADC转换值高8位
u16 DATAL = 0; //ADC转换值低8位
_Bool ADC_flag = 0; //ADC转换成功标志
u16 adc_cnt = 0;
//AD通道引脚初始化
void ADC_GPIO_Init( void )
{
PD_DDR &= ~( 1 << 2 ); //PD2 设置为输入 AIN3
PD_CR1 &= ~( 1 << 2 ); //PD2 设置为悬空输入
PD_DDR &= ~( 1 << 3 ); //PD3 设置为输入 AIN4
PD_CR1 &= ~( 1 << 3 ); //PD3 设置为悬空输入
}
//ch 为单片机的ADC 通道
//ADC输入通道初始化入口参数表示通道选择
void ADC_CH_Init( u8 ch )
{
char l = 0;
ADC_CR1 = 0x00; //fADC = fMASTER/2, 8Mhz 单次转换,禁止转换
ADC_CR2 = 0x00; //默认左对齐 读数据时先读高在读低
ADC_CR2 |= ( 1 << 6 ); //外部触发使能
ADC_CSR |= ch; //控制状态寄存器 选择要 AD输入通道 如:PD2(AIN3)
ADC_TDRL = ( 1 << ch ); //禁止相应通道 施密特触发功能 1左移ch位
ADC_CR1 |= 0x01; //使能ADC并开始转换
ADC_CSR |= ( 1 << 5 ); //EOCIE 使能转换结束中断 EOC中断使能
for( l = 0; l < 100; l++ ); //延时,保证ADC模块的上电完成 至少7us
ADC_CR1 = ADC_CR1 | 0x01; //再次将CR1寄存器的最低位置1 使能ADC 并开始转换
}
u16 value = 0;
//AD中断服务函数 中断号22
#pragma vector = 24 // IAR中的中断号,要在STVD中的中断号上加2
__interrupt void ADC_Handle( void )
{
ADC_CSR &= ~0x80; // 转换结束标志位清零 EOC
LED = 1;
//默认左对齐 读数据时先读高高8位 再读低8位
DATAH = ADC_DRH; // 读出ADC结果的高8位
DATAL = ADC_DRL; // 读出ADC结果的低8位
ADC_flag = 1; // ADC中断标志 置1
value = ( DATAH << 2 ) + DATAL ; //得到十位精度的数据 0--1024
adc_cnt++;
LED = 0;
}
ADC的初始化代码,比一般ADC的初始化代码只多了一条语句
ADC_CR2 |= ( 1 << 6 ); //外部触发使能
用来开启ADC的外部触发功能。
这里需要使能外部触发转换功能,设置外部触发事件为内部定时器1 TRG事件。不过EXTSEL位默认值就是00,所以这个位可以不用设置,只需要将EXTTRIG位设置为1。
下面初始化定时器1
#define FRE 1000 //frequency 频率 4M / 1000 = 4K
#define DUTY 500
void PWM_GPIO_Init( void )
{
PC_DDR |= ( 1 << 7 ); //PC7 推挽输出
PC_CR1 |= ( 1 << 7 );
PC_ODR_ODR7 = 0;
}
//使用定时器输出PWM触发ADC采样
//定时器1初始化
void TIM1_Init( void )
{
PWM_GPIO_Init();
TIM1_CCMR2 = ( 6 << 4 ) | ( 1 << 3 ) | ( 1 << 2 ); //TIM1 CH2 输出模式 PWM1
TIM1_CCER1 |= 0x10; //CC2为输出
TIM1_PSCRH = 0x00;
TIM1_PSCRL = 0x03; //16M/(1+3)=4 M
TIM1_ARRH = FRE >> 8; //设定自动重装载值高8位
TIM1_ARRL = FRE; //设置自动重装载值低8位
TIM1_CCR2H = DUTY >> 8; //捕获比较寄存器高8位
TIM1_CCR2L = DUTY; //捕获比较寄存器低8位 占空比值
TIM1_BKR = 0x80; //刹车寄存器 使能OC1输出 定时器不工作时输出无效电平
TIM1_CR2 |= ( 2 << 4 ); //使能信号,用于触发输出(TRGO)
TIM1_CR1 |= ( 0 << 5 ); //对齐模式
TIM1_CR1 |= 0x01; //允许定时器中断
}
定时器1的初始化代码也比正常输出PWM功能时多了一行代码。
TIM1_CR2 |= ( 2 << 4 ); //使能信号,用于触发输出(TRGO)
这里需要设置CR2寄存器中的MMS位。用来开启定时器的输出触发功能。
这样分别在ADC初始化和PWM初始化的代码中增加对CR2寄存器的设置,就可通过输出的PWM波来触发ADC的采样了。
最后在主函数中分别初始化ADC和定时器。
void main( void )
{
__asm( "sim" ); //禁止中断
SysClkInit();
delay_init( 16 );
LED_GPIO_Init();
ADC_GPIO_Init();
ADC_CH_Init(3);
TIM1_Init();
__asm( "rim" ); //开启中断
while( 1 )
{
}
}
接下来运行代码,查看运行结果,这里要注意一点,STM8输出PWM波时,需要设置选项字,如果选项字中TIM1_CH1和TIM1_CH2的功能未开启时,在输出引脚上是观察不到波形的。
在ADC中断中通过一个计数器来统计进入ADC中断的次数,在进入中断时将LED电平置高,在中断代码执行完成之后将LED电平置低。在示波器上观察PWM的波形和LED的波形。
黄色波形未PWM输出波形,绿色波形未LED输出波形。在每个PWM波的上升沿就触发一次ADC采样。
由于当前PWM输出是TIM1_CH2通道,所以在设置寄存器MMS位时,也可以将值设置为5.
经过实际测试,MMS位设置为2、3、5时程序都可以正常工作。
TIM1_CR2 |= ( 2 << 4 ); //使能信号,用于触发输出(TRGO)
TIM1_CR2 |= ( 3 << 4 ); //使能信号,用于触发输出(TRGO)
TIM1_CR2 |= ( 5 << 4 ); //使能信号,用于触发输出(TRGO)
通过改变CR1寄存器数据对齐模式和 CR2寄存器触发模式的值,可以改变ADC触发时间位置。
设置MMS值为3,CMS值为0.
输出波形如下:
设置MMS值为3,CMS值为2.
输出波形如下:
设置MMS值为3,CMS值为3.
输出波形如下:
设置MMS值为2,CMS值为3.
输出波形如下:
设置MMS值为2,CMS值为2.
输出波形如下:
设置MMS值为2,CMS值为0.
输出波形如下:
通过上面几组波形对比可以发现,ADC触发的时刻只能是在上升沿或者电平中点位置。不能后设置在任意位置触发,因为PWM的波形计数模式只有向上、向下或者中心对称模式。在向上或者向下计数模式中,中断一般只会触发一次,在中心对称模式计数时,中断会被触发两次,而中断触发的位置刚好是电平信号的中点位置。
通过自己的摸索终于将PWM触发ADC采样功能实现了,如果哪位朋友有更好的方法来实现,欢迎留言讨论。
版权说明 : 本文为转载文章, 版权归原作者所有 版权申明
原文链接 : https://hxydj.blog.csdn.net/article/details/121907350
内容来源于网络,如有侵权,请联系作者删除!