签到天数: 2213 天 [LV.Master]伴坛终老
|
堆栈及其使用(举例篇)
5 \( k% y! A/ I; ] 例如,在一个子程序中要用到寄存器A,X,Y和P。在进入子程序之后,首先要将这些寄存器的值存入堆栈保存起来,这称为保留现场。子程序的工作进行完准备返回主程序时,还要把原先保留起来的数据从堆栈中取出来,恢复各寄存器原先的状态,这称为恢复现场。保留现场与恢复现场时,对各寄存器进行操作的顺序是不能随意安排的,以免造成混乱。再有就是进栈与出栈的顺序恰好相反。
* G3 M" E: [3 t9 A/ g7 d T- r- D 子程序的起、止部分应当这样设计:8 s5 W) Z; V# P* M4 U
SUB PHP ;P必须首先进栈
/ v/ R2 b5 m4 R, s: w4 R PHA ;A进栈应在X,Y进栈之前5 m% V j6 [! J% x4 a5 B
TXA ! t0 k" G/ L( x" r, k _
PHA ;X进栈要通过累加器A( S+ S" a1 |) I/ Z& o- N
TYA , {+ F% e- K3 [# ^3 p5 N
PHA ;Y进栈也要通过累加器A- Z/ y. z7 \1 X7 G
....;子程序的工作部分
# p( G5 Z; i3 e$ P& ?3 B% U3 b* r PLA
8 ~4 @! O/ g, v* j TAY ;Y最后进栈故最先出栈,Y出栈经过A! c6 t# m1 o( n' ~( C* }8 Y% d
PLA
* ^" Q/ p" D4 O4 F3 G TAX ;X出栈也经过A
. c% e: i' p) U3 e PLA ;A出栈
8 X. Q. y' R+ K. o PLP ;P最后出栈,否则其他指令会影响P的某些标志位: `; Y) y q. f+ s% L) q
RTS ;恢复现场后可以返回主程序; R; O1 u/ L3 U0 E: Y5 J% P
6502在子程序执行完毕,遇到RTS指令时是怎样返回主程序的?我们知道,在这种非顺序执行的地方,必须给出跳转的目的地址,这个地址进入程序计数器PC后才能转向。那么这个跳转目的地址是如何得到的呢?原来,当执行JSR指令时,返回地址(也就是JSR指令码所在储存单元地址+2,或者说是JSR指令的第三字节的代码所在地址)自动从程序计数器PC压入堆栈。执行RTS指令时,要将返回地址从堆栈弹出送入程序计数器PC,PC值再自动加1,以指向下一条指令操作码的地址,因此必须记住这条规则:在执行RTS指令之前把在子程序中进栈的数据全部取出,但也不能从栈中取出过多的数据,否则会把返回地址丢掉。) @3 M, ]0 T/ `% Q) _
自顶向下的程序设计方法有一条非常重要的原则:子程序应该只有一个入口和一个出口,假如有多个出口,则容易出现的错误之一就是可能忘记将子程序中进栈的内容从堆栈中取出,造成返回地址混乱。解决的办法也很简单:不使用很多RTS指令,而在需要返回的分支处用JMP指令跳转到唯一的一个出口处。
# i* X/ S9 n# D! A 为了方便,把保留现场与恢复现场这两个工作分别设计成子程序,是经验不足的程序员可能出现而又难于侦错的错误之一。) Z" n- ?- T% O( a4 k) A* k
例:/ o5 Q6 u" Z$ D: Z; J( J9 s- q
SUB JSR SAVE ;# D3 c, [+ r: h: d, D: m
....... ;+ I+ `8 P% Y5 u4 Q! }7 u. K
JSR RESTORE ;: ^9 C; `9 K0 }4 R, I G" O
RTS! r! q- o3 x0 d1 _4 _0 J- W3 U
SAVE PHP ;
- f$ W7 M/ R' N j- v, g) u5 L& Z0 h PHA, b. K7 ^. z& s6 h7 P# N( w
TXA6 F+ C0 S7 ?/ ]) X9 t7 y4 D) w+ _
PHA
% y0 l T \& @( ?, x3 v5 J TYA
% z6 [( p" P0 l$ O' }8 |, y PHA0 @2 {, A' O Q. E5 t5 @
RTS
3 ]/ c# ~& k$ j. }. _, ]& nRESTORE PLA
, N; l0 ^ g$ x8 f; T TAY
% E S0 E0 W+ V, n2 t) a PLA
; O" i: J# U4 V& Y2 f7 m TAX- \ Q/ e4 f+ }6 q: D, H8 D% i
PLA
/ L6 Q" R% V7 ^8 e2 L" t PLP
9 e% x$ ^) V, H RTS
3 h# S4 |! O: r+ t t' R; Z 这样编写程序乍看起来是合理的,但实际执行的效果却与预想相差甚远:调用保留现场子程序的JSR指令指令把返回地址压入堆栈,然后转到SAVE子程序,将P、A、X、Y寄存器的值相继相继压入栈内,所以栈顶二数为Y、X的值。当SAVE子程序的返回指令RTS被执行时,6502从堆栈顶取出两个单元的内容作为返回地址送入PC计数器。而这两个单元中的内容根本不是返回地址,而是X、Y被保存起来的值,因此程序会返回到何处大概只有老天才能知道了。
; H7 O6 x' l9 _8 L" q 同样的,恢复现场子程序也不能使用,因为进入RESTORE子程序时堆栈顶部存放的是返回地址而不是X、Y寄存器的值。
; d' r) e; I) u+ i5 A4 n5 O 不过从这个错误,我们可以得到一点启发:堆栈可以作为中继站,使数据能够在PC计数器与其它寄存器之间交换。# |* b$ \; L& Y. c' ?2 x3 I; p
-------by sossb |
|