第四節(jié):PIC系列單片機(jī)程序存儲(chǔ)器及堆棧
PIC16C5X內(nèi)部有384~2K的只讀程序存貯器,下面論述其結(jié)構(gòu)和堆棧。
§1.4.1 程序存儲(chǔ)器結(jié)構(gòu)
PIC16C5X程序存儲(chǔ)器結(jié)構(gòu)如圖1.3所示:
從上圖可看出,PIC程序存儲(chǔ)器采用分頁(yè)結(jié)構(gòu),每頁(yè)長(zhǎng)0.5K。因此對(duì)于PIC16C52程序存儲(chǔ)器在1頁(yè)之內(nèi),而對(duì)于PIC16C54和PIC15C55程序存儲(chǔ)器容量為1頁(yè),PIC16C56和PIC16C57 的容量則分別為2頁(yè)和4頁(yè)。頁(yè)面地址由狀態(tài)寄存器f3的第5位和第6位(PA0、PA1)確定。程序轉(zhuǎn)移時(shí),在本頁(yè)內(nèi)可直接進(jìn)行;在需跨頁(yè)跳轉(zhuǎn)時(shí)(GOTO、CALL指令),則必須根據(jù)將要跳轉(zhuǎn)去的頁(yè)面,把f3中的PA0、PA1位置成相應(yīng)的值。具體請(qǐng)參考f3寄存器描述及§2.7.2 程序設(shè)計(jì)基礎(chǔ)的程序技巧例子。
§1.4.2 堆棧
PIC16C5X設(shè)有二層堆棧,堆棧1和堆棧2,供子程序調(diào)用。涉及堆棧操作的指令有二條。
1、CALL--在主程序中第一次執(zhí)行CALL指令時(shí),將PC值加1后推入堆棧1,堆棧1 原有的內(nèi)容則被推入堆棧2中。這時(shí)子程序中還可再做一次子程序嵌套,即再執(zhí)行一次 CALL指令。如果子程序調(diào)用多于二層時(shí),堆棧中只存放最近的二個(gè)返回地址。
當(dāng)執(zhí)行一條CALL指令時(shí),狀態(tài)寄存器f3中的將頁(yè)面尋址位PA1、PA0將被置入到PC的最高二位(第11位和第10位),而PC的第9位總是被置為0,如圖1.3所示。所以這時(shí)PC 值將是
這意味著在PIC16C5X中,子程序起始地址只能放在每個(gè)程序存儲(chǔ)頁(yè)面的上半頁(yè),即低地址的那一半(000-0FF、200-2FF、400-4FF、600-6FF)。注意,這里指的是子程序的起始地址,即子程序頭。而子程序體是可以延伸到下半頁(yè)去的。
對(duì)于PIC16C56和PIC16C57,由于程序空間分別為1K和2K,可能存在跨頁(yè)面子程序調(diào)用。所以調(diào)用子程序前必須先把f3中的PA0、PA1設(shè)置成該子程序所在的頁(yè)面地址。返回后再將其恢復(fù)成當(dāng)前的頁(yè)面值。當(dāng)然如果這時(shí)子程序是在同一頁(yè)面,則可省去這一過(guò)程。
2、RETLW--該指令把棧頂(堆棧1)的值寫入PC,同時(shí)還把堆棧2的值拷貝到堆棧1 去。子程序總是返回到調(diào)用它時(shí)所在的地方,不管它是處在什么頁(yè)面,也不管f3寄存器中的PA0、PA1這時(shí)是指在什么頁(yè)面。但是執(zhí)行RETLW(子程序返回)指令并不會(huì)改變f3中PA0、PA1的值,所以當(dāng)你從一次跨頁(yè)面的子程序調(diào)用返回時(shí),不要忘了恢復(fù)f3中的原先的PA0和PA1值。請(qǐng)參考上面關(guān)于CALL指令的敘述。
由于堆棧和PC的寬度是相同的,所以你可以在程序的任何地方執(zhí)行一條CALL 指令來(lái)調(diào)用子程序。但是對(duì)于跨頁(yè)面的調(diào)用,你要小心處理f3中的頁(yè)面地址位PA0和PA1,請(qǐng)參考第二章指令詳解中的CALL實(shí)例。