最近在学习Nand Flash驱动程序设计时,涉及到((volatile unsigned long )),通过查阅部分网络文章,对其用法和指针有了进一步理解,遂作此笔记。本文参考了yueleilei的文章--ARM定义特殊寄存器(*(volatile unsigned long *))的理解[点击打开链接](http://bbs.ednchina.com/BLOG_ARTICLE_3002274.HTM),在此感谢。
## 一、对 #define SREG (*(volatile unsigned char *)0x5F)的理解
嵌入式系统编程,要求程序员能够利用C语言访问固定的内存地址。既然是个地址,那么按照C语言的语法规则,这个表示地址的量应该是指针类型。所以,知道要访问的内存地址后,比如0x5F,第一步是要把它强制转换为指针类型(unsigned char *)0x5F,AVR的SREG是八位寄存器,所以0x5F强制转换为指向unsigned char类型。
(注意,这一步的理解很关键,很多文章说:( volatile unsigned long * )的意思是将后面的那个地址强制转换成 volatile unsigned long * ,当初看文章一直不理解这句话,其实解释为:将(volatile unsigned long * )后面跟的内容转化成一个指针,并且该指针指向一个易变的无符号整数,这样更易理解。)
volatile(可变的)这个关键字说明这变量可能会被意想不到地改变,这样编译器就不会去假设这个变量的值了。这种“意想不到地改变”,不是由程序去改变,而是由硬件去改变——意想不到。
第二步,对指针变量引用,就能操作指针所指向的地址的内容了
*(volatile unsigned CHAR *)0x5F
第三步,小心地把#define宏中的参数用括号括起来,这是一个很好的习惯,所以
#define SREG (*(volatile unsigned CHAR *)0x5F)
类似的,如果使用一个32位处理器,要对一个32位的内存地址进行访问,可以这样定义
#define RAM_ADDR (*(volatile unsigned LONG *)0x0000555F)
然后就可以用C语言对这个内存地址进行读写操作了
读:tmp = RAM_ADDR;
写:RAM_ADDR = 0x55;
## 二、举例
例:如何理解(*(volatile unsigned long *)0x56000010)
解答:
long是32位整型,unsigned指无符号数,左边的*表示取内容;
volatile表示易变的,告诉编译器不要优化,这个地址的内容不一定是在程序中改变的(可能是硬件上改变的,如寄存器中某些状态位的改变);
(volatile unsigned long *) 表示将后面跟的内容转化成一个指针,并且是指向一个易变的无符号整数。左边再加个,表示取该指针指向地址的内容。
总的意思是取那个内存单元(内存地址0x56000010)里存的数,并将这个数转化为无符号整数。