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