Commit 7f689041 authored by Kirill A. Shutemov's avatar Kirill A. Shutemov Committed by Ingo Molnar

x86/kexec: Add 5-level paging support

Handle additional page table level in the kexec code.
Signed-off-by: default avatarKirill A. Shutemov <kirill.shutemov@linux.intel.com>
Acked-by: default avatarThomas Gleixner <tglx@linutronix.de>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Andy Lutomirski <luto@amacapital.net>
Cc: Andy Lutomirski <luto@kernel.org>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Borislav Petkov <bp@alien8.de>
Cc: Brian Gerst <brgerst@gmail.com>
Cc: Dave Hansen <dave.hansen@intel.com>
Cc: Denys Vlasenko <dvlasenk@redhat.com>
Cc: H. Peter Anvin <hpa@zytor.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: linux-arch@vger.kernel.org
Cc: linux-mm@kvack.org
Link: http://lkml.kernel.org/r/20170317185515.8636-2-kirill.shutemov@linux.intel.comSigned-off-by: default avatarIngo Molnar <mingo@kernel.org>
parent 1f9ca184
...@@ -164,6 +164,7 @@ struct kimage_arch { ...@@ -164,6 +164,7 @@ struct kimage_arch {
}; };
#else #else
struct kimage_arch { struct kimage_arch {
p4d_t *p4d;
pud_t *pud; pud_t *pud;
pmd_t *pmd; pmd_t *pmd;
pte_t *pte; pte_t *pte;
......
...@@ -103,6 +103,7 @@ static void machine_kexec_page_table_set_one( ...@@ -103,6 +103,7 @@ static void machine_kexec_page_table_set_one(
pgd_t *pgd, pmd_t *pmd, pte_t *pte, pgd_t *pgd, pmd_t *pmd, pte_t *pte,
unsigned long vaddr, unsigned long paddr) unsigned long vaddr, unsigned long paddr)
{ {
p4d_t *p4d;
pud_t *pud; pud_t *pud;
pgd += pgd_index(vaddr); pgd += pgd_index(vaddr);
...@@ -110,7 +111,8 @@ static void machine_kexec_page_table_set_one( ...@@ -110,7 +111,8 @@ static void machine_kexec_page_table_set_one(
if (!(pgd_val(*pgd) & _PAGE_PRESENT)) if (!(pgd_val(*pgd) & _PAGE_PRESENT))
set_pgd(pgd, __pgd(__pa(pmd) | _PAGE_PRESENT)); set_pgd(pgd, __pgd(__pa(pmd) | _PAGE_PRESENT));
#endif #endif
pud = pud_offset(pgd, vaddr); p4d = p4d_offset(pgd, vaddr);
pud = pud_offset(p4d, vaddr);
pmd = pmd_offset(pud, vaddr); pmd = pmd_offset(pud, vaddr);
if (!(pmd_val(*pmd) & _PAGE_PRESENT)) if (!(pmd_val(*pmd) & _PAGE_PRESENT))
set_pmd(pmd, __pmd(__pa(pte) | _PAGE_TABLE)); set_pmd(pmd, __pmd(__pa(pte) | _PAGE_TABLE));
......
...@@ -36,6 +36,7 @@ static struct kexec_file_ops *kexec_file_loaders[] = { ...@@ -36,6 +36,7 @@ static struct kexec_file_ops *kexec_file_loaders[] = {
static void free_transition_pgtable(struct kimage *image) static void free_transition_pgtable(struct kimage *image)
{ {
free_page((unsigned long)image->arch.p4d);
free_page((unsigned long)image->arch.pud); free_page((unsigned long)image->arch.pud);
free_page((unsigned long)image->arch.pmd); free_page((unsigned long)image->arch.pmd);
free_page((unsigned long)image->arch.pte); free_page((unsigned long)image->arch.pte);
...@@ -43,6 +44,7 @@ static void free_transition_pgtable(struct kimage *image) ...@@ -43,6 +44,7 @@ static void free_transition_pgtable(struct kimage *image)
static int init_transition_pgtable(struct kimage *image, pgd_t *pgd) static int init_transition_pgtable(struct kimage *image, pgd_t *pgd)
{ {
p4d_t *p4d;
pud_t *pud; pud_t *pud;
pmd_t *pmd; pmd_t *pmd;
pte_t *pte; pte_t *pte;
...@@ -53,13 +55,21 @@ static int init_transition_pgtable(struct kimage *image, pgd_t *pgd) ...@@ -53,13 +55,21 @@ static int init_transition_pgtable(struct kimage *image, pgd_t *pgd)
paddr = __pa(page_address(image->control_code_page)+PAGE_SIZE); paddr = __pa(page_address(image->control_code_page)+PAGE_SIZE);
pgd += pgd_index(vaddr); pgd += pgd_index(vaddr);
if (!pgd_present(*pgd)) { if (!pgd_present(*pgd)) {
p4d = (p4d_t *)get_zeroed_page(GFP_KERNEL);
if (!p4d)
goto err;
image->arch.p4d = p4d;
set_pgd(pgd, __pgd(__pa(p4d) | _KERNPG_TABLE));
}
p4d = p4d_offset(pgd, vaddr);
if (!p4d_present(*p4d)) {
pud = (pud_t *)get_zeroed_page(GFP_KERNEL); pud = (pud_t *)get_zeroed_page(GFP_KERNEL);
if (!pud) if (!pud)
goto err; goto err;
image->arch.pud = pud; image->arch.pud = pud;
set_pgd(pgd, __pgd(__pa(pud) | _KERNPG_TABLE)); set_p4d(p4d, __p4d(__pa(pud) | _KERNPG_TABLE));
} }
pud = pud_offset(pgd, vaddr); pud = pud_offset(p4d, vaddr);
if (!pud_present(*pud)) { if (!pud_present(*pud)) {
pmd = (pmd_t *)get_zeroed_page(GFP_KERNEL); pmd = (pmd_t *)get_zeroed_page(GFP_KERNEL);
if (!pmd) if (!pmd)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment