aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86
diff options
context:
space:
mode:
authorAvi Kivity <avi@qumranet.com>2008-05-13 06:23:38 -0400
committerAvi Kivity <avi@qumranet.com>2008-07-20 05:41:43 -0400
commit4ecac3fd6dc2629ad76a658a486f081c44aef10e (patch)
treea078acb4c626432ac8cf887b911d912f4a316d06 /arch/x86
parent1b7fcd3263e5f12dba43d27b64e1578bec070c28 (diff)
KVM: Handle virtualization instruction #UD faults during reboot
KVM turns off hardware virtualization extensions during reboot, in order to disassociate the memory used by the virtualization extensions from the processor, and in order to have the system in a consistent state. Unfortunately virtual machines may still be running while this goes on, and once virtualization extensions are turned off, any virtulization instruction will #UD on execution. Fix by adding an exception handler to virtualization instructions; if we get an exception during reboot, we simply spin waiting for the reset to complete. If it's a true exception, BUG() so we can have our stack trace. Signed-off-by: Avi Kivity <avi@qumranet.com>
Diffstat (limited to 'arch/x86')
-rw-r--r--arch/x86/kvm/svm.c20
-rw-r--r--arch/x86/kvm/vmx.c25
2 files changed, 25 insertions, 20 deletions
diff --git a/arch/x86/kvm/svm.c b/arch/x86/kvm/svm.c
index 992ab7115871..9390a31c06f4 100644
--- a/arch/x86/kvm/svm.c
+++ b/arch/x86/kvm/svm.c
@@ -27,6 +27,8 @@
27 27
28#include <asm/desc.h> 28#include <asm/desc.h>
29 29
30#define __ex(x) __kvm_handle_fault_on_reboot(x)
31
30MODULE_AUTHOR("Qumranet"); 32MODULE_AUTHOR("Qumranet");
31MODULE_LICENSE("GPL"); 33MODULE_LICENSE("GPL");
32 34
@@ -129,17 +131,17 @@ static inline void push_irq(struct kvm_vcpu *vcpu, u8 irq)
129 131
130static inline void clgi(void) 132static inline void clgi(void)
131{ 133{
132 asm volatile (SVM_CLGI); 134 asm volatile (__ex(SVM_CLGI));
133} 135}
134 136
135static inline void stgi(void) 137static inline void stgi(void)
136{ 138{
137 asm volatile (SVM_STGI); 139 asm volatile (__ex(SVM_STGI));
138} 140}
139 141
140static inline void invlpga(unsigned long addr, u32 asid) 142static inline void invlpga(unsigned long addr, u32 asid)
141{ 143{
142 asm volatile (SVM_INVLPGA :: "a"(addr), "c"(asid)); 144 asm volatile (__ex(SVM_INVLPGA) :: "a"(addr), "c"(asid));
143} 145}
144 146
145static inline unsigned long kvm_read_cr2(void) 147static inline unsigned long kvm_read_cr2(void)
@@ -1758,17 +1760,17 @@ static void svm_vcpu_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
1758 /* Enter guest mode */ 1760 /* Enter guest mode */
1759 "push %%rax \n\t" 1761 "push %%rax \n\t"
1760 "mov %c[vmcb](%[svm]), %%rax \n\t" 1762 "mov %c[vmcb](%[svm]), %%rax \n\t"
1761 SVM_VMLOAD "\n\t" 1763 __ex(SVM_VMLOAD) "\n\t"
1762 SVM_VMRUN "\n\t" 1764 __ex(SVM_VMRUN) "\n\t"
1763 SVM_VMSAVE "\n\t" 1765 __ex(SVM_VMSAVE) "\n\t"
1764 "pop %%rax \n\t" 1766 "pop %%rax \n\t"
1765#else 1767#else
1766 /* Enter guest mode */ 1768 /* Enter guest mode */
1767 "push %%eax \n\t" 1769 "push %%eax \n\t"
1768 "mov %c[vmcb](%[svm]), %%eax \n\t" 1770 "mov %c[vmcb](%[svm]), %%eax \n\t"
1769 SVM_VMLOAD "\n\t" 1771 __ex(SVM_VMLOAD) "\n\t"
1770 SVM_VMRUN "\n\t" 1772 __ex(SVM_VMRUN) "\n\t"
1771 SVM_VMSAVE "\n\t" 1773 __ex(SVM_VMSAVE) "\n\t"
1772 "pop %%eax \n\t" 1774 "pop %%eax \n\t"
1773#endif 1775#endif
1774 1776
diff --git a/arch/x86/kvm/vmx.c b/arch/x86/kvm/vmx.c
index fff8e23433d6..b80b4d141637 100644
--- a/arch/x86/kvm/vmx.c
+++ b/arch/x86/kvm/vmx.c
@@ -30,6 +30,8 @@
30#include <asm/io.h> 30#include <asm/io.h>
31#include <asm/desc.h> 31#include <asm/desc.h>
32 32
33#define __ex(x) __kvm_handle_fault_on_reboot(x)
34
33MODULE_AUTHOR("Qumranet"); 35MODULE_AUTHOR("Qumranet");
34MODULE_LICENSE("GPL"); 36MODULE_LICENSE("GPL");
35 37
@@ -278,7 +280,7 @@ static inline void __invvpid(int ext, u16 vpid, gva_t gva)
278 u64 gva; 280 u64 gva;
279 } operand = { vpid, 0, gva }; 281 } operand = { vpid, 0, gva };
280 282
281 asm volatile (ASM_VMX_INVVPID 283 asm volatile (__ex(ASM_VMX_INVVPID)
282 /* CF==1 or ZF==1 --> rc = -1 */ 284 /* CF==1 or ZF==1 --> rc = -1 */
283 "; ja 1f ; ud2 ; 1:" 285 "; ja 1f ; ud2 ; 1:"
284 : : "a"(&operand), "c"(ext) : "cc", "memory"); 286 : : "a"(&operand), "c"(ext) : "cc", "memory");
@@ -290,7 +292,7 @@ static inline void __invept(int ext, u64 eptp, gpa_t gpa)
290 u64 eptp, gpa; 292 u64 eptp, gpa;
291 } operand = {eptp, gpa}; 293 } operand = {eptp, gpa};
292 294
293 asm volatile (ASM_VMX_INVEPT 295 asm volatile (__ex(ASM_VMX_INVEPT)
294 /* CF==1 or ZF==1 --> rc = -1 */ 296 /* CF==1 or ZF==1 --> rc = -1 */
295 "; ja 1f ; ud2 ; 1:\n" 297 "; ja 1f ; ud2 ; 1:\n"
296 : : "a" (&operand), "c" (ext) : "cc", "memory"); 298 : : "a" (&operand), "c" (ext) : "cc", "memory");
@@ -311,7 +313,7 @@ static void vmcs_clear(struct vmcs *vmcs)
311 u64 phys_addr = __pa(vmcs); 313 u64 phys_addr = __pa(vmcs);
312 u8 error; 314 u8 error;
313 315
314 asm volatile (ASM_VMX_VMCLEAR_RAX "; setna %0" 316 asm volatile (__ex(ASM_VMX_VMCLEAR_RAX) "; setna %0"
315 : "=g"(error) : "a"(&phys_addr), "m"(phys_addr) 317 : "=g"(error) : "a"(&phys_addr), "m"(phys_addr)
316 : "cc", "memory"); 318 : "cc", "memory");
317 if (error) 319 if (error)
@@ -378,7 +380,7 @@ static unsigned long vmcs_readl(unsigned long field)
378{ 380{
379 unsigned long value; 381 unsigned long value;
380 382
381 asm volatile (ASM_VMX_VMREAD_RDX_RAX 383 asm volatile (__ex(ASM_VMX_VMREAD_RDX_RAX)
382 : "=a"(value) : "d"(field) : "cc"); 384 : "=a"(value) : "d"(field) : "cc");
383 return value; 385 return value;
384} 386}
@@ -413,7 +415,7 @@ static void vmcs_writel(unsigned long field, unsigned long value)
413{ 415{
414 u8 error; 416 u8 error;
415 417
416 asm volatile (ASM_VMX_VMWRITE_RAX_RDX "; setna %0" 418 asm volatile (__ex(ASM_VMX_VMWRITE_RAX_RDX) "; setna %0"
417 : "=q"(error) : "a"(value), "d"(field) : "cc"); 419 : "=q"(error) : "a"(value), "d"(field) : "cc");
418 if (unlikely(error)) 420 if (unlikely(error))
419 vmwrite_error(field, value); 421 vmwrite_error(field, value);
@@ -621,7 +623,7 @@ static void vmx_vcpu_load(struct kvm_vcpu *vcpu, int cpu)
621 u8 error; 623 u8 error;
622 624
623 per_cpu(current_vmcs, cpu) = vmx->vmcs; 625 per_cpu(current_vmcs, cpu) = vmx->vmcs;
624 asm volatile (ASM_VMX_VMPTRLD_RAX "; setna %0" 626 asm volatile (__ex(ASM_VMX_VMPTRLD_RAX) "; setna %0"
625 : "=g"(error) : "a"(&phys_addr), "m"(phys_addr) 627 : "=g"(error) : "a"(&phys_addr), "m"(phys_addr)
626 : "cc"); 628 : "cc");
627 if (error) 629 if (error)
@@ -1030,13 +1032,14 @@ static void hardware_enable(void *garbage)
1030 MSR_IA32_FEATURE_CONTROL_LOCKED | 1032 MSR_IA32_FEATURE_CONTROL_LOCKED |
1031 MSR_IA32_FEATURE_CONTROL_VMXON_ENABLED); 1033 MSR_IA32_FEATURE_CONTROL_VMXON_ENABLED);
1032 write_cr4(read_cr4() | X86_CR4_VMXE); /* FIXME: not cpu hotplug safe */ 1034 write_cr4(read_cr4() | X86_CR4_VMXE); /* FIXME: not cpu hotplug safe */
1033 asm volatile (ASM_VMX_VMXON_RAX : : "a"(&phys_addr), "m"(phys_addr) 1035 asm volatile (ASM_VMX_VMXON_RAX
1036 : : "a"(&phys_addr), "m"(phys_addr)
1034 : "memory", "cc"); 1037 : "memory", "cc");
1035} 1038}
1036 1039
1037static void hardware_disable(void *garbage) 1040static void hardware_disable(void *garbage)
1038{ 1041{
1039 asm volatile (ASM_VMX_VMXOFF : : : "cc"); 1042 asm volatile (__ex(ASM_VMX_VMXOFF) : : : "cc");
1040 write_cr4(read_cr4() & ~X86_CR4_VMXE); 1043 write_cr4(read_cr4() & ~X86_CR4_VMXE);
1041} 1044}
1042 1045
@@ -2834,7 +2837,7 @@ static void vmx_vcpu_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
2834 "push %%edx; push %%ebp;" 2837 "push %%edx; push %%ebp;"
2835 "push %%ecx \n\t" 2838 "push %%ecx \n\t"
2836#endif 2839#endif
2837 ASM_VMX_VMWRITE_RSP_RDX "\n\t" 2840 __ex(ASM_VMX_VMWRITE_RSP_RDX) "\n\t"
2838 /* Check if vmlaunch of vmresume is needed */ 2841 /* Check if vmlaunch of vmresume is needed */
2839 "cmpl $0, %c[launched](%0) \n\t" 2842 "cmpl $0, %c[launched](%0) \n\t"
2840 /* Load guest registers. Don't clobber flags. */ 2843 /* Load guest registers. Don't clobber flags. */
@@ -2869,9 +2872,9 @@ static void vmx_vcpu_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
2869#endif 2872#endif
2870 /* Enter guest mode */ 2873 /* Enter guest mode */
2871 "jne .Llaunched \n\t" 2874 "jne .Llaunched \n\t"
2872 ASM_VMX_VMLAUNCH "\n\t" 2875 __ex(ASM_VMX_VMLAUNCH) "\n\t"
2873 "jmp .Lkvm_vmx_return \n\t" 2876 "jmp .Lkvm_vmx_return \n\t"
2874 ".Llaunched: " ASM_VMX_VMRESUME "\n\t" 2877 ".Llaunched: " __ex(ASM_VMX_VMRESUME) "\n\t"
2875 ".Lkvm_vmx_return: " 2878 ".Lkvm_vmx_return: "
2876 /* Save guest registers, load host registers, keep flags */ 2879 /* Save guest registers, load host registers, keep flags */
2877#ifdef CONFIG_X86_64 2880#ifdef CONFIG_X86_64