靈活使用PIC16C57單片機的PROGRAM MEMORY
當您使用Microchip公司的PIC16C57單片機在設計程式時(shí),是否被它的PROGRAM MEMORY需分PAGE使用,而PAGE之設定又影響到goto、call、addwf 2、movwf 2四個(gè)指令之執行結果而困擾不已呢?以下是小弟領(lǐng)悟出來(lái)的一點(diǎn)心得,在此野人獻曝,與大家分享。
PIC16C57單片機之PROGRAM MEMORY共有2 K words,分為四個(gè)PAGE,每個(gè)PAGE有512個(gè)words;goto、call、addwf 2、movwf 2四個(gè)指令之執行,會(huì )改變PROGRAM COUNTER內容,其結果也受STATUS WORD REGISTER f3之bit 6, 5的影響,如圖(一)所示;改變STATUS WORD REGISTER之bit 6, 5以下一律以"PAGE(的)控制"稱(chēng)呼。
除此之外,每個(gè)PAGE分為前半部及後半部,call、addwf 2、movwf 2三個(gè)指令執行結果只會(huì )跳到某一PAGE內的前半部,因為這三個(gè)指令執行後會(huì )使PROGRAM COUNTER的bit 8變?yōu)?,而goto指令則不受限制,可跳到一個(gè)PAGE之前、後半部。
PROGRAM MEMORY位址7FF是reset後第一個(gè)被執行的指令,正常情況下放入goto指令,而且STATUS WORD REGISTER bit 6, 5在 reset後全變?yōu)?,所以這個(gè)goto指令會(huì )使程式跳到PAGE 0去執行。至於要goto到何處?我們先以main 的label來(lái)代表,下文再繼續討論。
結構化的程式撰寫(xiě)是由主程式呼叫許多副程式組成的,為了能善用這四個(gè)PAGE,又不要讓程式在呼叫副程式時(shí)或做goto之前也要同時(shí)注意PAGE的設定值,我對PAGE 0做以下安排:PAGE 0的前半部先存放所有的第一階(即主程式直接呼叫的)副程式,接下來(lái)才存放主程式,上一段所提及的label—main就是安排在這個(gè)地方;整段主程式只能放在PAGE 0內,否則分置於兩個(gè)PAGE內的goto指令前需有不同的PAGE控制,此舉不但麻煩又易出錯,尤其是尚在發(fā)展更改中的程式,更難掌握。
當上述的第一階副程式太大或太多,以致無(wú)法全部放入PAGE 0的前半部時(shí),或是會(huì )造成主程式分跨於PAGE 0與PAGE 1時(shí),就要將部分副程式的"身體"移到別的PAGE中,但"頭"仍需保留在PAGE 0的前半部,"頭"是個(gè)重要的媒介,其任務(wù)就是要把副程式導引到正確的PAGE上執行;而被移走的副程式其"尾"(retlw之前)需加上指令,把PAGE的控制轉回PAGE 0;如此安排,主程式就不用管它所呼叫的副程式"主體"究竟位於哪個(gè)PAGE上了。請參考如下範例:
LIST p=16c57
;****************************************************************
ORG 0 ;page 0前半部
;****************************************************************
init: ;副程式的"頭
bsf STATUS,5 ;page selector point to page 1
goto sub1_1
;------------------------------------------------------------------------------------------------
tx: ;副程式的"頭
bsf STATUS,6 ;page selector point to page 2
goto sub2_1
;------------------------------------------------------------------------------------------------
rx: ;副程式的"頭
bsf STATUS,6 ;page selector point to page 2
goto sub2_2
;------------------------------------------------------------------------------------------------
rd_eeprom: ;副程式的"頭
bsf STATUS,5 ;page selector point to page 3
bsf STATUS,6
goto sub3_1
;------------------------------------------------------------------------------------------------
………………
;其它的第一階副程的"頭"
;=========================================================
main:
;主程式
………………
call init ;直接呼叫,不必管PAGE問(wèn)題
………………
call rd_eeprom ;直接呼叫,不必管PAGE問(wèn)題
………………
call tx ;直接呼叫,不必管PAGE問(wèn)題
………………
call rx ;直接呼叫,不必管PAGE問(wèn)題
………………
;****************************************************************
ORG 0x200 ;page 1前半部
;此前半部可用以放置第二階副程式,而且是僅接受PAGE 1程式的呼叫;同樣的,這些副程式進(jìn)入點(diǎn)也需全部位於PAGE 1的前半部,否則會(huì )被呼叫不到;如此規劃,對PAGE的控制才會(huì )單純。
;------------------------------------------------------------------------------------------------
rout1_1:
………………
retlw 0 ;如此規劃不必更改PAGE控制
;------------------------------------------------------------------------------------------------
rout1_2:
………………
retlw 0 ;如此規劃不必更改PAGE控制
;=========================================================
sub1_1: ;副程式的"身體"
………………
call rout1_1
………………
call rout1_2
………………
bcf STATUS,5 ;page selector point to page 0
retlw 0
;****************************************************************
ORG 0x400 ;page 2前半部
;此前半部的規劃與PAGE 1同
;------------------------------------------------------------------------------------------------
sub2_1: ;副程式的"身體"
………………
bcf STATUS,6 ;page selector point to page 0
retlw 0
;------------------------------------------------------------------------------------------------
sub2_2: ;副程式的"身體"
………………
bcf STATUS,6 ;page selector point to page 0
retlw 0
;****************************************************************
ORG 0x600 ;page 3前半部
;此前半部的規劃與PAGE 1同
;------------------------------------------------------------------------------------------------
sub3_1: ;副程式的"身體"
………………
bcf STATUS,5 ;page selector point to page 0
bcf STATUS,6
retlw 0
;****************************************************************
ORG 0x7FF ;reset vector
goto main
;****************************************************************
END
上例中將PAGE 1、2、3的前半部?jì)?yōu)先放置第二階副程式,而且此副程式只由同一PAGE中的第一階副程式所呼叫使用;若其它PAGE也有第一階副程式需用到此第二階副程式,那就把有此需要的第一階副程式搬來(lái)同一PAGE吧。如此用心良苦有以下幾個(gè)好處:1、第一階副程式可直接call第二階副程式,不用改變PAGE的控制,2、第二階副程式執行結束時(shí)只需retlw,也不用改變PAGE的控制,3、PAGE的管理單純,不易出錯,4、整個(gè)程式碼可減少bcf SATUS,5、bcf STATUS,6、bsf STATUS,5、bsf STATUS,6的數量;當主程式也有必要直接呼叫此第二階副程式時(shí),可以有兩種做法,第一種做法可以保持程式PAGE管理的一致性,就是為主程式另外設計一個(gè)第一階副程式,來(lái)間接呼叫這個(gè)第二階副程式,這會(huì )增加5個(gè)(若這個(gè)第二階副程式在PAGE 3則需再加2個(gè))指令及其執行時(shí)間,第二個(gè)做法就是破壞原則,由主程式直接呼叫,但在call指令的前後需加上PAGE控制,這種做法有其缺點(diǎn),就是當此第二階副程式被移到別的PAGE時(shí),主程式內所有為呼叫該第二階副程式所做之PAGE控制,全需因應修改,而且呼叫的次數愈多,PAGE控制也跟著(zhù)多,修改起來(lái)也更費事且易出錯;兩種做法各有其利弊,何者適用則全由閣下視情況自行判斷選擇了。
程式組譯(assemble)完一定要開(kāi)啟listing檔(副檔名.lst),看看有無(wú)跨PAGE的現象,有的話(huà)就要調整該PAGE內一部分的副程式到別的PAGE,同時(shí)被搬動(dòng)過(guò)的副程式的"頭"及"尾"也需因應修改,同時(shí)要注意的是,是否把相對應的第二階副程式也一齊搬動(dòng);另外要看看副程式(第一階或第二階)之進(jìn)入點(diǎn),有沒(méi)有落於每個(gè)PAGE的前半部(即Address的bit 8為0的區域),沒(méi)有的話(huà)也要調整以消除此現象,不然會(huì )造成呼叫不到的bug。
現在將以上程式PAGE安排原則總結如下:
一、PAGE 0的前半部先放第一階副程式的"頭"(若空間足夠也可放整個(gè)副程式),"頭"負責把PAGE控制轉到該副程式的"身體"所座落之PAGE。
二、接下來(lái)放置主程式,主程式需全部置於PAGE 0內。
三、第一階副程式的"身體"可置於任一PAGE內,該副程式的"尾"負責把PAGE控制轉回PAGE 0。
四、第二階副程式與呼叫它的第一階副程式放在同一PAGE內。
五、呼叫第二階副程式不用改變PAGE控制(例外:主程式的呼叫);第二階副程式的"尾"也不用改變PAGE控制。
六、所有的副程式,不管是第一階或第二階,其進(jìn)入點(diǎn)皆需位於每一PAGE的前半部。
如此安排後,對副程式的呼叫均不需處理PAGE控制,此控制是由副程式的"頭"及"尾"處理掉了;而主、副程式當中的goto指令就可安心使用,不虞會(huì )GOTO到錯誤的PAGE上。
以上安排是不是會(huì )讓您安心撰寫(xiě)程式而不用擔心PAGE的控制呢?如果您發(fā)現到更好的方法也請不吝指教,謝謝!
評論