- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我对Linux内核中的页表管理感到困惑吗?
在Linux内核空间中,打开页表之前。内核将以1-1映射机制在虚拟内存中运行。打开页面表后,内核将查询页面表以将虚拟地址转换为物理内存地址。
问题是:
这时,打开页表后,内核空间仍然是1GB(从0xC0000000-0xFFFFFFFF)?
并且在内核进程的页表中,仅映射范围为0xC0000000-0xFFFFFFFF的页表项(PTE)。超出此范围的PTE将不会被映射,因为内核代码永远不会跳到那里?
打开页表前后的映射地址是否相同?
例如。在打开页表之前,将虚拟地址0xC00000FF映射到物理地址0x000000FF,然后在打开页表之后,上面的映射不会更改。虚拟地址0xC00000FF仍映射到物理地址0x000000FF。唯一不同的是,打开页表后,CPU查阅了页表将虚拟地址转换为物理地址,而以前不需要这样做。
内核空间中的页表是全局的,将在系统中的所有进程(包括用户进程)之间共享。
这种机制在x86 32bit和ARM中是相同的。
最佳答案
以下讨论基于32位ARM Linux,内核源代码版本为3.9。
如果您完成设置初始页表(稍后将由功能paging_init
忽略)并打开MMU的过程,则可以解决所有问题。
当引导程序首次启动内核时,汇编函数stext
(在arch \ arm \ kernel \ head.s中)是第一个运行的函数。请注意,此刻MMU尚未打开。
其中,此功能stext
完成的两个导入作业是:
创建初始页面表格(稍后将被忽略)
函数paging_init
)
开启MMU
跳转到内核初始化代码的C部分并继续
在深入研究您的问题之前,先了解一下是有益的:
在打开MMU之前,CPU发出的每个地址都是物理的
地址
MMU打开后,CPU发出的每个地址都是虚拟地址
在打开MMU之前,应设置适当的页表,否则您的代码将“被炸掉”
按照惯例,Linux内核使用较高的1GB虚拟地址部分,而用户土地使用较低的3GB部分
现在棘手的部分是:
第一个技巧:使用与位置无关的代码。
汇编函数stext链接到地址“ PAGE_OFFSET + TEXT_OFFSET
”(0xCxxxxxxx),这是一个虚拟地址,但是,由于尚未打开MMU,因此运行汇编函数stext的实际地址为“ PHYS_OFFSET + TEXT_OFFSET
”(实际值取决于您的实际硬件),这是一个物理地址。
所以,这就是问题:函数stext
的程序“认为”它在地址0xCxxxxxxx之类运行,但实际上在地址(0x00000000 + some_offeset)中运行(例如,您的硬件将0x00000000配置为RAM的起点) 。因此,在打开MMU之前,需要非常仔细地编写汇编代码,以确保在执行过程中没有出错。实际上,使用了一种称为位置无关代码(PIC)的技术。
为了进一步解释上述内容,我提取了一些汇编代码片段:
ldr r13, =__mmap_switched @ address to jump to after MMU has been enabled
b __enable_mmu @ jump to function "__enable_mmu" to turn on MMU
ENTRY(__turn_mmu_on)
mcr p15, 0, r0, c1, c0, 0 @ write control reg to enable MMU====> This is where MMU is turned on, after this instruction, every address issued by CPU is "virtual address" which will be translated by MMU
mov r3, r13 @ r13 stores the (virtual) address to jump to after MMU has been enabled, which is (0xC0000000 + some_offset)
mov pc, r3 @ a long jump
ENDPROC(__turn_mmu_on)
__create_page_tables
)和用于设置最终页表的函数(C函数pageing_init)。
__create_page_tables
中进行初始映射。在函数
__create_page_tables
中,仅映射了很小一部分地址空间。
paging_init
中设置最终页表,并且在此功能中,映射了很大一部分地址空间。假设您只有512M RAM,对于大多数常见配置,此512M-RAM将按内核代码逐节映射(1节为1MB)。如果您的RAM很大(例如2GB),则仅一部分RAM将被直接映射。
关于linux-kernel - 启动期间Linux内核空间中的页表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16688540/
我需要从 kernel/system/do_kill.c (Minix) 将一些信息写入我的特殊日志文件(例如,/home/log.txt) . 我已经尝试过: int filedesc; filed
我正在经历 Uboot 和内核启动过程。 FDT(平面设备树)到底有什么用? 我读过的许多链接都指出,uboot 以 FDT 的形式将板和 SOC 配置信息传递给内核 https://wiki.fre
我的操作系统是Fedora 17。最近,出现内核污染警告“kernel bug at kernel/auditsc.c:1772!-abrt”:不应报告此问题(这可能是一个已知问题)。发生内核问题,但
为 Linux 编译内核模块的教程,使用不同的 Makefile 语法。 Example 1 obj-m += rpi-pwm.o Example 2 obj-m := nothing.o 有什么区别
我正在浏览 Linux 网络设备驱动程序代码,想知道是否可以从驱动程序代码中调用设备层代码。 --- a/drivers/net/ethernet/realtek/8139too.c +++ b/dr
出于工具目的,我尝试附加到 kprobe 事件,但我对 kprobe 事件不太熟悉。我读到注册的 kprobes 列表可以在 /sys/kernel/debug/kprobes/list 中找到,但是
我在其他地方使用 LinK+ 来开发 linux 内核模块。我的开发机器安装了 Linux Mint 18 操作系统,内核版本为 4.4.xx。为了进行测试,我想将内核模块部署到内核版本为 3.16.
我正在玩弄 Android Linux 内核。内核是P970 V30B内核。可用here .我想找出特定的 cmdline 参数到底做了什么。 我知道它是命令行参数,其中有一个参数我找不到它到底做了什
是否可以将 PTE 指向不同的物理页面? 假设我目前在某个进程 A 的上下文中处于内核模式,该进程当前将地址 400k 映射到物理页号。 5. 我可以将该地址 (400k) 映射到物理页号吗? 6 ?
我正在开发一个要在路由器上运行的内核模块。路由器型号为 Netgear 的 DGN2200v2。它在 MIPS 上运行 Linux 2.6.30。我的问题是,当我加载我的模块时,似乎我的 module
关闭。这个问题不符合 Stack Overflow guidelines 。它目前不接受答案。 想改进这个问题?更新问题,使其成为 Stack Overflow 的 on-topic。 6年前关闭。
在大型网格中执行的任务调用两个内核有区别吗 1. for(int i=0;i>>(MatrixA,MatrixB) } 2. dim3 dimBlock(16, 16); dim3 dimGrid(1
在 Linux 中,我生成了一个猜测 VM 并加载了另一个 Linux 实例。 VM 是通过 KVM/libvirt/qemu 生成的。 guest VM 被主机内核视为一个进程。让我们说由于某种原因
我想知道如何设置正确 MACH_TYPE或 arch_id对于内核。我搜索并找到了至少 2 个引用,其中内核会卡在“启动内核...”处。这些都给出了相同的答案。正确设置您的机器类型。但后来都没有提到如
kernel data inpage error蓝屏是一个不常见的问题,一般都是更新失败或者第三方软件冲突导致的,解决方法也非常简单,需要先安全模式进入系统,下面来看看详细的教程吧。 kern
当我用 app/console cache:clear 清除缓存时我收到错误: [Symfony\Component\DependencyInjection\Exception\ParameterNo
我正在玩 Raspberry 3,并尝试使用 U-Boot 启动 Linux 内核。 我构建了一个 Linux 内核(来自 github.com/raspberrypi)和 Busbox-Userla
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 这个问题是由于错别字或无法再重现的问题引起的。虽然类似的问题可能是on-topi
长话短说,我一直在做一个项目,当我使用时我注意到: 1.9.3p392 :001 > `gzip` IRB::Abort: abort then interrupt! from (irb):1
我有一个服务,我在其中注入(inject) TokenStorage 并想要获取当前用户。 /** * * @Service("liip_theme.theme_request_listener"
我是一名优秀的程序员,十分优秀!