lpc1114通用定时器-捕获功能
扫描二维码
随时随地手机看文章
上一节,我们讲了在CAP脚上计数,这一节,我们用捕获功能测量CAP引脚上的频率。原理是获取两次下降沿的时间间隔,这个时间间隔即是脉冲信号的周期。
新建一个工程,结构如下图所示:
在timer.h文件中,加入捕获测频的初始化函数T16B0_CAP_Init()的声明,如下所示:
#ifndef __NXPLPC11xx_TIME_H__
#define __NXPLPC11xx_TIME_H__
extern void T16B0_init(void);
extern void T16B0_delay_ms(uint16_t ms);
extern void T16B0_delay_us(uint16_t us);
extern void T16B0_cnt_init(void);
extern void T16B0_CAP_Init(void);
#endif
在timer.c文件中,加入T16B0_CAP_Init()函数的定义
void T16B0_CAP_Init(void)
{
LPC_SYSCON->SYSAHBCLKCTRL |= (1<<16); // 使能IOCON时钟
LPC_IOCON->PIO0_2 &= ~0x07;
LPC_IOCON->PIO0_2 |= 0x02; /* CT16B0 CAP0 */
LPC_SYSCON->SYSAHBCLKCTRL &= ~(1<<16); // 禁能IOCON时钟
LPC_SYSCON->SYSAHBCLKCTRL |= (0X1<<7); // 使能TIM16B0时钟
LPC_TMR16B0->TCR = 0x02; //复位定时器(bit1:写1复位)
LPC_TMR16B0->PR = SystemCoreClock/100000-1; //使10微妙TC+1
LPC_TMR16B0->IR = 0x1F; //CAP0中断复位
LPC_TMR16B0->CCR = 0x06; // 下降沿中断
LPC_TMR16B0->MR0 = 0XFFFF; // 匹配值
LPC_TMR16B0->MCR = 0X01; // 与MR0匹配产生中断
LPC_TMR16B0->TCR = 0x01;
NVIC_EnableIRQ(TIMER_16_0_IRQn); // 使能CT16B0中断
}
以上语句的说明,基本上都在前面几个章节介绍过了。
第12和13行的配置,是一个“不得不使用的技巧”。原因是LPC1114的定时器没有溢出中断,即当定时器值递增到最大值,再回到0计数,不会产生中断。所以,我们在这里,给匹配寄存器MR0写入定时器的最大值,然后设置定时器与MR0匹配后产生中断,即可实现溢出中断的效果。在这里产生溢出中断,是为了让引脚上没有脉冲信号的时候频率为0。假如没有溢出中断,你做的车速表将停留在刹车一瞬间的车速不归0,这是一件多么悲催的作品!
在main.c中,输入以下代码:
#include “lpc11xx.h”
#include “timer.h”
#include “uart.h”
uint16_t temp; //
uint16_t freq; //
// 非精确延时
void delay_ms(uint16_t ms)
{
uint16_t i,j;
for(i=0;i<5000;i++)
for(j=0;j } void CLKOUT_EN(uint8_t CLKOUT_DIV) { LPC_SYSCON->PDRUNCFG &= ~(0x1<<6); // 看门狗振荡器时钟上电(bit6) LPC_SYSCON->WDTOSCCTRL = 0X3F; // 0.6M/2*(1+31)=9375赫兹 LPC_SYSCON->SYSAHBCLKCTRL |= (1<<16); // 使能IOCON时钟 LPC_IOCON->PIO0_1=0XD1; // 把P0.1脚设置为CLKOUT引脚 LPC_SYSCON->SYSAHBCLKCTRL &= ~(1<<16); // 禁能IOCON时钟 LPC_SYSCON->CLKOUTDIV = CLKOUT_DIV; LPC_SYSCON->CLKOUTCLKSEL= 0X00000002; // CLKOUT时钟源选择为看门狗时钟 LPC_SYSCON->CLKOUTUEN =0; LPC_SYSCON->CLKOUTUEN =1; while (!(LPC_SYSCON->CLKOUTUEN & 0x01)); // 确定时钟源更新后向下执行 } void TIMER16_0_IRQHandler(void) { if((LPC_TMR16B0->IR&0x10)==0x10) // 如果是CAP引起的中断 { temp = LPC_TMR16B0->CR0; LPC_TMR16B0->TC = 0; freq = 100000/temp; // 把单位转换成赫兹 } else if((LPC_TMR16B0->IR&0X01)==0X01) // 如果是MR0匹配引起的中断,即溢出中断 { freq = 0; } LPC_TMR16B0->IR = 0X1F; // 清中断位 } int main() { UART_init(9600); T16B0_CAP_Init(); CLKOUT_EN(200);// 9375/200=46Hz while(1) { delay_ms(100); UART_send_byte(freq); UART_send_byte(freq>>8); } } 从main函数第一条语句开始看起。 第42行,打开串口并设置串口波特率为9600。 第43行,初始化“16位定时器0”的CAP功能。 第44行,使能CLKOUT_EN引脚,并输出46Hz的频率信号。(关于CLKOUT功能和此函数的介绍,请看第一章,这里我们只是用它来产生一个我们要测量的频率信号。) 第45~50行,间隔100毫秒,发送串口一次测量到的频率,打开串口调试助手,选择好串口号和波特率,选择为16进制接收。把开发板上的P0.1脚,即CLKOUT引脚和P0.2脚,即CAP引脚相连,即可在串口调试助手上看到测量出的频率值。 第26~39行是“16位定时器0”的中断服务函数。 第28行,判断是否是CAP引起的中断。 第30行,读取CR0寄存器的值。当CAP引脚上有下降沿中断产生,CR0就会自动获取当前定时器的值,存到里面。 第31行,把定时器的当前值清0。 第32行,计算频率。在初始化的时候,我们把定时器的TC值设置为10毫秒增1,每次发生下降沿中断,就会读取定时器的值,再清0,所以读取出来的值就是周期,周期=1/频率,现在的周期值单位是10毫秒,所以换成赫兹以后,就是100000/temp。