除了手动遍历页面目录条目之外,有没有合适的方法来获取逻辑地址的物理地址?我发现有一个 follow_page 函数可以很好地完成内置的巨大且透明的大页面支持 . 但它没有导出到内核模块(为什么???)...
follow_page
所以,我不能手动重新实现 follow_page 功能 .
好吧,它可能看起来像那样(从虚拟地址跟随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); } } } }
我认为您可以通过组合 /proc/[pid]/maps (给出进程的虚拟映射)和 /proc/[pid]/pagemap (为每个可寻址页面提供虚拟页面到物理页面映射)的间接方法实现虚拟 - >物理转换 . 首先,从 maps 找出您的进程的虚拟地址的映射(这样做是为了不搜索 pagemap 中的每个字节)然后检查页面映射中所需虚拟地址的物理映射(页面映射不是文本格式以下是格式Pagemap的详细说明 . 这应该为您提供精确的虚拟 - >物理映射
/proc/[pid]/maps
/proc/[pid]/pagemap
maps
pagemap
听起来你正在寻找virt_to_phys .
3 回答
好吧,它可能看起来像那样(从虚拟地址跟随PTE):
我认为您可以通过组合
/proc/[pid]/maps
(给出进程的虚拟映射)和/proc/[pid]/pagemap
(为每个可寻址页面提供虚拟页面到物理页面映射)的间接方法实现虚拟 - >物理转换 . 首先,从maps
找出您的进程的虚拟地址的映射(这样做是为了不搜索pagemap
中的每个字节)然后检查页面映射中所需虚拟地址的物理映射(页面映射不是文本格式以下是格式Pagemap的详细说明 . 这应该为您提供精确的虚拟 - >物理映射听起来你正在寻找virt_to_phys .