Linux内核32-读-拷贝-更新(RCU)

简介: Linux内核32-读-拷贝-更新(RCU)

每一种技术的出现必然是因为某种需求。正因为人的本性是贪婪的,所以科技的创新才能日新月异。


1 引言


seqlock锁只能允许一个写操作,但是有些时候我们可能需要多个写操作可以并发执行。所以,Linux内核引入了读-拷贝-更新技术(英文是Read-copy update,简称RCU),它是另外一种同步技术,主要用来保护被多个CPU读取的数据结构。RCU允许多个读操作和多个写操作并发执行。更重要的是,RCU是一种免锁算法,也就是说,它没有使用共享的锁或计数器保护数据结构(但是,这儿还是主要指的读操作是无锁算法。而对于多个写操作来说,需要使用lock保护避免多个CPU的并发访问。所以,其使用场合也是比较严格的,多个写操作中的锁开销不能大于读操作采用无锁算法省下的开销)。这相对于读写自旋锁和seqlock来说,具有很大的优势,毕竟锁的申请和释放对Cache行的”窥视”和失效也是一个很大的负担。

  1. Cache行的”窥视”,指的是因为每个CPU具有局部Cache,所以硬件snoop单元必须时时刻刻在”窥视”所有的Cache行,并对其不合法的数据进行失效处理,重新从内存获取数据替换到相应的Cache行中。而在这里,如果使用了共享的lock或者计数器,那么每次对其进行写操作,必然导致相应Cache行的失效。然后重新把使用这个lock的CPU的局部Cache进行更新。


2 RCU实现


既然RCU没有使用共享数据结构,那么它是如何神奇地实现同步技术的呢?其核心思想就是限制RCU的使用范围:


  1. 只有动态分配的、通过指针进行访问的数据结构。

  1. 进入RCU保护的临界代码段的内核控制路径不能休眠。


3 基本操作


  • 对于reader,RCU的基本操作为:
  • (1)调用rcu_read_lock(),进入RCU保护的临界代码段。等价于调用preempt_disable()
  • (2)调用rcu_dereference,获取RCU保护的数据指针。然后通过该指针读取数据。当然了,在此期间读操作不能发生休眠。
  • (3)调用rcu_read_unlock(),离开RCU保护的临界代码段。等价于调用preempt_enable()
  • 对于writer,RCU的基本操作为:
  • (1)拷贝一份旧数据到新数据,修改新数据。
  • (2)调用rcu_assign_pointer(),将RCU保护的指针修改为新数据的指针。
    因为指针的修改是一个原子操作,所以不会发生读写不一致的问题。但是,需要插入一个内存屏障保证只有在数据被修改完成后,其它CPU才能看见更新的指针。尤其是当使用了自旋锁保护RCU禁止多个写操作的并发访问的时候。
  • (3)调用synchronize_rcu,等待所有的读操作都离开临界代码段,完成同步。
    RCU技术的真正问题是当写操作更新了指针后,旧数据的存储空间不能立马释放。因为,这时候读操作可能还在读取旧数据,所以,必须等到所有的可能的读操作执行rcu_read_unlock()离开临界代码段后,旧数据的存储空间才能被释放。
  • (4)调用call_rcu(),完成旧数据存储空间的回收工作。
    该函数的参数是类型为rcu_head的描述符的地址。该描述符嵌入在要回收的数据结构的内部。该函数还有一个参数就是一个回调函数,当所有的CPU处于空闲状态的时候执行这个回调函数。这个函数通常是负责旧数据存储空间的释放工作。
    有一个问题需要注意的是,这个回调函数的执行是在另一个内核线程中执行。call_rcu()函数把回调函数的地址和其参数存储在rcu_head描述符中,然后将这个描述符插入到每个CPU的回调函数列表中(这儿又体现了per-CPU变量的重要性)。每个系统时间Tick,内核都会检查局部CPU是否处于空闲状态。当所有的CPU处于空闲状态的时候,一个特殊的tasklet就会执行所有的回调函数,这个tasklet描述符存储在每个CPU的rcu_tasklet变量中。


4 使用场合


RCU是从Linux2.6版本引入的,主要使用在网络层和虚拟文件系统层。

相关文章
|
24天前
|
Linux C语言
Linux内核队列queue.h
Linux内核队列queue.h
|
4天前
|
运维 监控 Linux
提升系统稳定性:Linux内核参数调优实战
【5月更文挑战第1天】 在运维领域,保障服务器的高效稳定运行是核心任务之一。Linux操作系统因其开源、可靠和灵活的特点被广泛应用于服务器中。本文将深入探讨通过调整Linux内核参数来优化系统性能,提升服务器的稳定性和响应能力。文章首先介绍了内核参数调优的必要性和基本原则,然后详细阐述了调优过程中的关键步骤,包括如何监控当前系统状态,确定性能瓶颈,选择合适的参数进行调优,以及调优后的测试与验证。最后,文中提供了一些常见问题的解决策略和调优的最佳实践。
20 5
|
4天前
|
算法 大数据 Linux
深入理解Linux内核的进程调度机制
【4月更文挑战第30天】操作系统的核心职能之一是有效地管理和调度进程,确保系统资源的合理分配和高效利用。在众多操作系统中,Linux因其开源和高度可定制的特点,在进程调度机制上展现出独特优势。本文将深入探讨Linux内核中的进程调度器——完全公平调度器(CFS),分析其设计理念、实现原理及面临的挑战,并探索未来可能的改进方向。
|
4天前
|
算法 Linux 调度
探索Linux内核:进程调度的奥秘
【4月更文挑战第30天】 在多任务操作系统中,进程调度是核心功能之一,它决定了处理器资源的分配。本文深入分析了Linux操作系统的进程调度机制,从调度器的基本原理到复杂的调度策略,以及它们如何影响系统性能和用户体验。通过剖析进程优先级、时间片分配以及实时性要求等方面,揭示了Linux如何在众多运行着的进程中做出快速而公平的决策,确保系统的高效与稳定运行。
|
5天前
|
算法 安全 Linux
深度解析:Linux内核内存管理机制
【4月更文挑战第30天】 在操作系统领域,内存管理是核心功能之一,尤其对于多任务操作系统来说更是如此。本文将深入探讨Linux操作系统的内核内存管理机制,包括物理内存的分配与回收、虚拟内存的映射以及页面替换算法等关键技术。通过对这些技术的详细剖析,我们不仅能够理解操作系统如何高效地利用有限的硬件资源,还能领会到系统设计中的性能与复杂度之间的权衡。
|
5天前
|
弹性计算 网络协议 Shell
自动优化Linux 内核参数
【4月更文挑战第29天】
5 1
|
6天前
|
弹性计算 网络协议 Linux
自动优化 Linux 内核参数
【4月更文挑战第28天】
15 0
|
17天前
|
算法 Linux 调度
深入理解Linux内核的进程调度机制
【4月更文挑战第17天】在多任务操作系统中,进程调度是核心功能之一,它决定了处理机资源的分配。本文旨在剖析Linux操作系统内核的进程调度机制,详细讨论其调度策略、调度算法及实现原理,并探讨了其对系统性能的影响。通过分析CFS(完全公平调度器)和实时调度策略,揭示了Linux如何在保证响应速度与公平性之间取得平衡。文章还将评估最新的调度技术趋势,如容器化和云计算环境下的调度优化。
|
23天前
|
算法 Linux 调度
深度解析:Linux内核的进程调度机制
【4月更文挑战第12天】 在多任务操作系统如Linux中,进程调度机制是系统的核心组成部分之一,它决定了处理器资源如何分配给多个竞争的进程。本文深入探讨了Linux内核中的进程调度策略和相关算法,包括其设计哲学、实现原理及对系统性能的影响。通过分析进程调度器的工作原理,我们能够理解操作系统如何平衡效率、公平性和响应性,进而优化系统表现和用户体验。
|
24天前
|
存储 缓存 Linux
深入理解Linux内核页表映射分页机制原理
深入理解Linux内核页表映射分页机制原理
36 0