Как получить физический адрес из логического в модуле ядра Linux? - PullRequest
9 голосов
/ 06 июня 2011

Есть ли какой-нибудь подходящий способ получить физический адрес по логическому, кроме как просматривать записи каталога страниц вручную? Я искал эту функциональность в исходных кодах ядра и обнаружил, что есть функция follow_page, которая хорошо справляется со встроенной поддержкой огромных и прозрачных огромных страниц. Но это не экспортируется в модули ядра (почему ???) ...

Итак, я не хочу изобретать колесо и думаю, что не очень хорошо переопределять функциональность follow_page вручную.

Ответы [ 3 ]

6 голосов
/ 07 июня 2011

Ну, это может выглядеть примерно так (следуйте 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 голосов
/ 07 июня 2011

Я думаю, что вы можете достичь виртуального-> физического перевода с помощью косвенного метода путем комбинации /proc/[pid]/maps (дает виртуальное отображение процесса) и /proc/[pid]/pagemap (дает отображение виртуальной страницы на физическую страницу для каждой адресуемой страницы),Во-первых, выясните соответствие виртуальных адресов вашего процесса из maps (это делается для того, чтобы вы не искали каждый байт в pagemap). Затем проверьте физическое сопоставление нужного виртуального адреса в pagemap (pagemapне в текстовом формате. Вот подробное объяснение формата Pagemap ) Это должно дать вам точное виртуальное -> физическое отображение

0 голосов
/ 06 июня 2011

Звучит так, будто вы ищете virt_to_phys.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...