C51的堆栈问题是很容易被忽略的问题,至少我之前是这样.今天早上查了一下,需要注意的问题有这么几点:
1.上电后自动设置堆栈指针SP的值(SP=0x2F)
C:0x0000 02006A LJMP C:006A
C:0x006A 787F MOV R0,#0x7F
C:0x006C E4 CLR A
C:0x006D F6 MOV @R0,A
C:0x006E D8FD DJNZ R0,C:006D
C:0x0070 75812F MOV SP(0x81),#0x2F
C:0x0073 020003 LJMP main(C:0003)
如代码所示,上电后直接跳转到某个地址,延时一段时间后,设置SP的值,然后才进入Main函数,执行我们定义的函数.
然而,我在STC12C5A60的数据手册中却看到:单片机复位后,堆栈指针SP为07H,指向了工作寄存器组0中的R7.也就是说SP的值是可以任意指定的,只要不与其它内存重叠.
2.C51的堆栈是向上生长的,这与ARM的向下生长不同.
3.STC12C5A60手册上还写到:用户初始化程序都应对SP设置初值,一般设在80H之后.
问题来了,80H--FFH不是特殊功能寄存器区吗?这样设置会产生重叠吗?答案是,不会重叠.
原因:高128字节RAM与特殊功能寄存器区都用80H--FFH,地址空间确实重叠,但物理上却是相互独立的.使用时通过不同的寻址方式加以区别,当内存使用时只能间接寻址,而特殊功能寄存器区只能寻址.用代码表示,就应该是这样的:
间接寻址:
MOV R0,#80H
MOV R1,@R0
直接寻址:
MOV R1,80H