首页 文章

如何从Linux内核模块中的逻辑地址获取物理地址?

提问于
浏览
9

除了手动遍历页面目录条目之外,有没有合适的方法来获取逻辑地址的物理地址?我发现有一个 follow_page 函数可以很好地完成内置的巨大且透明的大页面支持 . 但它没有导出到内核模块(为什么???)...

所以,我不能手动重新实现 follow_page 功能 .

3 回答

  • 0

    好吧,它可能看起来像那样(从虚拟地址跟随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);
                }
            }
        }
    }
    
  • 3

    我认为您可以通过组合 /proc/[pid]/maps (给出进程的虚拟映射)和 /proc/[pid]/pagemap (为每个可寻址页面提供虚拟页面到物理页面映射)的间接方法实现虚拟 - >物理转换 . 首先,从 maps 找出您的进程的虚拟地址的映射(这样做是为了不搜索 pagemap 中的每个字节)然后检查页面映射中所需虚拟地址的物理映射(页面映射不是文本格式以下是格式Pagemap的详细说明 . 这应该为您提供精确的虚拟 - >物理映射

  • 5

    听起来你正在寻找virt_to_phys .

相关问题