aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kvm/vmx.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/kvm/vmx.c')
-rw-r--r--arch/x86/kvm/vmx.c49
1 files changed, 33 insertions, 16 deletions
diff --git a/arch/x86/kvm/vmx.c b/arch/x86/kvm/vmx.c
index e8a5649f9c15..f3ab27b5a6b2 100644
--- a/arch/x86/kvm/vmx.c
+++ b/arch/x86/kvm/vmx.c
@@ -736,23 +736,45 @@ static void vmx_set_rflags(struct kvm_vcpu *vcpu, unsigned long rflags)
736 vmcs_writel(GUEST_RFLAGS, rflags); 736 vmcs_writel(GUEST_RFLAGS, rflags);
737} 737}
738 738
739static u32 vmx_get_interrupt_shadow(struct kvm_vcpu *vcpu, int mask)
740{
741 u32 interruptibility = vmcs_read32(GUEST_INTERRUPTIBILITY_INFO);
742 int ret = 0;
743
744 if (interruptibility & GUEST_INTR_STATE_STI)
745 ret |= X86_SHADOW_INT_STI;
746 if (interruptibility & GUEST_INTR_STATE_MOV_SS)
747 ret |= X86_SHADOW_INT_MOV_SS;
748
749 return ret & mask;
750}
751
752static void vmx_set_interrupt_shadow(struct kvm_vcpu *vcpu, int mask)
753{
754 u32 interruptibility_old = vmcs_read32(GUEST_INTERRUPTIBILITY_INFO);
755 u32 interruptibility = interruptibility_old;
756
757 interruptibility &= ~(GUEST_INTR_STATE_STI | GUEST_INTR_STATE_MOV_SS);
758
759 if (mask & X86_SHADOW_INT_MOV_SS)
760 interruptibility |= GUEST_INTR_STATE_MOV_SS;
761 if (mask & X86_SHADOW_INT_STI)
762 interruptibility |= GUEST_INTR_STATE_STI;
763
764 if ((interruptibility != interruptibility_old))
765 vmcs_write32(GUEST_INTERRUPTIBILITY_INFO, interruptibility);
766}
767
739static void skip_emulated_instruction(struct kvm_vcpu *vcpu) 768static void skip_emulated_instruction(struct kvm_vcpu *vcpu)
740{ 769{
741 unsigned long rip; 770 unsigned long rip;
742 u32 interruptibility;
743 771
744 rip = kvm_rip_read(vcpu); 772 rip = kvm_rip_read(vcpu);
745 rip += vmcs_read32(VM_EXIT_INSTRUCTION_LEN); 773 rip += vmcs_read32(VM_EXIT_INSTRUCTION_LEN);
746 kvm_rip_write(vcpu, rip); 774 kvm_rip_write(vcpu, rip);
747 775
748 /* 776 /* skipping an emulated instruction also counts */
749 * We emulated an instruction, so temporary interrupt blocking 777 vmx_set_interrupt_shadow(vcpu, 0);
750 * should be removed, if set.
751 */
752 interruptibility = vmcs_read32(GUEST_INTERRUPTIBILITY_INFO);
753 if (interruptibility & 3)
754 vmcs_write32(GUEST_INTERRUPTIBILITY_INFO,
755 interruptibility & ~3);
756} 778}
757 779
758static void vmx_queue_exception(struct kvm_vcpu *vcpu, unsigned nr, 780static void vmx_queue_exception(struct kvm_vcpu *vcpu, unsigned nr,
@@ -2400,12 +2422,6 @@ out:
2400 return ret; 2422 return ret;
2401} 2423}
2402 2424
2403void vmx_drop_interrupt_shadow(struct kvm_vcpu *vcpu)
2404{
2405 vmcs_clear_bits(GUEST_INTERRUPTIBILITY_INFO,
2406 GUEST_INTR_STATE_STI | GUEST_INTR_STATE_MOV_SS);
2407}
2408
2409static void enable_irq_window(struct kvm_vcpu *vcpu) 2425static void enable_irq_window(struct kvm_vcpu *vcpu)
2410{ 2426{
2411 u32 cpu_based_vm_exec_control; 2427 u32 cpu_based_vm_exec_control;
@@ -3649,6 +3665,8 @@ static struct kvm_x86_ops vmx_x86_ops = {
3649 .run = vmx_vcpu_run, 3665 .run = vmx_vcpu_run,
3650 .handle_exit = vmx_handle_exit, 3666 .handle_exit = vmx_handle_exit,
3651 .skip_emulated_instruction = skip_emulated_instruction, 3667 .skip_emulated_instruction = skip_emulated_instruction,
3668 .set_interrupt_shadow = vmx_set_interrupt_shadow,
3669 .get_interrupt_shadow = vmx_get_interrupt_shadow,
3652 .patch_hypercall = vmx_patch_hypercall, 3670 .patch_hypercall = vmx_patch_hypercall,
3653 .set_irq = vmx_inject_irq, 3671 .set_irq = vmx_inject_irq,
3654 .set_nmi = vmx_inject_nmi, 3672 .set_nmi = vmx_inject_nmi,
@@ -3658,7 +3676,6 @@ static struct kvm_x86_ops vmx_x86_ops = {
3658 .enable_nmi_window = enable_nmi_window, 3676 .enable_nmi_window = enable_nmi_window,
3659 .enable_irq_window = enable_irq_window, 3677 .enable_irq_window = enable_irq_window,
3660 .update_cr8_intercept = update_cr8_intercept, 3678 .update_cr8_intercept = update_cr8_intercept,
3661 .drop_interrupt_shadow = vmx_drop_interrupt_shadow,
3662 3679
3663 .set_tss_addr = vmx_set_tss_addr, 3680 .set_tss_addr = vmx_set_tss_addr,
3664 .get_tdp_level = get_ept_level, 3681 .get_tdp_level = get_ept_level,