diff options
Diffstat (limited to 'drivers/kvm/vmx.c')
-rw-r--r-- | drivers/kvm/vmx.c | 249 |
1 files changed, 158 insertions, 91 deletions
diff --git a/drivers/kvm/vmx.c b/drivers/kvm/vmx.c index dac2f93d1a07..96837d6ed50b 100644 --- a/drivers/kvm/vmx.c +++ b/drivers/kvm/vmx.c | |||
@@ -32,6 +32,37 @@ | |||
32 | MODULE_AUTHOR("Qumranet"); | 32 | MODULE_AUTHOR("Qumranet"); |
33 | MODULE_LICENSE("GPL"); | 33 | MODULE_LICENSE("GPL"); |
34 | 34 | ||
35 | struct vmcs { | ||
36 | u32 revision_id; | ||
37 | u32 abort; | ||
38 | char data[0]; | ||
39 | }; | ||
40 | |||
41 | struct vcpu_vmx { | ||
42 | struct kvm_vcpu *vcpu; | ||
43 | int launched; | ||
44 | struct kvm_msr_entry *guest_msrs; | ||
45 | struct kvm_msr_entry *host_msrs; | ||
46 | int nmsrs; | ||
47 | int save_nmsrs; | ||
48 | int msr_offset_efer; | ||
49 | #ifdef CONFIG_X86_64 | ||
50 | int msr_offset_kernel_gs_base; | ||
51 | #endif | ||
52 | struct vmcs *vmcs; | ||
53 | struct { | ||
54 | int loaded; | ||
55 | u16 fs_sel, gs_sel, ldt_sel; | ||
56 | int fs_gs_ldt_reload_needed; | ||
57 | }host_state; | ||
58 | |||
59 | }; | ||
60 | |||
61 | static inline struct vcpu_vmx *to_vmx(struct kvm_vcpu *vcpu) | ||
62 | { | ||
63 | return (struct vcpu_vmx*)vcpu->_priv; | ||
64 | } | ||
65 | |||
35 | static int init_rmode_tss(struct kvm *kvm); | 66 | static int init_rmode_tss(struct kvm *kvm); |
36 | 67 | ||
37 | static DEFINE_PER_CPU(struct vmcs *, vmxarea); | 68 | static DEFINE_PER_CPU(struct vmcs *, vmxarea); |
@@ -89,16 +120,33 @@ static const u32 vmx_msr_index[] = { | |||
89 | }; | 120 | }; |
90 | #define NR_VMX_MSR ARRAY_SIZE(vmx_msr_index) | 121 | #define NR_VMX_MSR ARRAY_SIZE(vmx_msr_index) |
91 | 122 | ||
92 | static inline u64 msr_efer_save_restore_bits(struct vmx_msr_entry msr) | 123 | static void load_msrs(struct kvm_msr_entry *e, int n) |
124 | { | ||
125 | int i; | ||
126 | |||
127 | for (i = 0; i < n; ++i) | ||
128 | wrmsrl(e[i].index, e[i].data); | ||
129 | } | ||
130 | |||
131 | static void save_msrs(struct kvm_msr_entry *e, int n) | ||
132 | { | ||
133 | int i; | ||
134 | |||
135 | for (i = 0; i < n; ++i) | ||
136 | rdmsrl(e[i].index, e[i].data); | ||
137 | } | ||
138 | |||
139 | static inline u64 msr_efer_save_restore_bits(struct kvm_msr_entry msr) | ||
93 | { | 140 | { |
94 | return (u64)msr.data & EFER_SAVE_RESTORE_BITS; | 141 | return (u64)msr.data & EFER_SAVE_RESTORE_BITS; |
95 | } | 142 | } |
96 | 143 | ||
97 | static inline int msr_efer_need_save_restore(struct kvm_vcpu *vcpu) | 144 | static inline int msr_efer_need_save_restore(struct kvm_vcpu *vcpu) |
98 | { | 145 | { |
99 | int efer_offset = vcpu->msr_offset_efer; | 146 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
100 | return msr_efer_save_restore_bits(vcpu->host_msrs[efer_offset]) != | 147 | int efer_offset = vmx->msr_offset_efer; |
101 | msr_efer_save_restore_bits(vcpu->guest_msrs[efer_offset]); | 148 | return msr_efer_save_restore_bits(vmx->host_msrs[efer_offset]) != |
149 | msr_efer_save_restore_bits(vmx->guest_msrs[efer_offset]); | ||
102 | } | 150 | } |
103 | 151 | ||
104 | static inline int is_page_fault(u32 intr_info) | 152 | static inline int is_page_fault(u32 intr_info) |
@@ -123,21 +171,23 @@ static inline int is_external_interrupt(u32 intr_info) | |||
123 | 171 | ||
124 | static int __find_msr_index(struct kvm_vcpu *vcpu, u32 msr) | 172 | static int __find_msr_index(struct kvm_vcpu *vcpu, u32 msr) |
125 | { | 173 | { |
174 | struct vcpu_vmx *vmx = to_vmx(vcpu); | ||
126 | int i; | 175 | int i; |
127 | 176 | ||
128 | for (i = 0; i < vcpu->nmsrs; ++i) | 177 | for (i = 0; i < vmx->nmsrs; ++i) |
129 | if (vcpu->guest_msrs[i].index == msr) | 178 | if (vmx->guest_msrs[i].index == msr) |
130 | return i; | 179 | return i; |
131 | return -1; | 180 | return -1; |
132 | } | 181 | } |
133 | 182 | ||
134 | static struct vmx_msr_entry *find_msr_entry(struct kvm_vcpu *vcpu, u32 msr) | 183 | static struct kvm_msr_entry *find_msr_entry(struct kvm_vcpu *vcpu, u32 msr) |
135 | { | 184 | { |
185 | struct vcpu_vmx *vmx = to_vmx(vcpu); | ||
136 | int i; | 186 | int i; |
137 | 187 | ||
138 | i = __find_msr_index(vcpu, msr); | 188 | i = __find_msr_index(vcpu, msr); |
139 | if (i >= 0) | 189 | if (i >= 0) |
140 | return &vcpu->guest_msrs[i]; | 190 | return &vmx->guest_msrs[i]; |
141 | return NULL; | 191 | return NULL; |
142 | } | 192 | } |
143 | 193 | ||
@@ -157,11 +207,12 @@ static void vmcs_clear(struct vmcs *vmcs) | |||
157 | static void __vcpu_clear(void *arg) | 207 | static void __vcpu_clear(void *arg) |
158 | { | 208 | { |
159 | struct kvm_vcpu *vcpu = arg; | 209 | struct kvm_vcpu *vcpu = arg; |
210 | struct vcpu_vmx *vmx = to_vmx(vcpu); | ||
160 | int cpu = raw_smp_processor_id(); | 211 | int cpu = raw_smp_processor_id(); |
161 | 212 | ||
162 | if (vcpu->cpu == cpu) | 213 | if (vcpu->cpu == cpu) |
163 | vmcs_clear(vcpu->vmcs); | 214 | vmcs_clear(vmx->vmcs); |
164 | if (per_cpu(current_vmcs, cpu) == vcpu->vmcs) | 215 | if (per_cpu(current_vmcs, cpu) == vmx->vmcs) |
165 | per_cpu(current_vmcs, cpu) = NULL; | 216 | per_cpu(current_vmcs, cpu) = NULL; |
166 | rdtscll(vcpu->host_tsc); | 217 | rdtscll(vcpu->host_tsc); |
167 | } | 218 | } |
@@ -172,7 +223,7 @@ static void vcpu_clear(struct kvm_vcpu *vcpu) | |||
172 | smp_call_function_single(vcpu->cpu, __vcpu_clear, vcpu, 0, 1); | 223 | smp_call_function_single(vcpu->cpu, __vcpu_clear, vcpu, 0, 1); |
173 | else | 224 | else |
174 | __vcpu_clear(vcpu); | 225 | __vcpu_clear(vcpu); |
175 | vcpu->launched = 0; | 226 | to_vmx(vcpu)->launched = 0; |
176 | } | 227 | } |
177 | 228 | ||
178 | static unsigned long vmcs_readl(unsigned long field) | 229 | static unsigned long vmcs_readl(unsigned long field) |
@@ -285,80 +336,81 @@ static void reload_tss(void) | |||
285 | static void load_transition_efer(struct kvm_vcpu *vcpu) | 336 | static void load_transition_efer(struct kvm_vcpu *vcpu) |
286 | { | 337 | { |
287 | u64 trans_efer; | 338 | u64 trans_efer; |
288 | int efer_offset = vcpu->msr_offset_efer; | 339 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
340 | int efer_offset = vmx->msr_offset_efer; | ||
289 | 341 | ||
290 | trans_efer = vcpu->host_msrs[efer_offset].data; | 342 | trans_efer = vmx->host_msrs[efer_offset].data; |
291 | trans_efer &= ~EFER_SAVE_RESTORE_BITS; | 343 | trans_efer &= ~EFER_SAVE_RESTORE_BITS; |
292 | trans_efer |= msr_efer_save_restore_bits( | 344 | trans_efer |= msr_efer_save_restore_bits(vmx->guest_msrs[efer_offset]); |
293 | vcpu->guest_msrs[efer_offset]); | ||
294 | wrmsrl(MSR_EFER, trans_efer); | 345 | wrmsrl(MSR_EFER, trans_efer); |
295 | vcpu->stat.efer_reload++; | 346 | vcpu->stat.efer_reload++; |
296 | } | 347 | } |
297 | 348 | ||
298 | static void vmx_save_host_state(struct kvm_vcpu *vcpu) | 349 | static void vmx_save_host_state(struct kvm_vcpu *vcpu) |
299 | { | 350 | { |
300 | struct vmx_host_state *hs = &vcpu->vmx_host_state; | 351 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
301 | 352 | ||
302 | if (hs->loaded) | 353 | if (vmx->host_state.loaded) |
303 | return; | 354 | return; |
304 | 355 | ||
305 | hs->loaded = 1; | 356 | vmx->host_state.loaded = 1; |
306 | /* | 357 | /* |
307 | * Set host fs and gs selectors. Unfortunately, 22.2.3 does not | 358 | * Set host fs and gs selectors. Unfortunately, 22.2.3 does not |
308 | * allow segment selectors with cpl > 0 or ti == 1. | 359 | * allow segment selectors with cpl > 0 or ti == 1. |
309 | */ | 360 | */ |
310 | hs->ldt_sel = read_ldt(); | 361 | vmx->host_state.ldt_sel = read_ldt(); |
311 | hs->fs_gs_ldt_reload_needed = hs->ldt_sel; | 362 | vmx->host_state.fs_gs_ldt_reload_needed = vmx->host_state.ldt_sel; |
312 | hs->fs_sel = read_fs(); | 363 | vmx->host_state.fs_sel = read_fs(); |
313 | if (!(hs->fs_sel & 7)) | 364 | if (!(vmx->host_state.fs_sel & 7)) |
314 | vmcs_write16(HOST_FS_SELECTOR, hs->fs_sel); | 365 | vmcs_write16(HOST_FS_SELECTOR, vmx->host_state.fs_sel); |
315 | else { | 366 | else { |
316 | vmcs_write16(HOST_FS_SELECTOR, 0); | 367 | vmcs_write16(HOST_FS_SELECTOR, 0); |
317 | hs->fs_gs_ldt_reload_needed = 1; | 368 | vmx->host_state.fs_gs_ldt_reload_needed = 1; |
318 | } | 369 | } |
319 | hs->gs_sel = read_gs(); | 370 | vmx->host_state.gs_sel = read_gs(); |
320 | if (!(hs->gs_sel & 7)) | 371 | if (!(vmx->host_state.gs_sel & 7)) |
321 | vmcs_write16(HOST_GS_SELECTOR, hs->gs_sel); | 372 | vmcs_write16(HOST_GS_SELECTOR, vmx->host_state.gs_sel); |
322 | else { | 373 | else { |
323 | vmcs_write16(HOST_GS_SELECTOR, 0); | 374 | vmcs_write16(HOST_GS_SELECTOR, 0); |
324 | hs->fs_gs_ldt_reload_needed = 1; | 375 | vmx->host_state.fs_gs_ldt_reload_needed = 1; |
325 | } | 376 | } |
326 | 377 | ||
327 | #ifdef CONFIG_X86_64 | 378 | #ifdef CONFIG_X86_64 |
328 | vmcs_writel(HOST_FS_BASE, read_msr(MSR_FS_BASE)); | 379 | vmcs_writel(HOST_FS_BASE, read_msr(MSR_FS_BASE)); |
329 | vmcs_writel(HOST_GS_BASE, read_msr(MSR_GS_BASE)); | 380 | vmcs_writel(HOST_GS_BASE, read_msr(MSR_GS_BASE)); |
330 | #else | 381 | #else |
331 | vmcs_writel(HOST_FS_BASE, segment_base(hs->fs_sel)); | 382 | vmcs_writel(HOST_FS_BASE, segment_base(vmx->host_state.fs_sel)); |
332 | vmcs_writel(HOST_GS_BASE, segment_base(hs->gs_sel)); | 383 | vmcs_writel(HOST_GS_BASE, segment_base(vmx->host_state.gs_sel)); |
333 | #endif | 384 | #endif |
334 | 385 | ||
335 | #ifdef CONFIG_X86_64 | 386 | #ifdef CONFIG_X86_64 |
336 | if (is_long_mode(vcpu)) { | 387 | if (is_long_mode(vcpu)) { |
337 | save_msrs(vcpu->host_msrs + vcpu->msr_offset_kernel_gs_base, 1); | 388 | save_msrs(vmx->host_msrs + |
389 | vmx->msr_offset_kernel_gs_base, 1); | ||
338 | } | 390 | } |
339 | #endif | 391 | #endif |
340 | load_msrs(vcpu->guest_msrs, vcpu->save_nmsrs); | 392 | load_msrs(vmx->guest_msrs, vmx->save_nmsrs); |
341 | if (msr_efer_need_save_restore(vcpu)) | 393 | if (msr_efer_need_save_restore(vcpu)) |
342 | load_transition_efer(vcpu); | 394 | load_transition_efer(vcpu); |
343 | } | 395 | } |
344 | 396 | ||
345 | static void vmx_load_host_state(struct kvm_vcpu *vcpu) | 397 | static void vmx_load_host_state(struct kvm_vcpu *vcpu) |
346 | { | 398 | { |
347 | struct vmx_host_state *hs = &vcpu->vmx_host_state; | 399 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
348 | 400 | ||
349 | if (!hs->loaded) | 401 | if (!vmx->host_state.loaded) |
350 | return; | 402 | return; |
351 | 403 | ||
352 | hs->loaded = 0; | 404 | vmx->host_state.loaded = 0; |
353 | if (hs->fs_gs_ldt_reload_needed) { | 405 | if (vmx->host_state.fs_gs_ldt_reload_needed) { |
354 | load_ldt(hs->ldt_sel); | 406 | load_ldt(vmx->host_state.ldt_sel); |
355 | load_fs(hs->fs_sel); | 407 | load_fs(vmx->host_state.fs_sel); |
356 | /* | 408 | /* |
357 | * If we have to reload gs, we must take care to | 409 | * If we have to reload gs, we must take care to |
358 | * preserve our gs base. | 410 | * preserve our gs base. |
359 | */ | 411 | */ |
360 | local_irq_disable(); | 412 | local_irq_disable(); |
361 | load_gs(hs->gs_sel); | 413 | load_gs(vmx->host_state.gs_sel); |
362 | #ifdef CONFIG_X86_64 | 414 | #ifdef CONFIG_X86_64 |
363 | wrmsrl(MSR_GS_BASE, vmcs_readl(HOST_GS_BASE)); | 415 | wrmsrl(MSR_GS_BASE, vmcs_readl(HOST_GS_BASE)); |
364 | #endif | 416 | #endif |
@@ -366,10 +418,10 @@ static void vmx_load_host_state(struct kvm_vcpu *vcpu) | |||
366 | 418 | ||
367 | reload_tss(); | 419 | reload_tss(); |
368 | } | 420 | } |
369 | save_msrs(vcpu->guest_msrs, vcpu->save_nmsrs); | 421 | save_msrs(vmx->guest_msrs, vmx->save_nmsrs); |
370 | load_msrs(vcpu->host_msrs, vcpu->save_nmsrs); | 422 | load_msrs(vmx->host_msrs, vmx->save_nmsrs); |
371 | if (msr_efer_need_save_restore(vcpu)) | 423 | if (msr_efer_need_save_restore(vcpu)) |
372 | load_msrs(vcpu->host_msrs + vcpu->msr_offset_efer, 1); | 424 | load_msrs(vmx->host_msrs + vmx->msr_offset_efer, 1); |
373 | } | 425 | } |
374 | 426 | ||
375 | /* | 427 | /* |
@@ -378,7 +430,8 @@ static void vmx_load_host_state(struct kvm_vcpu *vcpu) | |||
378 | */ | 430 | */ |
379 | static void vmx_vcpu_load(struct kvm_vcpu *vcpu) | 431 | static void vmx_vcpu_load(struct kvm_vcpu *vcpu) |
380 | { | 432 | { |
381 | u64 phys_addr = __pa(vcpu->vmcs); | 433 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
434 | u64 phys_addr = __pa(vmx->vmcs); | ||
382 | int cpu; | 435 | int cpu; |
383 | u64 tsc_this, delta; | 436 | u64 tsc_this, delta; |
384 | 437 | ||
@@ -387,16 +440,16 @@ static void vmx_vcpu_load(struct kvm_vcpu *vcpu) | |||
387 | if (vcpu->cpu != cpu) | 440 | if (vcpu->cpu != cpu) |
388 | vcpu_clear(vcpu); | 441 | vcpu_clear(vcpu); |
389 | 442 | ||
390 | if (per_cpu(current_vmcs, cpu) != vcpu->vmcs) { | 443 | if (per_cpu(current_vmcs, cpu) != vmx->vmcs) { |
391 | u8 error; | 444 | u8 error; |
392 | 445 | ||
393 | per_cpu(current_vmcs, cpu) = vcpu->vmcs; | 446 | per_cpu(current_vmcs, cpu) = vmx->vmcs; |
394 | asm volatile (ASM_VMX_VMPTRLD_RAX "; setna %0" | 447 | asm volatile (ASM_VMX_VMPTRLD_RAX "; setna %0" |
395 | : "=g"(error) : "a"(&phys_addr), "m"(phys_addr) | 448 | : "=g"(error) : "a"(&phys_addr), "m"(phys_addr) |
396 | : "cc"); | 449 | : "cc"); |
397 | if (error) | 450 | if (error) |
398 | printk(KERN_ERR "kvm: vmptrld %p/%llx fail\n", | 451 | printk(KERN_ERR "kvm: vmptrld %p/%llx fail\n", |
399 | vcpu->vmcs, phys_addr); | 452 | vmx->vmcs, phys_addr); |
400 | } | 453 | } |
401 | 454 | ||
402 | if (vcpu->cpu != cpu) { | 455 | if (vcpu->cpu != cpu) { |
@@ -503,13 +556,15 @@ static void vmx_inject_gp(struct kvm_vcpu *vcpu, unsigned error_code) | |||
503 | */ | 556 | */ |
504 | void move_msr_up(struct kvm_vcpu *vcpu, int from, int to) | 557 | void move_msr_up(struct kvm_vcpu *vcpu, int from, int to) |
505 | { | 558 | { |
506 | struct vmx_msr_entry tmp; | 559 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
507 | tmp = vcpu->guest_msrs[to]; | 560 | struct kvm_msr_entry tmp; |
508 | vcpu->guest_msrs[to] = vcpu->guest_msrs[from]; | 561 | |
509 | vcpu->guest_msrs[from] = tmp; | 562 | tmp = vmx->guest_msrs[to]; |
510 | tmp = vcpu->host_msrs[to]; | 563 | vmx->guest_msrs[to] = vmx->guest_msrs[from]; |
511 | vcpu->host_msrs[to] = vcpu->host_msrs[from]; | 564 | vmx->guest_msrs[from] = tmp; |
512 | vcpu->host_msrs[from] = tmp; | 565 | tmp = vmx->host_msrs[to]; |
566 | vmx->host_msrs[to] = vmx->host_msrs[from]; | ||
567 | vmx->host_msrs[from] = tmp; | ||
513 | } | 568 | } |
514 | 569 | ||
515 | /* | 570 | /* |
@@ -519,6 +574,7 @@ void move_msr_up(struct kvm_vcpu *vcpu, int from, int to) | |||
519 | */ | 574 | */ |
520 | static void setup_msrs(struct kvm_vcpu *vcpu) | 575 | static void setup_msrs(struct kvm_vcpu *vcpu) |
521 | { | 576 | { |
577 | struct vcpu_vmx *vmx = to_vmx(vcpu); | ||
522 | int save_nmsrs; | 578 | int save_nmsrs; |
523 | 579 | ||
524 | save_nmsrs = 0; | 580 | save_nmsrs = 0; |
@@ -547,13 +603,13 @@ static void setup_msrs(struct kvm_vcpu *vcpu) | |||
547 | move_msr_up(vcpu, index, save_nmsrs++); | 603 | move_msr_up(vcpu, index, save_nmsrs++); |
548 | } | 604 | } |
549 | #endif | 605 | #endif |
550 | vcpu->save_nmsrs = save_nmsrs; | 606 | vmx->save_nmsrs = save_nmsrs; |
551 | 607 | ||
552 | #ifdef CONFIG_X86_64 | 608 | #ifdef CONFIG_X86_64 |
553 | vcpu->msr_offset_kernel_gs_base = | 609 | vmx->msr_offset_kernel_gs_base = |
554 | __find_msr_index(vcpu, MSR_KERNEL_GS_BASE); | 610 | __find_msr_index(vcpu, MSR_KERNEL_GS_BASE); |
555 | #endif | 611 | #endif |
556 | vcpu->msr_offset_efer = __find_msr_index(vcpu, MSR_EFER); | 612 | vmx->msr_offset_efer = __find_msr_index(vcpu, MSR_EFER); |
557 | } | 613 | } |
558 | 614 | ||
559 | /* | 615 | /* |
@@ -589,7 +645,7 @@ static void guest_write_tsc(u64 guest_tsc) | |||
589 | static int vmx_get_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 *pdata) | 645 | static int vmx_get_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 *pdata) |
590 | { | 646 | { |
591 | u64 data; | 647 | u64 data; |
592 | struct vmx_msr_entry *msr; | 648 | struct kvm_msr_entry *msr; |
593 | 649 | ||
594 | if (!pdata) { | 650 | if (!pdata) { |
595 | printk(KERN_ERR "BUG: get_msr called with NULL pdata\n"); | 651 | printk(KERN_ERR "BUG: get_msr called with NULL pdata\n"); |
@@ -639,14 +695,15 @@ static int vmx_get_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 *pdata) | |||
639 | */ | 695 | */ |
640 | static int vmx_set_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 data) | 696 | static int vmx_set_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 data) |
641 | { | 697 | { |
642 | struct vmx_msr_entry *msr; | 698 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
699 | struct kvm_msr_entry *msr; | ||
643 | int ret = 0; | 700 | int ret = 0; |
644 | 701 | ||
645 | switch (msr_index) { | 702 | switch (msr_index) { |
646 | #ifdef CONFIG_X86_64 | 703 | #ifdef CONFIG_X86_64 |
647 | case MSR_EFER: | 704 | case MSR_EFER: |
648 | ret = kvm_set_msr_common(vcpu, msr_index, data); | 705 | ret = kvm_set_msr_common(vcpu, msr_index, data); |
649 | if (vcpu->vmx_host_state.loaded) | 706 | if (vmx->host_state.loaded) |
650 | load_transition_efer(vcpu); | 707 | load_transition_efer(vcpu); |
651 | break; | 708 | break; |
652 | case MSR_FS_BASE: | 709 | case MSR_FS_BASE: |
@@ -672,8 +729,8 @@ static int vmx_set_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 data) | |||
672 | msr = find_msr_entry(vcpu, msr_index); | 729 | msr = find_msr_entry(vcpu, msr_index); |
673 | if (msr) { | 730 | if (msr) { |
674 | msr->data = data; | 731 | msr->data = data; |
675 | if (vcpu->vmx_host_state.loaded) | 732 | if (vmx->host_state.loaded) |
676 | load_msrs(vcpu->guest_msrs, vcpu->save_nmsrs); | 733 | load_msrs(vmx->guest_msrs, vmx->save_nmsrs); |
677 | break; | 734 | break; |
678 | } | 735 | } |
679 | ret = kvm_set_msr_common(vcpu, msr_index, data); | 736 | ret = kvm_set_msr_common(vcpu, msr_index, data); |
@@ -1053,7 +1110,7 @@ static void vmx_set_cr4(struct kvm_vcpu *vcpu, unsigned long cr4) | |||
1053 | 1110 | ||
1054 | static void vmx_set_efer(struct kvm_vcpu *vcpu, u64 efer) | 1111 | static void vmx_set_efer(struct kvm_vcpu *vcpu, u64 efer) |
1055 | { | 1112 | { |
1056 | struct vmx_msr_entry *msr = find_msr_entry(vcpu, MSR_EFER); | 1113 | struct kvm_msr_entry *msr = find_msr_entry(vcpu, MSR_EFER); |
1057 | 1114 | ||
1058 | vcpu->shadow_efer = efer; | 1115 | vcpu->shadow_efer = efer; |
1059 | if (efer & EFER_LMA) { | 1116 | if (efer & EFER_LMA) { |
@@ -1244,6 +1301,7 @@ static void seg_setup(int seg) | |||
1244 | */ | 1301 | */ |
1245 | static int vmx_vcpu_setup(struct kvm_vcpu *vcpu) | 1302 | static int vmx_vcpu_setup(struct kvm_vcpu *vcpu) |
1246 | { | 1303 | { |
1304 | struct vcpu_vmx *vmx = to_vmx(vcpu); | ||
1247 | u32 host_sysenter_cs; | 1305 | u32 host_sysenter_cs; |
1248 | u32 junk; | 1306 | u32 junk; |
1249 | unsigned long a; | 1307 | unsigned long a; |
@@ -1385,18 +1443,18 @@ static int vmx_vcpu_setup(struct kvm_vcpu *vcpu) | |||
1385 | u32 index = vmx_msr_index[i]; | 1443 | u32 index = vmx_msr_index[i]; |
1386 | u32 data_low, data_high; | 1444 | u32 data_low, data_high; |
1387 | u64 data; | 1445 | u64 data; |
1388 | int j = vcpu->nmsrs; | 1446 | int j = vmx->nmsrs; |
1389 | 1447 | ||
1390 | if (rdmsr_safe(index, &data_low, &data_high) < 0) | 1448 | if (rdmsr_safe(index, &data_low, &data_high) < 0) |
1391 | continue; | 1449 | continue; |
1392 | if (wrmsr_safe(index, data_low, data_high) < 0) | 1450 | if (wrmsr_safe(index, data_low, data_high) < 0) |
1393 | continue; | 1451 | continue; |
1394 | data = data_low | ((u64)data_high << 32); | 1452 | data = data_low | ((u64)data_high << 32); |
1395 | vcpu->host_msrs[j].index = index; | 1453 | vmx->host_msrs[j].index = index; |
1396 | vcpu->host_msrs[j].reserved = 0; | 1454 | vmx->host_msrs[j].reserved = 0; |
1397 | vcpu->host_msrs[j].data = data; | 1455 | vmx->host_msrs[j].data = data; |
1398 | vcpu->guest_msrs[j] = vcpu->host_msrs[j]; | 1456 | vmx->guest_msrs[j] = vmx->host_msrs[j]; |
1399 | ++vcpu->nmsrs; | 1457 | ++vmx->nmsrs; |
1400 | } | 1458 | } |
1401 | 1459 | ||
1402 | setup_msrs(vcpu); | 1460 | setup_msrs(vcpu); |
@@ -1999,6 +2057,7 @@ static void vmx_flush_tlb(struct kvm_vcpu *vcpu) | |||
1999 | 2057 | ||
2000 | static int vmx_vcpu_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run) | 2058 | static int vmx_vcpu_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run) |
2001 | { | 2059 | { |
2060 | struct vcpu_vmx *vmx = to_vmx(vcpu); | ||
2002 | u8 fail; | 2061 | u8 fail; |
2003 | int r; | 2062 | int r; |
2004 | 2063 | ||
@@ -2123,7 +2182,7 @@ again: | |||
2123 | #endif | 2182 | #endif |
2124 | "setbe %0 \n\t" | 2183 | "setbe %0 \n\t" |
2125 | : "=q" (fail) | 2184 | : "=q" (fail) |
2126 | : "r"(vcpu->launched), "d"((unsigned long)HOST_RSP), | 2185 | : "r"(vmx->launched), "d"((unsigned long)HOST_RSP), |
2127 | "c"(vcpu), | 2186 | "c"(vcpu), |
2128 | [rax]"i"(offsetof(struct kvm_vcpu, regs[VCPU_REGS_RAX])), | 2187 | [rax]"i"(offsetof(struct kvm_vcpu, regs[VCPU_REGS_RAX])), |
2129 | [rbx]"i"(offsetof(struct kvm_vcpu, regs[VCPU_REGS_RBX])), | 2188 | [rbx]"i"(offsetof(struct kvm_vcpu, regs[VCPU_REGS_RBX])), |
@@ -2167,7 +2226,7 @@ again: | |||
2167 | if (unlikely(prof_on == KVM_PROFILING)) | 2226 | if (unlikely(prof_on == KVM_PROFILING)) |
2168 | profile_hit(KVM_PROFILING, (void *)vmcs_readl(GUEST_RIP)); | 2227 | profile_hit(KVM_PROFILING, (void *)vmcs_readl(GUEST_RIP)); |
2169 | 2228 | ||
2170 | vcpu->launched = 1; | 2229 | vmx->launched = 1; |
2171 | r = kvm_handle_exit(kvm_run, vcpu); | 2230 | r = kvm_handle_exit(kvm_run, vcpu); |
2172 | if (r > 0) { | 2231 | if (r > 0) { |
2173 | /* Give scheduler a change to reschedule. */ | 2232 | /* Give scheduler a change to reschedule. */ |
@@ -2232,10 +2291,12 @@ static void vmx_inject_page_fault(struct kvm_vcpu *vcpu, | |||
2232 | 2291 | ||
2233 | static void vmx_free_vmcs(struct kvm_vcpu *vcpu) | 2292 | static void vmx_free_vmcs(struct kvm_vcpu *vcpu) |
2234 | { | 2293 | { |
2235 | if (vcpu->vmcs) { | 2294 | struct vcpu_vmx *vmx = to_vmx(vcpu); |
2295 | |||
2296 | if (vmx->vmcs) { | ||
2236 | on_each_cpu(__vcpu_clear, vcpu, 0, 1); | 2297 | on_each_cpu(__vcpu_clear, vcpu, 0, 1); |
2237 | free_vmcs(vcpu->vmcs); | 2298 | free_vmcs(vmx->vmcs); |
2238 | vcpu->vmcs = NULL; | 2299 | vmx->vmcs = NULL; |
2239 | } | 2300 | } |
2240 | } | 2301 | } |
2241 | 2302 | ||
@@ -2246,33 +2307,39 @@ static void vmx_free_vcpu(struct kvm_vcpu *vcpu) | |||
2246 | 2307 | ||
2247 | static int vmx_create_vcpu(struct kvm_vcpu *vcpu) | 2308 | static int vmx_create_vcpu(struct kvm_vcpu *vcpu) |
2248 | { | 2309 | { |
2249 | struct vmcs *vmcs; | 2310 | struct vcpu_vmx *vmx; |
2250 | 2311 | ||
2251 | vcpu->guest_msrs = kmalloc(PAGE_SIZE, GFP_KERNEL); | 2312 | vmx = kzalloc(sizeof(*vmx), GFP_KERNEL); |
2252 | if (!vcpu->guest_msrs) | 2313 | if (!vmx) |
2253 | return -ENOMEM; | 2314 | return -ENOMEM; |
2254 | 2315 | ||
2255 | vcpu->host_msrs = kmalloc(PAGE_SIZE, GFP_KERNEL); | 2316 | vmx->guest_msrs = kmalloc(PAGE_SIZE, GFP_KERNEL); |
2256 | if (!vcpu->host_msrs) | 2317 | if (!vmx->guest_msrs) |
2257 | goto out_free_guest_msrs; | 2318 | goto out_free; |
2258 | 2319 | ||
2259 | vmcs = alloc_vmcs(); | 2320 | vmx->host_msrs = kmalloc(PAGE_SIZE, GFP_KERNEL); |
2260 | if (!vmcs) | 2321 | if (!vmx->host_msrs) |
2261 | goto out_free_msrs; | 2322 | goto out_free; |
2262 | 2323 | ||
2263 | vmcs_clear(vmcs); | 2324 | vmx->vmcs = alloc_vmcs(); |
2264 | vcpu->vmcs = vmcs; | 2325 | if (!vmx->vmcs) |
2265 | vcpu->launched = 0; | 2326 | goto out_free; |
2327 | |||
2328 | vmcs_clear(vmx->vmcs); | ||
2329 | |||
2330 | vmx->vcpu = vcpu; | ||
2331 | vcpu->_priv = vmx; | ||
2266 | 2332 | ||
2267 | return 0; | 2333 | return 0; |
2268 | 2334 | ||
2269 | out_free_msrs: | 2335 | out_free: |
2270 | kfree(vcpu->host_msrs); | 2336 | if (vmx->host_msrs) |
2271 | vcpu->host_msrs = NULL; | 2337 | kfree(vmx->host_msrs); |
2338 | |||
2339 | if (vmx->guest_msrs) | ||
2340 | kfree(vmx->guest_msrs); | ||
2272 | 2341 | ||
2273 | out_free_guest_msrs: | 2342 | kfree(vmx); |
2274 | kfree(vcpu->guest_msrs); | ||
2275 | vcpu->guest_msrs = NULL; | ||
2276 | 2343 | ||
2277 | return -ENOMEM; | 2344 | return -ENOMEM; |
2278 | } | 2345 | } |