diff options
Diffstat (limited to 'arch/x86/kvm')
-rw-r--r-- | arch/x86/kvm/svm.c | 52 |
1 files changed, 40 insertions, 12 deletions
diff --git a/arch/x86/kvm/svm.c b/arch/x86/kvm/svm.c index 67fad6641d55..5e55a1bdd13d 100644 --- a/arch/x86/kvm/svm.c +++ b/arch/x86/kvm/svm.c | |||
@@ -1390,6 +1390,39 @@ static inline int nested_svm_intr(struct vcpu_svm *svm) | |||
1390 | return 0; | 1390 | return 0; |
1391 | } | 1391 | } |
1392 | 1392 | ||
1393 | static void *nested_svm_map(struct vcpu_svm *svm, u64 gpa, enum km_type idx) | ||
1394 | { | ||
1395 | struct page *page; | ||
1396 | |||
1397 | down_read(¤t->mm->mmap_sem); | ||
1398 | page = gfn_to_page(svm->vcpu.kvm, gpa >> PAGE_SHIFT); | ||
1399 | up_read(¤t->mm->mmap_sem); | ||
1400 | |||
1401 | if (is_error_page(page)) | ||
1402 | goto error; | ||
1403 | |||
1404 | return kmap_atomic(page, idx); | ||
1405 | |||
1406 | error: | ||
1407 | kvm_release_page_clean(page); | ||
1408 | kvm_inject_gp(&svm->vcpu, 0); | ||
1409 | |||
1410 | return NULL; | ||
1411 | } | ||
1412 | |||
1413 | static void nested_svm_unmap(void *addr, enum km_type idx) | ||
1414 | { | ||
1415 | struct page *page; | ||
1416 | |||
1417 | if (!addr) | ||
1418 | return; | ||
1419 | |||
1420 | page = kmap_atomic_to_page(addr); | ||
1421 | |||
1422 | kunmap_atomic(addr, idx); | ||
1423 | kvm_release_page_dirty(page); | ||
1424 | } | ||
1425 | |||
1393 | static struct page *nested_svm_get_page(struct vcpu_svm *svm, u64 gpa) | 1426 | static struct page *nested_svm_get_page(struct vcpu_svm *svm, u64 gpa) |
1394 | { | 1427 | { |
1395 | struct page *page; | 1428 | struct page *page; |
@@ -1597,13 +1630,16 @@ static inline void copy_vmcb_control_area(struct vmcb *dst_vmcb, struct vmcb *fr | |||
1597 | dst->lbr_ctl = from->lbr_ctl; | 1630 | dst->lbr_ctl = from->lbr_ctl; |
1598 | } | 1631 | } |
1599 | 1632 | ||
1600 | static int nested_svm_vmexit_real(struct vcpu_svm *svm, void *arg1, | 1633 | static int nested_svm_vmexit(struct vcpu_svm *svm) |
1601 | void *arg2, void *opaque) | ||
1602 | { | 1634 | { |
1603 | struct vmcb *nested_vmcb = (struct vmcb *)arg1; | 1635 | struct vmcb *nested_vmcb; |
1604 | struct vmcb *hsave = svm->nested.hsave; | 1636 | struct vmcb *hsave = svm->nested.hsave; |
1605 | struct vmcb *vmcb = svm->vmcb; | 1637 | struct vmcb *vmcb = svm->vmcb; |
1606 | 1638 | ||
1639 | nested_vmcb = nested_svm_map(svm, svm->nested.vmcb, KM_USER0); | ||
1640 | if (!nested_vmcb) | ||
1641 | return 1; | ||
1642 | |||
1607 | /* Give the current vmcb to the guest */ | 1643 | /* Give the current vmcb to the guest */ |
1608 | disable_gif(svm); | 1644 | disable_gif(svm); |
1609 | 1645 | ||
@@ -1678,15 +1714,7 @@ static int nested_svm_vmexit_real(struct vcpu_svm *svm, void *arg1, | |||
1678 | /* Exit nested SVM mode */ | 1714 | /* Exit nested SVM mode */ |
1679 | svm->nested.vmcb = 0; | 1715 | svm->nested.vmcb = 0; |
1680 | 1716 | ||
1681 | return 0; | 1717 | nested_svm_unmap(nested_vmcb, KM_USER0); |
1682 | } | ||
1683 | |||
1684 | static int nested_svm_vmexit(struct vcpu_svm *svm) | ||
1685 | { | ||
1686 | nsvm_printk("VMexit\n"); | ||
1687 | if (nested_svm_do(svm, svm->nested.vmcb, 0, | ||
1688 | NULL, nested_svm_vmexit_real)) | ||
1689 | return 1; | ||
1690 | 1718 | ||
1691 | kvm_mmu_reset_context(&svm->vcpu); | 1719 | kvm_mmu_reset_context(&svm->vcpu); |
1692 | kvm_mmu_load(&svm->vcpu); | 1720 | kvm_mmu_load(&svm->vcpu); |