除了手动浏览页面目录条目外,有没有合适的方法来获取逻辑地址的物理地址?我在内核的源代码中寻找了这个功能,并发现有一个follow_page
函数可以很好地完成内置的巨大且透明的大页面支持。但是它没有导出到内核模块(为什么???)......
所以,我不想发明轮子,我认为手动重新实现follow_page
功能并不是很好。
答案 0 :(得分:6)
好吧,它可能看起来像那样(从虚拟地址跟随PTE):
void follow_pte(struct mm_struct * mm, unsigned long address, pte_t * entry)
{
pgd_t * pgd = pgd_offset(mm, address);
printk("follow_pte() for %lx\n", address);
entry->pte = 0;
if (!pgd_none(*pgd) && !pgd_bad(*pgd)) {
pud_t * pud = pud_offset(pgd, address);
struct vm_area_struct * vma = find_vma(mm, address);
printk(" pgd = %lx\n", pgd_val(*pgd));
if (pud_none(*pud)) {
printk(" pud = empty\n");
return;
}
if (pud_huge(*pud) && vma->vm_flags & VM_HUGETLB) {
entry->pte = pud_val(*pud);
printk(" pud = huge\n");
return;
}
if (!pud_bad(*pud)) {
pmd_t * pmd = pmd_offset(pud, address);
printk(" pud = %lx\n", pud_val(*pud));
if (pmd_none(*pmd)) {
printk(" pmd = empty\n");
return;
}
if (pmd_huge(*pmd) && vma->vm_flags & VM_HUGETLB) {
entry->pte = pmd_val(*pmd);
printk(" pmd = huge\n");
return;
}
if (pmd_trans_huge(*pmd)) {
entry->pte = pmd_val(*pmd);
printk(" pmd = trans_huge\n");
return;
}
if (!pmd_bad(*pmd)) {
pte_t * pte = pte_offset_map(pmd, address);
printk(" pmd = %lx\n", pmd_val(*pmd));
if (!pte_none(*pte)) {
entry->pte = pte_val(*pte);
printk(" pte = %lx\n", pte_val(*pte));
} else {
printk(" pte = empty\n");
}
pte_unmap(pte);
}
}
}
}
答案 1 :(得分:3)
我认为您可以通过/proc/[pid]/maps
(给出进程的虚拟映射)和/proc/[pid]/pagemap
(给虚拟页面到物理页面映射)的组合,通过间接方法实现虚拟>物理转换对于每个可寻址的页面)。首先,从maps
找出进程虚拟地址的映射(这样做是为了不搜索pagemap
中的每个字节)然后检查所需虚拟地址的物理映射。 pagemap(页面地图不是文本格式。以下是格式Pagemap的详细说明)
这应该为您提供精确的虚拟 - >物理映射
答案 2 :(得分:0)
听起来你正在寻找virt_to_phys
。