aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEddie Dong <eddie.dong@intel.com>2007-05-17 11:55:15 -0400
committerAvi Kivity <avi@qumranet.com>2007-07-16 05:05:41 -0400
commita75beee6e4f5d2f0ae6e28cd626b2f157e93afd2 (patch)
treeef5fcd31527f5e211526f65180be2a6f73a1783e
parentb3f37707b05e9ce82d5bec660e9d0b15452ee9a0 (diff)
KVM: VMX: Avoid saving and restoring msrs on lightweight vmexit
In a lightweight exit (where we exit and reenter the guest without scheduling or exiting to userspace in between), we don't need various msrs on the host, and avoiding shuffling them around reduces raw exit time by 8%. i386 compile fix by Daniel Hecken <dh@bahntechnik.de>. Signed-off-by: Yaozu (Eddie) Dong <eddie.dong@intel.com> Signed-off-by: Avi Kivity <avi@qumranet.com>
-rw-r--r--drivers/kvm/kvm.h4
-rw-r--r--drivers/kvm/vmx.c128
2 files changed, 76 insertions, 56 deletions
diff --git a/drivers/kvm/kvm.h b/drivers/kvm/kvm.h
index fc4a6c1235f0..c252efed49d9 100644
--- a/drivers/kvm/kvm.h
+++ b/drivers/kvm/kvm.h
@@ -288,6 +288,10 @@ struct kvm_vcpu {
288 u64 apic_base; 288 u64 apic_base;
289 u64 ia32_misc_enable_msr; 289 u64 ia32_misc_enable_msr;
290 int nmsrs; 290 int nmsrs;
291 int save_nmsrs;
292#ifdef CONFIG_X86_64
293 int msr_offset_kernel_gs_base;
294#endif
291 struct vmx_msr_entry *guest_msrs; 295 struct vmx_msr_entry *guest_msrs;
292 struct vmx_msr_entry *host_msrs; 296 struct vmx_msr_entry *host_msrs;
293 297
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