今晚真是個不眠之夜啊,很久沒用C++對單片機編程了,而且以前一直在用IAR For AVR編程,現在用IAR 8051來寫STC的程序,一直都是開最高優化級別,沒發現問題過.今天發現了一個苦B的問題.我在全局定義了一個bool變量,在定時器中斷函數里改變它的值,然后主程序判斷這個值來確定是否進入空閑模式.可是一開優化,馬上變味了.打開反編譯軟件查看生成的代碼,一個while()的循環竟然變成了死循環,無論怎么改變C語法都無法改變生成的結果.想到以前在寫Microsoft C++程序的時候用到過一個關鍵字:volatile , 意思是容易被意想不到改變的變量.添加在變量的聲明以后,while循環回來了.IAR編譯器我真是越來越喜歡了.雖然有時程序會莫名其妙的走樣,可是它的代碼優化方面還是無可比擬的.
C++原程序:volatile bool bStart;
while(bStart == 0)
PCON_bit.IDL = 1;// 進入空閑模式
意思是如果沒有程序改變bStart這個變量的話,CPU就繼續休息.如果有程序改變的話就退出循環執行一次主函數.
在沒加volatile時生成的匯編代碼:
Q022C: ORL PCON,#01H
SJMP Q022C
壞了,while 循環影子都沒了
聲明volatile后的代碼:
Q0225: MOV R0,#56H
MOV A,@R0
JZ Q022D
LJMP Q0196
;===========================================
Q022D: ORL PCON,#01H
SJMP Q0225
這回它判斷@R0里的變量值了,56H里的值就是bStart,可愛的sjmp指令也回來了.
好在以前還學習過Microsoft的編程,IAR編譯器才是真正的標準C編譯器,比那些KEIL,ICC好多了,只要我們寫程序的時候嚴謹些,知道CPU這個時候會干什么, 就可以寫出漂亮的程序.
|