- 论坛徽章:
- 11
|
本帖最后由 zylthinking 于 2012-06-27 11:07 编辑
@塑料袋 @tempname2 @帅绝人寰 帮我看看, 搞不定了- #define lock(lkp) do{ \
- while(!__sync_bool_compare_and_swap(lkp, 0, 1)){ \
- sched_yield(); \
- } \
- }while(0)
- #define try_lock(lkp) ({ \
- (__sync_bool_compare_and_swap(lkp, 0, 1) ? 0 : -1); \
- })
- #define unlock(lkp) do{ \
- *(lkp) = 0; \
- }while(0)
复制代码 lkp 在不是 volatile 的情况下, 是不是不安全?
一直以来都是这么用的, 似乎也问题, 今天突然觉得不安了
主要是 unlock 直接就是 *(lkp) = 0; 行不行?
1。 似乎不能防止编译器重拍指令, 比如
int lk = 0;
int n = 0;
lock(&lk);
++n;
unlock(&lk);
编译器有可能重排成
lock(&lk);
unlock(&lk);
++n;
2。 就算不重拍, 那么乱序呢?
执行完 unlock(&lk) 后会是怎么一种情景?
我的理解是 ++n 肯定已经执行完了, 但是可能还在 load/store unit 里面, 没有更新到寄存器
那么这个时候发生线程切换, 再次读此值, 应该会直接从 load/store unit 内直接读取到, 因此不会有太大问题; 但还是需要 @塑料袋 确认一下,
关键是如果 *(lkp) = 0; 已经进入了 load/store unit, 则可以保证 ++n 至少在 load/store unit 内, 或者已经到了寄存器中???
另外, 如果另一个线程读 n 时, 产生的汇编应该是从内存读, 那么它怎么知道内存中某单元中的值已经在 load/store unit 中了?
似乎首先要确认的是 ++n 的结果在 load/store unit 中时, 写的目标到底是存储器还是寄存器;
如果是寄存器, 也就是 n 被优化到寄存器中了, 那么CPU 还能直接从 load/store unit 中读吗? 但如果不从其中读而直接读内存, 那读出的就是一个错误的值
3。 另一个是C层面的问题了@OwnWaterloo @starwing83 @幻の上帝
文件1:
volatile int n = 0;
文件2:
extern int n;
++n;
那么在文件2中, n 会不会还会当作 volatile 来对待
不知道会不会又是所谓 UB 行为, 但 gcc 4.2 这样做是可以的, 甚至连一个警告都没有 |
|