diff options
-rw-r--r-- | arch/x86/xen/mmu.c | 44 |
1 files changed, 39 insertions, 5 deletions
diff --git a/arch/x86/xen/mmu.c b/arch/x86/xen/mmu.c index f5e86eee4e0e..e8e34938c57d 100644 --- a/arch/x86/xen/mmu.c +++ b/arch/x86/xen/mmu.c | |||
@@ -1408,7 +1408,6 @@ static void __xen_write_cr3(bool kernel, unsigned long cr3) | |||
1408 | xen_mc_callback(set_current_cr3, (void *)cr3); | 1408 | xen_mc_callback(set_current_cr3, (void *)cr3); |
1409 | } | 1409 | } |
1410 | } | 1410 | } |
1411 | |||
1412 | static void xen_write_cr3(unsigned long cr3) | 1411 | static void xen_write_cr3(unsigned long cr3) |
1413 | { | 1412 | { |
1414 | BUG_ON(preemptible()); | 1413 | BUG_ON(preemptible()); |
@@ -1434,6 +1433,45 @@ static void xen_write_cr3(unsigned long cr3) | |||
1434 | xen_mc_issue(PARAVIRT_LAZY_CPU); /* interrupts restored */ | 1433 | xen_mc_issue(PARAVIRT_LAZY_CPU); /* interrupts restored */ |
1435 | } | 1434 | } |
1436 | 1435 | ||
1436 | #ifdef CONFIG_X86_64 | ||
1437 | /* | ||
1438 | * At the start of the day - when Xen launches a guest, it has already | ||
1439 | * built pagetables for the guest. We diligently look over them | ||
1440 | * in xen_setup_kernel_pagetable and graft as appropiate them in the | ||
1441 | * init_level4_pgt and its friends. Then when we are happy we load | ||
1442 | * the new init_level4_pgt - and continue on. | ||
1443 | * | ||
1444 | * The generic code starts (start_kernel) and 'init_mem_mapping' sets | ||
1445 | * up the rest of the pagetables. When it has completed it loads the cr3. | ||
1446 | * N.B. that baremetal would start at 'start_kernel' (and the early | ||
1447 | * #PF handler would create bootstrap pagetables) - so we are running | ||
1448 | * with the same assumptions as what to do when write_cr3 is executed | ||
1449 | * at this point. | ||
1450 | * | ||
1451 | * Since there are no user-page tables at all, we have two variants | ||
1452 | * of xen_write_cr3 - the early bootup (this one), and the late one | ||
1453 | * (xen_write_cr3). The reason we have to do that is that in 64-bit | ||
1454 | * the Linux kernel and user-space are both in ring 3 while the | ||
1455 | * hypervisor is in ring 0. | ||
1456 | */ | ||
1457 | static void __init xen_write_cr3_init(unsigned long cr3) | ||
1458 | { | ||
1459 | BUG_ON(preemptible()); | ||
1460 | |||
1461 | xen_mc_batch(); /* disables interrupts */ | ||
1462 | |||
1463 | /* Update while interrupts are disabled, so its atomic with | ||
1464 | respect to ipis */ | ||
1465 | this_cpu_write(xen_cr3, cr3); | ||
1466 | |||
1467 | __xen_write_cr3(true, cr3); | ||
1468 | |||
1469 | xen_mc_issue(PARAVIRT_LAZY_CPU); /* interrupts restored */ | ||
1470 | |||
1471 | pv_mmu_ops.write_cr3 = &xen_write_cr3; | ||
1472 | } | ||
1473 | #endif | ||
1474 | |||
1437 | static int xen_pgd_alloc(struct mm_struct *mm) | 1475 | static int xen_pgd_alloc(struct mm_struct *mm) |
1438 | { | 1476 | { |
1439 | pgd_t *pgd = mm->pgd; | 1477 | pgd_t *pgd = mm->pgd; |
@@ -2102,11 +2140,7 @@ static const struct pv_mmu_ops xen_mmu_ops __initconst = { | |||
2102 | .write_cr2 = xen_write_cr2, | 2140 | .write_cr2 = xen_write_cr2, |
2103 | 2141 | ||
2104 | .read_cr3 = xen_read_cr3, | 2142 | .read_cr3 = xen_read_cr3, |
2105 | #ifdef CONFIG_X86_32 | ||
2106 | .write_cr3 = xen_write_cr3_init, | 2143 | .write_cr3 = xen_write_cr3_init, |
2107 | #else | ||
2108 | .write_cr3 = xen_write_cr3, | ||
2109 | #endif | ||
2110 | 2144 | ||
2111 | .flush_tlb_user = xen_flush_tlb, | 2145 | .flush_tlb_user = xen_flush_tlb, |
2112 | .flush_tlb_kernel = xen_flush_tlb, | 2146 | .flush_tlb_kernel = xen_flush_tlb, |