CppGuide社区 CppGuide社区
首页
  • 最新谷歌C++风格指南(含C++17/20)
  • C++17详解
  • C++20完全指南
  • C++23快速入门
  • C++语言面试问题集锦
  • 🔥C/C++后端开发常见面试题解析 (opens new window)
  • 网络编程面试题 (opens new window)
  • 网络编程面试题 答案详解 (opens new window)
  • 聊聊WebServer作面试项目那些事儿 (opens new window)
  • 字节跳动面试官现身说 (opens new window)
  • 技术简历指南 (opens new window)
  • 🔥交易系统开发岗位求职与面试指南 (opens new window)
  • 第1章 高频C++11重难点知识解析
  • 第2章 Linux GDB高级调试指南
  • 第3章 C++多线程编程从入门到进阶
  • 第4章 C++网络编程重难点解析
  • 第5章 网络通信故障排查常用命令
  • 第6章 高性能网络通信协议设计精要
  • 第7章 高性能服务结构设计
  • 第8章 Redis网络通信模块源码分析
  • 第9章 后端服务重要模块设计探索
  • 🚀 全部章节.pdf 下载 (opens new window)
  • 源码分析系列

    • leveldb源码分析
    • libevent源码分析
    • Memcached源码分析
    • TeamTalk源码分析
    • 优质源码分享 (opens new window)
    • 🔥远程控制软件gh0st源码分析
  • 从零手写C++项目系列

    • C++游戏编程入门(零基础学C++)
    • 🔥使用C++17从零开发一个调试器 (opens new window)
    • 🔥使用C++20从零构建一个完整的低延迟交易系统 (opens new window)
    • 🔥使用C++从零写一个C语言编译器 (opens new window)
    • 从零用C语言写一个Redis
  • Windows 10系统编程
  • Go语言特性

    • Go开发实用指南
    • Go系统接口编程
    • 高效Go并发编程
    • Go性能调优
    • Go项目架构设计
  • Go项目实战

    • 使用Go从零开发一个数据库
    • 🔥使用Go从零开发一个编译器 (opens new window)
    • 🔥使用Go从零开发一个解释器 (opens new window)
    • 🔥用Go从零写一个编排器(类Kubernetes) (opens new window)
  • Rust编程

    • Rust编程指南
  • 数据库

    • SQL零基础指南
    • MySQL开发与调试指南
  • Linux内核

    • 心中的内核 —— 在阅读内核代码之前先理解内核
    • 🔥Linux 5.x内核开发与调试 完全指南 (opens new window)
    • TCP源码实现超详细注释版.pdf (opens new window)
GitHub (opens new window)
首页
  • 最新谷歌C++风格指南(含C++17/20)
  • C++17详解
  • C++20完全指南
  • C++23快速入门
  • C++语言面试问题集锦
  • 🔥C/C++后端开发常见面试题解析 (opens new window)
  • 网络编程面试题 (opens new window)
  • 网络编程面试题 答案详解 (opens new window)
  • 聊聊WebServer作面试项目那些事儿 (opens new window)
  • 字节跳动面试官现身说 (opens new window)
  • 技术简历指南 (opens new window)
  • 🔥交易系统开发岗位求职与面试指南 (opens new window)
  • 第1章 高频C++11重难点知识解析
  • 第2章 Linux GDB高级调试指南
  • 第3章 C++多线程编程从入门到进阶
  • 第4章 C++网络编程重难点解析
  • 第5章 网络通信故障排查常用命令
  • 第6章 高性能网络通信协议设计精要
  • 第7章 高性能服务结构设计
  • 第8章 Redis网络通信模块源码分析
  • 第9章 后端服务重要模块设计探索
  • 🚀 全部章节.pdf 下载 (opens new window)
  • 源码分析系列

    • leveldb源码分析
    • libevent源码分析
    • Memcached源码分析
    • TeamTalk源码分析
    • 优质源码分享 (opens new window)
    • 🔥远程控制软件gh0st源码分析
  • 从零手写C++项目系列

    • C++游戏编程入门(零基础学C++)
    • 🔥使用C++17从零开发一个调试器 (opens new window)
    • 🔥使用C++20从零构建一个完整的低延迟交易系统 (opens new window)
    • 🔥使用C++从零写一个C语言编译器 (opens new window)
    • 从零用C语言写一个Redis
  • Windows 10系统编程
  • Go语言特性

    • Go开发实用指南
    • Go系统接口编程
    • 高效Go并发编程
    • Go性能调优
    • Go项目架构设计
  • Go项目实战

    • 使用Go从零开发一个数据库
    • 🔥使用Go从零开发一个编译器 (opens new window)
    • 🔥使用Go从零开发一个解释器 (opens new window)
    • 🔥用Go从零写一个编排器(类Kubernetes) (opens new window)
  • Rust编程

    • Rust编程指南
  • 数据库

    • SQL零基础指南
    • MySQL开发与调试指南
  • Linux内核

    • 心中的内核 —— 在阅读内核代码之前先理解内核
    • 🔥Linux 5.x内核开发与调试 完全指南 (opens new window)
    • TCP源码实现超详细注释版.pdf (opens new window)
GitHub (opens new window)
  • 心中的内核——在阅读内核代码之前先理解内核 专栏说明
  • 01 内核不是进程,而是系统
  • 02 为进程服务:内核的首要职责
  • 03 代码之前的概念图
  • 04 作为分层系统的内核:虚拟、映射、隔离、控制
  • 05 单体形式,协同行为:真正的内核模型
  • 06 内核对象揭示设计——函数仅执行设计
  • 07 无冲突的代码——内核如何在并发风暴中保持安全
  • 08 间接的力量——一个内核如何为所有进程服务
  • 09 内核的设备模型:硬件如何成为/dev
  • 10 内核如何看待内存:不是映射,而是责任
  • 11 内存不是一个地方,而是一个系统
  • 12 内核始终存在——你知道它在哪里吗?
  • 13 不只是代码执行:内核实际执行的内容
  • 14 boot结束之处:内核开始之处
  • 15 从vmlinuz到eBPF:Linux内核内部实际运行的内容
  • 16 无状态CPU,有状态内核:执行如何被协调
  • 17 内核构建的内容——逐层构建
  • 18 内核执行路径:在哪里运行,以及为何重要
  • 19 追踪执行的模板
  • 20 中断不是干扰,而是设计
  • 21 执行是逻辑的,位置是物理的
  • 22 不仅仅是一段代码:每个内核路径内部的过程
  • 23 内核如何自我通信——内部通信工具
  • 24 内核模块仅通过导出符号相互认知
  • 25 搭建组件之间的桥梁
  • 26 libc之外:用户空间与内核的真实通信方式
  • 27 CPU不移动数据——但没有CPU,什么都无法移动
  • 28 时间与精度:内核眼中的CPU执行
  • 29 内核在虚拟化中的角色:理解KVM
  • 30 两个世界,一个CPU:虚拟化中的root操作和非root操作
  • 31 内核与VirtIO:无需模拟的网络驱动程序
  • 32 一切仍由操作系统掌控
  • 33 对齐即理解
  • 34 如果内核不是由 Linus 创建和维护,会怎样?
  • 35 配置并非定制,而是内核的身份标识
  • 36 内存生命周期与塑造它的角色
  • 37 中断如何在不变中演变
  • 38 并发之外的同步机制
  • 39 这从不关乎炒作,始终关乎硬件
  • 40 从意图到I/O:内核如何看待文件、磁盘和设备
  • 41 心中的内核——效率至上而非历史遗留原因:为什么内核仍用C语言开发

20 中断不是干扰,而是设计

# 20 中断不是干扰,而是设计

“中断”一词意味着打断、干扰,是意外之事。但在内核以及其底层架构中,中断并非如此。它不是混乱,不是冲突,而是系统主张行动权的方式,无论当前运行的是什么。它是结构化的、可预期的,受设计约束。

在内核处理中断之前,它早已做出决策:中断将去往何处、如何处理,以及谁不负责处理它。

每条中断线都注册有处理程序。内核设置向量表、初始化本地和I/O APIC、分配优先级、屏蔽或取消屏蔽中断线,并将每个源路由到逻辑CPU。这些不是反应,而是声明。系统预先构建了时间、设备和其他CPU可能进行干预的确切路径。

当中断发生时(无论是定时器滴答、网络数据包还是来自另一个核心的关闭请求),CPU切换到内核模式。它保存当前执行状态并开始执行处理程序。但此处理程序不属于它所中断的任务。该任务可能在用户空间,可能在系统调用中途,可能处于空闲状态,这都无关紧要。中断跨越该边界,却不成为其一部分。

内核精确处理这种区别。处理程序在中断上下文中运行,使用当前任务的内核栈,但从不声明任务的身份。它不修改任务状态,不改变其调度状态,不留下任何痕迹。

这就是中断不能睡眠的原因。不仅因为它必须快速,还因为它不能成为任务的一部分。睡眠意味着此执行可以被暂停并在调度规则下恢复,就好像它是一个线程。但它不是,它根本不是线程的一部分,而是系统从外部进行的干预。

当需要更多工作时,内核会移交任务。它委托给软中断或将函数入队到工作队列。这些路径可以安全调度、阻塞和拥有线程。中断路径则不然,它的定义是无上下文、无所有权、无延续。

当处理程序完成时,内核决定接下来发生什么。如果需要重新调度任务,它会切换;如果不需要,被中断的任务会恢复。无论哪种方式,栈都是完整的,边界得到尊重。被中断的内容和响应的内容之间没有泄漏。

这就是内核获得控制权的方式。不仅是它可以被中断,还在于它可以响应而不被纠缠。中断不属于它所抢占的逻辑,它的存在是为了确保系统对不能等待用户代码注意到的事件保持响应。

因此,“中断”这个名字可能听起来像是一种干扰,但设计讲述了不同的故事。

它不是流程的干扰,

而是流程之外的路径——结构化、精确且有边界。

上次更新: 2025/06/17, 19:51:40
19 追踪执行的模板
21 执行是逻辑的,位置是物理的

← 19 追踪执行的模板 21 执行是逻辑的,位置是物理的→

最近更新
01
第二章 关键字static及其不同用法
03-27
02
第一章 auto与类型推导
03-27
03
第四章 Lambda函数
03-27
更多文章>
Copyright © 2024-2025 沪ICP备2023015129号 张小方 版权所有
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式