linux内核 —— 读写信号量实验

简介: linux内核 —— 读写信号量实验

内核版本:5.14

代码路径:

  • kernel/locking/rwsem.c
  • include/linux/rwsem.h

概述

读写信号量具有如下特点:

  • 是一种睡眠锁
  • 可以有多个read持有读信号量
  • 只允许一个write持有持有写信号量
  • read和write之间互斥
  • write和write之间互斥
  • 以严格的FIFO顺序处理等待读/写信号量的所有进程。如果read或write进程发现信号量关闭,这些进程就被插入到信号量等待队列链表的末尾。
  • 当信号量被释放时,检查处于等待队列链表第一个位置的进程。第一个进程被唤醒。如果时一个写者进程,等待队列上的其他的进程就继续睡眠。如果是一个读者进程,那么紧跟第一个进程的其他所有读者进程也被唤醒并获得信号量。不过,在写者进程之后排队的读者进程继续睡眠

数据结构

struct rw_semaphore {
  atomic_long_t count;
  atomic_long_t owner;
#ifdef CONFIG_RWSEM_SPIN_ON_OWNER
  struct optimistic_spin_queue osq; /* spinner MCS lock */
#endif
  raw_spinlock_t wait_lock;
  struct list_head wait_list;
};
  • owner
    bit0:是否是读者持有信号,是的话为1
    bit1:是否可以在读者持有的信号量上自旋等待
    bit2~63:当读者持有信号时,owner中会记录获取到信号的最后一个读者进程的task_struct。如果写者持有信号,那么owner记录的是该写者进程的task_struct
  • count
    bit0:是否是写者持有信号
    bit1:是否有进程阻塞在等待队列中
    bit2:handoff
    bit8~62:读者的数量
    bit63:read fail

实验

实现一个申请和释放读写信号量rwsem_test的内核模块,然后通过应用来控制申请或者释放,期间使用crash工具查看rwsem_test信号量的内部状态。

  • 获取读写信号量rwsem_test的地址
root@ubuntu-vm:~# crash /mnt/linux-5.14/vmlinux
KERNEL: /mnt/linux-5.14/vmlinux
DUMPFILE: /dev/mem
CPUS: 12
DATE: Sat Mar 26 09:34:29 CST 2022
UPTIME: 00:29:52
LOAD AVERAGE: 0.72, 0.23, 0.08
TASKS: 176
NODENAME: ubuntu-vm
RELEASE: 5.14.0+
VERSION: #3 SMP Fri Mar 25 08:57:39 PDT 2022
MACHINE: x86_64  (3599 Mhz)
MEMORY: 16 GB
PID: 578
COMMAND: "crash"
TASK: ffff8de0c595ec80  [THREAD_INFO: ffff8de0c595ec80]
CPU: 2
STATE: TASK_RUNNING (ACTIVE)
crash> sym rwsem_test
ffffffffa5645b00 (d) rwsem_test
  • 查看读写信号量的状态
crash> rw_semaphore.count,owner,wait_list -x ffffffffa5645b00
count = {
    counter = 0x0
  },
  owner = {
    counter = 0x0
  },
wait_list = {
    next = 0xffffffffa5645b18 <rwsem_test+24>,
    prev = 0xffffffffa5645b18 <rwsem_test+24>
  }

可以看到初始状态count和owner都是0

  • 查看等待队列的地址
crash> rw_semaphore.wait_list -ox ffffffffa5645b00
struct rw_semaphore {
  [ffffffffa5645b18] struct list_head wait_list;
}
  • 遍历等待队列

方式一:

crash> list -o rwsem_waiter.list -O rw_semaphore.wait_list -s rwsem_waiter.task,type -h ffffffffa5645b00
(empty)

方式二:

crash> list -o rwsem_waiter.list  -s rwsem_waiter.task,type -H ffffffffa5645b18
(empty)

或者: 由于list在rwsem_waiter中的偏移量为0

crash> list  -s rwsem_waiter.task,type -H ffffffffa5645b18
(empty)
  • 实验数据

说明:

r: 申请读信号
w:申请写信号
R:释放读信号
W:释放写信号
操作序列 状态 队列 备注
初始 count:0 owner:0
r1 count:0x100
owner:0xffff9ba7465b9f01
count表示读者数量为1
owner的bit0为1,表示被读者持有,读者的task_struct为0xffff9ba7465b9f00
w1 count:0x1
owner:0xffff9ba7465b9f00
count的bit0为1,表示写者持有
owner记录的时这个写者的task_struct:0xffff9ba7465b9f00
r1R1 count:0x0
owner:0xffff9ba7465b9f01
w1W1 count:0
owner:0
r1r2 count:0x200
owner:0xffff9ba7465bae81
r1r2R1 count:0x100
owner:0xffff9ba7465bae81
r1r2R1R2 count:0x0
owner:0xffff9ba7465bae81
r1r1 count:0x200
owner:0xffff9ba7465b9f01
w1w2 count:0x3
owner:0xffff9ba7465b9f00
task = 0xffff9ba7465bae80,
type = RWSEM_WAITING_FOR_WRITE
w1w2W1 count:0x1
owner:0xffff9ba7465bae80
w1w2W1W2 count:0
owner:0
r1w1 count:0x102
owner:0xffff9ba7465b9f03
task = 0xffff9ba7465bae80,
type = RWSEM_WAITING_FOR_WRITE
r1w1r2 count:0x102
owner:0xffff9ba7465b9f03
task = 0xffff9ba7465bae80,
type = RWSEM_WAITING_FOR_WRITE
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_READ
r1w1r2R1 count:0x3
owner:0xffff9ba7465bae80
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_READ
r1w1r2R1W1 count:0x100
owner:0xffff9ba748febe01
r1w1r2R1W1R2 count:0x0
owner:0xffff9ba748febe01
w1r1 count:0x3
owner:0xffff9ba7465b9f00
task = 0xffff9ba7465bae80,
type = RWSEM_WAITING_FOR_READ
w1r1W1 count:0x100
owner:0xffff9ba7465bae81
w1r1w2 count:0x3
owner:0xffff9ba7465b9f00
task = 0xffff9ba7465bae80,
type = RWSEM_WAITING_FOR_READ
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_WRITE
w1r1w2W1 count:0x102
owner:0xffff9ba7465bae81
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_WRITE
w1r1w2W1R1 count:0x1
owner:0xffff9ba748febe00
w1r1w2W1R1W2 count:0
owner:0
r1w1w2 count:0x102
owner:0xffff9ba7465b9f03
task = 0xffff9ba7465bae80,
type = RWSEM_WAITING_FOR_WRITE
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_WRITE
r1w1w2R1 count:0x3
owner:0xffff9ba7465bae80
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_WRITE
r1w1w2R1W1 count:0x1
owner:0xffff9ba748febe00
r1w1w2R1W1W2 count:0
owner:0
r1r2w1 count:0x202
owner:0xffff9ba7465bae83
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_WRITE
r1r2w1R2 count:0x102
owner:0xffff9ba7465bae83
task = 0xffff9ba748febe00,
type = RWSEM_WAITING_FOR_WRITE
r1r2w1R2R1 count:0x1
owner:0xffff9ba748febe00
r1r2w1R2R1W1 count:0
owner:0

玩完。

相关文章
|
3天前
|
算法 Linux 调度
深入理解Linux内核调度器:从基础到优化####
本文旨在通过剖析Linux操作系统的心脏——内核调度器,为读者揭开其高效管理CPU资源的神秘面纱。不同于传统的摘要概述,本文将直接以一段精简代码片段作为引子,展示一个简化版的任务调度逻辑,随后逐步深入,详细探讨Linux内核调度器的工作原理、关键数据结构、调度算法演变以及性能调优策略,旨在为开发者与系统管理员提供一份实用的技术指南。 ####
19 4
|
4天前
|
缓存 负载均衡 Linux
深入理解Linux内核调度器
本文探讨了Linux操作系统核心组件之一——内核调度器的工作原理和设计哲学。不同于常规的技术文章,本摘要旨在提供一种全新的视角来审视Linux内核的调度机制,通过分析其对系统性能的影响以及在多核处理器环境下的表现,揭示调度器如何平衡公平性和效率。文章进一步讨论了完全公平调度器(CFS)的设计细节,包括它如何处理不同优先级的任务、如何进行负载均衡以及它是如何适应现代多核架构的挑战。此外,本文还简要概述了Linux调度器的未来发展方向,包括对实时任务支持的改进和对异构计算环境的适应性。
23 6
|
5天前
|
缓存 Linux 开发者
Linux内核中的并发控制机制:深入理解与应用####
【10月更文挑战第21天】 本文旨在为读者提供一个全面的指南,探讨Linux操作系统中用于实现多线程和进程间同步的关键技术——并发控制机制。通过剖析互斥锁、自旋锁、读写锁等核心概念及其在实际场景中的应用,本文将帮助开发者更好地理解和运用这些工具来构建高效且稳定的应用程序。 ####
21 5
|
5天前
|
算法 Unix Linux
深入理解Linux内核调度器:原理与优化
本文探讨了Linux操作系统的心脏——内核调度器(Scheduler)的工作原理,以及如何通过参数调整和代码优化来提高系统性能。不同于常规摘要仅概述内容,本摘要旨在激发读者对Linux内核调度机制深层次运作的兴趣,并简要介绍文章将覆盖的关键话题,如调度算法、实时性增强及节能策略等。
|
5天前
|
缓存 运维 网络协议
深入Linux内核架构:操作系统的核心奥秘
深入Linux内核架构:操作系统的核心奥秘
22 2
|
网络协议 NoSQL Linux
阿里云 Linux 内核优化实战(sysctl.conf 和 ulimits )
一、sysctl.conf优化Linux系统内核参数的配置文件为 /etc/sysctl.conf 和 /etc/sysctl.d/ 目录。其读取顺序为: /etc/sysctl.d/ 下面的文件按照字母排序;然后读取 /etc/sysctl.conf 。
8597 1
|
5天前
|
缓存 资源调度 安全
深入探索Linux操作系统的心脏——内核配置与优化####
本文作为一篇技术性深度解析文章,旨在引领读者踏上一场揭秘Linux内核配置与优化的奇妙之旅。不同于传统的摘要概述,本文将以实战为导向,直接跳入核心内容,探讨如何通过精细调整内核参数来提升系统性能、增强安全性及实现资源高效利用。从基础概念到高级技巧,逐步揭示那些隐藏在命令行背后的强大功能,为系统管理员和高级用户打开一扇通往极致性能与定制化体验的大门。 --- ###
26 9
|
8天前
|
机器学习/深度学习 负载均衡 算法
深入探索Linux内核调度机制的优化策略###
本文旨在为读者揭开Linux操作系统中至关重要的一环——CPU调度机制的神秘面纱。通过深入浅出地解析其工作原理,并探讨一系列创新优化策略,本文不仅增强了技术爱好者的理论知识,更为系统管理员和软件开发者提供了实用的性能调优指南,旨在促进系统的高效运行与资源利用最大化。 ###
|
7天前
|
监控 网络协议 算法
Linux内核优化:提升系统性能与稳定性的策略####
本文深入探讨了Linux操作系统内核的优化策略,旨在通过一系列技术手段和最佳实践,显著提升系统的性能、响应速度及稳定性。文章首先概述了Linux内核的核心组件及其在系统中的作用,随后详细阐述了内存管理、进程调度、文件系统优化、网络栈调整及并发控制等关键领域的优化方法。通过实际案例分析,展示了这些优化措施如何有效减少延迟、提高吞吐量,并增强系统的整体健壮性。最终,文章强调了持续监控、定期更新及合理配置对于维持Linux系统长期高效运行的重要性。 ####
|
10天前
|
人工智能 算法 大数据
Linux内核中的调度算法演变:从O(1)到CFS的优化之旅###
本文深入探讨了Linux操作系统内核中进程调度算法的发展历程,聚焦于O(1)调度器向完全公平调度器(CFS)的转变。不同于传统摘要对研究背景、方法、结果和结论的概述,本文创新性地采用“技术演进时间线”的形式,简明扼要地勾勒出这一转变背后的关键技术里程碑,旨在为读者提供一个清晰的历史脉络,引领其深入了解Linux调度机制的革新之路。 ###