宏定义-地址内容
 eg:SOC中内部物理地址0xC000A001;其中[9:8]位域表示GPIO口的功能选择.
 版本1
 在C语言中,
 0xC000A001:表示数值
 (unsigned int *)0xC000A001:数值0xC000A001经过(unsigned int *)强制转换成:表示地址。
 *(unsigned int *)0xC000A001:表示取出地址(unsigned int *)0xC000A001指向的内存空间里的内容。
 *(unsigned int *)0xC000A001 = *(unsigned int *)0xC000A001;
 等号右边:表示取出地址(unsigned int *)0xC000A001里面的值(数据)
 赋值给
 等号左边:地址(unsigned int *)0xC000A001指向的内存空间
 操作[9:8]功能选择位域:
 *(unsigned int *)0xC000A001 = *(unsigned int *)0xC000A001 & (~(0x3 << 8));
 版本2
 #define GPIOALT (*(volatile unsigned int *)0xC000A001)
 *(unsigned int *)0xC000A001 = *(unsigned int *)0xC000A001 & (~(0x3 << 8));
 可简写成:
 GPIOALT = GPIOALT & (~(0x3 << 8));或 GPIOALT &= (~(0x3 << 8));
版本3
 #define NUM 保留的地址个数
 typedef unsigned int uint;
 typedef strcut{
 uint OUT;
 uint ALT;
 uint resd[NUM];
 uint ENABL;
 }gpio;
 #define GPIOA (*(* volatile gpio)0xC000A000)
 #define GPIOB (*(* volatile gpio)0xC000B000)
 #define GPIOC (*(* volatile gpio)0xC000C000)
 #define GPIOD (*(* volatile gpio)0xC000D000)
 #define GPIOE (*(* volatile gpio)0xC000E000)
 //使用方法:
 GPIOA.OUT &= (~(0x3 << 10));
 ...
 volatile-关键字功能
 volatile防止编译器对代码进行优化:因为编译器为了提高带代码的执行效率,会将经常访问的变量放在高速缓存cache中,每次读数据都是从cache里读取数据,不是从内存中读,当内存数据变化时,cache的值来不及改变,会造成取值错误。