~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/Documentation/translations/zh_CN/locking/mutex-design.rst

Version: ~ [ linux-6.12-rc7 ] ~ [ linux-6.11.7 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.60 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.116 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.171 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.229 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.285 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.323 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.12 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  1 .. SPDX-License-Identifier: GPL-2.0
  2 .. include:: ../disclaimer-zh_CN.rst
  3 
  4 :Original: Documentation/locking/mutex-design.rst
  5 
  6 :翻译:
  7 
  8   唐艺舟 Tang Yizhou <tangyeechou@gmail.com>
  9 
 10 ================
 11 通用互斥锁子系统
 12 ================
 13 
 14 :初稿:
 15 
 16   Ingo Molnar <mingo@redhat.com>
 17 
 18 :更新:
 19 
 20   Davidlohr Bueso <davidlohr@hp.com>
 21 
 22 什么是互斥锁?
 23 --------------
 24 
 25 在Linux内核中,互斥锁(mutex)指的是一个特殊的加锁原语,它在共享内存系统上
 26 强制保证序列化,而不仅仅是指在学术界或类似的理论教科书中出现的通用术语“相互
 27 排斥”。互斥锁是一种睡眠锁,它的行为类似于二进制信号量(semaphores),在
 28 2006年被引入时[1],作为后者的替代品。这种新的数据结构提供了许多优点,包括更
 29 简单的接口,以及在当时更少的代码量(见缺陷)。
 30 
 31 [1] https://lwn.net/Articles/164802/
 32 
 33 实现
 34 ----
 35 
 36 互斥锁由“struct mutex”表示,在include/linux/mutex.h中定义,并在
 37 kernel/locking/mutex.c中实现。这些锁使用一个原子变量(->owner)来跟踪
 38 它们生命周期内的锁状态。字段owner实际上包含的是指向当前锁所有者的
 39 `struct task_struct *` 指针,因此如果无人持有锁,则它的值为空(NULL)。
 40 由于task_struct的指针至少按L1_CACHE_BYTES对齐,低位(3)被用来存储额外
 41 的状态(例如,等待者列表非空)。在其最基本的形式中,它还包括一个等待队列和
 42 一个确保对其序列化访问的自旋锁。此外,CONFIG_MUTEX_SPIN_ON_OWNER=y的
 43 系统使用一个自旋MCS锁(->osq,译注:MCS是两个人名的合并缩写),在下文的
 44 (ii)中描述。
 45 
 46 准备获得一把自旋锁时,有三种可能经过的路径,取决于锁的状态:
 47 
 48 (i) 快速路径:试图通过调用cmpxchg()修改锁的所有者为当前任务,以此原子化地
 49     获取锁。这只在无竞争的情况下有效(cmpxchg()检查值是否为0,所以3个状态
 50     比特必须为0)。如果锁处在竞争状态,代码进入下一个可能的路径。
 51 
 52 (ii) 中速路径:也就是乐观自旋,当锁的所有者正在运行并且没有其它优先级更高的
 53      任务(need_resched,需要重新调度)准备运行时,当前任务试图自旋来获得
 54      锁。原理是,如果锁的所有者正在运行,它很可能不久就会释放锁。互斥锁自旋体
 55      使用MCS锁排队,这样只有一个自旋体可以竞争互斥锁。
 56 
 57      MCS锁(由Mellor-Crummey和Scott提出)是一个简单的自旋锁,它具有一些
 58      理想的特性,比如公平,以及每个CPU在试图获得锁时在一个本地变量上自旋。
 59      它避免了常见的“检测-设置”自旋锁实现导致的(CPU核间)缓存行回弹
 60      (cacheline bouncing)这种昂贵的开销。一个类MCS锁是为实现睡眠锁的
 61      乐观自旋而专门定制的。这种定制MCS锁的一个重要特性是,它有一个额外的属性,
 62      当自旋体需要重新调度时,它们能够退出MCS自旋锁队列。这进一步有助于避免
 63      以下场景:需要重新调度的MCS自旋体将继续自旋等待自旋体所有者,即将获得
 64      MCS锁时却直接进入慢速路径。
 65 
 66 (iii) 慢速路径:最后的手段,如果仍然无法获得锁,该任务会被添加到等待队列中,
 67       休眠直到被解锁路径唤醒。在通常情况下,它以TASK_UNINTERRUPTIBLE状态
 68       阻塞。
 69 
 70 虽然从形式上看,内核互斥锁是可睡眠的锁,路径(ii)使它实际上成为混合类型。通过
 71 简单地不中断一个任务并忙着等待几个周期,而不是立即睡眠,这种锁已经被认为显著
 72 改善一些工作负载的性能。注意,这种技术也被用于读写信号量(rw-semaphores)。
 73 
 74 语义
 75 ----
 76 
 77 互斥锁子系统检查并强制执行以下规则:
 78 
 79     - 每次只有一个任务可以持有该互斥锁。
 80     - 只有锁的所有者可以解锁该互斥锁。
 81     - 不允许多次解锁。
 82     - 不允许递归加锁/解锁。
 83     - 互斥锁只能通过API进行初始化(见下文)。
 84     - 一个任务不能在持有互斥锁的情况下退出。
 85     - 持有锁的内存区域不得被释放。
 86     - 被持有的锁不能被重新初始化。
 87     - 互斥锁不能用于硬件或软件中断上下文,如小任务(tasklet)和定时器。
 88 
 89 当CONFIG DEBUG_MUTEXES被启用时,这些语义将被完全强制执行。此外,互斥锁
 90 调试代码还实现了一些其它特性,使锁的调试更容易、更快速:
 91 
 92     - 当打印到调试输出时,总是使用互斥锁的符号名称。
 93     - 加锁点跟踪,函数名符号化查找,系统持有的全部锁的列表,打印出它们。
 94     - 所有者跟踪。
 95     - 检测自我递归的锁并打印所有相关信息。
 96     - 检测多任务环形依赖死锁,并打印所有受影响的锁和任务(并且只限于这些任务)。
 97 
 98 
 99 接口
100 ----
101 静态定义互斥锁::
102 
103    DEFINE_MUTEX(name);
104 
105 动态初始化互斥锁::
106 
107    mutex_init(mutex);
108 
109 以不可中断方式(uninterruptible)获取互斥锁::
110 
111    void mutex_lock(struct mutex *lock);
112    void mutex_lock_nested(struct mutex *lock, unsigned int subclass);
113    int  mutex_trylock(struct mutex *lock);
114 
115 以可中断方式(interruptible)获取互斥锁::
116 
117    int mutex_lock_interruptible_nested(struct mutex *lock,
118                                        unsigned int subclass);
119    int mutex_lock_interruptible(struct mutex *lock);
120 
121 当原子变量减为0时,以可中断方式(interruptible)获取互斥锁::
122 
123    int atomic_dec_and_mutex_lock(atomic_t *cnt, struct mutex *lock);
124 
125 释放互斥锁::
126 
127    void mutex_unlock(struct mutex *lock);
128 
129 检测是否已经获取互斥锁::
130 
131    int mutex_is_locked(struct mutex *lock);
132 
133 缺陷
134 ----
135 
136 与它最初的设计和目的不同,'struct mutex' 是内核中最大的锁之一。例如:在
137 x86-64上它是32字节,而 'struct semaphore' 是24字节,rw_semaphore是
138 40字节。更大的结构体大小意味着更多的CPU缓存和内存占用。
139 
140 
141 何时使用互斥锁
142 --------------
143 
144 总是优先选择互斥锁而不是任何其它锁原语,除非互斥锁的严格语义不合适,和/或临界区
145 阻止锁被共享。

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php