优化屏障和内存屏障

优化屏障

        编译器编译源代码时,会将源代码进行优化,将源代码的指令进行重排序,以适合于CPU的并行执行。然而,内核同步必须避免指令重新排序,优化屏障(Optimization barrier)避免编译器的重排序优化操作,保证编译程序时在优化屏障之前的指令不会在优化屏障之后执行。

        Linux用宏barrier实现优化屏障,gcc编译器的优化屏障宏定义列出如下(在include/linux/compiler-gcc.h中): 
#define barrier() __asm__ __volatile__("": : :"memory") 

        上述定义中,“__asm__”表示插入了汇编语言程序,“__volatile__”表示阻止编译器对该值进行优化,确保变量使用了用户定义的精确地址,而不是装有同一信息的一些别名。“memory”表示指令修改了内存单元。 


内存屏障

         软件可通过读写屏障强制内存访问次序。读写屏障像一堵墙,所有在设置读写屏障之前发起的内存访问,必须先于在设置屏障之后发起的内存访问之前完成,确保内存访问按程序的顺序完成。

         读写屏障通过处理器构架的特殊指令mfence(内存屏障)、lfence(读屏障)和sfence(写屏障)完成,见《x86-64构架规范》一章。另外,在x86-64处理器中,对硬件进行操作的汇编语言指令是“串行的”,也具有内存屏障的作用,如:对I/O端口进行操作的所有指令、带lock前缀的指令以及写控制寄存器、系统寄存器或调试寄存器的所有指令(如:cli和sti)。

         Linux内核提供的内存屏障API函数说明如下表。内存屏障可用于多处理器和单处理器系统,如果仅用于多处理器系统,就使用smp_xxx函数,在单处理器系统上,它们什么都不要

内存屏障API函数说明 内存屏障的宏定义功能说明 
mb()                      适用于多处理器和单处理器的内存屏障。 
rmb()                     适用于多处理器和单处理器的读内存屏障。 
wmb()                   适用于多处理器和单处理器的写内存屏障。 
smp_mb()            适用于多处理器的内存屏障。 
smp_rmb()           适用于多处理器的读内存屏障。 
smp_wmb()         适用于多处理器的写内存屏障。


适合于多处理器和单处理器的内存屏障宏定义列出如下(在include/asm-x86/system.h中): 

[csharp]  view plain copy
  1. #ifdef CONFIG_X86_32  
  2. /*指令“lock; addl $0,0(%%esp)”表示加锁,把0加到栈顶的内存单元,该指令操作本身无意义,但这些指令起到内存屏障的作用,让前面的指令执行完成。具有XMM2特征的CPU已有内存屏障指令,就直接使用该指令*/  
  3. #define mb() alternative("lock; addl $0,0(%%esp)", "mfence", X86_FEATURE_XMM2)  
  4. #define rmb() alternative("lock; addl $0,0(%%esp)", "lfence", X86_FEATURE_XMM2)  
  5. #define wmb() alternative("lock; addl $0,0(%%esp)", "sfence", X86_FEATURE_XMM)  
  6. #else  
  7. #define mb() asm volatile("mfence":::"memory")  
  8. #define rmb() asm volatile("lfence":::"memory")  
  9. #define wmb() asm volatile("sfence" ::: "memory")  
  10. #endif  

[csharp]  view plain copy
  1. /*刷新后面的读所依赖的所有挂起读操作,在x86-64构架上不需要*/  
  2. #define read_barrier_depends() do { } while (0)   

        宏定义ead_barrier_depends()刷新后面的读所依赖的所有挂起读操作,后面的读操作依赖于正处理的读操作返回的数据。在x86-64构架上不需要此宏。它表明:在此屏障之前,没有来自内存区域数据所依赖的读曾经重排序。所有的读操作处理此原语,保证在跟随此原语的任何读操作此原语之前访问内存(但不需要其他CPU的cache)。此原语在大多数CPU上有比rmb()更轻的份量。


        本地CPU和编译器遵循内存屏障的排序限制,仅内存屏障原语保证排序,即使数据有依赖关系,也不能保证排序。例如:下面代码将强迫排序,因为*q的读操作依赖于p的读操作,并且这两个读操作被read_barrier_depends()分开。在CPU 0和CPU 1上执行的程序语句分别列出如下: 


CPU 0                                      CPU 1 
b = 2;
memory_barrier();
p = &b;                                      q = p;
                                                  read_barrier_depends();
                                                  d = *q;


下面的代码没有强制排序,因为在a和b的读操作之间没有依赖关系,因此,在一些CPU上,如:Alpha,y将设置为3,x设置为0。类似这种没有数据依赖关系的读操作,需要排序应使用rmb()。

CPU 0                                        CPU 1

a = 2;
memory_barrier();
b = 3;                                         y = b;
                                                   read_barrier_depends();
                                                   x = a;

适合于多处理器的内存屏障宏定义列出如下(在include/asm-x86/system.h中): 

[csharp]  view plain copy
  1. #ifdef CONFIG_SMP  
  2. #define smp_mb() mb()  
  3. #ifdef CONFIG_X86_PPRO_FENCE  
  4. # define smp_rmb() rmb()  
  5. #else  
  6. # define smp_rmb() barrier()  
  7. #endif  
  8. #ifdef CONFIG_X86_OOSTORE  
  9. # define smp_wmb() wmb()  
  10. #else  
  11. # define smp_wmb() barrier()  
  12. #endif  
  13. #define smp_read_barrier_depends() read_barrier_depends()  
  14. #define set_mb(var, value) do { (void)xchg(&var, value); } while (0)  
  15. #else  
  16. #define smp_mb() barrier()  
  17. #define smp_rmb() barrier()  
  18. #define smp_wmb() barrier()  
  19. #define smp_read_barrier_depends() do { } while (0)  
  20. #define set_mb(var, value) do { var = value; barrier(); } while (0)  
  21. #endif   


函数rdtsc_barrier用于加内存屏障阻止RDTSC猜测,当在一个定义的代码区域使用读取时间戳计数器(Read Time-Stamp Counter,RDTSC)函数(或者函数get_cycles或vread)时,必须加内存屏障阻止RDTSC猜测。其列出如下:


[csharp]  view plain copy
  1. static inline void rdtsc_barrier(void)  
  2. {  
  3.     alternative(ASM_NOP3, "mfence", X86_FEATURE_MFENCE_RDTSC);  
  4.     alternative(ASM_NOP3, "lfence", X86_FEATURE_LFENCE_RDTSC);  
  5.   
  6. }  



转帖:http://blog.csdn.net/xujianqun/article/details/7800813

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值