#define ngx_atomic_cmp_set(lock, old, set)
((ngx_atomic_uint_t) InterlockedCompareExchange((long *) lock, set, old)
== old)
#define ngx_trylock(lock) (*(lock) == 0 && ngx_atomic_cmp_set(lock, 0, 1))
ngx_trylock 的意思是如果 初始状态*lock 为 0,且经过原子操作成功赋值为 1 之后,说明 lock 成功。
是这样的吗?
PS: 顺便问下,如何阅读一个开源项目源码,比如 nginx,如何在 win 下 debug 起来?
1
EchoUtopia 2018-04-11 17:12:18 +08:00 via Android
cas ?
|
2
v2byy OP @EchoUtopia 什么意思?
|
3
owenliang 2018-04-11 17:23:58 +08:00
自旋锁有什么难以理解的?
|
4
Shura 2018-04-11 17:44:06 +08:00
其实 ngx_trylock 的作用就是用 ngx_atomic_cmp_set 加锁而已,定义成宏只是为了增加代码可读性。
“ Core: use ngx_trylock() where possible.It makes code much more readable.”,From: http://nginx152.rssing.com/chan-6700897/all_p1165.html |
5
v2byy OP @Shura 恩,这个我明白,我是想问这里的加锁是怎么实现的?
我的理解是这样操作的结果是将*lock 赋值为 1 了,然后其他再调用 ngx_trylock 的时候发现*lock 不为 0,直接 return,这样来实现锁定吗? |
6
shilyx 2018-04-11 18:22:13 +08:00 8
这种锁核心在于 InterlockedCompareExchange,在 x86 架构下,他会对应到一条 cpu 指令。
一般的,比较后再赋值会对应多条 cpu 指令,这样多线程切换时,很有可能上下文恰好切换到指令序列之间,那么比较和赋值的实际情况就有可能乱套了。 InterlockedCompareExchange 由于是一条长指令,因此无论线程如何切换、如何暂停,都不会执行一半。 特别的,对于多核情况,InterlockedCompareExchange 会短暂的锁定内存总线来达到原子性。 |