Mutex(又叫 Lock),在多线程中,作为同步的基本类型,用来保证没有两个线程或进程同时在他们的关键区域.因为 Mutex 这种排它性,很多人认为 Mutex 开销很大,尽量避免使用它.就如这篇分析完共享数据问题后,进一步分析说明 Avoiding locks 来解决这个问题.但 Mutex 真的开销如此大,还是被大家误解了?Matthew Dillon 写道,”Most people have the misconception that locks are slow.”, Jeff Preshing 也 写了这篇”Locks Aren’t Slow; Lock Contention Is”.
那么接下来做 3 个关于 Mutex 的 Benchmark,具体分析一下 Mutex 的开销如何,最后并利用原子操作和 semaphore 实现一个 lightweight Mutex.
一个 Mutex 仅仅从 Lock 到 Unlock 具体开销是多少,是不是占用很多时间,从 Always Use a Lightweight Mutex 从可以看到在 windows 中有两种 Mutex:Muetx 和 Critical Section, 重量级和轻量级的区别,两者的时间开销相差 25 倍多,所以一直使用轻量级的 Mutex.
这篇文章在高强度下 lock 的性能:每个线程做任何事情都占用 lock(高冲突),lock 占用极短的时间 (高频率).值得一读,但是在实际应用中,基本避免如此使用 locks.这里对 Mutex Contention 和 Mutex Frequency 都做最好和最坏场景的使用测试.
Mutex 被灌以避免使用也因为其他原因.现在有很多大家熟知的 lock-free programming 技术.Lock-free 编程非常具有挑战性,但在实际场景中获得巨大的性能.既然有 lock-free 的技术吸引我们使用它们,那么 locks 就显得索然无味了.
但也不能因此忽略 lock.因为在实际很多场景,它仍然是利器.
Lightweight Mutex Benchmark
Linux 下的 POSIX thread 是轻量级的 Mutex.基于 Linux 特有的 futex 技术,当没有其他线程竞争锁时它被优化过.使用如下简单的例子,测试一个单线程 lock 和 unlock,所有代码在 Github 上.
1 2 3 4 5 6 7 |
|
插入相应的时间代码,算出 10 万次的单线程 lock/unlock 平均时间.在不同的处理器下,结果如下:
如果假设一个线程每分钟获取 1e5 次 mutex,并且没有其他线程与它竞争.基于如下的图,可预计 0.2%到 0.4%的开销.不算差.在比较低频率下,开销基本忽略不计.之后 Build own lightweight mutex,会利用 semaphore 和一个原子操作,实现一个 lightweight mutex.
POSIX thread 与 Windows Critical Section 不同,它不仅支持线程间的同步, 还支持进程间的同步.实例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
Mutex Contention Benchmark
在测试中,产生一个不断生成随机数的线程,使用自己编制的线程安全的 Mersenne Twister 实现代码.每过一段时间,它获取和释放一个锁,获取和释放锁之间的时间每次是随机的,但是总的平均时间是提前设计好的.这个随机的过程就是个泊松分布过程,计算出产生一个随机数的平均时间 6.25 ns 在 2.93 GHz i7 上,把它作为运行单位.利用 Poisson Process 的算法决定运行多少个运行单位在获取和释放锁之间.并利用 High Resolution TimeAPI 计算时间.这个线程的代码如下,所有代码在 Github 上:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 |
|
这里模拟获取和释放 15000 次锁每秒,从 1 个线程运行到 2 个线程,最后到 4 个线程.并且验证占用锁的时间,从 0%到 100%的每次运行时间占用锁.把 1 个线程的完成的工作量作为基准数据,其他的去除以它,计算相对增益.基本测试方案如下:
1 2 3 4 5 |
|
从图中看出,随着锁占用的时间增加,并行性越来越差,直到最后占用 60%以后,单线程运行的更好.可以说,短时间的占用锁的时间,以 10%以内,系统达到很高的并行性.虽然并不是完美的,但是也接近.锁总体很快.
把这个结果放到实际中,Jeff Preshing 在 这篇 提到,实际的游戏程序中,15000 的锁每秒来自 3 个线程,占用锁的时间相对 2%.在图中很适中的区域.
Mutex Frequency Benchmark
尽管一个 lightweight mutex 有开销,但如上测试在 2.40GHz i5 上,lock/unlock 锁开销约 34.2ns ,因此 15000 锁每秒开销很低以致不是严重影响结果.那么把锁的每秒频率提高呢?
只创建 2 个线程,进行一系列的锁的每秒频率测试在 2.40GHz i5 上,从占用锁时间 10 ns(1e8/s)到 100 us(1e4/s),用单线程的占用锁时间 10 ms 作为基准工作量,其他与它比较,测试方案如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
如预想一样,对于非常高频率的锁,锁的开销开始减少实际工作量.在网络上,可以找到很多同样的测试.图中下边的线条,对于这样高的频率,也就是占用锁的时间很短,就一些 CPU 的指令,这样的情况下,当锁之间的工作如此简单,那么一个 lock-free 的实现更适合.
我们获得了一大块关于锁的性能:从它进行很好的情况,到缓慢应用的情况.在考虑实际锁的使用情况,不能说所有锁都是慢的.必须承认,很容易乱用锁,但不用太担心,任何的瓶颈问题都会在细心的 profiling 中发现.当你考虑锁是如何的稳定, 相对容易的理解它们(与 lock-free 技术相比),锁有时候其实很好用.
Build own lightweight mutex
我们也可以实现自己的简单轻量级的 mutex,但仅仅作为教育手段,理解 mutex 一些内在实现细节,实际现在操作系统都提供轻量级的 mutex,千万不要自己实现一个并实际使用,直接只用操作系统提供的即可.
网络上有很多种方法在用户层写自己的 mutex:
- roll-your-own-lightweight-mutex 利用 Windows 提供的 semaphore 和 atomic 操作实现的 mutex.
- Review of many Mutex implementations 很长的一篇文章,总结了很多种 mutex 的实现细节.
这里利用 Benaphore 技术,在 Linux 平台上利用 semaphore 和 atomic 操作实现自己的 C++版本的 lightweight mutex.这里并没有用 C++11 的原子库.所有代码在 Github 上.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
|
__sync_add_and_fetch
是一个由 GCC 内部提供的 atomic read-modify-write (RMW) 操作,它把 1 加到某个数并且返回新的数,在同一时间所有操作由一个线程原子操作完成,其他线程不能干涉,只能在后等待.这里counter_
初始化为 0,第一个线程调用Lock
将得到 1 从__sync_add_and_fetch
,然后跳过sem_wait
,一旦这个线程占用这个锁,
之后线程都将递增counter_
,获得大于 1 的数,从而调用sem_wait
等待.
之后,第一个线程完成自己的操作,调用Unlock
,__sync_sub_and_fetch
的返回值大于 1 说明有其他线程在等待这个 mutex,调用sem_post
唤醒其他线程.
底层分析与性能
上面使用了__sync_add_and_fetch
,它编译成lock xadd
指令如下.在没有竞争下的 lock/unlock 操作性能与 pthread mutex 相当.但是在 mutex 多线程竞争情况下,这个 mutex 性能没有 pthread mutex 好.
增强 Mutex 支持递归
上面简单的 lightweight mutex 的局限性是它不能递归.也就是同一个线程试图获取同样的锁两次以上,将造成死锁(deadlock).递归锁在函数调用自己时很有用.比如在内存管理代码中,可能会遇到如下代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
|
Lock
是个封装好的 C++宏,用来获取锁和自动结果当退出函数.
可以看到,当传递NULL
给Realloc
,锁被Realloc
函数获取,然后第二次被获取当Alloc
被调用.
把它扩展成可递归的锁如下,加入 2 个新成员变量,owner_
,存储当前占有线程的
ID(TID),和recursion_
,存储递归的层数.基本代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 |
|
如之前一样,第一个线程调用Lock
,设置owner_
为自己的 TID,增加
recursion_
到 1.如果同一个线程再次调用Lock
,它将同时增加
recursion_
和counter_
.
之后,第一个线程完成自己的操作,调用Unlock
,同时减少recursion_
和counter_
,
仅仅调用sem_post
唤醒其他线程当recursion_
减少到0
.如果
recursion_
仍然大于 0,意味着当前的线程仍然占有此锁在外层程序.
最后进行压力测试,建立一些线程,每个随机获取锁,随机的递归层次.代码在 Github 上.
一些细节问题:
* 在Unlock
中,设置owner_
为 0 在调用__sync_sub_and_fetch
之前,否则可能发生死锁(deadlock).比如,有两个线程 TID 是 111 和 222.
1. 线程 111 完成操作调用Unlock
,先调用__sync_sub_and_fetch
把counter_
减到 0
2. 在设置owner_
为 0 被中断,线程 222 得到运行,它调用Lock
,发现counter_
为 0,跳过sem_wait
,设置owner_=222
,完成Lock
操作.
3. 线程 222 被中断调出,线程 111 重新得到运行,设置owner_
为 0,然后完成Unlock
操作.
4. 因为此时owner_
为 0,线程 222 不能在递归占用锁,一旦它再次获取锁,形成死锁.
-
在
Unlock
中,recursion_
被拷贝到本地变量一次,之后只本地变量,比如没有在__sync_sub_and_fetch
之后重新读取她.因为在那之后它能被其他线程已经改变. -
recursion_
和owner_
没有原子操作.因为它们在调用Lock
的__sync_add_and_fetch
和调用Unlock
的__sync_sub_and_fetch
之间,线程占有锁,独占recursion_
和owner_
的读写操作,并拥有所有的 acquire and release semantics.对recursion_
和owner_
使用原子操作没必要.因为在 X86/84 的平台上,__sync_add_and_fetch
生成lock xadd
的指令,保证全部的 memory barrier,也就保证 acquire and release semantics.
Mutex VS Spinlock
提到 Mutex,往往会提到 Spinlock,因为在使用 Lock 时,会遇到如何在 Mutex 与 Spinlock 之间选择.那么接下来对比一下两者.
定义
Mutex: 如果一个线程试图获取一个 mutex,但是没有成功,因为 mutex 已经被占用, 它将进入睡眠,让其他进程运行,直到 mutex 被其他进程释放.
Spinlock: 如果一个线程试图获取一个 Spinlock, 但是没有成功,它将持续试着去获取它,直到它最终成功获取,因为它将不允许其他线程运行(然而,操作系统将强制调度其他线程).
各自缺点
Mutex: Mutex 将使得线程睡眠,然后再唤醒它们,两者都是开销比较大的操作,也就是 context switch 的开销.如果锁只是被其他线程占用非常短的时间,那么时间花在使的线程睡眠并唤醒它可能超过它使用 spinlock 持续获取锁的时间.
Spinlock: Spinlock 持续获取锁,浪费很多 CPU 时间,如果锁被其他线程占用很长时间,那么它将浪费很多时间,不如使得线程进入睡眠,让出 CPU.Spinlock 的确能优化 context switches 但会在没有 threads priority inversion 的平台上产生副作用.(但一个高优先级的线程自旋一个锁来等待一个低优先级的线程释放这个锁,就会造成死锁).在没有 Preemption 的 Uniprocessor,使用 spinlock 是没有意义的,当前只有一个线程运行,没有必要保护关键区域,也没有其他线程同时运行,释放锁给它.
所以在 Linux 下,Spinlock 在 kernel 这样实现:
- 没有打开
CONFIG_SMP
和CONFIG_PREEMPT
,spinlock 实现代码是空的. - 没有打开
CONFIG_SMP
,打开CONFIG_PREEMPT
,spinlock 仅仅是简单的关闭 preemption,足够来防止任何的 races. - 打开
CONFIG_SMP
,打开CONFIG_PREEMPT
,spinlock 实现如下代码,不断检查 lock 是否被其他线程释放:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
|
总结
Criteria | Muutex | Spinlock |
---|---|---|
机制 | 尝试获取锁.若可得到就占有.若不能,就进入睡眠等待. | 尝试获取锁.若可得到就占有.若不能,持续尝试直到获取. |
什么时候使用 | 当线程进入睡眠没有伤害.或需要等待一段足够长的时间才能获取锁. | 当线程不应该进入睡眠如中断处理等.当只需等待非常短的时间就能获取锁. |
缺点 | 引起 context switch 和 scheduling 开销. | 线程不做任何事情在获取到锁前.浪费 CPU 运行. |
大多数操作系统(包括 Solaris,Mac OS X 和 FreeBSD)使用混合的机制叫”adaptive mutex”或”hybrid mutex”.一个 hybrid mutex 首先行为和 spinlock 一样,如果不能获取锁,持续尝试获取,但过了一定的时间,它就和 mutex 一样,让线程进入睡眠.1.
-
http://stackoverflow.com/questions/5869825/when-should-one-use-a-spinlock-instead-of-mutex↩