C语言中volatile关键字

一、前言——程序的优化
1)硬件级别上的优化
a.由于内存访问速度远不及CPU处理速度,为提高机器整体性能,在硬件上引入硬件高速缓存Cache,加速对内存的访问。
b.在现代CPU中指令的执行并不一定严格按照顺序执行,没有相关性的指令可以乱序执行,以充分利用CPU的指令流水线,提高执行速度。
2)软件级的优化
a.编写代码时由程序员优化
b.由编译器进行优化。编译器优化常用的方法有:将内存变量缓存到寄存器;调整指令顺序充分利用CPU指令流水线,常见的是重新排序读写指令。
二、volatile详解
1.编译器有一种技术叫做数据流分析,分析程序中的变量在哪里赋值、在哪里使用、在哪里失效,分析结果可以用于常量合并,常量传播等优化,进一步可以消除一些代码。但有时这些优化不是程序所需要的,这时可以用volatile关键字禁止做这些优化。volatile的本意是“易变的” 因为访问寄存器要比访问内存单元快的多,所以编译器一般都会作减少存取内存的优化,但有可能会读脏数据。当要求使用volatile声明变量值的时候,系统总是重新从它所在的内存读取数据,即使它前面的指令刚刚从该处读取过数据。
2.简而言之,volatile变量的作用就两个:

a、告诉编译器不要进行优化
比如:要往某一地址送两指令:
int *ip =...; //设备地址
*ip = 1; //第一个指令
*ip = 2; //第二个指令
以上程序compiler可能做优化而成:
int *ip = ...;
*ip = 2;
结果第一个指令丢失。如果用volatile, compiler就不允许做任何的优化,从而保证程序的原意:
volatile int *ip = ...;
*ip = 1;
*ip = 2;
即使你要compiler做优化,它也不会把两次付值语句间化为一。它只能做其它的优化。
b、告诉系统始终从内存中取变量的值,而不是从缓存中取变量的值。
比如:中断服务程序中修改的供其它程序检测的变量

static int i=0;
int main(void)
{
	...
    while (1)
	{
		if (i) 
		{
			dosomething();
		}
	}
}
/* Interrupt service routine. */
void ISR_2(void)
{
    i=1;
}

程序的本意是希望ISR_2中断产生时,在main函数中调用dosomething函数,但是,由于编译器判断在main函数里面没有修改过i,因此可能只执行一次对从i到某寄存器的读操作,然后每次if判断都只使用这个寄存器里面的“i副本”,导致dosomething永远也不会被调用。如果将变量加上volatile修饰,则编译器保证对此变量的读写操作都不会被优化(肯定执行)。此例中i也应该如此说明。
3.一般来说,volatile用在如下的地方:
a.中断服务程序中修改的供其它程序检测的变量要加volatile,如上面的例子。
b.多线程应用中被多个任务共享的变量要加volatile
如:A线程将变量复制入寄存器,然后进入循环,反复检测寄存器的值是否满足一定条件(它期待B线程改变变量的值)在此种情况下,当B线程改变了变量的值时,改变的是其内存中的值,已改变的值对其在寄存器中的值没有影响。所以A线程进入死循环。此时必须使用volatile。
c.并行设备的硬件寄存器(如:状态寄存器)
d.存储器映射的硬件寄存器通常也要加voliate,因为每次对它的读写都可能有不同意义。
如:假设要对一个设备进行初始化,此设备的某一个寄存器为0xff800000。

int  *output = (unsigned  int *)0xff800000;//定义一个IO端口;
int   init(void)
{
     int i;
     for(i=0;i< 10;i++)
    {
      	*output = i;
    }
}
经过编译器优化后,编译器认为前面循环半天都是废话,对最后的结果毫无影响,因为最终只是将output这个指针赋值为9,所以编译器最后给你编译编译的代码结果相当于:
int  init(void)
{
      *output = 9;
}
如果你对此外部设备进行初始化的过程是必须是像上面代码一样顺序的对其赋值,显然优化过程并不能达到目的。反之如果你不是对此端口反复写操作,而是反复读操作,其结果是一样的,编译器在优化后,也许你的代码对此地址的读操作只做了一次。然而从代码角度看是没有任何问题的。这时候就该使用volatile通知编译器这个变量是一个不稳定的,在遇到此变量时候不要优化。
例如:
volatile  int *output=(volatile unsigned int *)0xff800000;//定义一个I/O端口
4.几个问题
1).一个参数既可以是const还可以是volatile吗?
可以的,例如只读的状态寄存器。它是volatile因为它可能被意想不到地改变。它是const因为程序不应该试图去修改它。
2).一个指针可以是volatile 吗?
可以,当一个中服务子程序修该一个指向一个buffer的指针时。
3).下面的函数有什么错误:
int square(volatile int *ptr)
{

 return *ptr * *ptr;

}
该程序的目的是用来返指针*ptr指向值的平方,但是,由于*ptr指向一个volatile型参数,编译器将产生类似下面的代码:
int square(volatile int *ptr)
{
 int a,b;
 a = *ptr;
 b = *ptr;
 return a * b;
}
由于*ptr的值可能被意想不到地该变,因此a和b可能是不同的。结果,这段代码可能返不是你所期望的平方值!正确的代码如下:
long square(volatile int *ptr)
{
 int a;
 a = *ptr;
 return a * a;
}
注意:频繁地使用volatile很可能会增加代码尺寸和降低性能,因此要合理的使用volatile。因为volatile抑制了优化,因此应尽量减少对它的引用操作,最好只是对其进行简单的赋值。如:
volatile char *pcWr_g;
 ...
while(...)
  *pcWr_g ++=UDR;
 ...
考虑改写成:
volatile char *pcWr_g;
char *pcTemp;
 ...
pcTemp = pcWr_g;
while(...)
    *pcTemp++=UDR;
pcWr_g = pcTemp;
 ...
一般而言,这样改写后效率更高些,可以在循环中只针对寄存器操作。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值