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语言开发

16 无状态CPU,有状态内核:执行如何被协调

# 16 无状态CPU,有状态内核:执行如何被协调

在机器层面,CPU从根本上是无状态的。它一条接一条地执行指令,使用寄存器和内存的内容,而不感知任务、所有权或历史。在任何周期中,它只处理呈现给它的内容,不了解之前或之后的情况。CPU不跟踪正在执行哪个任务或它属于哪里;结构和连续性完全由内核维护。

这种设计是有意为之的。保持CPU无状态可保留速度、简洁性和通用性。CPU通过其格式化的指令、指令指针(IP)、栈指针(SP)和通用寄存器来暴露其功能。IP决定下一条指令从哪里获取;SP控制临时数据的推入和弹出。它们共同定义了代码和数据的流动。CPU本身不管理上下文或保留连续性;它精确地遵循这些流动,没有记忆。

在Linux中,内核是真正有状态的实体。它记录并管理每个执行路径的完整上下文,包括CPU状态、内存映射和调度信息。每个执行路径(无论是用户空间进程、进程内的线程还是内核线程)都由一个task_struct表示,其中包含精确暂停和稍后恢复执行所需的一切。

每个任务都有一个私有的内核栈。对于用户程序,这个栈与用户空间栈是分开的。当任务进入内核模式(通过系统调用、页面错误或中断)时,CPU切换到私有内核栈,允许内核安全地保存寄存器并管理临时数据。没有它,特权操作将不可靠。

上下文切换是内核用于在执行路径之间移动的机制。它将当前任务的完整CPU状态保存到其task_struct中,恢复另一个任务的状态,并让CPU从那里继续执行。CPU对此并不知晓,它只是从新状态继续执行。这使得许多独立任务可以共享单个核心,在同一时间只有一条指令运行的情况下创造出并行执行的假象。

在人类时间尺度上,上下文切换发生得如此之快,以至于多个程序似乎在同时进行。但在CPU层面,每个周期都严格专用于一个任务。内核对任务状态和调度决策的精心管理为本身无状态的CPU带来了结构。

CPU不记忆,它执行。内核记住一切。它推进每个执行路径,在无数次切换中保留身份和连续性,在幕后协调系统。

CPU带来执行的精确性;内核带来时间上的连续性——它们共同将简单的执行转化为协调的系统行为。

上次更新: 2025/06/17, 19:51:40
15 从vmlinuz到eBPF:Linux内核内部实际运行的内容
17 内核构建的内容——逐层构建

← 15 从vmlinuz到eBPF:Linux内核内部实际运行的内容 17 内核构建的内容——逐层构建→

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