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. Linux下httpd服务与Apache服务的查看和启动

    转:http://jingyan.baidu.com/article/63f236282d43170209ab3d43.html 这里简要介绍Linux环境中Apache也就是httpd服务的启动,查 ...

  2. TextView字体大小及颜色设置

    TextView设置文字大小及颜色: 1.1)通过xml配置 <TextView      android:layout_width="match_parent"     a ...

  3. JvisualVM、JMC监控远程服务器

    修改服务器上jmxremote.access与jmxremote.password,输入命令: find -name jmxremote.access进入该jmxremote.access文件所在目录 ...

  4. MCMC(二)马尔科夫链

    MCMC(一)蒙特卡罗方法 MCMC(二)马尔科夫链 MCMC(三)M-H采样和Gibbs采样(待填坑) 在MCMC(一)蒙特卡罗方法中,我们讲到了如何用蒙特卡罗方法来随机模拟求解一些复杂的连续积分或 ...

  5. java实现8种排序算法(详细)

    八种排序分别是:直接插入排序.希尔排序.冒泡排序.快速排序.直接选择排序.堆排序.归并排序.基数排序. 希尔排序在时间性能上优于直接插入排序,但希尔排序是一种不稳定排序. 快速排序的时间性能也优于冒泡 ...

  6. Omi新成员omi-router正式发布

    原文链接-https://github.com/AlloyTeam/omi/blob/master/tutorial omi-router omi-router是Omi框架专属的router插件,文件 ...

  7. Python3.6安装

    1.安装依赖包#yum install -y openssl openssl-devel openssl-static#yum groupinstall -y "Development to ...

  8. OOP的完美点缀—AOP之SpringAOP实现原理

    OOP的完美点缀-AOP之SpringAOP实现原理 前言 OOP与AOP OOP(Object Oriented Programming,面向对象编程),通过封装.继承将程序抽象为各个层次的对象,进 ...

  9. Java并发基础:进程和线程之由来

    转载自:http://www.cnblogs.com/dolphin0520/p/3910667.html 在前面,已经介绍了Java的基础知识,现在我们来讨论一点稍微难一点的问题:Java并发编程. ...

  10. hdu 2157 How many ways?? (可达矩阵)

    题意:给你一个有向图,从A 点到 B点恰好经过k个点的方案数 (k < 20), 可以走重复边 思路:利用离散数学中的可达矩阵,可达矩阵的K次幂便是从i到j走K步能到达的方案数 代码: #inc ...