Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1495852
  • 博文数量: 267
  • 博客积分: 3010
  • 博客等级: 少校
  • 技术积分: 3089
  • 用 户 组: 普通用户
  • 注册时间: 2011-06-05 17:09
个人简介

尊天命,尽人事

文章分类

全部博文(267)

文章存档

2017年(6)

2015年(4)

2014年(27)

2013年(52)

2012年(59)

2011年(120)

分类: LINUX

2011-12-12 19:59:40

#define set_current_state(state_value) /   
set_mb(current->state, (state_value))   
 
#define set_mb(var, value) do { var = value; mb(); } while (0)   
#define mb() __asm__ __volatile__ ("lock; addl $0,0(%%esp)" : : : "memory") 

语句解释:
1.set_mb(),mb(),barrier()函数追踪到底,就是__asm__ __volatile__("lock; addl $0,0(%%esp)":::"memory"),而这行代码就是内存屏障。
2.__asm__:用于指示编译器在此插入汇编语句
3.volatile/__volatile__:基本可以做到两件事情:
  (1).阻止编译器为了提高速度将一个变量缓存到寄存器内而不写回内存。
  (2).阻止编译器调整操作volatile变量的指令顺序。  
4.memory强制gcc编译器假设RAM所有内存单元均被汇编指令修改,这样cpu中的registers和cache中已缓存的内存单元中的数据将作废。cpu将不得不在需要的时候重新读取内存中的数据。这就阻止了cpu又将registers,cache中的数据用于去优化指令,而避免去访问内存。  
5.lock前缀表示将后面这句汇编语句:"addl $0,0(%%esp)"作为cpu的一个内存屏障。  
6.addl $0,0(%%esp)表示将数值0加到esp寄存器中,而该寄存器指向栈顶的内存单元。加上一个0,esp寄存器的数值依然不变。即这是一条无用的汇编指令。在此利用这条无价值的汇编指令来配合lock指令,在__asm__,__volatile__,memory的作用下,用作cpu的内存屏障。

个人思考:
    volatile关键字来让编译器保证变量计算结果写回到内存之中,而没有用volatile关键字约束的变量的计算结果可能保存在寄存器之中,这种限制完全在我们的代码之中控制;语句中的memory强制gcc编译器假设RAM所有内存单元均被汇编指令修改,cpu将不得不在需要的时候重新读取内存中的数据。这两条关键字保证在多线程的环境下数据的一致性。

遗留问题:
    void aaa()
    {
     int a = 1; //没有使用volatile,
     a++;       //a的计算结果保存在寄存器中的话
     mb();      //寄存器失效
     
     printf("%d/n", a); (问题:a等于几呢)
    }

解决办法:
 把汇编看看,然后编译,看看gcc编译处理的汇编代码。

阅读(1063) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~