Linux的新式线程同步原语——Futex在我的上一篇文章《本地POSIX线程库》中,提到了Futex一词,发现好多读者误以为这是我的笔误,将Mutex错写为Futex了。其实Futex是Linux的一种全新的线程同步原语。本文将为您解读高效的Futex。
Futex是fast userspace mutex的缩写,意思是快速用户空间互斥体。Linux内核把它们作为快速的用户空间的锁和信号量的预制构件提供给开发者。Futex非常基础,借助其自身的优异性能,构建更高级别的锁的抽象,如POSIX互斥体。大多数程序员并不需要直接使用Futex,它一般用来实现像NPTL这样的系统库。
Futex其实就是可以由不同进程所共享使用的一块内存。在这些进程中,并不需要有相同的地址。一个光秃秃的Futex,与信号量的语义是相同的;它有一个可以被原子增减的计数器;进程可以等待这个计数器值变为正数。
Futex的操作完全是在用户空间,不需要进行上下文切换。内核仅在发生竞争的时候作一个公断。它的本质结构是一个对齐的整数,仅由原子的汇编指令操作。进程们可以通过mmap,在共享段中访问它,或许是由于进程们共享了内存空间,通常就把这样的应用程序称为多线程程序。
任何Futex的操作都起始于用户空间,但是在必要的时候还是需要使用某些系统调用与内核通讯的。
要“up”一个Futex,需执行正确的汇编指令使主CPU原子的递增这个整数。然后,检查它是否已从0变为1,这说明没有等待进程,操作完成。这是无竞争情况,这很快而且应该很普遍。
在竞争情况下,原子增量从-1(或者是其他的负数)开始变化。如果是这样的话,说明有等待进程。用户空间应立即将计数器设置为1,并通知内核唤醒那些正使用FUTEX_WAKE操作的等待进程。
正等待一个Futex时,“down”它是一个相反的操作。原子递减这个计数器,并检查它是否变为0,使操作完成,Futex无竞争。在所有其他的情况下,进程会设置计数器为-1,并请求内核等待其他进程“up”Futex。这是通过FUTEX_WAIT操作完成的。
Futex是由Hubertus Franke(IBM Thomas J. Watson研究中心),Matthew Kirkwood,Ingo Molnar (Red Hat)和Rusty Russell (IBM Linux科技中心)设计并维护的。最初的Futex的支持是从Linux2.5.7开始的,但是以上述语义有些不同。当前的语义是从Linux2.5.40获得的,在Linux2.5.70至2.6.7,已经可以获得更多的附加功能了。