單片機(jī)匯編程序設(shè)計(jì)之時(shí)間片
曾在某個(gè)論壇上看過(guò)這樣一句話(huà)——給每個(gè)模塊都分配一個(gè)時(shí)間,這樣才能寫(xiě)好匯編。
當(dāng)時(shí)覺(jué)著有些道理,卻不能很深地體會(huì)。
記得剛學(xué)單片機(jī)匯編的時(shí)候,印象最深的莫過(guò)于循環(huán)點(diǎn)亮一排led.。先點(diǎn)亮一個(gè)、延時(shí)、計(jì)數(shù)值不為零則移位(寄存器)點(diǎn)亮下一個(gè)......十分類(lèi)似于c中的
while(n--){
//...led=1;
delay_ms(500);
}
請(qǐng)注意‘延時(shí)’,在這段時(shí)間內(nèi)單片機(jī)不可以做其它事。如果此時(shí)要檢測(cè)一個(gè)按鍵是否按下,按鍵會(huì)顯得十分不靈活。當(dāng)然可以用中斷的方式檢測(cè)按鍵(硬件消抖),然而事實(shí)上,當(dāng)系統(tǒng)比較‘龐大’時(shí),中斷的資源是十分寶貴的,不到急需的時(shí)候最好別使用。
為了讓mcu能在led延時(shí)期間做其他事,可以使用查詢(xún)的方式判斷l(xiāng)ed是否需要切換,比如開(kāi)啟一個(gè)定時(shí)器,判斷計(jì)數(shù)值是否>=規(guī)定的延時(shí)值,相等,則點(diǎn)亮下一個(gè)led;不等,則退出,讓單片機(jī)做其它事(檢測(cè)按鍵)。事實(shí)上,這樣做可以解決問(wèn)題,但同時(shí)帶來(lái)一大堆麻煩:定時(shí)器的數(shù)量可能不夠用(主程序中有多處使用定時(shí)器)、需要設(shè)定一大堆標(biāo)志位....
我想,應(yīng)該這樣做。
1、點(diǎn)亮led和按鍵讀取分別定義成兩個(gè)模塊(本來(lái)就是兩個(gè)模塊:-))
2、在每個(gè)模塊的入口處定義一個(gè)計(jì)數(shù)寄存器。這個(gè)寄存器的計(jì)數(shù)方式及作用是這樣的:
(仍以點(diǎn)亮led為例) 每隔10ms,計(jì)數(shù)寄存器+1。當(dāng)計(jì)數(shù)寄存器的值>=50,計(jì)數(shù)值清零,點(diǎn)亮下一個(gè)led;計(jì)數(shù)值不為零,退出。
3、開(kāi)啟一個(gè)定時(shí)器,并允許中斷,每次50us。
4、最后,定義一個(gè)時(shí)間管理 子程序。使用變量sys_slice,用于從中斷中獲取10ms的計(jì)數(shù)值。
主體思路是這樣。貌似不夠清晰(詞不達(dá)意真痛苦),所以用‘完整’的匯編代碼再敘述一遍。
1、系統(tǒng)要求:循環(huán)點(diǎn)亮led(若干)、檢測(cè)按鍵(有軟件消抖功能)
2、偽代碼
r_sys_slice equ 0x10 ;系統(tǒng)時(shí)間片
r_sw_cyc_cnt equ 0x11 ;按鍵檢測(cè)周期
r_led_cyc_cnt equ 0x12 ;led點(diǎn)亮?xí)r間
org 0
jmp sys_init
org 8 ;定時(shí)器中斷入口
jmp timer0_isev
org 0x30
sys_init:
call ...
...
main: ;主程序
call ck_sw
call flash_led
call sys_time
jmp main
評(píng)論