linux 内存恒等映射

news2024/11/24 0:29:17

目录

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;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/157725.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

在centos系统上安装mongodb数据库

在centos系统上安装mongodb数据库 本文章基于centos8系统; 如何查看当前系统是centos的那个版本 终端输入命令 cat /etc/redhat-release 开始安装官网当前的数据库版本6.0; 查看官方文档 创建 mongodb yum 源头,写入基本信息 vim /etc/yum.repos.d/mongodb-org-6.0.repo [mon…

手把手教你如何创建和美化图表,老板直接给我升职!

一图胜千言。说到图表&#xff0c;想必很多人都被网上酷炫的图表震惊过。比如下面这样的可视化图表&#xff0c;看起来&#xff0c;很高大上有没有。但是&#xff0c;拆解开来&#xff0c;就是由一个个基础图表演变而来的。所以可不要小瞧了基础图表的制作。今天我就教你如何用…

使用ORM方式查询Mongodb里的数据,再也不用记Mongodb的语法(ORM Bee)

使用ORM方式查询Mongodb里的数据,再也不用记Mongodb的语法&#xff08;ORM Bee)Mongodb的语法可读性差&#xff0c;要写复杂查询&#xff0c;要求技术能力高&#xff1b;Java驱动&#xff0c;还要使用另一种语法&#xff1b;学习成本太高了。可以使用ORM方式&#xff0c;轻松搞…

【redis6】第三章(五大常用数据类型)

redis命令列表&#xff1a;http://www.redis.cn/commands.html Redis键 set < key >< value >添加键值对 [rootlocalhost ~]# cd /usr/local/bin [rootlocalhost bin]# redis-server /etc/redis.conf [rootlocalhost bin]# redis-cli 127.0.0.1:6379> set k1 a…

电脑怎么恢复出厂设置?Win10系统恢复出厂设置的方法

电脑经常出现报错的情况&#xff0c;而且还有运行缓慢、卡顿的问题&#xff0c;这代表你的电脑可能出现了老化。想要恢复正常&#xff0c;你可以选择把电脑恢复到出厂设置&#xff0c;从而恢复到原始状态。电脑怎么恢复出厂设置&#xff1f;下面以Win10系统电脑为例&#xff0c…

CTK Plugin Framework插件框架学习--插件通信【事件监听】

文章目录一、前言二、事件三、类通信3.1、新建接收插件3.2、新建发送插件3.3、启用插件四、信号槽通信4.1、新建接收插件4.2、新建发送插件4.3、启用插件五、类通信和信号槽通信的区别六、插件依赖七、获取元数据一、前言 CTK框架中的事件监听&#xff0c;其实就是观察者模式&…

Dolphin Streaming实时计算,助力商家端算法第二增长曲线

丨目录&#xff1a;1.背景2.业务问题3.业界解决方案4.技术方案5.应用示例6.业务收益7.总结1. 背景随着业务朝向精细化经营增长&#xff0c;阿里妈妈商家端营销产品更加聚焦客户投放体验&#xff0c;旨在帮助商家提升经营效果&#xff0c;在变化的市场中找到确定增长。近年来&am…

【OpenGL学习】OpenGL介绍

一、OpenGL是什么&#xff1f; 在早期一般认为OpenGL是一个API(Application Programming Interface, 应用程序编程接口)&#xff0c;包含了一系列可以操作图形、图像的函数。然而&#xff0c;OpenGL本身并不是一个API&#xff0c;它仅仅是一个由Khronos组织制定并维护的规范(S…

商城模式“分享购”意味着什么?关键逻辑是什么?

商城模式“分享购”近期炽手可热的卖家打赏模式电子商务平台&#xff0c;根据申请注册交易赠予贡献度&#xff0c;推动者每日权重计算分派领到固定不动积分兑换数量空投物资&#xff0c;积分兑换前去出售买卖转现&#xff0c;店家根据选购积分兑换完成对客户的交易打赏主播&…

【遥感综述】

遥感综述小集合 Image fusion meets deep learning: A survey and perspective&#xff08;张浩&#xff0c;马佳义&#xff09; Sharpening fusion 多光谱图像锐化和高光谱图像锐化是两种典型的锐化融合任务。 Multi-spectral sharpening多光谱锐化 多光谱锐化是将低空间…

Android开发必修—Activity,View,Window三者之间的密切联系【深度解析】

简介 Activity就像工匠&#xff0c;Window就像是窗户&#xff0c;View就像是窗花&#xff0c;LayoutInflater像剪刀&#xff0c;Xml配置像窗花图纸。 Android根据他们不同的职能让他们各斯其活&#xff0c;同时也相互配合展示给我们灵活、精致的界面。 一张图理清所有层级关系&…

量子计算机的推出会影响网络安全吗

在谈量子计算机之前&#xff0c;首先得了解什么是量子计算机。量子计算机是一类遵循量子力学规律进行高速数学和逻辑运算、存储及处理量子信息的物理装置。当某个装置处理和计算的是量子信息&#xff0c;运行的是量子算法时&#xff0c;它就是量子计算机。量子计算机在分析大量…

飞凌OK3568-C嵌入式Linux开发板开箱体验

本篇来介绍一款具有AI计算能力的Linux开发板&#xff1a;OK3568-C。 1 OK3568-C介绍 OK3568-C是飞凌嵌入式设计的一款开发板&#xff0c;板子的包装如下&#xff0c;盒子内有两层&#xff0c;上层是开发板&#xff0c;下层是配件。 1.1 核心板介绍 开发板的SOC采用Rockchip的…

MySQL进阶:索引的使用及理解

MySQL索引前言什么是索引?索引的优缺点?MySql索引添加索引的sql语句添加PRIMARY KEY&#xff08;主键索引&#xff09;添加UNIQUE(唯一索引)添加INDEX(普通索引)添加FULLTEXT(全文索引)添加多列索引索引的底层数据结构哈希索引BTree树索引为什么索引会提高查找速度?最左前缀…

数据结构与算法2—链表

1. 链表 线性表的链式存储结构就是用一组任意的存储单元&#xff08;可以是不连续的&#xff09;存储线性表的数据元素。采用链式存储结构的表示的线性表简称链表。链式存储方式可用于表示线性结构&#xff0c;也可用于表示非线性结构。 链表通常有两个域 data域——存放结点…

北大硕士LeetCode算法专题课--递归和回溯

算法专题课 北大硕士LeetCode算法专题课-栈、队列相关问题_骨灰级收藏家的博客-CSDN博客 北大硕士LeetCode算法专题课--链表相关问题_骨灰级收藏家的博客-CSDN博客 北大硕士LeetCode算法专题课-查找相关问题_骨灰级收藏家的博客-CSDN博客 北大硕士LeetCode算法专题课-字符串…

一.【linux】常用操作命令

目录 1 . 查看当前登录系统用户信息 【who】 2 . 查看终端用户whoami 3 . 查看日期date 4 . 查看系统信息uname 5 . 日历cal 6 . 计算器bc 8 . 显示或修改计算机主机名的命令hostname 9 . 在显示器上显示文字的命令echo 10 . 清除计算机屏幕上的信息命令clear 11 . 查…

【尚硅谷】Java数据结构与算法笔记04 - 栈

文章目录一、实际需求二、栈的介绍三、应用场景四、数组模拟栈4.1 思路分析4.2 Java代码实现五、栈实现综合计算器5.1 中缀表达式5.2 后缀表达式一、实际需求 二、栈的介绍 栈的英文为(stack)栈是一个先入后出(FILO-First In Last Out)的有序列表。栈(stack)是限制线性表中元素…

nacos 2.1.0集群生产环境多节点部署

nacos 2.1.0集群生产环境多节点部署 版本 2.1.0版本发布日期 2022-04-29官网 集群部署说明GitHub GitHub - alibaba/nacos: an easy-to-use dynamic service discovery, configuration and service management platform for building cloud native applications. 下载地址&…

大数据面试题(七):Flume核心高频面试题

文章目录 Flume核心高频面试题 一、Flume有哪些组件&#xff0c;flume的source、channel、sink具体是做什么的 二、你是如何实现flume数据传输的监控的 三、Flume的source&#xff0c;sink&#xff0c;channel的作用&#xff1f;你们source是什么类型&#xff1f; 四、Flu…