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

26 libc之外:用户空间与内核的真实通信方式

# 26 libc之外:用户空间与内核的真实通信方式

当大多数人思考用户空间如何与Linux内核通信时,他们会想到libc。这是有道理的——libc提供了open、read、write和malloc等常见函数,它带来了便利性、可移植性和一致的API。但libc只是一个层,并非唯一的接口。

在其之下是一个更广泛且经过精心设计的系统。libc封装了系统调用,但这些系统调用可以直接调用,无需借助库。更重要的是,并非所有的内核交互都完全依赖系统调用。

Linux内核通过多种接口展示自身。像/proc和/sys这样的虚拟文件系统提供了对内部状态和配置的结构化访问;ioctl支持特定于设备的控制路径,这些路径不适合标准的读/写模型;mmap允许用户空间和内核空间之间直接内存映射,以实现高效I/O;ptrace等工具为调试提供了底层进程控制;Netlink套接字在用户空间和内核子系统之间实现了结构化的异步通信;而eBPF则引入了一个可编程运行时,可在预定义的钩子处安全地将逻辑注入内核。

这些接口的存在是有原因的。内核并非强制使用单一路径,而是支持一系列交互方式,每种方式都适合不同的目的。脚本可能从/proc读取数据,对性能要求严格的服务可能依赖mmap或io_uring,跟踪工具可能附加eBPF程序来实时观察内核行为。

在内部,这些路径汇聚到共享逻辑。无论是处理系统调用、文件读取还是Netlink消息,内核都使用通用的调度表、内部抽象和子系统来处理请求。表面可能不同,但基础始终是统一的。

这种灵活性并非偶然,它反映了塑造内核开发的原则:不破坏用户空间、保持兼容性、为可扩展性而设计。即使内部发生演变,接口也保持稳定;机制优于固定策略;模块化使子系统能够独立发展,同时与整体保持一致。

这正是Linux同时具备稳定性和适应性的原因。旧工具继续有效,新工具获得发展空间。多种接口并存并非碎片化,而是有意的设计。libc仍然是最常见的内核入口路径,但它只是众多路径之一,所有路径都被设计得安全、有目的性且精确。

这不是偶然,而是设计。

上次更新: 2025/06/17, 19:51:40
25 搭建组件之间的桥梁
27 CPU不移动数据——但没有CPU,什么都无法移动

← 25 搭建组件之间的桥梁 27 CPU不移动数据——但没有CPU,什么都无法移动→

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