news 2026/6/10 15:54:58

Linux 内核学习(16) --- linux x86-64 虚拟地址空间和区域

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Linux 内核学习(16) --- linux x86-64 虚拟地址空间和区域

目录

      • x86_64 的虚拟地址空间
      • 直接映射区和 vmalloc 区域
      • I/O 统一编址与 ioremap
      • alloc_pages 内存

x86_64 的虚拟地址空间



直接映射区和 vmalloc 区域

内核虚拟地址空间直接映射区:

  • 直接映射区是内核虚拟地址空间中一段连续的区域,将物理内存按固定偏移(通常是PAGE_OFFSET)线性地映射到虚拟地址空间
  • x86_64架构上,物理地址0x0对应的内核虚拟地址通常是0xffff888000000000(即PAGE_OFFSET

设计内核直接映射区原则:

  • 高效访问物理内存:通过简单的加/减偏移即可完成物理地址 ↔ 虚拟地址的转换,无需查页表
  • 用于大块连续物理内存的管理:内核初始化时会将大部分物理内存(如低端内存)通过直接映射区映射,便于快速分配(如伙伴系统分配的页)
  • 性能优化:避免频繁创建/销毁页表项,提高内存访问速度

局限性:
要求物理内存连续:直接映射区只能映射物理上连续的内存页
地址空间有限:在 32 位系统上,直接映射区大小受限(如896MB),无法覆盖全部物理内存(尤其在大内存系统中)
无法处理不连续物理内存:当需要大块虚拟连续但物理不连续的内存时,直接映射区无能为力

vmalloc内存区域:

  • vmalloc是内核提供的一个函数,它分配的内存在虚拟地址上连续,但物理地址可以不连续
  • 这些内存通过页表动态映射到vmalloc区域,通常位于直接映射区之上(在 64 位系统中位于较高地址)

设计原则
解决物理内存碎片问题:当系统运行一段时间后,物理内存可能碎片化,难以分配大块连续物理内存,vmalloc允许用多个不连续的物理页拼成一个虚拟连续区域
支持大内存分配:适用于需要大块虚拟地址空间(如模块加载、I/O 映射、某些驱动缓冲区)但不要求物理连续的场景

局限性
性能开销:每次访问都需要查页表;分配时需修改页表,可能触发TLB刷新
地址空间碎片:频繁的vmalloc/free可能导致vmalloc区域本身碎片化
不能用于原子上下文:因为可能睡眠(需分配页表等)

举例说明:

  1. kmalloc: 小内存分配通常使用直接映射区(slab/slob/slub分配器基于伙伴系统,返回直接映射的页)
  2. vmalloc: 加载内核(代码/数据可能较大而且无法保证物理连续)、某些 GPU/设备驱动的缓冲区
  3. ioreamp: 将设备I/O内存映射到vmalloc区域,因为设备地址不属于普通物理内存

I/O 统一编址与 ioremap

ARM架构处理器中,I/O地址通常是与内存统一编址的(memory-mapped I/O,简称MMIO

ARM架构 没有 为I/O操作提供独立的I/O地址空间(不像x86架构那样有专门的in/out指令和独立的I/O端口地址空间,
相反,ARM使用 内存映射 I/O(Memory-Mapped I/O, MMIO) 的方式

外设的寄存器(如UARTGPIO、定时器等)被映射到系统的物理内存地址空间中
CPU通过普通的加载(load) 和 存储(store) 指令(如LDRSTR)来访问这些地址,从而读写外设寄存器
这些地址通常位于系统地址空间的特定区域(例如,SoC厂商会指定外设寄存器位于0x400000000x5FFFFFFF等范围)

  1. 如何在内核空间读写 I/O 地址

ioremap(phys_addr, size)将一段物理I/O地址映射到内核的虚拟地址空间;
返回一个虚拟地址(void __iomem* 类型),后续通过readl()writel()ioread32()iowrite32()等专用函数访问;
这些函数确保正确的内存访问语义(如禁止编译器优化、保证访问顺序、处理弱内存模型等)

举例如下:

// 使用 ioremap 的情况void__iomem*reg=ioremap(0xFED40000,4);// 映射4字节writel(0x12345678,reg);// 正确!通过虚拟地址访问

使用的物理空间如下:

物理地址空间:[0x00000000 - 0x3FFFFFFF]:系统 RAM(已线性映射)[0xFED40000 - 0xFED40FFF]:PCI 设备配置空间[0xFEC00000 - 0xFEC00FFF]:APIC 寄存器[0xFEE00000 - 0xFEE00FFF]:IOAPIC 寄存器 虚拟地址空间(内核部分):[0xffff888000000000]:直接映射区(RAM)[0xffffc90000000000]:vmalloc/ioremap 区域 ← ioremap 在这里分配虚拟地址 │ ↓ 建立页表映射[0xFED40000]物理 I/O 地址

alloc_pages 内存

alloc_pages()分配的物理内存,其对应的内核虚拟地址位于直接映射区(Direct Mapping Region),也叫线性映射区

alloc_pages()系列函数(包括__get_free_pageskmalloc的底层也用它)的主要目的是分配连续的物理页面

为了能让内核代码访问这些物理内存,必须将它们映射到内核的虚拟地址空间

直接映射是一种最简单的映射方式:在系统启动早期,内核会建立一个从物理地址到虚拟地址的固定偏移映射。在大多数架构(如 x86-64)上,这个偏移是固定的(例如0xffff888000000000)

它在内核虚拟地址空间中的典型起始地址是__PAGE_OFFSET(例如0xffff888000000000

地址转换:
当你调用alloc_pages(GFP_KERNEL, order) 并获得一个struct page*指针后,可以通过page_to_virt(page)宏来获取这块内存对应的内核虚拟地址

虽然alloc_pages()默认返回直接映射区的虚拟地址,但通过使用不同的标志,可以改变其行为,使其映射到其他区域:

  • GFP_HIGHUSER:当用于用户空间内存分配时(例如通过page fault处理程序),分配的物理页面最终会被映射到用户进程的虚拟地址空间,而不是内核的直接映射区。

  • __GFP_HIGHMEM(在 32 位系统上重要):在 32 位系统上,直接映射区大小有限(通常约896MB),无法映射所有物理内存超出部分称为“高端内存”,使用此标志分配的页面,其物理地址可能没有固定的内核虚拟地址。内核需要通过 kmap 等临时映射机制来访问它们
    但在 64 位系统上,由于虚拟地址空间极其庞大,已经废除了高端内存的概念,所有物理内存都可以线性映射

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 11:26:04

学霸同款2026继续教育AI论文写作软件TOP10:选对工具轻松过关

学霸同款2026继续教育AI论文写作软件TOP10:选对工具轻松过关 2026年继续教育AI论文写作工具测评:为何需要一份精准榜单? 随着人工智能技术在教育领域的深入应用,越来越多的继续教育学员开始依赖AI写作工具提升论文撰写效率。然而&…

作者头像 李华
网站建设 2026/6/10 13:43:51

[特殊字符]️_开发效率与运行性能的平衡艺术[20260112162407]

作为一名经历过无数项目开发的工程师,我深知开发效率与运行性能之间的平衡是多么重要。在快节奏的互联网行业,我们既需要快速交付功能,又需要保证系统性能。今天我要分享的是如何在开发效率和运行性能之间找到最佳平衡点的实战经验。 &#…

作者头像 李华
网站建设 2026/6/7 10:51:48

顶尖AI竟输给三岁宝宝,BabyVision测试暴露多模态模型硬伤

来源:机器之心01|“看懂世界” 这关,大模型还没上幼儿园过去一年,大模型在语言与文本推理上突飞猛进:论文能写、难题能解、甚至在顶级学术 / 竞赛类题目上屡屡刷新上限。但一个更关键的问题是:当问题不再能…

作者头像 李华
网站建设 2026/6/10 11:12:43

使用 IChatReducer 进行聊天记录缩减

序言在多轮对话场景中,随着聊天次数增加,发送给大语言模型(LLM)的上下文会持续膨胀,带来 Token 成本上升与上下文溢出风险。 Microsoft Agent Framework 将这一问题抽象为 Chat Reduction(聊天记录缩减&…

作者头像 李华
网站建设 2026/6/10 11:10:45

面试 Java 基础八股文十问十答第七期

面试 Java 基础八股文十问十答第七期 作者:程序员小白条,个人博客 相信看了本文后,对你的面试是有一定帮助的! ⭐点赞⭐收藏⭐不迷路!⭐ 1)Tomcat 是什么? Tomcat 是一个开源的、轻量级的应用服务器&am…

作者头像 李华