VirtualBox

source: vbox/trunk/src/VBox/VMM/include/HMInternal.h@ 45841

Last change on this file since 45841 was 45804, checked in by vboxsync, 12 years ago

VMX: Added CFGM key to disable unrestricted execution.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 34.8 KB
Line 
1/* $Id: HMInternal.h 45804 2013-04-29 12:03:31Z vboxsync $ */
2/** @file
3 * HM - Internal header file.
4 */
5
6/*
7 * Copyright (C) 2006-2013 Oracle Corporation
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
18#ifndef ___HMInternal_h
19#define ___HMInternal_h
20
21#include <VBox/cdefs.h>
22#include <VBox/types.h>
23#include <VBox/vmm/em.h>
24#include <VBox/vmm/stam.h>
25#include <VBox/dis.h>
26#include <VBox/vmm/hm.h>
27#include <VBox/vmm/hm_vmx.h>
28#include <VBox/vmm/pgm.h>
29#include <VBox/vmm/cpum.h>
30#include <iprt/memobj.h>
31#include <iprt/cpuset.h>
32#include <iprt/mp.h>
33#include <iprt/avl.h>
34
35#if HC_ARCH_BITS == 64 || defined(VBOX_WITH_HYBRID_32BIT_KERNEL) || defined (VBOX_WITH_64_BITS_GUESTS)
36/* Enable 64 bits guest support. */
37# define VBOX_ENABLE_64_BITS_GUESTS
38#endif
39
40#ifdef VBOX_WITH_OLD_VTX_CODE
41# define VMX_USE_CACHED_VMCS_ACCESSES
42#elif HC_ARCH_BITS == 32 && defined(VBOX_ENABLE_64_BITS_GUESTS) && !defined(VBOX_WITH_HYBRID_32BIT_KERNEL)
43# define VMX_USE_CACHED_VMCS_ACCESSES
44#endif
45
46/** @def HM_PROFILE_EXIT_DISPATCH
47 * Enables profiling of the VM exit handler dispatching. */
48#if 0
49# define HM_PROFILE_EXIT_DISPATCH
50#endif
51
52/* The MSR auto load/store does not work for KERNEL_GS_BASE MSR, thus we
53 * handle this MSR manually. See @bugref{6208}. This is clearly visible while
54 * booting Solaris 11 (11.1 b19) VMs with 2 Cpus.
55 *
56 * Note: don't forget to update the assembly files while modifying this!
57 */
58# define VBOX_WITH_AUTO_MSR_LOAD_RESTORE
59
60RT_C_DECLS_BEGIN
61
62
63/** @defgroup grp_hm_int Internal
64 * @ingroup grp_hm
65 * @internal
66 * @{
67 */
68
69
70/** Maximum number of exit reason statistics counters. */
71#define MAX_EXITREASON_STAT 0x100
72#define MASK_EXITREASON_STAT 0xff
73#define MASK_INJECT_IRQ_STAT 0xff
74
75/** @name Changed flags
76 * These flags are used to keep track of which important registers that
77 * have been changed since last they were reset.
78 * @{
79 */
80#ifdef VBOX_WITH_OLD_VTX_CODE
81# define HM_CHANGED_GUEST_FPU RT_BIT(0)
82# define HM_CHANGED_GUEST_CR0 RT_BIT(1)
83# define HM_CHANGED_GUEST_CR3 RT_BIT(2)
84# define HM_CHANGED_GUEST_CR4 RT_BIT(3)
85# define HM_CHANGED_GUEST_GDTR RT_BIT(4)
86# define HM_CHANGED_GUEST_IDTR RT_BIT(5)
87# define HM_CHANGED_GUEST_LDTR RT_BIT(6)
88# define HM_CHANGED_GUEST_TR RT_BIT(7)
89# define HM_CHANGED_GUEST_MSR RT_BIT(8)
90# define HM_CHANGED_GUEST_SEGMENT_REGS RT_BIT(9)
91# define HM_CHANGED_GUEST_DEBUG RT_BIT(10)
92# define HM_CHANGED_HOST_CONTEXT RT_BIT(11)
93# define HM_CHANGED_ALL_GUEST ( HM_CHANGED_GUEST_SEGMENT_REGS \
94 | HM_CHANGED_GUEST_CR0 \
95 | HM_CHANGED_GUEST_CR3 \
96 | HM_CHANGED_GUEST_CR4 \
97 | HM_CHANGED_GUEST_GDTR \
98 | HM_CHANGED_GUEST_IDTR \
99 | HM_CHANGED_GUEST_LDTR \
100 | HM_CHANGED_GUEST_TR \
101 | HM_CHANGED_GUEST_MSR \
102 | HM_CHANGED_GUEST_DEBUG \
103 | HM_CHANGED_GUEST_FPU)
104#else
105# define HM_CHANGED_GUEST_RIP RT_BIT(0)
106# define HM_CHANGED_GUEST_RSP RT_BIT(1)
107# define HM_CHANGED_GUEST_RFLAGS RT_BIT(2)
108# define HM_CHANGED_GUEST_CR0 RT_BIT(3)
109# define HM_CHANGED_GUEST_CR3 RT_BIT(4)
110# define HM_CHANGED_GUEST_CR4 RT_BIT(5)
111# define HM_CHANGED_GUEST_GDTR RT_BIT(6)
112# define HM_CHANGED_GUEST_IDTR RT_BIT(7)
113# define HM_CHANGED_GUEST_LDTR RT_BIT(8)
114# define HM_CHANGED_GUEST_TR RT_BIT(9)
115# define HM_CHANGED_GUEST_SEGMENT_REGS RT_BIT(10)
116# define HM_CHANGED_GUEST_DEBUG RT_BIT(11)
117# define HM_CHANGED_GUEST_SYSENTER_CS_MSR RT_BIT(12)
118# define HM_CHANGED_GUEST_SYSENTER_EIP_MSR RT_BIT(13)
119# define HM_CHANGED_GUEST_SYSENTER_ESP_MSR RT_BIT(14)
120# define HM_CHANGED_VMX_GUEST_AUTO_MSRS RT_BIT(15)
121# define HM_CHANGED_VMX_GUEST_ACTIVITY_STATE RT_BIT(16)
122# define HM_CHANGED_VMX_GUEST_APIC_STATE RT_BIT(17)
123# define HM_CHANGED_VMX_ENTRY_CTLS RT_BIT(18)
124# define HM_CHANGED_VMX_EXIT_CTLS RT_BIT(19)
125
126# define HM_CHANGED_HOST_CONTEXT RT_BIT(20)
127
128# define HM_CHANGED_ALL_GUEST ( HM_CHANGED_GUEST_RIP \
129 | HM_CHANGED_GUEST_RSP \
130 | HM_CHANGED_GUEST_RFLAGS \
131 | HM_CHANGED_GUEST_CR0 \
132 | HM_CHANGED_GUEST_CR3 \
133 | HM_CHANGED_GUEST_CR4 \
134 | HM_CHANGED_GUEST_GDTR \
135 | HM_CHANGED_GUEST_IDTR \
136 | HM_CHANGED_GUEST_LDTR \
137 | HM_CHANGED_GUEST_TR \
138 | HM_CHANGED_GUEST_SEGMENT_REGS \
139 | HM_CHANGED_GUEST_DEBUG \
140 | HM_CHANGED_GUEST_SYSENTER_CS_MSR \
141 | HM_CHANGED_GUEST_SYSENTER_EIP_MSR \
142 | HM_CHANGED_GUEST_SYSENTER_ESP_MSR \
143 | HM_CHANGED_VMX_GUEST_AUTO_MSRS \
144 | HM_CHANGED_VMX_GUEST_ACTIVITY_STATE \
145 | HM_CHANGED_VMX_GUEST_APIC_STATE \
146 | HM_CHANGED_VMX_ENTRY_CTLS \
147 | HM_CHANGED_VMX_EXIT_CTLS)
148#endif
149
150#define HM_CHANGED_ALL (HM_CHANGED_ALL_GUEST | HM_CHANGED_HOST_CONTEXT)
151/** @} */
152
153/** Maximum number of page flushes we are willing to remember before considering a full TLB flush. */
154#define HM_MAX_TLB_SHOOTDOWN_PAGES 8
155
156/** Size for the EPT identity page table (1024 4 MB pages to cover the entire address space). */
157#define HM_EPT_IDENTITY_PG_TABLE_SIZE PAGE_SIZE
158/** Size of the TSS structure + 2 pages for the IO bitmap + end byte. */
159#define HM_VTX_TSS_SIZE (sizeof(VBOXTSS) + 2 * PAGE_SIZE + 1)
160/** Total guest mapped memory needed. */
161#define HM_VTX_TOTAL_DEVHEAP_MEM (HM_EPT_IDENTITY_PG_TABLE_SIZE + HM_VTX_TSS_SIZE)
162
163/** Enable for TPR guest patching. */
164#define VBOX_HM_WITH_GUEST_PATCHING
165
166/** HM SSM version
167 */
168#ifdef VBOX_HM_WITH_GUEST_PATCHING
169# define HM_SSM_VERSION 5
170# define HM_SSM_VERSION_NO_PATCHING 4
171#else
172# define HM_SSM_VERSION 4
173# define HM_SSM_VERSION_NO_PATCHING 4
174#endif
175#define HM_SSM_VERSION_2_0_X 3
176
177/**
178 * Global per-cpu information. (host)
179 */
180typedef struct HMGLOBLCPUINFO
181{
182 /** The CPU ID. */
183 RTCPUID idCpu;
184 /** The memory object */
185 RTR0MEMOBJ hMemObj;
186 /** Current ASID (AMD-V) / VPID (Intel). */
187 uint32_t uCurrentAsid;
188 /** TLB flush count. */
189 uint32_t cTlbFlushes;
190 /** Whether to flush each new ASID/VPID before use. */
191 bool fFlushAsidBeforeUse;
192 /** Configured for VT-x or AMD-V. */
193 bool fConfigured;
194 /** Set if the VBOX_HWVIRTEX_IGNORE_SVM_IN_USE hack is active. */
195 bool fIgnoreAMDVInUseError;
196 /** In use by our code. (for power suspend) */
197 volatile bool fInUse;
198} HMGLOBLCPUINFO;
199/** Pointer to the per-cpu global information. */
200typedef HMGLOBLCPUINFO *PHMGLOBLCPUINFO;
201
202typedef enum
203{
204 HMPENDINGIO_INVALID = 0,
205 HMPENDINGIO_PORT_READ,
206 HMPENDINGIO_PORT_WRITE,
207 HMPENDINGIO_STRING_READ,
208 HMPENDINGIO_STRING_WRITE,
209 /** The usual 32-bit paranoia. */
210 HMPENDINGIO_32BIT_HACK = 0x7fffffff
211} HMPENDINGIO;
212
213
214typedef enum
215{
216 HMTPRINSTR_INVALID,
217 HMTPRINSTR_READ,
218 HMTPRINSTR_READ_SHR4,
219 HMTPRINSTR_WRITE_REG,
220 HMTPRINSTR_WRITE_IMM,
221 HMTPRINSTR_JUMP_REPLACEMENT,
222 /** The usual 32-bit paranoia. */
223 HMTPRINSTR_32BIT_HACK = 0x7fffffff
224} HMTPRINSTR;
225
226typedef struct
227{
228 /** The key is the address of patched instruction. (32 bits GC ptr) */
229 AVLOU32NODECORE Core;
230 /** Original opcode. */
231 uint8_t aOpcode[16];
232 /** Instruction size. */
233 uint32_t cbOp;
234 /** Replacement opcode. */
235 uint8_t aNewOpcode[16];
236 /** Replacement instruction size. */
237 uint32_t cbNewOp;
238 /** Instruction type. */
239 HMTPRINSTR enmType;
240 /** Source operand. */
241 uint32_t uSrcOperand;
242 /** Destination operand. */
243 uint32_t uDstOperand;
244 /** Number of times the instruction caused a fault. */
245 uint32_t cFaults;
246 /** Patch address of the jump replacement. */
247 RTGCPTR32 pJumpTarget;
248} HMTPRPATCH;
249/** Pointer to HMTPRPATCH. */
250typedef HMTPRPATCH *PHMTPRPATCH;
251
252/**
253 * Switcher function, HC to the special 64-bit RC.
254 *
255 * @param pVM Pointer to the VM.
256 * @param offCpumVCpu Offset from pVM->cpum to pVM->aCpus[idCpu].cpum.
257 * @returns Return code indicating the action to take.
258 */
259typedef DECLCALLBACK(int) FNHMSWITCHERHC(PVM pVM, uint32_t offCpumVCpu);
260/** Pointer to switcher function. */
261typedef FNHMSWITCHERHC *PFNHMSWITCHERHC;
262
263/**
264 * HM VM Instance data.
265 * Changes to this must checked against the padding of the hm union in VM!
266 */
267typedef struct HM
268{
269 /** Set when we've initialized VMX or SVM. */
270 bool fInitialized;
271
272 /** Set if nested paging is enabled. */
273 bool fNestedPaging;
274
275 /** Set if nested paging is allowed. */
276 bool fAllowNestedPaging;
277
278 /** Set if large pages are enabled (requires nested paging). */
279 bool fLargePages;
280
281 /** Set if we can support 64-bit guests or not. */
282 bool fAllow64BitGuests;
283
284 /** Set if an IO-APIC is configured for this VM. */
285 bool fHasIoApic;
286
287 /** Set when TPR patching is allowed. */
288 bool fTRPPatchingAllowed;
289
290 /** Set when we initialize VT-x or AMD-V once for all CPUs. */
291 bool fGlobalInit;
292
293 /** Set when TPR patching is active. */
294 bool fTPRPatchingActive;
295 bool u8Alignment[7];
296
297 /** And mask for copying register contents. */
298 uint64_t u64RegisterMask;
299
300 /** Maximum ASID allowed. */
301 uint32_t uMaxAsid;
302
303 /** The maximum number of resumes loops allowed in ring-0 (safety precaution).
304 * This number is set much higher when RTThreadPreemptIsPending is reliable. */
305 uint32_t cMaxResumeLoops;
306
307 /** Guest allocated memory for patching purposes. */
308 RTGCPTR pGuestPatchMem;
309 /** Current free pointer inside the patch block. */
310 RTGCPTR pFreeGuestPatchMem;
311 /** Size of the guest patch memory block. */
312 uint32_t cbGuestPatchMem;
313 uint32_t uPadding1;
314
315#if HC_ARCH_BITS == 32 && defined(VBOX_ENABLE_64_BITS_GUESTS) && !defined(VBOX_WITH_HYBRID_32BIT_KERNEL)
316 /** 32 to 64 bits switcher entrypoint. */
317 R0PTRTYPE(PFNHMSWITCHERHC) pfnHost32ToGuest64R0;
318 RTR0PTR uPadding2;
319#endif
320
321 struct
322 {
323 /** Set by the ring-0 side of HM to indicate VMX is supported by the
324 * CPU. */
325 bool fSupported;
326
327 /** Set when we've enabled VMX. */
328 bool fEnabled;
329
330 /** Set if VPID is supported. */
331 bool fVpid;
332
333 /** Set if VT-x VPID is allowed. */
334 bool fAllowVpid;
335
336 /** Set if unrestricted guest execution is in use (real and protected mode without paging). */
337 bool fUnrestrictedGuest;
338
339 /** Set if unrestricted guest execution is allowed to be used. */
340 bool fAllowUnrestricted;
341
342 /** Whether we're using the preemption timer or not. */
343 bool fUsePreemptTimer;
344 /** The shift mask employed by the VMX-Preemption timer. */
345 uint8_t cPreemptTimerShift;
346
347 /** Virtual address of the TSS page used for real mode emulation. */
348 R3PTRTYPE(PVBOXTSS) pRealModeTSS;
349
350 /** Virtual address of the identity page table used for real mode and protected mode without paging emulation in EPT mode. */
351 R3PTRTYPE(PX86PD) pNonPagingModeEPTPageTable;
352
353 /** R0 memory object for the APIC-access page. */
354 RTR0MEMOBJ hMemObjApicAccess;
355 /** Physical address of the APIC-access page. */
356 RTHCPHYS HCPhysApicAccess;
357 /** Virtual address of the APIC-access page. */
358 R0PTRTYPE(uint8_t *) pbApicAccess;
359
360#ifdef VBOX_WITH_CRASHDUMP_MAGIC
361 RTR0MEMOBJ hMemObjScratch;
362 RTHCPHYS HCPhysScratch;
363 R0PTRTYPE(uint8_t *) pbScratch;
364#endif
365
366#ifndef VBOX_WITH_OLD_VTX_CODE
367 unsigned uFlushTaggedTlb;
368#else
369 /** Ring 0 handlers for VT-x. */
370 DECLR0CALLBACKMEMBER(void, pfnFlushTaggedTlb, (PVM pVM, PVMCPU pVCpu));
371#endif
372
373#if HC_ARCH_BITS == 32 && defined(VBOX_ENABLE_64_BITS_GUESTS)
374 uint32_t u32Alignment;
375#endif
376 /** Host CR4 value (set by ring-0 VMX init) */
377 uint64_t hostCR4;
378
379 /** Host EFER value (set by ring-0 VMX init) */
380 uint64_t hostEFER;
381
382 /** VMX MSR values */
383 struct
384 {
385 uint64_t feature_ctrl;
386 uint64_t vmx_basic_info;
387 VMX_CAPABILITY vmx_pin_ctls;
388 VMX_CAPABILITY vmx_proc_ctls;
389 VMX_CAPABILITY vmx_proc_ctls2;
390 VMX_CAPABILITY vmx_exit;
391 VMX_CAPABILITY vmx_entry;
392 uint64_t vmx_misc;
393 uint64_t vmx_cr0_fixed0;
394 uint64_t vmx_cr0_fixed1;
395 uint64_t vmx_cr4_fixed0;
396 uint64_t vmx_cr4_fixed1;
397 uint64_t vmx_vmcs_enum;
398 uint64_t vmx_ept_vpid_caps;
399 } msr;
400
401 /** Flush types for invept & invvpid; they depend on capabilities. */
402 VMX_FLUSH_EPT enmFlushEpt;
403 VMX_FLUSH_VPID enmFlushVpid;
404 } vmx;
405
406 struct
407 {
408 /** Set by the ring-0 side of HM to indicate SVM is supported by the
409 * CPU. */
410 bool fSupported;
411 /** Set when we've enabled SVM. */
412 bool fEnabled;
413 /** Set if erratum 170 affects the AMD cpu. */
414 bool fAlwaysFlushTLB;
415 /** Set when the hack to ignore VERR_SVM_IN_USE is active. */
416 bool fIgnoreInUseError;
417
418 /** R0 memory object for the IO bitmap (12kb). */
419 RTR0MEMOBJ hMemObjIOBitmap;
420 /** Physical address of the IO bitmap (12kb). */
421 RTHCPHYS HCPhysIOBitmap;
422 /** Virtual address of the IO bitmap. */
423 R0PTRTYPE(void *) pvIOBitmap;
424
425 /* HWCR MSR (for diagnostics) */
426 uint64_t msrHwcr;
427
428 /** SVM revision. */
429 uint32_t u32Rev;
430
431 /** SVM feature bits from cpuid 0x8000000a */
432 uint32_t u32Features;
433 } svm;
434
435 /**
436 * AVL tree with all patches (active or disabled) sorted by guest instruction address
437 */
438 AVLOU32TREE PatchTree;
439 uint32_t cPatches;
440 HMTPRPATCH aPatches[64];
441
442 struct
443 {
444 uint32_t u32AMDFeatureECX;
445 uint32_t u32AMDFeatureEDX;
446 } cpuid;
447
448 /** Saved error from detection */
449 int32_t lLastError;
450
451 /** HMR0Init was run */
452 bool fHMR0Init;
453 bool u8Alignment1[7];
454
455 STAMCOUNTER StatTprPatchSuccess;
456 STAMCOUNTER StatTprPatchFailure;
457 STAMCOUNTER StatTprReplaceSuccess;
458 STAMCOUNTER StatTprReplaceFailure;
459} HM;
460/** Pointer to HM VM instance data. */
461typedef HM *PHM;
462
463/* Maximum number of cached entries. */
464#define VMCSCACHE_MAX_ENTRY 128
465
466/* Structure for storing read and write VMCS actions. */
467typedef struct VMCSCACHE
468{
469#ifdef VBOX_WITH_CRASHDUMP_MAGIC
470 /* Magic marker for searching in crash dumps. */
471 uint8_t aMagic[16];
472 uint64_t uMagic;
473 uint64_t u64TimeEntry;
474 uint64_t u64TimeSwitch;
475 uint64_t cResume;
476 uint64_t interPD;
477 uint64_t pSwitcher;
478 uint32_t uPos;
479 uint32_t idCpu;
480#endif
481 /* CR2 is saved here for EPT syncing. */
482 uint64_t cr2;
483 struct
484 {
485 uint32_t cValidEntries;
486 uint32_t uAlignment;
487 uint32_t aField[VMCSCACHE_MAX_ENTRY];
488 uint64_t aFieldVal[VMCSCACHE_MAX_ENTRY];
489 } Write;
490 struct
491 {
492 uint32_t cValidEntries;
493 uint32_t uAlignment;
494 uint32_t aField[VMCSCACHE_MAX_ENTRY];
495 uint64_t aFieldVal[VMCSCACHE_MAX_ENTRY];
496 } Read;
497#ifdef DEBUG
498 struct
499 {
500 RTHCPHYS HCPhysCpuPage;
501 RTHCPHYS HCPhysVmcs;
502 RTGCPTR pCache;
503 RTGCPTR pCtx;
504 } TestIn;
505 struct
506 {
507 RTHCPHYS HCPhysVmcs;
508 RTGCPTR pCache;
509 RTGCPTR pCtx;
510 uint64_t eflags;
511 uint64_t cr8;
512 } TestOut;
513 struct
514 {
515 uint64_t param1;
516 uint64_t param2;
517 uint64_t param3;
518 uint64_t param4;
519 } ScratchPad;
520#endif
521} VMCSCACHE;
522/** Pointer to VMCSCACHE. */
523typedef VMCSCACHE *PVMCSCACHE;
524
525/** VMX StartVM function. */
526typedef DECLCALLBACK(int) FNHMVMXSTARTVM(RTHCUINT fResume, PCPUMCTX pCtx, PVMCSCACHE pCache, PVM pVM, PVMCPU pVCpu);
527/** Pointer to a VMX StartVM function. */
528typedef R0PTRTYPE(FNHMVMXSTARTVM *) PFNHMVMXSTARTVM;
529
530/** SVM VMRun function. */
531typedef DECLCALLBACK(int) FNHMSVMVMRUN(RTHCPHYS pVMCBHostPhys, RTHCPHYS pVMCBPhys, PCPUMCTX pCtx, PVM pVM, PVMCPU pVCpu);
532/** Pointer to a SVM VMRun function. */
533typedef R0PTRTYPE(FNHMSVMVMRUN *) PFNHMSVMVMRUN;
534
535/**
536 * HM VMCPU Instance data.
537 */
538typedef struct HMCPU
539{
540 /** Old style FPU reporting trap mask override performed (optimization) */
541 bool fFPUOldStyleOverride;
542 /** Set if we don't have to flush the TLB on VM entry. */
543 bool fResumeVM;
544 /** Set if we need to flush the TLB during the world switch. */
545 bool fForceTLBFlush;
546 /** Set when we're using VT-x or AMD-V at that moment. */
547 bool fActive;
548 /** Set when the TLB has been checked until we return from the world switch. */
549 volatile bool fCheckedTLBFlush;
550 uint8_t u8Alignment[3];
551
552 /** World switch exit counter. */
553 volatile uint32_t cWorldSwitchExits;
554 /** HM_CHANGED_* flags. */
555 uint32_t fContextUseFlags;
556 /** Id of the last cpu we were executing code on (NIL_RTCPUID for the first time) */
557 RTCPUID idLastCpu;
558 /** TLB flush count */
559 uint32_t cTlbFlushes;
560 /** Current ASID in use by the VM */
561 uint32_t uCurrentAsid;
562 uint32_t u32Alignment;
563
564 /** Host's TSC_AUX MSR (used when RDTSCP doesn't cause VM-exits). */
565 uint64_t u64HostTscAux;
566
567 struct
568 {
569 /** Physical address of the VM control structure (VMCS). */
570 RTHCPHYS HCPhysVmcs;
571 /** R0 memory object for the VM control structure (VMCS). */
572 RTR0MEMOBJ hMemObjVmcs;
573 /** Virtual address of the VM control structure (VMCS). */
574 R0PTRTYPE(void *) pvVmcs;
575 /** Ring 0 handlers for VT-x. */
576 PFNHMVMXSTARTVM pfnStartVM;
577#if HC_ARCH_BITS == 32
578 uint32_t u32Alignment1;
579#endif
580
581 /** Current VMX_VMCS32_CTRL_PIN_EXEC_CONTROLS. */
582 uint32_t u32PinCtls;
583 /** Current VMX_VMCS32_CTRL_PROC_EXEC_CONTROLS. */
584 uint32_t u32ProcCtls;
585 /** Current VMX_VMCS32_CTRL_PROC_EXEC2_CONTROLS. */
586 uint32_t u32ProcCtls2;
587 /** Current VMX_VMCS32_CTRL_EXIT_CONTROLS. */
588 uint32_t u32ExitCtls;
589 /** Current VMX_VMCS32_CTRL_ENTRY_CONTROLS. */
590 uint32_t u32EntryCtls;
591 /** Physical address of the virtual APIC page for TPR caching. */
592 RTHCPHYS HCPhysVirtApic;
593 /** R0 memory object for the virtual APIC page for TPR caching. */
594 RTR0MEMOBJ hMemObjVirtApic;
595 /** Virtual address of the virtual APIC page for TPR caching. */
596 R0PTRTYPE(uint8_t *) pbVirtApic;
597#if HC_ARCH_BITS == 32
598 uint32_t u32Alignment2;
599#endif
600
601 /** Current CR0 mask. */
602 uint64_t cr0_mask;
603 /** Current CR4 mask. */
604 uint64_t cr4_mask;
605 /** Current exception bitmap. */
606 uint32_t u32XcptBitmap;
607 /** The updated-guest-state mask. */
608 uint32_t fUpdatedGuestState;
609 /** Current EPTP. */
610 RTHCPHYS HCPhysEPTP;
611
612 /** Physical address of the MSR bitmap. */
613 RTHCPHYS HCPhysMsrBitmap;
614 /** R0 memory object for the MSR bitmap. */
615 RTR0MEMOBJ hMemObjMsrBitmap;
616 /** Virtual address of the MSR bitmap. */
617 R0PTRTYPE(void *) pvMsrBitmap;
618
619#ifdef VBOX_WITH_AUTO_MSR_LOAD_RESTORE
620 /** Physical address of the VM-entry MSR-load and VM-exit MSR-store area (used
621 * for guest MSRs). */
622 RTHCPHYS HCPhysGuestMsr;
623 /** R0 memory object of the VM-entry MSR-load and VM-exit MSR-store area
624 * (used for guest MSRs). */
625 RTR0MEMOBJ hMemObjGuestMsr;
626 /** Virtual address of the VM-entry MSR-load and VM-exit MSR-store area (used
627 * for guest MSRs). */
628 R0PTRTYPE(void *) pvGuestMsr;
629
630 /** Physical address of the VM-exit MSR-load area (used for host MSRs). */
631 RTHCPHYS HCPhysHostMsr;
632 /** R0 memory object for the VM-exit MSR-load area (used for host MSRs). */
633 RTR0MEMOBJ hMemObjHostMsr;
634 /** Virtual address of the VM-exit MSR-load area (used for host MSRs). */
635 R0PTRTYPE(void *) pvHostMsr;
636
637 /** Number of automatically loaded/restored guest MSRs during the world switch. */
638 uint32_t cGuestMsrs;
639 uint32_t uAlignment;
640#endif /* VBOX_WITH_AUTO_MSR_LOAD_RESTORE */
641
642 /** The cached APIC-base MSR used for identifying when to map the HC physical APIC-access page. */
643 uint64_t u64MsrApicBase;
644 /** Last use TSC offset value. (cached) */
645 uint64_t u64TSCOffset;
646 /** VMCS cache. */
647 VMCSCACHE VMCSCache;
648
649 /** Real-mode emulation state. */
650 struct
651 {
652 X86DESCATTR uAttrCS;
653 X86DESCATTR uAttrDS;
654 X86DESCATTR uAttrES;
655 X86DESCATTR uAttrFS;
656 X86DESCATTR uAttrGS;
657 X86DESCATTR uAttrSS;
658 X86EFLAGS eflags;
659 uint32_t fRealOnV86Active;
660 } RealMode;
661
662 struct
663 {
664 uint64_t u64VMCSPhys;
665 uint32_t u32VMCSRevision;
666 uint32_t u32InstrError;
667 uint32_t u32ExitReason;
668 RTCPUID idEnteredCpu;
669 RTCPUID idCurrentCpu;
670 uint32_t padding;
671 } lasterror;
672
673#ifdef VBOX_WITH_OLD_VTX_CODE
674 /** The last seen guest paging mode (by VT-x). */
675 PGMMODE enmLastSeenGuestMode;
676 /** Current guest paging mode (as seen by HMR3PagingModeChanged). */
677 PGMMODE enmCurrGuestMode;
678 /** Previous guest paging mode (as seen by HMR3PagingModeChanged). */
679 PGMMODE enmPrevGuestMode;
680#else
681 /** Set if guest was executing in real mode (extra checks). */
682 bool fWasInRealMode;
683#endif
684 } vmx;
685
686 struct
687 {
688 /** R0 memory object for the host VM control block (VMCB). */
689 RTR0MEMOBJ hMemObjVMCBHost;
690 /** Physical address of the host VM control block (VMCB). */
691 RTHCPHYS HCPhysVMCBHost;
692 /** Virtual address of the host VM control block (VMCB). */
693 R0PTRTYPE(void *) pvVMCBHost;
694
695 /** R0 memory object for the VM control block (VMCB). */
696 RTR0MEMOBJ hMemObjVMCB;
697 /** Physical address of the VM control block (VMCB). */
698 RTHCPHYS HCPhysVMCB;
699 /** Virtual address of the VM control block (VMCB). */
700 R0PTRTYPE(void *) pvVMCB;
701
702 /** Ring 0 handlers for VT-x. */
703 PFNHMSVMVMRUN pfnVMRun;
704
705 /** R0 memory object for the MSR bitmap (8kb). */
706 RTR0MEMOBJ hMemObjMsrBitmap;
707 /** Physical address of the MSR bitmap (8kb). */
708 RTHCPHYS HCPhysMsrBitmap;
709 /** Virtual address of the MSR bitmap. */
710 R0PTRTYPE(void *) pvMsrBitmap;
711 } svm;
712
713 /** Event injection state. */
714 struct
715 {
716 uint32_t fPending;
717 uint32_t u32ErrCode;
718 uint32_t cbInstr;
719 uint32_t u32Padding; /**< Explicit alignment padding. */
720 uint64_t u64IntrInfo;
721 RTGCUINTPTR GCPtrFaultAddress;
722 } Event;
723
724 /** IO Block emulation state. */
725 struct
726 {
727 bool fEnabled;
728 uint8_t u8Align[7];
729
730 /** RIP at the start of the io code we wish to emulate in the recompiler. */
731 RTGCPTR GCPtrFunctionEip;
732
733 uint64_t cr0;
734 } EmulateIoBlock;
735
736 struct
737 {
738 /** Pending IO operation type. */
739 HMPENDINGIO enmType;
740 uint32_t uPadding;
741 RTGCPTR GCPtrRip;
742 RTGCPTR GCPtrRipNext;
743 union
744 {
745 struct
746 {
747 uint32_t uPort;
748 uint32_t uAndVal;
749 uint32_t cbSize;
750 } Port;
751 uint64_t aRaw[2];
752 } s;
753 } PendingIO;
754
755 /** The PAE PDPEs used with Nested Paging (only valid when
756 * VMCPU_FF_HM_UPDATE_PAE_PDPES is set). */
757 X86PDPE aPdpes[4];
758
759 /** Current shadow paging mode. */
760 PGMMODE enmShadowMode;
761
762 /** The CPU ID of the CPU currently owning the VMCS. Set in
763 * HMR0Enter and cleared in HMR0Leave. */
764 RTCPUID idEnteredCpu;
765
766 /** To keep track of pending TLB shootdown pages. (SMP guest only) */
767 struct
768 {
769 RTGCPTR aPages[HM_MAX_TLB_SHOOTDOWN_PAGES];
770 uint32_t cPages;
771 uint32_t u32Padding; /**< Explicit alignment padding. */
772 } TlbShootdown;
773
774 /** For saving stack space, the disassembler state is allocated here instead of
775 * on the stack. */
776 DISCPUSTATE DisState;
777
778 STAMPROFILEADV StatEntry;
779 STAMPROFILEADV StatExit1;
780 STAMPROFILEADV StatExit2;
781 STAMPROFILEADV StatExitIO;
782 STAMPROFILEADV StatExitMovCRx;
783 STAMPROFILEADV StatExitXcptNmi;
784 STAMPROFILEADV StatLoadGuestState;
785 STAMPROFILEADV StatInGC;
786
787#if HC_ARCH_BITS == 32 && defined(VBOX_ENABLE_64_BITS_GUESTS) && !defined(VBOX_WITH_HYBRID_32BIT_KERNEL)
788 STAMPROFILEADV StatWorldSwitch3264;
789#endif
790 STAMPROFILEADV StatPoke;
791 STAMPROFILEADV StatSpinPoke;
792 STAMPROFILEADV StatSpinPokeFailed;
793
794 STAMCOUNTER StatIntInject;
795
796 STAMCOUNTER StatExitShadowNM;
797 STAMCOUNTER StatExitGuestNM;
798 STAMCOUNTER StatExitShadowPF; /* Misleading, currently used for MMIO #PFs as well. */
799 STAMCOUNTER StatExitShadowPFEM;
800 STAMCOUNTER StatExitGuestPF;
801 STAMCOUNTER StatExitGuestUD;
802 STAMCOUNTER StatExitGuestSS;
803 STAMCOUNTER StatExitGuestNP;
804 STAMCOUNTER StatExitGuestGP;
805 STAMCOUNTER StatExitGuestDE;
806 STAMCOUNTER StatExitGuestDB;
807 STAMCOUNTER StatExitGuestMF;
808 STAMCOUNTER StatExitGuestBP;
809 STAMCOUNTER StatExitGuestXF;
810 STAMCOUNTER StatExitGuestXcpUnk;
811 STAMCOUNTER StatExitInvlpg;
812 STAMCOUNTER StatExitInvd;
813 STAMCOUNTER StatExitWbinvd;
814 STAMCOUNTER StatExitPause;
815 STAMCOUNTER StatExitCpuid;
816 STAMCOUNTER StatExitRdtsc;
817 STAMCOUNTER StatExitRdtscp;
818 STAMCOUNTER StatExitRdpmc;
819 STAMCOUNTER StatExitRdrand;
820 STAMCOUNTER StatExitCli;
821 STAMCOUNTER StatExitSti;
822 STAMCOUNTER StatExitPushf;
823 STAMCOUNTER StatExitPopf;
824 STAMCOUNTER StatExitIret;
825 STAMCOUNTER StatExitInt;
826 STAMCOUNTER StatExitCRxWrite[16];
827 STAMCOUNTER StatExitCRxRead[16];
828 STAMCOUNTER StatExitDRxWrite;
829 STAMCOUNTER StatExitDRxRead;
830 STAMCOUNTER StatExitRdmsr;
831 STAMCOUNTER StatExitWrmsr;
832 STAMCOUNTER StatExitClts;
833 STAMCOUNTER StatExitXdtrAccess;
834 STAMCOUNTER StatExitHlt;
835 STAMCOUNTER StatExitMwait;
836 STAMCOUNTER StatExitMonitor;
837 STAMCOUNTER StatExitLmsw;
838 STAMCOUNTER StatExitIOWrite;
839 STAMCOUNTER StatExitIORead;
840 STAMCOUNTER StatExitIOStringWrite;
841 STAMCOUNTER StatExitIOStringRead;
842 STAMCOUNTER StatExitIntWindow;
843 STAMCOUNTER StatExitMaxResume;
844 STAMCOUNTER StatExitExtInt;
845 STAMCOUNTER StatExitPreemptTimer;
846 STAMCOUNTER StatExitTprBelowThreshold;
847 STAMCOUNTER StatExitTaskSwitch;
848 STAMCOUNTER StatExitMtf;
849 STAMCOUNTER StatExitApicAccess;
850 STAMCOUNTER StatIntReinject;
851 STAMCOUNTER StatPendingHostIrq;
852
853 STAMCOUNTER StatFlushPage;
854 STAMCOUNTER StatFlushPageManual;
855 STAMCOUNTER StatFlushPhysPageManual;
856 STAMCOUNTER StatFlushTlb;
857 STAMCOUNTER StatFlushTlbManual;
858 STAMCOUNTER StatFlushTlbWorldSwitch;
859 STAMCOUNTER StatNoFlushTlbWorldSwitch;
860 STAMCOUNTER StatFlushAsid;
861 STAMCOUNTER StatFlushNestedPaging;
862 STAMCOUNTER StatFlushTlbInvlpgVirt;
863 STAMCOUNTER StatFlushTlbInvlpgPhys;
864 STAMCOUNTER StatTlbShootdown;
865 STAMCOUNTER StatTlbShootdownFlush;
866
867 STAMCOUNTER StatSwitchGuestIrq;
868 STAMCOUNTER StatSwitchHmToR3FF;
869 STAMCOUNTER StatSwitchExitToR3;
870 STAMCOUNTER StatSwitchLongJmpToR3;
871
872 STAMCOUNTER StatTscOffset;
873 STAMCOUNTER StatTscIntercept;
874 STAMCOUNTER StatTscInterceptOverFlow;
875
876 STAMCOUNTER StatExitReasonNpf;
877 STAMCOUNTER StatDRxArmed;
878 STAMCOUNTER StatDRxContextSwitch;
879 STAMCOUNTER StatDRxIoCheck;
880
881 STAMCOUNTER StatLoadMinimal;
882 STAMCOUNTER StatLoadFull;
883
884#if HC_ARCH_BITS == 32 && defined(VBOX_ENABLE_64_BITS_GUESTS) && !defined(VBOX_WITH_HYBRID_32BIT_KERNEL)
885 STAMCOUNTER StatFpu64SwitchBack;
886 STAMCOUNTER StatDebug64SwitchBack;
887#endif
888
889#ifdef VBOX_WITH_STATISTICS
890 R3PTRTYPE(PSTAMCOUNTER) paStatExitReason;
891 R0PTRTYPE(PSTAMCOUNTER) paStatExitReasonR0;
892 R3PTRTYPE(PSTAMCOUNTER) paStatInjectedIrqs;
893 R0PTRTYPE(PSTAMCOUNTER) paStatInjectedIrqsR0;
894#endif
895#ifdef HM_PROFILE_EXIT_DISPATCH
896 STAMPROFILEADV StatExitDispatch;
897#endif
898} HMCPU;
899/** Pointer to HM VM instance data. */
900typedef HMCPU *PHMCPU;
901
902
903#ifdef IN_RING0
904
905VMMR0DECL(PHMGLOBLCPUINFO) HMR0GetCurrentCpu(void);
906VMMR0DECL(PHMGLOBLCPUINFO) HMR0GetCurrentCpuEx(RTCPUID idCpu);
907
908
909#ifdef VBOX_STRICT
910VMMR0DECL(void) HMDumpRegs(PVM pVM, PVMCPU pVCpu, PCPUMCTX pCtx);
911VMMR0DECL(void) HMR0DumpDescriptor(PCX86DESCHC pDesc, RTSEL Sel, const char *pszMsg);
912#else
913# define HMDumpRegs(a, b ,c) do { } while (0)
914# define HMR0DumpDescriptor(a, b, c) do { } while (0)
915#endif
916
917# ifdef VBOX_WITH_KERNEL_USING_XMM
918DECLASM(int) hmR0VMXStartVMWrapXMM(RTHCUINT fResume, PCPUMCTX pCtx, PVMCSCACHE pCache, PVM pVM, PVMCPU pVCpu, PFNHMVMXSTARTVM pfnStartVM);
919DECLASM(int) hmR0SVMRunWrapXMM(RTHCPHYS pVMCBHostPhys, RTHCPHYS pVMCBPhys, PCPUMCTX pCtx, PVM pVM, PVMCPU pVCpu, PFNHMSVMVMRUN pfnVMRun);
920# endif
921
922# ifdef VBOX_WITH_HYBRID_32BIT_KERNEL
923/**
924 * Gets 64-bit GDTR and IDTR on darwin.
925 * @param pGdtr Where to store the 64-bit GDTR.
926 * @param pIdtr Where to store the 64-bit IDTR.
927 */
928DECLASM(void) hmR0Get64bitGdtrAndIdtr(PX86XDTR64 pGdtr, PX86XDTR64 pIdtr);
929
930/**
931 * Gets 64-bit CR3 on darwin.
932 * @returns CR3
933 */
934DECLASM(uint64_t) hmR0Get64bitCR3(void);
935# endif
936
937#endif /* IN_RING0 */
938
939/** @} */
940
941RT_C_DECLS_END
942
943#endif
944
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