如何理解 nignx 中的一行代码?

2018-04-11 17:09:34 +08:00
 v2byy

#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 起来?

3567 次点击
所在节点    NGINX
8 条回复
EchoUtopia
2018-04-11 17:12:18 +08:00
cas ?
v2byy
2018-04-11 17:14:04 +08:00
@EchoUtopia 什么意思?
owenliang
2018-04-11 17:23:58 +08:00
自旋锁有什么难以理解的?
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
v2byy
2018-04-11 17:57:47 +08:00
@Shura 恩,这个我明白,我是想问这里的加锁是怎么实现的?

我的理解是这样操作的结果是将*lock 赋值为 1 了,然后其他再调用 ngx_trylock 的时候发现*lock 不为 0,直接 return,这样来实现锁定吗?
shilyx
2018-04-11 18:22:13 +08:00
这种锁核心在于 InterlockedCompareExchange,在 x86 架构下,他会对应到一条 cpu 指令。
一般的,比较后再赋值会对应多条 cpu 指令,这样多线程切换时,很有可能上下文恰好切换到指令序列之间,那么比较和赋值的实际情况就有可能乱套了。
InterlockedCompareExchange 由于是一条长指令,因此无论线程如何切换、如何暂停,都不会执行一半。
特别的,对于多核情况,InterlockedCompareExchange 会短暂的锁定内存总线来达到原子性。
Showfom
2018-04-11 18:30:35 +08:00
@shilyx 我什么时候才能和你一样优秀
shilyx
2018-04-26 12:54:21 +08:00
@Showfom 或许到时候我会更优秀了吧。。。闪了

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/445988

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX