linux 内存恒等映射

2024-01-18 08:20
文章标签 linux 内存 映射 恒等

本文主要是介绍linux 内存恒等映射,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

arm64内存管理

恒等映射

页表定义,采用4级分页模型

创建页表

内存属性


从进程角度看,需要内存的地方

1、进程本身,代码段、数据段用来存储程序本身需要的数据

2、栈空间:用来保存函数调用关系、局部变量、函数参数、函数返回值等

3、堆空间:动态分配的内存

arm64内存管理

MMU包括TLB和页表遍历单元。

TLB是一个高速缓存,用于缓存页表转换的结果,从而缩短页表查询的时间。

在得到物理地址后,首先要查询该物理地址的内容是否在高速缓存中。

恒等映射

        物理地址=虚拟地址

从bootloader跳转到操作系统入口时,MMU是关闭的,意味着不能使用高速缓存。

在关闭MMU的情况下,处理器访问的地址都是物理地址,当打开MMU时,访问的是虚拟地址。

恒等映射的目的:保证处理器在开启MMU前后可以连续取指令,因为处理器大多是流水线体系结构。

恒等映射的创建

低512M内存映射到虚拟地址0~512M地址空间;采用4KB大小的页面和48位地址宽度来创建恒等映射。

页表定义,采用4级分页模型

  • 页全局目录(Page Global Directory,PGD)
  • 页上级目录(Page Upper Directory,PUD)
  • 页中间目录  (Page Middle Directory,PMD)
  • 页表 (Page Table,PT)

  • PGD的偏移量 39

页表的大小和页表项数量

#define PGDIR_SHIFT 39
#define PGDIR_SIZE (1UL << PGDIR_SHIFT)
#define PGDIR_MASK (~(PGDIR_SIZE-1))
#define PTRS_PER_PGD (1 << (VA_BITS - PGDIR_SHIFT))
  • PGDIR_SHIFT 表示PGD页表在虚拟地址中的起始偏移量
  • PGDIR_SIZE 表示一个PGD页表项所能映射的区域大小
  • PGDIR_MASK 用来屏蔽虚拟地址中PUD索引,PMD索引以及PT索引字段的位
  • PTS_PER_PGD 表示PGD页表中页表项的个数。其中VA_BITS表示虚拟地址位宽(48位)

其他宏定义类似的含义。 

/* PUD */
#define PUD_SHIFT 30
#define PUD_SIZE (1UL << PUD_SHIFT)
#define PUD_MASK (~(PUD_SIZE-1))
#define PTRS_PER_PUD (1 << (PGDIR_SHIFT - PUD_SHIFT))/* PMD */
#define PMD_SHIFT 21
#define PMD_SIZE (1UL << PMD_SHIFT)
#define PMD_MASK (~(PMD_SIZE-1))
#define PTRS_PER_PMD (1 << (PUD_SHIFT - PMD_SHIFT))/* PTE */
#define PTE_SHIFT 12
#define PTE_SIZE (1UL << PTE_SHIFT)
#define PTE_MASK (~(PTE_SIZE-1))
#define PTRS_PER_PTE (1 << (PMD_SHIFT - PTE_SHIFT))

 PTE属性

#define PTE_TYPE_MASK  (3UL << 0)
#define PTE_TYPE_FAULT (0UL << 0)
#define PTE_TYPE_PAGE  (3UL << 0)
#define PTE_TABLE_BIT  (1UL << 1)
#define PTE_USER       (1UL << 6) /* AP[1] */
#define PTE_RDONLY     (1UL << 7) /* AP[2] */
#define PTE_SHARED     (3UL << 8) /* SH[1:0], inner shareable */
#define PTE_AF         (1UL << 10)      /* Access Flag */
#define PTE_NG         (1UL << 11)      /* nG */
#define PTE_DBM        (1UL << 51)      /* Dirty Bit Management */
#define PTE_CONT       (1UL << 52)      /* Contiguous range */
#define PTE_PXN        (1UL << 53)      /* Privileged XN */
#define PTE_UXN        (1UL << 54)      /* User XN */
#define PTE_HYP_XN  (1UL << 54) /* HYP XN */

根据内存的属性,页表项的属性

不同类型的页面采用哪种属性?

  • 操作系统的代码段和数据段都应该采用普通内存

页表数据结构

typedef struct
{unsigned long long pte;
}pte_t;#define pte_val(x) ((x).pte)
#define __pmd(x) (pte_t) ((x))

创建页表

页表创建是软件完成,页表的遍历是MMU自动完成,在打开MMU之前,软件需要手动创建4级页表。

在链接脚本中的数据段中,预留4KB大小的内存空间给PGD页表

SECTIONS
{ _data = .;.data : { *(.data) }. = ALIGN(4096);idmap_pg_dir = .;. += 4096;_edata = .;
}

内存属性

  • 代码段,代码段有只读、可执行属性,因此他们必须映射到PAGE_KERNEL_ROX属性
  • 数据段以及剩下内存。普通属性,映射到PAGE_KERNEL
  • 寄存器空间地址,寄存器地址空间属于设备类型内存,映射到PORT_DEVICE_nGnRnE属性。
static void create_identical_mapping(void)
{unsigned long start;unsigned long end;/*map text*/start = (unsigned long)_text_boot;end = (unsigned long)_etext;__create_pgd_mapping((pgd_t *)idmap_pg_dir, start, start,end - start, PAGE_KERNEL_ROX,early_pgtable_alloc,0);/*map memory*/start = PAGE_ALIGN((unsigned long)_etext);end = TOTAL_MEMORY;__create_pgd_mapping((pgd_t *)idmap_pg_dir, start, start,end - start, PAGE_KERNEL,early_pgtable_alloc,0);
}static void create_mmio_mapping(void)
{__create_pgd_mapping((pgd_t *)idmap_pg_dir, PBASE, PBASE,DEVICE_SIZE, PROT_DEVICE_nGnRnE,early_pgtable_alloc,0);
}
//PGD 创建页表的基地址
/*
port 内存属性
*/static void __create_pgd_mapping(pgd_t *pgdir, unsigned long phys,unsigned long virt, unsigned long size,unsigned long prot,unsigned long (*alloc_pgtable)(void),unsigned long flags)
{pgd_t *pgdp = pgd_offset_raw(pgdir, virt);unsigned long addr, end, next;phys &= PAGE_MASK;addr = virt & PAGE_MASK;end = PAGE_ALIGN(virt + size);do {next = pgd_addr_end(addr, end);alloc_init_pud(pgdp, addr, next, phys,prot, alloc_pgtable, flags);phys += next - addr;} while (pgdp++, addr = next, addr != end);}

以PGDIR_SIZE为步长遍历内存区域[virt,virt+size];

调用alloc_init_pud初始化PGD页表项内容和PUD

pgd_addr_end以PGDIR_SIZE为步长

static void alloc_init_pud(pgd_t *pgdp, unsigned long addr,unsigned long end, unsigned long phys,unsigned long prot,unsigned long (*alloc_pgtable)(void),unsigned long flags)
{pgd_t pgd = *pgdp;pud_t *pudp;unsigned long next;if (pgd_none(pgd)) {unsigned long pud_phys;pud_phys = alloc_pgtable();set_pgd(pgdp, __pgd(pud_phys | PUD_TYPE_TABLE));pgd = *pgdp;}pudp = pud_offset_phys(pgdp, addr);do {next = pud_addr_end(addr, end);alloc_init_pmd(pudp, addr, next, phys,prot, alloc_pgtable, flags);phys += next - addr;} while (pudp++, addr = next, addr != end);
}
  • 如果pgd页表项的内容为空,说明下一级页表还没创建,需要动态分配下一级页表。
  • 使用alloc_pgtable函数分配一个4KB页面用于PUD页表,
  • PUD页表基地址pud_phys与相关属性PUD_TYPE_TABLE组成一个PGD页表项,然后通过set_pgd函数设置到相应的PGD页表项中。

pud_offset_phys()函数通过addr和PGD页表项来找到对应的PUD页表项。

alloc_init_pmd 创建下一级页表。

页表建立过程如下

__create_pgd_mapping->alloc_init_pud->alloc_init_pmd->alloc_init_pte

static unsigned long early_pgtable_alloc(void)
{unsigned long phys;phys = get_free_page();memset((void *)phys, 0, PAGE_SIZE);return phys;
}
unsigned long get_free_page(void)
{int i;for (i = 0; i < NR_PAGES; i++) {if (mem_map[i] == 0) {mem_map[i] = 1;return LOW_MEMORY + i * PAGE_SIZE;}}return 0;
}

这篇关于linux 内存恒等映射的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/618493

相关文章

Linux镜像文件制作方式

《Linux镜像文件制作方式》本文介绍了Linux镜像文件制作的过程,包括确定磁盘空间布局、制作空白镜像文件、分区与格式化、复制引导分区和其他分区... 目录1.确定磁盘空间布局2.制作空白镜像文件3.分区与格式化1) 分区2) 格式化4.复制引导分区5.复制其它分区1) 挂载2) 复制bootfs分区3)

Linux服务器数据盘移除并重新挂载的全过程

《Linux服务器数据盘移除并重新挂载的全过程》:本文主要介绍在Linux服务器上移除并重新挂载数据盘的整个过程,分为三大步:卸载文件系统、分离磁盘和重新挂载,每一步都有详细的步骤和注意事项,确保... 目录引言第一步:卸载文件系统第二步:分离磁盘第三步:重新挂载引言在 linux 服务器上移除并重新挂p

Linux下屏幕亮度的调节方式

《Linux下屏幕亮度的调节方式》文章介绍了Linux下屏幕亮度调节的几种方法,包括图形界面、手动调节(使用ACPI内核模块)和外接显示屏调节,以及自动调节软件(CaliseRedshift和Reds... 目录1 概述2 手动调节http://www.chinasem.cn2.1 手动屏幕调节2.2 外接显

Linux(centos7)虚拟机没有IP问题及解决方案

《Linux(centos7)虚拟机没有IP问题及解决方案》文章介绍了在CentOS7中配置虚拟机网络并使用Xshell连接虚拟机的步骤,首先,检查并配置网卡ens33的ONBOOT属性为yes,然后... 目录输入查看ZFhrxIP命令:ip addr查看,没有虚拟机IP修改ens33配置文件重启网络Xh

Java JAR 启动内存参数配置指南(从基础设置到性能优化)

《JavaJAR启动内存参数配置指南(从基础设置到性能优化)》在启动Java可执行JAR文件时,合理配置JVM内存参数是保障应用稳定性和性能的关键,本文将系统讲解如何通过命令行参数、环境变量等方式... 目录一、核心内存参数详解1.1 堆内存配置1.2 元空间配置(MetASPace)1.3 线程栈配置1.

linux实现对.jar文件的配置文件进行修改

《linux实现对.jar文件的配置文件进行修改》文章讲述了如何使用Linux系统修改.jar文件的配置文件,包括进入文件夹、编辑文件、保存并退出编辑器,以及重新启动项目... 目录linux对.jar文件的配置文件进行修改第一步第二步 第三步第四步总结linux对.jar文件的配置文件进行修改第一步进

SpringMVC配置、映射与参数处理​入门案例详解

《SpringMVC配置、映射与参数处理​入门案例详解》文章介绍了SpringMVC框架的基本概念和使用方法,包括如何配置和编写Controller、设置请求映射规则、使用RestFul风格、获取请求... 目录1.SpringMVC概述2.入门案例①导入相关依赖②配置web.XML③配置SpringMVC

Elasticsearch 的索引管理与映射配置实战指南

《Elasticsearch的索引管理与映射配置实战指南》在本文中,我们深入探讨了Elasticsearch中索引与映射的基本概念及其重要性,通过详细的操作示例,我们了解了如何创建、更新和删除索引,... 目录一、索引操作(一)创建索引(二)删除索引(三)关闭索引(四)打开索引(五)索引别名二、映射操作(一

linux ssh如何实现增加访问端口

《linuxssh如何实现增加访问端口》Linux中SSH默认使用22端口,为了增强安全性或满足特定需求,可以通过修改SSH配置来增加或更改SSH访问端口,具体步骤包括修改SSH配置文件、增加或修改... 目录1. 修改 SSH 配置文件2. 增加或修改端口3. 保存并退出编辑器4. 更新防火墙规则使用uf

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出