aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/kvm/vmx.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/kvm/vmx.c')
-rw-r--r--drivers/kvm/vmx.c128
1 files changed, 72 insertions, 56 deletions
diff --git a/drivers/kvm/vmx.c b/drivers/kvm/vmx.c
index a05bfa085877..872ca0381fbe 100644
--- a/drivers/kvm/vmx.c
+++ b/drivers/kvm/vmx.c
@@ -85,19 +85,6 @@ static const u32 vmx_msr_index[] = {
85}; 85};
86#define NR_VMX_MSR ARRAY_SIZE(vmx_msr_index) 86#define NR_VMX_MSR ARRAY_SIZE(vmx_msr_index)
87 87
88#ifdef CONFIG_X86_64
89static unsigned msr_offset_kernel_gs_base;
90#define NR_64BIT_MSRS 4
91/*
92 * avoid save/load MSR_SYSCALL_MASK and MSR_LSTAR by std vt
93 * mechanism (cpu bug AA24)
94 */
95#define NR_BAD_MSRS 2
96#else
97#define NR_64BIT_MSRS 0
98#define NR_BAD_MSRS 0
99#endif
100
101static inline int is_page_fault(u32 intr_info) 88static inline int is_page_fault(u32 intr_info)
102{ 89{
103 return (intr_info & (INTR_INFO_INTR_TYPE_MASK | INTR_INFO_VECTOR_MASK | 90 return (intr_info & (INTR_INFO_INTR_TYPE_MASK | INTR_INFO_VECTOR_MASK |
@@ -118,13 +105,23 @@ static inline int is_external_interrupt(u32 intr_info)
118 == (INTR_TYPE_EXT_INTR | INTR_INFO_VALID_MASK); 105 == (INTR_TYPE_EXT_INTR | INTR_INFO_VALID_MASK);
119} 106}
120 107
121static struct vmx_msr_entry *find_msr_entry(struct kvm_vcpu *vcpu, u32 msr) 108static int __find_msr_index(struct kvm_vcpu *vcpu, u32 msr)
122{ 109{
123 int i; 110 int i;
124 111
125 for (i = 0; i < vcpu->nmsrs; ++i) 112 for (i = 0; i < vcpu->nmsrs; ++i)
126 if (vcpu->guest_msrs[i].index == msr) 113 if (vcpu->guest_msrs[i].index == msr)
127 return &vcpu->guest_msrs[i]; 114 return i;
115 return -1;
116}
117
118static struct vmx_msr_entry *find_msr_entry(struct kvm_vcpu *vcpu, u32 msr)
119{
120 int i;
121
122 i = __find_msr_index(vcpu, msr);
123 if (i >= 0)
124 return &vcpu->guest_msrs[i];
128 return NULL; 125 return NULL;
129} 126}
130 127
@@ -307,10 +304,10 @@ static void vmx_save_host_state(struct kvm_vcpu *vcpu)
307 304
308#ifdef CONFIG_X86_64 305#ifdef CONFIG_X86_64
309 if (is_long_mode(vcpu)) { 306 if (is_long_mode(vcpu)) {
310 save_msrs(vcpu->host_msrs + msr_offset_kernel_gs_base, 1); 307 save_msrs(vcpu->host_msrs + vcpu->msr_offset_kernel_gs_base, 1);
311 load_msrs(vcpu->guest_msrs, NR_BAD_MSRS);
312 } 308 }
313#endif 309#endif
310 load_msrs(vcpu->guest_msrs, vcpu->save_nmsrs);
314} 311}
315 312
316static void vmx_load_host_state(struct kvm_vcpu *vcpu) 313static void vmx_load_host_state(struct kvm_vcpu *vcpu)
@@ -337,12 +334,8 @@ static void vmx_load_host_state(struct kvm_vcpu *vcpu)
337 334
338 reload_tss(); 335 reload_tss();
339 } 336 }
340#ifdef CONFIG_X86_64 337 save_msrs(vcpu->guest_msrs, vcpu->save_nmsrs);
341 if (is_long_mode(vcpu)) { 338 load_msrs(vcpu->host_msrs, vcpu->save_nmsrs);
342 save_msrs(vcpu->guest_msrs, NR_BAD_MSRS);
343 load_msrs(vcpu->host_msrs, NR_BAD_MSRS);
344 }
345#endif
346} 339}
347 340
348/* 341/*
@@ -464,41 +457,74 @@ static void vmx_inject_gp(struct kvm_vcpu *vcpu, unsigned error_code)
464} 457}
465 458
466/* 459/*
460 * Swap MSR entry in host/guest MSR entry array.
461 */
462void move_msr_up(struct kvm_vcpu *vcpu, int from, int to)
463{
464 struct vmx_msr_entry tmp;
465 tmp = vcpu->guest_msrs[to];
466 vcpu->guest_msrs[to] = vcpu->guest_msrs[from];
467 vcpu->guest_msrs[from] = tmp;
468 tmp = vcpu->host_msrs[to];
469 vcpu->host_msrs[to] = vcpu->host_msrs[from];
470 vcpu->host_msrs[from] = tmp;
471}
472
473/*
467 * Set up the vmcs to automatically save and restore system 474 * Set up the vmcs to automatically save and restore system
468 * msrs. Don't touch the 64-bit msrs if the guest is in legacy 475 * msrs. Don't touch the 64-bit msrs if the guest is in legacy
469 * mode, as fiddling with msrs is very expensive. 476 * mode, as fiddling with msrs is very expensive.
470 */ 477 */
471static void setup_msrs(struct kvm_vcpu *vcpu) 478static void setup_msrs(struct kvm_vcpu *vcpu)
472{ 479{
473 int nr_skip, nr_good_msrs; 480 int index, save_nmsrs;
474 481
475 if (is_long_mode(vcpu)) 482 save_nmsrs = 0;
476 nr_skip = NR_BAD_MSRS; 483#ifdef CONFIG_X86_64
477 else 484 if (is_long_mode(vcpu)) {
478 nr_skip = NR_64BIT_MSRS; 485 index = __find_msr_index(vcpu, MSR_SYSCALL_MASK);
479 nr_good_msrs = vcpu->nmsrs - nr_skip; 486 if (index >= 0)
487 move_msr_up(vcpu, index, save_nmsrs++);
488 index = __find_msr_index(vcpu, MSR_LSTAR);
489 if (index >= 0)
490 move_msr_up(vcpu, index, save_nmsrs++);
491 index = __find_msr_index(vcpu, MSR_CSTAR);
492 if (index >= 0)
493 move_msr_up(vcpu, index, save_nmsrs++);
494 index = __find_msr_index(vcpu, MSR_KERNEL_GS_BASE);
495 if (index >= 0)
496 move_msr_up(vcpu, index, save_nmsrs++);
497 /*
498 * MSR_K6_STAR is only needed on long mode guests, and only
499 * if efer.sce is enabled.
500 */
501 index = __find_msr_index(vcpu, MSR_K6_STAR);
502 if ((index >= 0) && (vcpu->shadow_efer & EFER_SCE))
503 move_msr_up(vcpu, index, save_nmsrs++);
504 }
505#endif
506 vcpu->save_nmsrs = save_nmsrs;
480 507
481 /*
482 * MSR_K6_STAR is only needed on long mode guests, and only
483 * if efer.sce is enabled.
484 */
485 if (find_msr_entry(vcpu, MSR_K6_STAR)) {
486 --nr_good_msrs;
487#ifdef CONFIG_X86_64 508#ifdef CONFIG_X86_64
488 if (is_long_mode(vcpu) && (vcpu->shadow_efer & EFER_SCE)) 509 vcpu->msr_offset_kernel_gs_base =
489 ++nr_good_msrs; 510 __find_msr_index(vcpu, MSR_KERNEL_GS_BASE);
490#endif 511#endif
512 index = __find_msr_index(vcpu, MSR_EFER);
513 if (index >= 0)
514 save_nmsrs = 1;
515 else {
516 save_nmsrs = 0;
517 index = 0;
491 } 518 }
492
493 vmcs_writel(VM_ENTRY_MSR_LOAD_ADDR, 519 vmcs_writel(VM_ENTRY_MSR_LOAD_ADDR,
494 virt_to_phys(vcpu->guest_msrs + nr_skip)); 520 virt_to_phys(vcpu->guest_msrs + index));
495 vmcs_writel(VM_EXIT_MSR_STORE_ADDR, 521 vmcs_writel(VM_EXIT_MSR_STORE_ADDR,
496 virt_to_phys(vcpu->guest_msrs + nr_skip)); 522 virt_to_phys(vcpu->guest_msrs + index));
497 vmcs_writel(VM_EXIT_MSR_LOAD_ADDR, 523 vmcs_writel(VM_EXIT_MSR_LOAD_ADDR,
498 virt_to_phys(vcpu->host_msrs + nr_skip)); 524 virt_to_phys(vcpu->host_msrs + index));
499 vmcs_write32(VM_EXIT_MSR_STORE_COUNT, nr_good_msrs); /* 22.2.2 */ 525 vmcs_write32(VM_EXIT_MSR_STORE_COUNT, save_nmsrs);
500 vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, nr_good_msrs); /* 22.2.2 */ 526 vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, save_nmsrs);
501 vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, nr_good_msrs); /* 22.2.2 */ 527 vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, save_nmsrs);
502} 528}
503 529
504/* 530/*
@@ -595,14 +621,6 @@ static int vmx_set_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 data)
595 case MSR_GS_BASE: 621 case MSR_GS_BASE:
596 vmcs_writel(GUEST_GS_BASE, data); 622 vmcs_writel(GUEST_GS_BASE, data);
597 break; 623 break;
598 case MSR_LSTAR:
599 case MSR_SYSCALL_MASK:
600 msr = find_msr_entry(vcpu, msr_index);
601 if (msr)
602 msr->data = data;
603 if (vcpu->vmx_host_state.loaded)
604 load_msrs(vcpu->guest_msrs, NR_BAD_MSRS);
605 break;
606#endif 624#endif
607 case MSR_IA32_SYSENTER_CS: 625 case MSR_IA32_SYSENTER_CS:
608 vmcs_write32(GUEST_SYSENTER_CS, data); 626 vmcs_write32(GUEST_SYSENTER_CS, data);
@@ -620,6 +638,8 @@ static int vmx_set_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 data)
620 msr = find_msr_entry(vcpu, msr_index); 638 msr = find_msr_entry(vcpu, msr_index);
621 if (msr) { 639 if (msr) {
622 msr->data = data; 640 msr->data = data;
641 if (vcpu->vmx_host_state.loaded)
642 load_msrs(vcpu->guest_msrs,vcpu->save_nmsrs);
623 break; 643 break;
624 } 644 }
625 return kvm_set_msr_common(vcpu, msr_index, data); 645 return kvm_set_msr_common(vcpu, msr_index, data);
@@ -1331,10 +1351,6 @@ static int vmx_vcpu_setup(struct kvm_vcpu *vcpu)
1331 vcpu->host_msrs[j].reserved = 0; 1351 vcpu->host_msrs[j].reserved = 0;
1332 vcpu->host_msrs[j].data = data; 1352 vcpu->host_msrs[j].data = data;
1333 vcpu->guest_msrs[j] = vcpu->host_msrs[j]; 1353 vcpu->guest_msrs[j] = vcpu->host_msrs[j];
1334#ifdef CONFIG_X86_64
1335 if (index == MSR_KERNEL_GS_BASE)
1336 msr_offset_kernel_gs_base = j;
1337#endif
1338 ++vcpu->nmsrs; 1354 ++vcpu->nmsrs;
1339 } 1355 }
1340 1356