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<ms;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。