51单片机、stm32中sfr和define的定义问题求解!!!

在51单片机中为什么使用特殊功能寄存器,像P0口,一定要用sfr P0 = 0x80;来定义。
而用#define P0 (*(volatile unsigned char*)0x80)后不行,我PROTEUS仿真试过此时不能使用P0口了。
而在STM32中则用define定义
例如:#define GPIOB_CRL *(volatile unsigned long *)0x40010C00 则可正确地使用该寄存器。
求中肯的解释。
同是地址,为什么在51中和在STM32中使用时,定义就不同了???
希望能给个详细点的解释

51单片机中,如果你这么定义:

#define P0 (*(volatile unsigned char*)0x80)
那么执行P0=0x25的时候,汇编就变成
MOV DPTR,#0080H
MOV A,#25H
MOVX @DPTR,A
很明显,你就不是往端口P0赋值0x25,而是像外部总线上的地址0x0080写数据0x25了,了解?
51单片机的寄存器存在于内部空间0x80-0xFF,这空间属于直接寻址,不是访问外部总线。所以只能用 sfr P0=0x80来定义。

STM32单片机就不一样,所有的外设寄存器都处于统一寻址的外部空间中,所以只能用这种方法定义其地址,访问时也类似MOVX指令的执行。追问

我反汇编了一下P1口的
P1=0;
C:0x000D 7890 MOV R0,#P1(0x90)
C:0x000F E4 CLR A
C:0x0010 F6 MOV @R0,A
C:0x0011 08 INC R0
C:0x0012 F6 MOV @R0,A
是这样的,不是外部吧,是内部的吧
正常的情况下是
C:0x0154 8F80 MOV P0(0x80),R7,不需要间接寻址
大概懂了

追答

是的。

温馨提示:答案为网友推荐,仅供参考
第1个回答  2012-10-29
在STM32中有个固件库,已经把那些端口的地址全部存储在一个向量表中,你用的那个DEFINE就是内部已经定义好的。你本来就是使用的固件库就可以对端口的地址进行正确的端口操作,然而在51中,这些端口是不能这样操作的,只能把它都当做特殊的寄存器给他附地址。
第2个回答  2012-10-28
因为平台不同,C语言大的框架是相通的,但细节上有所区别
第3个回答  2012-10-28
51特殊功能寄存只能直接寻址,不能间接寻址
第4个回答  2012-10-28
规定追问

我知道。。。