对(*(volatile unsigned long *)) 的理解
(*(volatile unsigned long *)) 这个语句对于不同的计算机体系结构,设备可能是端口映射,也可能是内存映射的。如果系统结构支持独立的IO地址空间,并且是端口映射,就必须使用汇编语言完成实际对设备的控制,因为C语言并没有提供真正的“端口”的概念。如果是内存映射,那就方便多了。
以 #define IOPIN (*((volatile unsigned long *)0xE0028000))为例:作为一个宏定义语句,define是定义一个变量或常量的伪指令。首先 (volatile unsigned long *) 的意思是将后面那个地址强制转换成 volatile unsigned long * , unsigned long *是无符号长整型,volatile是一个类型限定符,如const一样,当使用volatile限定时,表示这个变量是依赖系统实现的,意味着这个变量会被其他程序或者计算机硬件修改,由于地址依赖于硬件,volatile就表示他的值会依赖于硬件。
volatile 类型是这样的,其数据确实可能在未知的情况下发生变化。比如,硬件设备的终端更改了它,现在硬件设备往往也有自己的私有内存地址,比如显存,他们一般是通过映象的方式,反映到一段特定的内存地址当中,这样,在某些条件下,程序就可以直接访问这些私有内存了。另外,比如共享的内存地址,多个程序都对它操作的时候。你的程序并不知道,这个内存何时被改变了。如果不加这个voliatile修饰,程序是利用catch当中的数据,那个可能是过时的了,加了 voliatile,就在需要用的时候,程序重新去那个
地址去提取,保证是最新的。归纳起来如下:
1. volatile变量可变允许除了程序之外的比如硬件来修改他的内容
2. 访问该数据任何时候都会直接访问该地址处内容,即通过cache提高访问速度的优化被取消
对于((volatile unsigned long *) 0xE0028000)为随硬件需要定义的一种地址,前面加上“*”指针,为直接指向该地址,整个定义约定符号IOPIN代替,调用的时候直接对指向的地址寄存器写内容既可。这实际上就是内存映射机制的方便性了。其中volatile关键字是嵌入式系统开发的一个重要特点。上述表达式拆开来分析,首先(volatile unsigned long *) 0xE0028000的意思是把0xE0028000强制转换成volatile unsigned long类型的指针,暂记为p,那么就是#define A *p,即A为P指针指向位置的内容了。这里就是通过内存寻址访问到寄存器A,可以读/写操作。
对于(volatile unsigned char *)0x20可以再分析一下,它是由两部分组成:
1)(unsigned char *)0x20,0x20只是个值,前面加(unsigned char *)表示0x20是个地址,而且这个地址类型是unsigned char ,意思是说读写这个地址时,要写进unsigned char 的值,读出也是unsigned char 。
2)volatile,关键字volatile 确保本条指令不会因C 编译器的优化而被省略,且要求每次直接读值。例如用while((unsigned char
*)0x20)时,有时系统可能不真正去读0x20的值,而是用第一次读出的值,如果这样,那这个循环可能是个死循环。用了volatile 则要求每次都去读0x20的实际值。
那么(volatile unsigned char *)0x20是一个固定的指针,是不可变的,不是变量。而char *u则是个指针变量。再在前面加"*":*(volatile unsigned char *)0x20则变成了变量(普通的unsigned char变量,不是指针变量),如果#define i (*(volatile unsigned char *)0x20),那么与unsigned char i是一样了,只不过前面的i的地址是固定的。
那么问题就可解答了,(*(volatile unsigned char *)0x20)可看作是一个普通变量,这个变量有固定的地址,指向0x20。而0x20只是个常量,不是指针更不是变量。
相关推荐
在上述代码中,我们可以看到`(*(volatile unsigned long *) 0xE0028000)`这种写法,这是什么意思呢?这是一种强制类型转换,强制将`0xE0028000`这个地址转换成`volatile unsigned long *`类型的指针。这里的`...
#define CLK_CFG1 ( *((volatile unsigned long *)0xE1400004) )//0X #define FB_CLK_SEL1 ( *((volatile unsigned long *)0xE140002C) ) #define SWAP_CFG1 ( *((volatile unsigned long *)0xE1400028) ) #...
#define GPA0CON (*((volatile unsigned long*)0xE0200000)) #define GPA1CON (*((volatile unsigned long*)0xE0200020)) #define ULCON0 (*((volatile unsigned long*)0xE2900000)) #define UCON0 (*((volatile ...
#define UTRSTAT0 (*(volatile unsigned long*)0x50000010) #define UTXH0 (*(volatile unsigned char*)0x50000020) #define URXH0 (*(volatile unsigned char*)0x50000024) #define TXD0_READY 0x2 #define RXD0_...
volatile long state; /* 进程状态 */ unsigned long flags; /* 进程标志 */ int sigpending; /* 是否有信号未处理的标志 */ mm_segment_t addr_limit; /* 地址上限 */ struct exec_domain *exec_domain; /* ...
了解 task_struct 结构对理解 Linux 任务调度非常重要。通过分析 task_struct 结构,我们可以了解 Linux 任务调度的机制和实现细节。 task_struct 结构的每个成员变量都有其特定的含义和用途。例如,state 变量记录...
#define GPBCON (*(volatile unsigned long*)0x56000010) #define GPBDAT (*(volatile unsigned long*)0x56000014) ``` #### 四、定义端口功能 在S3C2440的GPIO端口定义中,“00”表示输入,“01”表示输出。例如,...
5. **unsigned long**:四字节,值域为0~4294967295。 6. **signed long**:四字节,值域为-2147483648~2147483647。 7. **float**:四字节,值域约为±1.175494E-38~±3.402823E+38。 8. **bit**:表示位,值域为0...
#define GPBCON (*volatile unsigned long *)0x56000010 #define GPBDAT (*volatile unsigned long *)0x56000014 #define GPB5_out (1(5*2)) GPBCON = GPB5_out; GPBDAT &= ~(1); ``` 此外,除了直接访问寄存器,...
- **Unsigned long long**: 无符号非常长的整型,占8字节。 - **Float**: 单精度浮点型,占4字节。 - **Double**: 双精度浮点型,占8字节。 #### 构造型数据类型 除了基本数据类型外,C语言还支持一些复杂的构造...
(*(volatile unsigned long*)0x80000001) = address; // 锁存地址,CS高忽略RD&WE GPIO = 0; // 片选enable (*(volatile unsigned long*)0x80000000) = data; // 输出数据 GPIO = 1; } ``` 3. **读数据函数**...
unsigned long have_console; unsigned long reloc_off; /* 重定位偏移量 */ unsigned long env_addr; /* 环境变量地址 */ unsigned long env_valid; /* 环境变量是否有效 */ unsigned long fb_base; /* 帧缓冲...
#define HWREG(x) (*((volatile unsigned long *)(x))) #define HWREGH(x) (*((volatile unsigned short *)(x))) #define HWREGB(x) (*((volatile unsigned char *)(x))) ``` **参数说明** - `x`: 表示要访问的...
- **Long**:声明长整型变量。 - **Register**:声明使用CPU寄存器的变量。 - **Return**:用于从函数返回值。 - **Short**:声明短整型变量。 - **Signed**:声明有符号数。 - **Sizeof**:计算表达式的大小。 - **...
- **unsigned long long**:占8字节,用于存储非负的非常大的整数。 ##### 浮点型 - **float**:占4字节,用于存储单精度浮点数,有效范围约为3.4E-38至3.4E+38。 - **double**:占8字节,用于存储双精度浮点数,...
* 整数:char、short、int、long、unsigned char、unsigned short、unsigned int、unsigned long * 浮点数:float、double * 字符型:char * 字符串:char [] 不同的数据类型有不同的取值范围和内存占用空间。例如...
stacked_r0 = ((unsigned long)hardfault_args[0]); stacked_r1 = ((unsigned long)hardfault_args[1]); stacked_r2 = ((unsigned long)hardfault_args[2]); stacked_r3 = ((unsigned long)hardfault_args[3]); ...
/****************************************************************************** * 文件名称:ili932x.c * 摘 要:支持ILI9320和ILI9325...#define Clr_Rst (*((volatile unsigned long *) 0x40011414) = RstPin)