多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
最近在学习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)里存的数,并将这个数转化为无符号整数。