VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR0/HWACCMR0A.asm@ 9510

Last change on this file since 9510 was 9484, checked in by vboxsync, 17 years ago

Save & restore CSTAR, STAR, SFMASK & KERNEL_GSBASE MSRs (VT-x)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 22.3 KB
Line 
1; $Id: HWACCMR0A.asm 9484 2008-06-06 14:02:24Z vboxsync $
2;; @file
3; VMXM - R0 vmx helpers
4;
5
6;
7; Copyright (C) 2006-2007 Sun Microsystems, Inc.
8;
9; This file is part of VirtualBox Open Source Edition (OSE), as
10; available from http://www.virtualbox.org. This file is free software;
11; you can redistribute it and/or modify it under the terms of the GNU
12; General Public License (GPL) as published by the Free Software
13; Foundation, in version 2 as it comes in the "COPYING" file of the
14; VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15; hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16;
17; Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18; Clara, CA 95054 USA or visit http://www.sun.com if you need
19; additional information or have any questions.
20;
21
22;*******************************************************************************
23;* Header Files *
24;*******************************************************************************
25%include "VBox/asmdefs.mac"
26%include "VBox/err.mac"
27%include "VBox/hwacc_vmx.mac"
28%include "VBox/cpum.mac"
29%include "VBox/x86.mac"
30
31%ifdef RT_OS_OS2 ;; @todo fix OMF support in yasm and kick nasm out completely.
32 %macro vmwrite 2,
33 int3
34 %endmacro
35 %define vmlaunch int3
36 %define vmresume int3
37 %define vmsave int3
38 %define vmload int3
39 %define vmrun int3
40 %define clgi int3
41 %define stgi int3
42 %macro invlpga 2,
43 int3
44 %endmacro
45%endif
46
47;; This is too risky wrt. stability, performance and correctness.
48;%define VBOX_WITH_DR6_EXPERIMENT 1
49
50;; @def MYPUSHAD
51; Macro generating an equivalent to pushad
52
53;; @def MYPOPAD
54; Macro generating an equivalent to popad
55
56;; @def MYPUSHSEGS
57; Macro saving all segment registers on the stack.
58; @param 1 full width register name
59; @param 2 16-bit regsiter name for \a 1.
60
61;; @def MYPOPSEGS
62; Macro restoring all segment registers on the stack
63; @param 1 full width register name
64; @param 2 16-bit regsiter name for \a 1.
65
66%ifdef RT_ARCH_AMD64
67 ; Save a host and load the corresponding guest MSR (trashes rdx & rcx)
68 %macro LOADGUESTMSR 2
69 mov rcx, %1
70 rdmsr
71 push rdx
72 push rax
73 xor rdx, rdx
74 mov rax, qword [xSI + %2]
75 wrmsr
76 %endmacro
77
78 ; Save a guest and load the corresponding host MSR (trashes rdx & rcx)
79 %macro LOADHOSTMSR 2
80 mov rcx, %1
81 rdmsr
82 mov qword [xSI + %2], rax
83 pop rax
84 pop rdx
85 wrmsr
86 %endmacro
87
88 %ifdef ASM_CALL64_GCC
89 %macro MYPUSHAD 0
90 push r15
91 push r14
92 push r13
93 push r12
94 push rbx
95 %endmacro
96 %macro MYPOPAD 0
97 pop rbx
98 pop r12
99 pop r13
100 pop r14
101 pop r15
102 %endmacro
103
104 %else ; ASM_CALL64_MSC
105 %macro MYPUSHAD 0
106 push r15
107 push r14
108 push r13
109 push r12
110 push rbx
111 push rsi
112 push rdi
113 %endmacro
114 %macro MYPOPAD 0
115 pop rdi
116 pop rsi
117 pop rbx
118 pop r12
119 pop r13
120 pop r14
121 pop r15
122 %endmacro
123 %endif
124
125; trashes, rax, rdx & rcx
126 %macro MYPUSHSEGS 2
127 mov %2, es
128 push %1
129 mov %2, ds
130 push %1
131
132 ; Special case for FS; Windows and Linux either don't use it or restore it when leaving kernel mode, Solaris OTOH doesn't and we must save it.
133 mov ecx, MSR_K8_FS_BASE
134 rdmsr
135 push rdx
136 push rax
137 push fs
138
139 ; Special case for GS; OSes typically use swapgs to reset the hidden base register for GS on entry into the kernel. The same happens on exit
140 mov ecx, MSR_K8_GS_BASE
141 rdmsr
142 push rdx
143 push rax
144 push gs
145 %endmacro
146
147; trashes, rax, rdx & rcx
148 %macro MYPOPSEGS 2
149 ; Note: do not step through this code with a debugger!
150 pop gs
151 pop rax
152 pop rdx
153 mov ecx, MSR_K8_GS_BASE
154 wrmsr
155
156 pop fs
157 pop rax
158 pop rdx
159 mov ecx, MSR_K8_FS_BASE
160 wrmsr
161 ; Now it's safe to step again
162
163 pop %1
164 mov ds, %2
165 pop %1
166 mov es, %2
167 %endmacro
168
169%else ; RT_ARCH_X86
170 %macro MYPUSHAD 0
171 pushad
172 %endmacro
173 %macro MYPOPAD 0
174 popad
175 %endmacro
176
177 %macro MYPUSHSEGS 2
178 push ds
179 push es
180 push fs
181 push gs
182 %endmacro
183 %macro MYPOPSEGS 2
184 pop gs
185 pop fs
186 pop es
187 pop ds
188 %endmacro
189%endif
190
191
192BEGINCODE
193
194;/**
195; * Prepares for and executes VMLAUNCH/VMRESUME (32 bits guest mode)
196; *
197; * @returns VBox status code
198; * @param fResume vmlauch/vmresume
199; * @param pCtx Guest context
200; */
201BEGINPROC VMXR0StartVM32
202 push xBP
203 mov xBP, xSP
204
205 pushf
206 cli
207
208 ;/* First we have to save some final CPU context registers. */
209%ifdef RT_ARCH_AMD64
210 mov rax, qword .vmlaunch_done
211 push rax
212%else
213 push .vmlaunch_done
214%endif
215 mov eax, VMX_VMCS_HOST_RIP ;/* return address (too difficult to continue after VMLAUNCH?) */
216 vmwrite xAX, [xSP]
217 ;/* Note: assumes success... */
218 add xSP, xS
219
220 ;/* Manual save and restore:
221 ; * - General purpose registers except RIP, RSP
222 ; *
223 ; * Trashed:
224 ; * - CR2 (we don't care)
225 ; * - LDTR (reset to 0)
226 ; * - DRx (presumably not changed at all)
227 ; * - DR7 (reset to 0x400)
228 ; * - EFLAGS (reset to RT_BIT(1); not relevant)
229 ; *
230 ; */
231
232 ;/* Save all general purpose host registers. */
233 MYPUSHAD
234
235 ;/* Save the Guest CPU context pointer. */
236%ifdef RT_ARCH_AMD64
237 %ifdef ASM_CALL64_GCC
238 ; fResume already in rdi
239 ; pCtx already in rsi
240 %else
241 mov rdi, rcx ; fResume
242 mov rsi, rdx ; pCtx
243 %endif
244%else
245 mov edi, [ebp + 8] ; fResume
246 mov esi, [ebp + 12] ; pCtx
247%endif
248
249 ;/* Save segment registers */
250 ; Note: MYPUSHSEGS trashes rdx & rcx, so we moved it here (msvc amd64 case)
251 MYPUSHSEGS xAX, ax
252
253 ; Save the pCtx pointer
254 push xSI
255
256 ; Save LDTR
257 xor eax, eax
258 sldt ax
259 push xAX
260
261 ; VMX only saves the base of the GDTR & IDTR and resets the limit to 0xffff; we must restore the limit correctly!
262 sub xSP, xS*2
263 sgdt [xSP]
264
265 sub xSP, xS*2
266 sidt [xSP]
267
268%ifdef VBOX_WITH_DR6_EXPERIMENT
269 ; Restore DR6 - experiment, not safe!
270 mov xBX, [xSI + CPUMCTX.dr6]
271 mov dr6, xBX
272%endif
273
274 ; Restore CR2
275 mov ebx, [xSI + CPUMCTX.cr2]
276 mov cr2, xBX
277
278 mov eax, VMX_VMCS_HOST_RSP
279 vmwrite xAX, xSP
280 ;/* Note: assumes success... */
281 ;/* Don't mess with ESP anymore!! */
282
283 ;/* Restore Guest's general purpose registers. */
284 mov eax, [xSI + CPUMCTX.eax]
285 mov ebx, [xSI + CPUMCTX.ebx]
286 mov ecx, [xSI + CPUMCTX.ecx]
287 mov edx, [xSI + CPUMCTX.edx]
288 mov ebp, [xSI + CPUMCTX.ebp]
289
290 ; resume or start?
291 cmp xDI, 0 ; fResume
292 je .vmlauch_lauch
293
294 ;/* Restore edi & esi. */
295 mov edi, [xSI + CPUMCTX.edi]
296 mov esi, [xSI + CPUMCTX.esi]
297
298 vmresume
299 jmp .vmlaunch_done; ;/* here if vmresume detected a failure. */
300
301.vmlauch_lauch:
302 ;/* Restore edi & esi. */
303 mov edi, [xSI + CPUMCTX.edi]
304 mov esi, [xSI + CPUMCTX.esi]
305
306 vmlaunch
307 jmp .vmlaunch_done; ;/* here if vmlaunch detected a failure. */
308
309ALIGNCODE(16)
310.vmlaunch_done:
311 jc near .vmxstart_invalid_vmxon_ptr
312 jz near .vmxstart_start_failed
313
314 ; Restore base and limit of the IDTR & GDTR
315 lidt [xSP]
316 add xSP, xS*2
317 lgdt [xSP]
318 add xSP, xS*2
319
320 push xDI
321 mov xDI, [xSP + xS * 2] ; pCtx
322
323 mov [ss:xDI + CPUMCTX.eax], eax
324 mov [ss:xDI + CPUMCTX.ebx], ebx
325 mov [ss:xDI + CPUMCTX.ecx], ecx
326 mov [ss:xDI + CPUMCTX.edx], edx
327 mov [ss:xDI + CPUMCTX.esi], esi
328 mov [ss:xDI + CPUMCTX.ebp], ebp
329%ifdef RT_ARCH_AMD64
330 pop xAX ; the guest edi we pushed above
331 mov dword [ss:xDI + CPUMCTX.edi], eax
332%else
333 pop dword [ss:xDI + CPUMCTX.edi] ; the guest edi we pushed above
334%endif
335
336%ifdef VBOX_WITH_DR6_EXPERIMENT
337 ; Save DR6 - experiment, not safe!
338 mov xAX, dr6
339 mov [ss:xDI + CPUMCTX.dr6], xAX
340%endif
341
342 pop xAX ; saved LDTR
343 lldt ax
344
345 add xSP, xS ; pCtx
346
347 ; Restore segment registers
348 MYPOPSEGS xAX, ax
349
350 ; Restore general purpose registers
351 MYPOPAD
352
353 mov eax, VINF_SUCCESS
354
355.vmstart_end:
356 popf
357 pop xBP
358 ret
359
360
361.vmxstart_invalid_vmxon_ptr:
362 ; Restore base and limit of the IDTR & GDTR
363 lidt [xSP]
364 add xSP, xS*2
365 lgdt [xSP]
366 add xSP, xS*2
367
368 pop xAX ; saved LDTR
369 lldt ax
370
371 add xSP, xS ; pCtx
372
373 ; Restore segment registers
374 MYPOPSEGS xAX, ax
375
376 ; Restore all general purpose host registers.
377 MYPOPAD
378 mov eax, VERR_VMX_INVALID_VMXON_PTR
379 jmp .vmstart_end
380
381.vmxstart_start_failed:
382 ; Restore base and limit of the IDTR & GDTR
383 lidt [xSP]
384 add xSP, xS*2
385 lgdt [xSP]
386 add xSP, xS*2
387
388 pop xAX ; saved LDTR
389 lldt ax
390
391 add xSP, xS ; pCtx
392
393 ; Restore segment registers
394 MYPOPSEGS xAX, ax
395
396 ; Restore all general purpose host registers.
397 MYPOPAD
398 mov eax, VERR_VMX_UNABLE_TO_START_VM
399 jmp .vmstart_end
400
401ENDPROC VMXR0StartVM32
402
403%ifdef RT_ARCH_AMD64
404;/**
405; * Prepares for and executes VMLAUNCH/VMRESUME (32 bits guest mode)
406; *
407; * @returns VBox status code
408; * @param fResume vmlauch/vmresume
409; * @param pCtx Guest context
410; */
411BEGINPROC VMXR0StartVM64
412 push xBP
413 mov xBP, xSP
414
415 pushf
416 cli
417
418 ;/* First we have to save some final CPU context registers. */
419 mov rax, qword .vmlaunch64_done
420 push rax
421 mov rax, VMX_VMCS_HOST_RIP ;/* return address (too difficult to continue after VMLAUNCH?) */
422 vmwrite rax, [xSP]
423 ;/* Note: assumes success... */
424 add xSP, xS
425
426 ;/* Manual save and restore:
427 ; * - General purpose registers except RIP, RSP
428 ; *
429 ; * Trashed:
430 ; * - CR2 (we don't care)
431 ; * - LDTR (reset to 0)
432 ; * - DRx (presumably not changed at all)
433 ; * - DR7 (reset to 0x400)
434 ; * - EFLAGS (reset to RT_BIT(1); not relevant)
435 ; *
436 ; */
437
438 ;/* Save all general purpose host registers. */
439 MYPUSHAD
440
441 ;/* Save the Guest CPU context pointer. */
442%ifdef ASM_CALL64_GCC
443 ; fResume already in rdi
444 ; pCtx already in rsi
445%else
446 mov rdi, rcx ; fResume
447 mov rsi, rdx ; pCtx
448%endif
449
450 ;/* Save segment registers */
451 ; Note: MYPUSHSEGS trashes rdx & rcx, so we moved it here (msvc amd64 case)
452 MYPUSHSEGS xAX, ax
453
454 ; Save the host LSTAR, CSTAR, SFMASK & KERNEL_GSBASE MSRs and restore the guest MSRs
455 ; @todo use the automatic load feature for MSRs
456 LOADGUESTMSR MSR_K8_LSTAR, CPUMCTX.msrLSTAR
457 LOADGUESTMSR MSR_K8_CSTAR, CPUMCTX.msrCSTAR
458 LOADGUESTMSR MSR_K8_SF_MASK, CPUMCTX.msrSFMASK
459 LOADGUESTMSR MSR_K8_KERNEL_GS_BASE, CPUMCTX.msrKERNELGSBASE
460
461 ; Save the pCtx pointer
462 push xSI
463
464 ; Save LDTR
465 xor eax, eax
466 sldt ax
467 push xAX
468
469 ; VMX only saves the base of the GDTR & IDTR and resets the limit to 0xffff; we must restore the limit correctly!
470 sub xSP, xS*2
471 sgdt [xSP]
472
473 sub xSP, xS*2
474 sidt [xSP]
475
476%ifdef VBOX_WITH_DR6_EXPERIMENT
477 ; Restore DR6 - experiment, not safe!
478 mov xBX, [xSI + CPUMCTX.dr6]
479 mov dr6, xBX
480%endif
481
482 ; Restore CR2
483 mov rbx, qword [xSI + CPUMCTX.cr2]
484 mov cr2, rbx
485
486 mov eax, VMX_VMCS_HOST_RSP
487 vmwrite xAX, xSP
488 ;/* Note: assumes success... */
489 ;/* Don't mess with ESP anymore!! */
490
491 ;/* Restore Guest's general purpose registers. */
492 mov rax, qword [xSI + CPUMCTX.eax]
493 mov rbx, qword [xSI + CPUMCTX.ebx]
494 mov rcx, qword [xSI + CPUMCTX.ecx]
495 mov rdx, qword [xSI + CPUMCTX.edx]
496 mov rbp, qword [xSI + CPUMCTX.ebp]
497 mov r8, qword [xSI + CPUMCTX.r8]
498 mov r9, qword [xSI + CPUMCTX.r9]
499 mov r10, qword [xSI + CPUMCTX.r10]
500 mov r11, qword [xSI + CPUMCTX.r11]
501 mov r12, qword [xSI + CPUMCTX.r12]
502 mov r13, qword [xSI + CPUMCTX.r13]
503 mov r14, qword [xSI + CPUMCTX.r14]
504 mov r15, qword [xSI + CPUMCTX.r15]
505
506 ; resume or start?
507 cmp xDI, 0 ; fResume
508 je .vmlauch64_lauch
509
510 ;/* Restore edi & esi. */
511 mov rdi, qword [xSI + CPUMCTX.edi]
512 mov rsi, qword [xSI + CPUMCTX.esi]
513
514 vmresume
515 jmp .vmlaunch64_done; ;/* here if vmresume detected a failure. */
516
517.vmlauch64_lauch:
518 ;/* Restore rdi & rsi. */
519 mov rdi, qword [xSI + CPUMCTX.edi]
520 mov rsi, qword [xSI + CPUMCTX.esi]
521
522 vmlaunch
523 jmp .vmlaunch64_done; ;/* here if vmlaunch detected a failure. */
524
525ALIGNCODE(16)
526.vmlaunch64_done:
527 jc near .vmxstart64_invalid_vmxon_ptr
528 jz near .vmxstart64_start_failed
529
530 ; Restore base and limit of the IDTR & GDTR
531 lidt [xSP]
532 add xSP, xS*2
533 lgdt [xSP]
534 add xSP, xS*2
535
536 push xDI
537 mov xDI, [xSP + xS * 2] ; pCtx
538
539 mov qword [xDI + CPUMCTX.eax], rax
540 mov qword [xDI + CPUMCTX.ebx], rbx
541 mov qword [xDI + CPUMCTX.ecx], rcx
542 mov qword [xDI + CPUMCTX.edx], rdx
543 mov qword [xDI + CPUMCTX.esi], rsi
544 mov qword [xDI + CPUMCTX.ebp], rbp
545 mov qword [xDI + CPUMCTX.r8], r8
546 mov qword [xDI + CPUMCTX.r9], r9
547 mov qword [xDI + CPUMCTX.r10], r10
548 mov qword [xDI + CPUMCTX.r11], r11
549 mov qword [xDI + CPUMCTX.r12], r12
550 mov qword [xDI + CPUMCTX.r13], r13
551 mov qword [xDI + CPUMCTX.r14], r14
552 mov qword [xDI + CPUMCTX.r15], r15
553
554 pop xAX ; the guest edi we pushed above
555 mov qword [xDI + CPUMCTX.edi], rax
556
557%ifdef VBOX_WITH_DR6_EXPERIMENT
558 ; Save DR6 - experiment, not safe!
559 mov xAX, dr6
560 mov [xDI + CPUMCTX.dr6], xAX
561%endif
562
563 pop xAX ; saved LDTR
564 lldt ax
565
566 add xSP, xS ; pCtx
567
568 ; Save the guest LSTAR, CSTAR, SFMASK & KERNEL_GSBASE MSRs and restore the host MSRs
569 ; @todo use the automatic load feature for MSRs
570 LOADHOSTMSR MSR_K8_KERNEL_GS_BASE, CPUMCTX.msrKERNELGSBASE
571 LOADHOSTMSR MSR_K8_SF_MASK, CPUMCTX.msrSFMASK
572 LOADHOSTMSR MSR_K8_CSTAR, CPUMCTX.msrCSTAR
573 LOADHOSTMSR MSR_K8_LSTAR, CPUMCTX.msrLSTAR
574
575 ; Restore segment registers
576 MYPOPSEGS xAX, ax
577
578 ; Restore general purpose registers
579 MYPOPAD
580
581 mov eax, VINF_SUCCESS
582
583.vmstart64_end:
584 popf
585 pop xBP
586 ret
587
588
589.vmxstart64_invalid_vmxon_ptr:
590 ; Restore base and limit of the IDTR & GDTR
591 lidt [xSP]
592 add xSP, xS*2
593 lgdt [xSP]
594 add xSP, xS*2
595
596 pop xAX ; saved LDTR
597 lldt ax
598
599 add xSP, xS ; pCtx
600
601 ; Save the guest LSTAR, CSTAR, SFMASK & KERNEL_GSBASE MSRs and restore the host MSRs
602 ; @todo use the automatic load feature for MSRs
603 LOADHOSTMSR MSR_K8_KERNEL_GS_BASE, CPUMCTX.msrKERNELGSBASE
604 LOADHOSTMSR MSR_K8_SF_MASK, CPUMCTX.msrSFMASK
605 LOADHOSTMSR MSR_K8_CSTAR, CPUMCTX.msrCSTAR
606 LOADHOSTMSR MSR_K8_LSTAR, CPUMCTX.msrLSTAR
607
608 ; Restore segment registers
609 MYPOPSEGS xAX, ax
610
611 ; Restore all general purpose host registers.
612 MYPOPAD
613 mov eax, VERR_VMX_INVALID_VMXON_PTR
614 jmp .vmstart64_end
615
616.vmxstart64_start_failed:
617 ; Restore base and limit of the IDTR & GDTR
618 lidt [xSP]
619 add xSP, xS*2
620 lgdt [xSP]
621 add xSP, xS*2
622
623 pop xAX ; saved LDTR
624 lldt ax
625
626 add xSP, xS ; pCtx
627
628 ; Save the guest LSTAR, CSTAR, SFMASK & KERNEL_GSBASE MSRs and restore the host MSRs
629 ; @todo use the automatic load feature for MSRs
630 LOADHOSTMSR MSR_K8_KERNEL_GS_BASE, CPUMCTX.msrKERNELGSBASE
631 LOADHOSTMSR MSR_K8_SF_MASK, CPUMCTX.msrSFMASK
632 LOADHOSTMSR MSR_K8_CSTAR, CPUMCTX.msrCSTAR
633 LOADHOSTMSR MSR_K8_LSTAR, CPUMCTX.msrLSTAR
634
635 ; Restore segment registers
636 MYPOPSEGS xAX, ax
637
638 ; Restore all general purpose host registers.
639 MYPOPAD
640 mov eax, VERR_VMX_UNABLE_TO_START_VM
641 jmp .vmstart64_end
642ENDPROC VMXR0StartVM64
643
644;/**
645; * Executes VMWRITE
646; *
647; * @returns VBox status code
648; * @param idxField x86: [ebp + 08h] msc: rcx gcc: rdi VMCS index
649; * @param pData x86: [ebp + 0ch] msc: rdx gcc: rsi VM field value
650; */
651BEGINPROC VMXWriteVMCS64
652%ifdef ASM_CALL64_GCC
653 mov eax, 0ffffffffh
654 and rdi, rax
655 xor rax, rax
656 vmwrite rdi, rsi
657%else
658 mov eax, 0ffffffffh
659 and rcx, rax
660 xor rax, rax
661 vmwrite rcx, rdx
662%endif
663 jnc .valid_vmcs
664 mov eax, VERR_VMX_INVALID_VMCS_PTR
665 ret
666.valid_vmcs:
667 jnz .the_end
668 mov eax, VERR_VMX_INVALID_VMCS_FIELD
669.the_end:
670 ret
671ENDPROC VMXWriteVMCS64
672
673;/**
674; * Executes VMREAD
675; *
676; * @returns VBox status code
677; * @param idxField VMCS index
678; * @param pData Ptr to store VM field value
679; */
680;DECLASM(int) VMXReadVMCS64(uint32_t idxField, uint64_t *pData);
681BEGINPROC VMXReadVMCS64
682%ifdef ASM_CALL64_GCC
683 mov eax, 0ffffffffh
684 and rdi, rax
685 xor rax, rax
686 vmread [rsi], rdi
687%else
688 mov eax, 0ffffffffh
689 and rcx, rax
690 xor rax, rax
691 vmread [rdx], rcx
692%endif
693 jnc .valid_vmcs
694 mov eax, VERR_VMX_INVALID_VMCS_PTR
695 ret
696.valid_vmcs:
697 jnz .the_end
698 mov eax, VERR_VMX_INVALID_VMCS_FIELD
699.the_end:
700 ret
701ENDPROC VMXReadVMCS64
702
703
704;/**
705; * Executes VMXON
706; *
707; * @returns VBox status code
708; * @param HCPhysVMXOn Physical address of VMXON structure
709; */
710;DECLASM(int) VMXEnable(RTHCPHYS HCPhysVMXOn);
711BEGINPROC VMXEnable
712%ifdef RT_ARCH_AMD64
713 xor rax, rax
714 %ifdef ASM_CALL64_GCC
715 push rdi
716 %else
717 push rcx
718 %endif
719 vmxon [rsp]
720%else
721 xor eax, eax
722 vmxon [esp + 4]
723%endif
724 jnc .good
725 mov eax, VERR_VMX_INVALID_VMXON_PTR
726 jmp .the_end
727
728.good:
729 jnz .the_end
730 mov eax, VERR_VMX_GENERIC
731
732.the_end:
733%ifdef RT_ARCH_AMD64
734 add rsp, 8
735%endif
736 ret
737ENDPROC VMXEnable
738
739
740;/**
741; * Executes VMXOFF
742; */
743;DECLASM(void) VMXDisable(void);
744BEGINPROC VMXDisable
745 vmxoff
746 ret
747ENDPROC VMXDisable
748
749
750;/**
751; * Executes VMCLEAR
752; *
753; * @returns VBox status code
754; * @param HCPhysVMCS Physical address of VM control structure
755; */
756;DECLASM(int) VMXClearVMCS(RTHCPHYS HCPhysVMCS);
757BEGINPROC VMXClearVMCS
758%ifdef RT_ARCH_AMD64
759 xor rax, rax
760 %ifdef ASM_CALL64_GCC
761 push rdi
762 %else
763 push rcx
764 %endif
765 vmclear [rsp]
766%else
767 xor eax, eax
768 vmclear [esp + 4]
769%endif
770 jnc .the_end
771 mov eax, VERR_VMX_INVALID_VMCS_PTR
772.the_end:
773%ifdef RT_ARCH_AMD64
774 add rsp, 8
775%endif
776 ret
777ENDPROC VMXClearVMCS
778
779
780;/**
781; * Executes VMPTRLD
782; *
783; * @returns VBox status code
784; * @param HCPhysVMCS Physical address of VMCS structure
785; */
786;DECLASM(int) VMXActivateVMCS(RTHCPHYS HCPhysVMCS);
787BEGINPROC VMXActivateVMCS
788%ifdef RT_ARCH_AMD64
789 xor rax, rax
790 %ifdef ASM_CALL64_GCC
791 push rdi
792 %else
793 push rcx
794 %endif
795 vmptrld [rsp]
796%else
797 xor eax, eax
798 vmptrld [esp + 4]
799%endif
800 jnc .the_end
801 mov eax, VERR_VMX_INVALID_VMCS_PTR
802.the_end:
803%ifdef RT_ARCH_AMD64
804 add rsp, 8
805%endif
806 ret
807ENDPROC VMXActivateVMCS
808
809%endif ; RT_ARCH_AMD64
810
811
812;/**
813; * Prepares for and executes VMRUN
814; *
815; * @returns VBox status code
816; * @param HCPhysVMCB Physical address of host VMCB
817; * @param HCPhysVMCB Physical address of guest VMCB
818; * @param pCtx Guest context
819; */
820BEGINPROC SVMVMRun
821%ifdef RT_ARCH_AMD64 ; fake a cdecl stack frame - I'm lazy, sosume.
822 %ifdef ASM_CALL64_GCC
823 push rdx
824 push rsi
825 push rdi
826 %else
827 push r8
828 push rdx
829 push rcx
830 %endif
831 push 0
832%endif
833 push xBP
834 mov xBP, xSP
835 pushf
836
837 ;/* Manual save and restore:
838 ; * - General purpose registers except RIP, RSP, RAX
839 ; *
840 ; * Trashed:
841 ; * - CR2 (we don't care)
842 ; * - LDTR (reset to 0)
843 ; * - DRx (presumably not changed at all)
844 ; * - DR7 (reset to 0x400)
845 ; */
846
847 ;/* Save all general purpose host registers. */
848 MYPUSHAD
849
850 ;/* Save the Guest CPU context pointer. */
851 mov xSI, [xBP + xS*2 + RTHCPHYS_CB*2] ; pCtx
852 push xSI ; push for saving the state at the end
853
854 ; Restore CR2
855 mov ebx, [xSI + CPUMCTX.cr2]
856 mov cr2, xBX
857
858 ; save host fs, gs, sysenter msr etc
859 mov xAX, [xBP + xS*2] ; pVMCBHostPhys (64 bits physical address; x86: take low dword only)
860 push xAX ; save for the vmload after vmrun
861 vmsave
862
863 ; setup eax for VMLOAD
864 mov xAX, [xBP + xS*2 + RTHCPHYS_CB] ; pVMCBPhys (64 bits physical address; take low dword only)
865
866 ;/* Restore Guest's general purpose registers. */
867 ;/* EAX is loaded from the VMCB by VMRUN */
868 mov ebx, [xSI + CPUMCTX.ebx]
869 mov ecx, [xSI + CPUMCTX.ecx]
870 mov edx, [xSI + CPUMCTX.edx]
871 mov edi, [xSI + CPUMCTX.edi]
872 mov ebp, [xSI + CPUMCTX.ebp]
873 mov esi, [xSI + CPUMCTX.esi]
874
875 ; Clear the global interrupt flag & execute sti to make sure external interrupts cause a world switch
876 clgi
877 sti
878
879 ; load guest fs, gs, sysenter msr etc
880 vmload
881 ; run the VM
882 vmrun
883
884 ;/* EAX is in the VMCB already; we can use it here. */
885
886 ; save guest fs, gs, sysenter msr etc
887 vmsave
888
889 ; load host fs, gs, sysenter msr etc
890 pop xAX ; pushed above
891 vmload
892
893 ; Set the global interrupt flag again, but execute cli to make sure IF=0.
894 cli
895 stgi
896
897 pop xAX ; pCtx
898
899 mov [ss:xAX + CPUMCTX.ebx], ebx
900 mov [ss:xAX + CPUMCTX.ecx], ecx
901 mov [ss:xAX + CPUMCTX.edx], edx
902 mov [ss:xAX + CPUMCTX.esi], esi
903 mov [ss:xAX + CPUMCTX.edi], edi
904 mov [ss:xAX + CPUMCTX.ebp], ebp
905
906 ; Restore general purpose registers
907 MYPOPAD
908
909 mov eax, VINF_SUCCESS
910
911 popf
912 pop xBP
913%ifdef RT_ARCH_AMD64
914 add xSP, 4*xS
915%endif
916 ret
917ENDPROC SVMVMRun
918
919
920;;
921; Executes INVLPGA
922;
923; @param pPageGC msc:ecx gcc:edi x86:[esp+04] Virtual page to invalidate
924; @param uASID msc:edx gcc:esi x86:[esp+08] Tagged TLB id
925;
926;DECLASM(void) SVMInvlpgA(RTGCPTR pPageGC, uint32_t uASID);
927BEGINPROC SVMInvlpgA
928%ifdef RT_ARCH_AMD64
929 %ifdef ASM_CALL64_GCC
930 mov eax, edi ;; @todo 64-bit guest.
931 mov ecx, esi
932 %else
933 mov eax, ecx ;; @todo 64-bit guest.
934 mov ecx, edx
935 %endif
936%else
937 mov eax, [esp + 4]
938 mov ecx, [esp + 8]
939%endif
940 invlpga [xAX], ecx
941 ret
942ENDPROC SVMInvlpgA
943
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette