futex是linux内核为用户空间实现锁等同步机制而设计的同步排队(队列queueing)服务。在futex.c的注释中,futex起源于"Fast Userspace Mutex",但它并非只是一个Mutex,而是用户空间线程(pthread)同步机制的底层服务。它的基本设计有三点,1.同步对象可以在进程之使用,并且保证同步对象的内部状态访问有序,2.当同步对象没有竞争的情况下,可以在用户空间快速完成(fastpath),3.当同步对象发生竞争时,必须进入内核使用futex系统提供的排队机制进行仲裁(slowpath),仲裁胜出的任务(进程或线程)必须返回到用户空间完成同步对象的其它状态的修改 。

futex在用户空间只是一个地址对齐的整形,它是用户空间的同步委托给内核的futex系统,futex访问的同步对象内部状态。

内核中的futex系统设计了三个基本结构。futex_key,futex_queue, 以及futex_hash_bucket。

futex_key:是一个使用用户空间中futex的地址,以及地址的映射对象作为键的标识,它是futex可以在进程间共享使用的保证。

futex_hash_bucket:是一个等待队列,内核内建一般为cpu个数的8倍的等待队列。futex根据它的futex_key进行hash后的值,共享对于编号的等待队列。

futex_queue:代表一个任务(进程或线程)在某个futex上的等待关系,是等待队列的链表入口,链入到futex_hash_bucket队列中。

注意的是,每个futex_hash_bucket都有一个spinlock进行同步访问,它实现的队列是prio list,基于priority(优先级)的有序队列。

futex系统提供了基本的同步接口futex_wait(uaddr, flags, val, timeout, bitset)和futex_wake(uaddr, flags, nr_wake, bitset)。

futex_wait:就是排队然后阻塞(睡眠调度)。查找对应队列,队列入队操作,睡眠调度。

futex_wake:就是唤醒阻塞队列的任务(进程或线程)。查找对应队列,队列出队操作,唤醒任务。

调用futex系统调用时,必须传入futex的地址uaddr,以及地址的内存属性flags,同时才能标识出这个futex。futex系统根据futex的地址属性决定生成的key是否需要第二个因素,映射的对象(mm或inode)。当使用文件映射页面,进行进程间共享时,key不能使用虚拟地址进行标识,而要使用映射的物理页面号进行标识。所以在这里大致可以看到,锁的开销随跨越的边界越大而增加。访问页面管理必须进行同步,访问文件系统也必须进行同步。

所以通过uaddr和flags进行futex系统调用,就可以在主机内得到唯一的标识,让futex_wake可以找回进行futex_wait的futex。

futex设计为Fast Userspace Mutex,就是可以让用户空间的实现的Lock和Unlock操作拥有fastpath快速路径,只在没有锁竞争的时候。

如何使用futex实现在用户空间快速的锁,以pthread的lowlevellock为例。

pthread的lowlevellock将futex的锁状态定义为3种:

0,无锁,可以进行快速上锁,不需要使用futex系统调用。

1,上锁,但没有锁竞争。如果这时有其它任务需要上锁,就必须标记futex为状态锁竞争,然后使用futex系统服务。

2,锁竞争,有其它任务将要或正在内核的futex系统中排队等待着锁的使用。

在无锁状态进行Lock,以及非锁竞争的持锁状态进行Unlock都不必须进行futex系统调用。

上锁和解锁,对锁状态的修改在用户空间中完成,锁竞争通过futex系统调用完成。futex系统调用并不进行上锁和解锁操作,只负责排队服务,也就是说futex系统调用并不理会也不必要知道用户空间是如何定义futex的锁状态的。

futex系统的排队服务,通过hash链表,每一个等待都会分配到由futex_key计算出的hash链表上进行排队。hash链表是共享的链表,多个futex的等待可能会共享同一个hash链表,而一个futex的等待只会分配到相同的hash链表上。因为futex_key是唯一的。每一个hash链表都是一个prio有序链表,等待在排队时需要根据阻塞的任务的优先级进行排序插入。

上面是futex提供的最基本的功能,pthread_mutex使用futex_wait和futex_wake实现它的lock和unlock操作。

futex.c文件的注释告诉我们,futex从纳入到linux内核开始就不断增加它的功能,使它更加通用,更好地支持用户空间编程的同步服务。

futex requeueing 随futex纳入到linux内核添加。

robust futex,更加健壮的futex,持有基于futex的锁的任务退出而没有释放,使阻塞于这个锁的其它任务得以解放。

pi-futex,使用rt_mutex代理排队服务,同时使用rt_mutex的PI优先级继承算法,增加futex的功能。

private futexes,优化进程内的futex,因为futex最初被设计,局限于当时的内核,是基于多进程考虑的。

Requeue-PI,一种对non-pi-futex到pi-futex之间的requeue进行优化。

requeue这一功能广泛使用在pthread_cond,换过来说就是为实现pthread_cond而设计的。

linux 内核的futex的更多相关文章

  1. linux 内核的futex pi-support,即pi-futex使用rt_mutex委托

    futex的pi-support,也就是为futex添加pi算法解决优先级逆转的能力,使用pi-support的futex又称为pi-futex.在linux内核的同步机制中,有一个pi算法的成例,就 ...

  2. linux 内核的futex - requeue 以及 requeue-pi

    futex为更好支持pthread_cond的实现(,最主要是broadcast),设计了requeue功能,并以futex系统调用提供操作接口,包括一对配对的操作 futex_wait_requeu ...

  3. linux内核级同步机制--futex

    在面试中关于多线程同步,你必须要思考的问题 一文中,我们知道glibc的pthread_cond_timedwait底层是用linux futex机制实现的. 理想的同步机制应该是没有锁冲突时在用户态 ...

  4. 用Qemu模拟vexpress-a9 (三)--- 实现用u-boot引导Linux内核

    环境介绍 Win7 64 + Vmware 11 + ubuntu14.04 32 u-boot 版本:u-boot-2015-04 Linux kernel版本:linux-3.16.y busyb ...

  5. linux内核数据结构学习总结

    目录 . 进程相关数据结构 ) struct task_struct ) struct cred ) struct pid_link ) struct pid ) struct signal_stru ...

  6. 戴文的Linux内核专题:07内核配置(3)

    转自Linux中国 OK,我们还继续配置内核.还有更多功能等待着去配置. 下一个问题(Enable ELF core dumps (ELF_CORE))询问的是内核是否可以生成内核转储文件.这会使内核 ...

  7. linux内核编程笔记【原创】

    以下为本人学习笔记,如有转载请注明出处,谢谢 DEFINE_MUTEX(buzzer_mutex); mutex_lock(&buzzer_mutex); mutex_unlock(& ...

  8. Linux 内核综述

    一.什么是Linux内核: 内核->操作系统中最重要的部分,内核将在系统引导时被装载进RAM,其中包含了很多关键的例程,以操作系统.内核是OS最为关键的部分,人们常将OS(操作系统)与内核等同. ...

  9. 基于tiny4412的Linux内核移植(支持device tree)(三)

    作者信息 作者: 彭东林 邮箱:pengdonglin137@163.com QQ:405728433 平台简介 开发板:tiny4412ADK + S700 + 4GB Flash 要移植的内核版本 ...

随机推荐

  1. TCP/IP笔记(二)TCP/IP简介

    上回,主要介绍了下协议和OSI参考模型,并简单了解下网络构成要素,这回该说说TCP/IP了 互联网与TCP/IP的关系   互联网进行通信时,需要相应的网络协议,TCP/IP原本就是为使用互联网而开发 ...

  2. JS Math.round()方法原理

    请先测试代码: <!doctype html> <html lang="en"> <head> <meta charset="U ...

  3. iOS坐标转换失败?UIKit的屠神坑

    使用UIKit的坐标转换方法convertxxx,千万要注意: 一个控件可以转换子控件上的某个点,到另外一个控件上 但是不能转换自己本身的点,到另外一个控件上,否则会数量加倍 所以,一个控件若想转换本 ...

  4. (转)KMP算法实现。超级赞!见过的最容易理解的

    网上有很多讲解KMP算法的博客,我就不浪费时间再写一份了.直接推荐一个当初我入门时看的博客吧:http://www.cnblogs.com/yjiyjige/p/3263858.html这位同学用详细 ...

  5. 老司机带路——15个Android撸代码常见的坑

    老司机为何能够成为老司机,不是因为开车开得多,而是撸多了… 0x00 使用 startActivityForResult 后在 onActivityResult 中没有正确回调到 Activity.R ...

  6. Unity 3D Framework Designing(7)——IoC工厂理念先行

    一谈到 『IoC』,有经验的程序员马上会联想到控制反转,将创建对象的责任反转给工厂.IoC是依赖注入 『DI』 的核心,大名鼎鼎的Spring框架就是一个非常卓越的的控制反转.依赖注入框架.遗憾的是, ...

  7. java 解压 zip 包并删除

    需求是这样的,  在服务器上有 运营上传的zip 包,内容是用户的照片,我需要做的是 获取这些照片上传,并保存到 数据库. 这里面的 上传照片,保存数据库都不难,主要问题是解压zip包,和删除zip ...

  8. android 学习笔记(1)

    内容来源:高成珍.钟元生<Android编程经典案例>学习笔记 表格布局——TableLayout 表格布局以行和列的方式来管理界面的布局,但并不能明确声明包含几行几列.可通过TableR ...

  9. (iOS)开发中收集的小方法

    1.颜色转变成图片 - (UIImage *)createImageWithColor:(UIColor *)color {     CGRect rect = CGRectMake(0.0f, 0. ...

  10. 简单 fibonacci 函数

    public static int fibonacci(int n){  if(n<=1) return 1;  else {     return fibonacci(n-1)+fibonac ...