内核的 Documentation/x86/x86_64/mm.txt 说:
ffff880000000000 - ffffc7ffffffffff (=64 TB) direct mapping of all phys. memory
所以我假设在这个区域中不应该有大于实际物理 DRAM 大小的地址映射。但是在具有 16GB DRAM 和内核 v4.2.8 的 x86_64 PC 上,我从 /sys/kernel/debug/kernel_page_tables 得到了这个结果:
---[ Low Kernel Mapping ]---
......
0xffff88008b3ff000-0xffff88008b400000 4K RW GLB NX pte
0xffff88008b400000-0xffff8800c0000000 844M pmd
0xffff8800c0000000-0xffff880100000000 1G pud
0xffff880100000000-0xffff880400000000 12G RW PSE GLB NX pud
0xffff880400000000-0xffff88043dc00000 988M RW PSE GLB NX pmd
0xffff88043dc00000-0xffff88043dc25000 148K RW GLB NX pte
0xffff88043dc25000-0xffff88043dc27000 8K ro GLB NX pte
.......
0xffff88045c200000-0xffff88046f000000 302M RW PSE GLB NX pmd
0xffff88046f000000-0xffff880480000000 272M pmd
0xffff880480000000-0xffff888000000000 494G pud
0xffff888000000000-0xffffc90000000000 66048G pgd
您会看到虚拟地址为 0xffff88043dc00000 的页面条目,在此类虚拟地址上使用 virt_to_phys() 将得到 0x43dc00000,这显然是非法的,因为它大于实际的 DRAM 大小(16GB 只是 0x400000000)。
那么这些大型虚拟地址的规则是什么,我怎样才能获得正确的物理地址呢?
多谢!