上一節(jié),我們講了在CAP腳上計數(shù),這一節(jié),我們用捕獲功能測量CAP引腳上的頻率。原理是獲取兩次下降沿的時間間隔,這個時間間隔即是脈沖信號的周期。
新建一個工程,結(jié)構(gòu)如下圖所示:
在timer.h文件中,加入捕獲測頻的初始化函數(shù)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()函數(shù)的定義
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; //復(fù)位定時器(bit1:寫1復(fù)位)
LPC_TMR16B0->PR = SystemCoreClock/100000-1; //使10微妙TC+1
LPC_TMR16B0->IR = 0x1F; //CAP0中斷復(fù)位
LPC_TMR16B0->CCR = 0x06; // 下降沿中斷
LPC_TMR16B0->MR0 = 0XFFFF; // 匹配值
LPC_TMR16B0->MCR = 0X01; // 與MR0匹配產(chǎn)生中斷
LPC_TMR16B0->TCR = 0x01;
NVIC_EnableIRQ(TIMER_16_0_IRQn); // 使能CT16B0中斷
}
以上語句的說明,基本上都在前面幾個章節(jié)介紹過了。
第12和13行的配置,是一個“不得不使用的技巧”。原因是LPC1114的定時器沒有溢出中斷,即當(dāng)定時器值遞增到最大值,再回到0計數(shù),不會產(chǎn)生中斷。所以,我們在這里,給匹配寄存器MR0寫入定時器的最大值,然后設(shè)置定時器與MR0匹配后產(chǎn)生中斷,即可實現(xiàn)溢出中斷的效果。在這里產(chǎn)生溢出中斷,是為了讓引腳上沒有脈沖信號的時候頻率為0。假如沒有溢出中斷,你做的車速表將停留在剎車一瞬間的車速不歸0,這是一件多么悲催的作品!
在main.c中,輸入以下代碼:
#include “l(fā)pc11xx.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腳設(shè)置為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)); // 確定時鐘源更新后向下執(zhí)行 } void TIMER16_0_IRQHandler(void) { if((LPC_TMR16B0->IR&0x10)==0x10) // 如果是CAP引起的中斷 { temp = LPC_TMR16B0->CR0; LPC_TMR16B0->TC = 0; freq = 100000/temp; // 把單位轉(zhuǎn)換成赫茲 } 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函數(shù)第一條語句開始看起。 第42行,打開串口并設(shè)置串口波特率為9600。 第43行,初始化“16位定時器0”的CAP功能。 第44行,使能CLKOUT_EN引腳,并輸出46Hz的頻率信號。(關(guān)于CLKOUT功能和此函數(shù)的介紹,請看第一章,這里我們只是用它來產(chǎn)生一個我們要測量的頻率信號。) 第45~50行,間隔100毫秒,發(fā)送串口一次測量到的頻率,打開串口調(diào)試助手,選擇好串口號和波特率,選擇為16進制接收。把開發(fā)板上的P0.1腳,即CLKOUT引腳和P0.2腳,即CAP引腳相連,即可在串口調(diào)試助手上看到測量出的頻率值。 第26~39行是“16位定時器0”的中斷服務(wù)函數(shù)。 第28行,判斷是否是CAP引起的中斷。 第30行,讀取CR0寄存器的值。當(dāng)CAP引腳上有下降沿中斷產(chǎn)生,CR0就會自動獲取當(dāng)前定時器的值,存到里面。 第31行,把定時器的當(dāng)前值清0。 第32行,計算頻率。在初始化的時候,我們把定時器的TC值設(shè)置為10毫秒增1,每次發(fā)生下降沿中斷,就會讀取定時器的值,再清0,所以讀取出來的值就是周期,周期=1/頻率,現(xiàn)在的周期值單位是10毫秒,所以換成赫茲以后,就是100000/temp。