嵌入式Linux之虚拟内存地址空间布局(Virtual Memory Space)
虚拟内存地址空间
Linux内核属于微内核的范畴,内核控制计算机的硬件资源,运行在特权模式;用户态应用程序运行在普通用户模式,无法直接访问硬件资源,必须依托于内核提供的资源,如CPU资源、Memory资源、I/O资源等。
Linux采用沙箱机制,每一个进程运行在独立的虚拟地址空间,最大限度避免单个进程异常导致整个系统崩溃。
每一个进程的虚拟地址空间分为内核虚拟地址空间和用户虚拟地址空间两部分,内核虚拟地址空间为内核态代码和内核堆栈,所有进程的内核虚拟地址空间是复用的;用户虚拟地址空间则是各进程的代码段、数据段、BSS段、mmap段(动态加载库以及各库运行数据段、运行栈等)、堆、栈、环境变量等信息。
Linux中可以通过cat /proc/<pid>/maps命令查看指定进程的用户虚拟地址空间的映射,也可以通过pmap -x <pid>查看,该命令实质上是基于/proc/<pid>/maps实现的。
ARM32
用户虚拟地址空间范围:0x0000 0000 ~ 0xbfff ffff
内核虚拟地址空间范围:0xc000 0000 ~ 0xffff ffff
ARM64
ARMv7架构(32位CPU)早期CPU采用32位物理地址空间,支持最大物理内存为4GB。但是随着内存容量和软件的扩张,4GB物理内存寻址范围已经不够用了,所以引入了LPAE(Large Physical Address Extension)机制,用户可以通过CPU寄存器扩展物理地址宽度。
ARMv8架构(64位CPU)开始,ARM公司从长远地角度考虑,优化了LPAE机制,可以支持32~48位物理地址寻址,满足了物理内存寻址的要求。同时做了兼容32位应用程序的考虑。
ARMv8在硬件体系结构上支持4级的4KB页大小的页表转换,也支持3级的64KB页大小的页表转换。
在Linux ARM64中,如果页大小为4KB,使用3级或者4级页表转换,用户虚拟地址空间和内核虚拟地址空间都支持39位(512GB)或者48位(256TB)寻址范围。
如果页大小为64KB,就只有2级页表转换,用户虚拟地址空间和内核虚拟地址空间支持42位(4TB)寻址范围。
上面这张表格,整理了ARMv7和ARMv8的物理地址位宽、用户虚拟地址空间范围和内核虚拟地址空间范围。
ARMv8架构开始,Linux内核新增了几个内核配置选项用来设定虚拟地址位宽和页表层级以及页大小:
1)CONFIG_ARM64_64K_PAGES 可以使能64KB大页,如果没有使能,则延用4KB页;
2)CONFIG_COMPAT 设置是否兼容32位应用程序;
#ifdef CONFIG_COMPAT
#define TASK_SIZE_32 UL(0x100000000)
#define TASK_SIZE (test_thread_flag(TIF_32BIT) ? \
TASK_SIZE_32 : TASK_SIZE_64)
#else
#define TASK_SIZE TASK_SIZE_64
#endif /* CONFIG_COMPAT */
3)CONFIG_ARM64_VA_BITS 设置虚拟地址位宽,常用的虚拟地址位宽为39位或者48位;(实际生效的是内核VA_BITS宏)
ARM64内核虚拟地址空间布局可以通过arch/arm64/include/asm/memory.h文件查看,也可以查看也可以看看 arch/arm64/mm/init.c 和 arch/arm64/include/asm/pgtable.h。
pr_notice("Virtual kernel memory layout:\n"
" vmalloc : 0x%16lx - 0x%16lx (%6ld MB)\n"
#ifdef CONFIG_SPARSEMEM_VMEMMAP
" vmemmap : 0x%16lx - 0x%16lx (%6ld MB)\n"
#endif
" modules : 0x%16lx - 0x%16lx (%6ld MB)\n"
" memory : 0x%16lx - 0x%16lx (%6ld MB)\n"
" .init : 0x%p" " - 0x%p" " (%6ld kB)\n"
" .text : 0x%p" " - 0x%p" " (%6ld kB)\n"
" .data : 0x%p" " - 0x%p" " (%6ld kB)\n",
MLM(VMALLOC_START, VMALLOC_END),
#ifdef CONFIG_SPARSEMEM_VMEMMAP
MLM((unsigned long)virt_to_page(PAGE_OFFSET),
(unsigned long)virt_to_page(high_memory)),
#endif
MLM(MODULES_VADDR, MODULES_END),
MLM(PAGE_OFFSET, (unsigned long)high_memory), MLK_ROUNDUP(__init_begin, __init_end),
MLK_ROUNDUP(_text, _etext),
MLK_ROUNDUP(_sdata, _edata));
实例如下:
ARM64用户虚拟地址空间布局和ARM32类似,但是地址范围略有差异。
ARM64兼容ARM 32位应用程序,所以ARM 32位应用程序可以无需修改运行在ARMv8上。为了运行ARM 32位应用程序,Linux内核仍然从init进程创建一个64位的进程(clone系统调用)、但是将其用户虚拟地址空间限制到4GB。通过这种方式,64位的Linux内核可以同时运行32位和64位应用程序。
需要注意到,ARM64上32位应用程序仍然拥有512GB的内核地址空间,并且不与内核共享自己的4GB用户虚拟地址空间;但是ARM32上,32位应用程序只有3GB用户虚拟地址空间。
嵌入式Linux之虚拟内存地址空间布局(Virtual Memory Space)的更多相关文章
- JVM virtual memory
This has been a long-standing complaint with Java, but it's largely meaningless, and usually based o ...
- Extended paging tables to map guest physical memory addresses from virtual memory page tables to host physical memory addresses in a virtual machine system
A processor including a virtualization system of the processor with a memory virtualization support ...
- Linux Process Virtual Memory
目录 . 简介 . 进程虚拟地址空间 . 内存映射的原理 . 数据结构 . 对区域的操作 . 地址空间 . 内存映射 . 反向映射 .堆的管理 . 缺页异常的处理 . 用户空间缺页异常的校正 . 内核 ...
- 深入虚拟内存(Virtual Memory,VM)
我们应该知道物理内存(Physical Memory)指的是硬件上的内存,即 RAM.它通常指的是插在主板上的内存条,给进程提供临时数据存储的设备.因为 CPU 可以直接从物理内存中读取数据和指令,所 ...
- Kafka:ZK+Kafka+Spark Streaming集群环境搭建(十三)kafka+spark streaming打包好的程序提交时提示虚拟内存不足(Container is running beyond virtual memory limits. Current usage: 119.5 MB of 1 GB physical memory used; 2.2 GB of 2.1 G)
异常问题:Container is running beyond virtual memory limits. Current usage: 119.5 MB of 1 GB physical mem ...
- LDAP实例异常停止日志提示虚拟内存virtual memory不足
[05/Oct/2014:20:50:37 +0800] - ERROR<5135> - Resource Limit - conn=-1 op=-1 msgId=-1 - Memory ...
- 嵌入式linux启动信息完全注释
嵌入式linux启动信息完全注释 from:http://www.embedlinux.cn/ShowPost.asp?ThreadID=377 摘要 我们在这里讨论的是对嵌入式linux系统的启动过 ...
- 《嵌入式Linux基础教程学习笔记一》
常用书目下载地址:http://www.cnblogs.com/pengdonglin137/p/3688029.html 第二章 1.进程上下文和中断上下文(Page20) 当应用程序执行系统调用, ...
- 《嵌入式Linux内存使用与性能优化》笔记
这本书有两个关切点:系统内存(用户层)和性能优化. 这本书和Brendan Gregg的<Systems Performance>相比,无论是技术层次还是更高的理论都有较大差距.但是这不影 ...
随机推荐
- linux中断处理上下部分
一.linux中断处理为什么要分为上下部 1.1. 中断处理的上半部(top half,又叫顶半部)和处理的下半部(bottom half,又叫底半部) 1.1. linux中断处理不参与调度,故中断 ...
- 嗨翻C语言--这里没有蠢问题(一)
问:card_name[0]是什么意思?答:它是用户输入的第一个字符.如果用户输入了10,那么card_name[0]就将是1.问:总是得用/*和*/写注释吗?答:如果你的编译器支持C99标准,就可以 ...
- 03: 使用docker搭建Harbor私有镜像仓库
1.1 harbor介绍 1.Harbor简介 1. Harbor是一个用于存储和分发Docker镜像的企业级Registry服务器. 2. 镜像的存储harbor使用的是官方的docker regi ...
- [LeetCode] 137. 只出现一次的数字 II
题目链接 : https://leetcode-cn.com/problems/single-number-ii/ 题目描述: 给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现了三 ...
- Spring如何读取xml配置文件的
我们通过一个小案例来看xml解析过程. 1. 导包 <dependencies> <!-- xml解析工具 --> <dependency> <groupId ...
- makeinfo - 翻译 Texinfo 文档
SYNOPSIS 总览 makeinfo [OPTION]... TEXINFO-FILE... DESCRIPTION 描述 将 Texinfo 源文档翻译为各种其他格式,默认是可以用 Emacs ...
- Verilog-2001标准在2001年就发布了
,不过翻了一些Verilog书籍,对Verilog-2001的新增特性很少有提及,即使提到了,也只是寥寥数语带过,其实在Verilog-2001中做了很多有用的改进,给编程带来很大的帮助,有必要详 ...
- VB TreeView控件使用详解(有趣的示例)
第一小时:学习直接用代码将数据填充到树控件中. 为什么要先学习直接用代码将数据填充到树控件中?因为这种方法是最简单的,代码也最容易理解,学习树控件,先将这个学会,已经掌握了一半,所以先不要急着想怎么将 ...
- ssh跳板到其他服务器
https://my.oschina.net/foreverich/blog/657075 http://mingxinglai.com/cn/2015/07/ssh-proxycommand/ 查看 ...
- ES6 Module(模块)
1.export命令 一个模块就是一个独立的文件. 该文件内部的所有变量,外部无法获取. 如果你希望外部能够读取模块内部的某个变量,就必须使用export关键字输出该变量. 下面是一个 JS 文件,里 ...