linux内核 - pte_xxx()API在模块编程中不起作用

时间:2016-12-09 16:20:42

标签: c linux memory linux-kernel

我正在研究linux内存管理。

我尝试创建一个类似于 mmap。

的模块

但是当我使用pte_xxx() API时,会发生内核恐慌。

我的架构是x86-64,linux内核版本是4.4

所以,我认为是4级分页。

pgd_xxx()API,pud_xxx()API和pmd_xxx()API似乎没问题。

然而,当我调用pte_xxx API()时,发生内核恐慌。

具体来说,pgd_val(* pgd),pud_val(* pud)和pmd_val(* pmd)不会导致内核崩溃。

pte_val(* pte)导致内核恐慌。

我确认pte_offset_map(pmd,address)不会引起恐慌。

似乎pte_val()有问题。

我不知道发生了什么。

似乎我编码不当。

请评论。谢谢。

void callback_fn(struct mm_struct *mm)
{
    unsigned long address = mm->mmap->vm_start;
    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)) {

                    //  I think pte_val() cause panic.
                    entry->pte = pte_val(*pte);
                    printk("    pte = %lx\n", pte_val(*pte));
                } else {
                    printk("    pte = empty\n");
                }
                pte_unmap(pte);
            }
        }
    }
}

1 个答案:

答案 0 :(得分:0)

抱歉,我的源代码另一部分(程序)发生内核恐慌。

以上代码效果很好。遗憾。