用TB捕獲脈寬想用TBCCRO捕獲脈沖低電平寬度,思路是:tbccr0捕獲到下降沿中斷,則記下tbccro的值,并改為上升沿觸發(fā);捕獲到上升沿中斷,則記下tbccro的值,改為下降沿觸發(fā)。 硬件:?jiǎn)纹瑱C(jī):MSP430F149 晶振:32K,8M 輸入信號(hào):通過(guò)無(wú)線接收到低電平10ms,高電平7.5ms, 輸入口:P4.0(TB0) 要求:捕獲低電平的脈寬軟件: 1. 初步思路:通過(guò)定時(shí)器TBCCR0作為捕獲模塊對(duì)外部輸入信號(hào)進(jìn)行捕獲:先設(shè)為下降沿捕獲,如果捕獲到,馬上修改為上升沿捕獲,并馬上TBR清零開(kāi)始計(jì)數(shù);如果不過(guò)到上升沿,馬細(xì)奈陸笛,并把TBCCR0的數(shù)據(jù)記下來(lái),此即為脈沖低電平寬度。 2. 使用TI公司的c語(yǔ)言例程稍做修改程序可以運(yùn)行。 3. 出現(xiàn)問(wèn)題:程序能捕獲到上升下降沿,并且捕獲到的width總是忽大忽小,毫無(wú)規(guī)律。 程序改來(lái)改去毫無(wú)進(jìn)展,頭開(kāi)始慢慢大了~~~ 4.師兄過(guò)來(lái)看看說(shuō),怎么沒(méi)有開(kāi)晶振啊?我說(shuō)沒(méi)用到8M的,也就沒(méi)專門開(kāi)晶振~不過(guò)既然說(shuō)起,要不干脆換個(gè)晶振試試,于是加了段程序,并把TB改成用MCLK(8M): void InitSys() { unsigned int iq0; //使用XT2振蕩器 BCSCTL1&=~XT2OFF; //打開(kāi)XT2振蕩器 do { IFG1 &= ~OFIFG; // 清除振蕩器失效標(biāo)志 for (iq0 = 0xFF; iq0 > 0; iq0--); // 延時(shí),等待XT2起振 } while ((IFG1 & OFIFG) != 0); // 判斷XT2是否起振 BCSCTL2 =SELM_2+SELS; //選擇MCLK=SMCLK為XT2 } 奇怪的事情發(fā)生了,程序一直卡在此處的延時(shí)程序語(yǔ)句中,怎么回事,難道晶振打不開(kāi)?突然想到查查硬件,才發(fā)現(xiàn)8M晶振一個(gè)管腳松了!颍ぁ鵃$…… 焊好8M晶振后,程序可以繼續(xù)運(yùn)行了. 5.又發(fā)現(xiàn)問(wèn)題:雖然程序可以正常運(yùn)行了,width采集到的數(shù)據(jù)也不再忽大忽小了,開(kāi)始很規(guī)律的在14500左右變化,可一算,14500*(1/8000000)=1.8125ms,跟輸入信號(hào)脈寬不一致,用示波器測(cè)輸入端確實(shí)是10ms???~~ 6.突然想10ms的數(shù)據(jù)如果采集到應(yīng)該為10ms/(1/8000000)=80000,這個(gè)數(shù)據(jù)早就超過(guò)TBR的值了。那TBR溢出后就會(huì)從0重新開(kāi)始計(jì)時(shí),那顯示的數(shù)據(jù)就應(yīng)該正好是65500+14500=80000!也就是說(shuō),我得到的數(shù)據(jù)是對(duì)的,只是沒(méi)有考慮TBR溢出的情況! 7.既然問(wèn)題發(fā)現(xiàn)了,就好辦了~TB的TBCTL不是可以時(shí)鐘分頻功能嗎?設(shè)置1/8分頻后,時(shí)鐘為1M,這樣10ms的脈寬應(yīng)該得到10ms/(1000000)=10000! 程序修改好后一運(yùn)行,果然~阿彌托佛 源程序如下: #include<msp430x14x.h> unsigned int width[10]={0,0,0,0,0,0,0,0,0,0}; unsigned int i=0; void main() { WDTCTL=WDTPW+WDTHOLD; //關(guān)閉看門狗 P4SEL =BIT0; //P4.0作為捕獲模塊功能的輸入端輸入方波 //-------開(kāi)晶振XT2--------- BCSCTL1&=~XT2OFF; //打開(kāi)XT2振蕩器 do { IFG1 &= ~OFIFG; // 清除振蕩器失效標(biāo)志 for (i=256;i>0;i--); // 延時(shí),等待XT2起振 } while ((IFG1 & OFIFG) != 0); // 判斷XT2是否起振 BCSCTL2 =SELM_2+SELS; //選擇MCLK=SMCLK為XT2 //----------------------------- TBCCTL0&=~(CCIS1+CCIS0); // 捕獲源為P4.0,即CCI0A(也是CCI0B) TBCCTL0 =CM_2+SCS+CAP; //下降沿捕獲,同步捕獲,工作在捕獲模式 TBCCTL0 =CCIE; //允許捕獲比較模塊提出中斷請(qǐng)求 TBCTL =ID_3; TBCTL =TBSSEL_2; //選擇時(shí)鐘MCLK TBCTL =TBCLR; //定時(shí)器清零, //定時(shí)器開(kāi)始計(jì)數(shù)(連續(xù)計(jì)數(shù)模式0~0xFFFF) TBCTL =MC_2; _EINT(); while(1); } //―――――定時(shí)器TB的CCR0的中斷:用于檢測(cè)脈沖上升與下降沿―――― #pragma vector=TIMERB0_VECTOR __interrupt void TimerB0(void) { if(TBCCTL0&CM1) //捕獲到下降沿 { TBCTL =TBCLR; TBCCTL0=(TBCCTL0&(~CM1)) CM0; //改為上升沿捕獲:CM1置零,CM0置一 } else if(TBCCTL0&CM0) //捕獲到上升沿 { width[i++]=TBCCR0; //記錄下結(jié)束時(shí)間 TBCCTL0=(TBCCTL0&(~CM0)) CM1; //改為下降沿捕獲:CM0置零,CM1置一 if(i==10) i=0; } } 教訓(xùn): 1.程序的模塊化設(shè)計(jì)很重要。每次寫(xiě)程序,最好遵循如下規(guī)矩: 關(guān)看門狗;WDTCTL=WDTPW+WDTHOLD; 開(kāi)晶振:都把ACLK= XT1(32k),MCLK=SMCLK=XT2(8M);并且能用8M最好用8M,這樣比較準(zhǔn)確。 晶振的檢測(cè)方法:XT2可以通過(guò)程序中的掃描標(biāo)志位實(shí)現(xiàn);蛘咴O(shè)置P1.4(SMCLK),P2.0(ACLK),然后用示波器檢查 主程序:使用自己寫(xiě)的模板。 2.如果在一個(gè)問(wèn)題上卡住了,就不斷細(xì)化深入下去,直到觸到其本質(zhì),就看你能把這個(gè)問(wèn)題細(xì)化到什么程度! 3.任何數(shù)字或信息都有他隱含的本質(zhì)信息,都能直接或間接反映其本質(zhì)。就看你能否抓住這個(gè)數(shù)字,想到他對(duì)本質(zhì)的反映。 |
|