VirtualBox

source: vbox/trunk/include/VBox/vmm/hm_svm.h@ 70352

Last change on this file since 70352 was 70352, checked in by vboxsync, 7 years ago

VMM/HM,CPUM: SVM VMCB layout adjustments.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 44.4 KB
Line 
1/** @file
2 * HM - SVM (AMD-V) Structures and Definitions. (VMM)
3 */
4
5/*
6 * Copyright (C) 2006-2017 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___VBox_vmm_svm_h
27#define ___VBox_vmm_svm_h
28
29#include <VBox/types.h>
30#include <VBox/err.h>
31#include <iprt/assert.h>
32#include <iprt/asm.h>
33
34#ifdef RT_OS_SOLARIS
35# undef ES
36# undef CS
37# undef DS
38# undef SS
39# undef FS
40# undef GS
41#endif
42
43/** @defgroup grp_hm_svm SVM (AMD-V) Types and Definitions
44 * @ingroup grp_hm
45 * @{
46 */
47
48/** @name SVM generic / convenient defines.
49 * @{
50 */
51/** Number of pages required for the VMCB. */
52#define SVM_VMCB_PAGES 1
53/** Number of pages required for the MSR permission bitmap. */
54#define SVM_MSRPM_PAGES 2
55/** Number of pages required for the IO permission bitmap. */
56#define SVM_IOPM_PAGES 3
57/** @} */
58
59/*
60 * Ugly!
61 * When compiling the recompiler, its own svm.h defines clash with
62 * the following defines. Avoid just the duplicates here as we still
63 * require other definitions and structures in this header.
64 */
65#ifndef IN_REM_R3
66/** @name SVM_EXIT_XXX - SVM Basic Exit Reasons.
67 * @{
68 */
69/** Invalid guest state in VMCB. */
70# define SVM_EXIT_INVALID (uint64_t)(-1)
71/** Read from CR0-CR15. */
72# define SVM_EXIT_READ_CR0 0x0
73# define SVM_EXIT_READ_CR1 0x1
74# define SVM_EXIT_READ_CR2 0x2
75# define SVM_EXIT_READ_CR3 0x3
76# define SVM_EXIT_READ_CR4 0x4
77# define SVM_EXIT_READ_CR5 0x5
78# define SVM_EXIT_READ_CR6 0x6
79# define SVM_EXIT_READ_CR7 0x7
80# define SVM_EXIT_READ_CR8 0x8
81# define SVM_EXIT_READ_CR9 0x9
82# define SVM_EXIT_READ_CR10 0xA
83# define SVM_EXIT_READ_CR11 0xB
84# define SVM_EXIT_READ_CR12 0xC
85# define SVM_EXIT_READ_CR13 0xD
86# define SVM_EXIT_READ_CR14 0xE
87# define SVM_EXIT_READ_CR15 0xF
88/** Writes to CR0-CR15. */
89# define SVM_EXIT_WRITE_CR0 0x10
90# define SVM_EXIT_WRITE_CR1 0x11
91# define SVM_EXIT_WRITE_CR2 0x12
92# define SVM_EXIT_WRITE_CR3 0x13
93# define SVM_EXIT_WRITE_CR4 0x14
94# define SVM_EXIT_WRITE_CR5 0x15
95# define SVM_EXIT_WRITE_CR6 0x16
96# define SVM_EXIT_WRITE_CR7 0x17
97# define SVM_EXIT_WRITE_CR8 0x18
98# define SVM_EXIT_WRITE_CR9 0x19
99# define SVM_EXIT_WRITE_CR10 0x1A
100# define SVM_EXIT_WRITE_CR11 0x1B
101# define SVM_EXIT_WRITE_CR12 0x1C
102# define SVM_EXIT_WRITE_CR13 0x1D
103# define SVM_EXIT_WRITE_CR14 0x1E
104# define SVM_EXIT_WRITE_CR15 0x1F
105/** Read from DR0-DR15. */
106# define SVM_EXIT_READ_DR0 0x20
107# define SVM_EXIT_READ_DR1 0x21
108# define SVM_EXIT_READ_DR2 0x22
109# define SVM_EXIT_READ_DR3 0x23
110# define SVM_EXIT_READ_DR4 0x24
111# define SVM_EXIT_READ_DR5 0x25
112# define SVM_EXIT_READ_DR6 0x26
113# define SVM_EXIT_READ_DR7 0x27
114# define SVM_EXIT_READ_DR8 0x28
115# define SVM_EXIT_READ_DR9 0x29
116# define SVM_EXIT_READ_DR10 0x2A
117# define SVM_EXIT_READ_DR11 0x2B
118# define SVM_EXIT_READ_DR12 0x2C
119# define SVM_EXIT_READ_DR13 0x2D
120# define SVM_EXIT_READ_DR14 0x2E
121# define SVM_EXIT_READ_DR15 0x2F
122/** Writes to DR0-DR15. */
123# define SVM_EXIT_WRITE_DR0 0x30
124# define SVM_EXIT_WRITE_DR1 0x31
125# define SVM_EXIT_WRITE_DR2 0x32
126# define SVM_EXIT_WRITE_DR3 0x33
127# define SVM_EXIT_WRITE_DR4 0x34
128# define SVM_EXIT_WRITE_DR5 0x35
129# define SVM_EXIT_WRITE_DR6 0x36
130# define SVM_EXIT_WRITE_DR7 0x37
131# define SVM_EXIT_WRITE_DR8 0x38
132# define SVM_EXIT_WRITE_DR9 0x39
133# define SVM_EXIT_WRITE_DR10 0x3A
134# define SVM_EXIT_WRITE_DR11 0x3B
135# define SVM_EXIT_WRITE_DR12 0x3C
136# define SVM_EXIT_WRITE_DR13 0x3D
137# define SVM_EXIT_WRITE_DR14 0x3E
138# define SVM_EXIT_WRITE_DR15 0x3F
139/* Exception 0-31. */
140# define SVM_EXIT_EXCEPTION_0 0x40
141# define SVM_EXIT_EXCEPTION_1 0x41
142# define SVM_EXIT_EXCEPTION_2 0x42
143# define SVM_EXIT_EXCEPTION_3 0x43
144# define SVM_EXIT_EXCEPTION_4 0x44
145# define SVM_EXIT_EXCEPTION_5 0x45
146# define SVM_EXIT_EXCEPTION_6 0x46
147# define SVM_EXIT_EXCEPTION_7 0x47
148# define SVM_EXIT_EXCEPTION_8 0x48
149# define SVM_EXIT_EXCEPTION_9 0x49
150# define SVM_EXIT_EXCEPTION_10 0x4A
151# define SVM_EXIT_EXCEPTION_11 0x4B
152# define SVM_EXIT_EXCEPTION_12 0x4C
153# define SVM_EXIT_EXCEPTION_13 0x4D
154# define SVM_EXIT_EXCEPTION_14 0x4E
155# define SVM_EXIT_EXCEPTION_15 0x4F
156# define SVM_EXIT_EXCEPTION_16 0x50
157# define SVM_EXIT_EXCEPTION_17 0x51
158# define SVM_EXIT_EXCEPTION_18 0x52
159# define SVM_EXIT_EXCEPTION_19 0x53
160# define SVM_EXIT_EXCEPTION_20 0x54
161# define SVM_EXIT_EXCEPTION_21 0x55
162# define SVM_EXIT_EXCEPTION_22 0x56
163# define SVM_EXIT_EXCEPTION_23 0x57
164# define SVM_EXIT_EXCEPTION_24 0x58
165# define SVM_EXIT_EXCEPTION_25 0x59
166# define SVM_EXIT_EXCEPTION_26 0x5A
167# define SVM_EXIT_EXCEPTION_27 0x5B
168# define SVM_EXIT_EXCEPTION_28 0x5C
169# define SVM_EXIT_EXCEPTION_29 0x5D
170# define SVM_EXIT_EXCEPTION_30 0x5E
171# define SVM_EXIT_EXCEPTION_31 0x5F
172/** Physical maskable interrupt. */
173# define SVM_EXIT_INTR 0x60
174/** Non-maskable interrupt. */
175# define SVM_EXIT_NMI 0x61
176/** System Management interrupt. */
177# define SVM_EXIT_SMI 0x62
178/** Physical INIT signal. */
179# define SVM_EXIT_INIT 0x63
180/** Virtual interrupt. */
181# define SVM_EXIT_VINTR 0x64
182/** Write to CR0 that changed any bits other than CR0.TS or CR0.MP. */
183# define SVM_EXIT_CR0_SEL_WRITE 0x65
184/** IDTR read. */
185# define SVM_EXIT_IDTR_READ 0x66
186/** GDTR read. */
187# define SVM_EXIT_GDTR_READ 0x67
188/** LDTR read. */
189# define SVM_EXIT_LDTR_READ 0x68
190/** TR read. */
191# define SVM_EXIT_TR_READ 0x69
192/** IDTR write. */
193# define SVM_EXIT_IDTR_WRITE 0x6A
194/** GDTR write. */
195# define SVM_EXIT_GDTR_WRITE 0x6B
196/** LDTR write. */
197# define SVM_EXIT_LDTR_WRITE 0x6C
198/** TR write. */
199# define SVM_EXIT_TR_WRITE 0x6D
200/** RDTSC instruction. */
201# define SVM_EXIT_RDTSC 0x6E
202/** RDPMC instruction. */
203# define SVM_EXIT_RDPMC 0x6F
204/** PUSHF instruction. */
205# define SVM_EXIT_PUSHF 0x70
206/** POPF instruction. */
207# define SVM_EXIT_POPF 0x71
208/** CPUID instruction. */
209# define SVM_EXIT_CPUID 0x72
210/** RSM instruction. */
211# define SVM_EXIT_RSM 0x73
212/** IRET instruction. */
213# define SVM_EXIT_IRET 0x74
214/** software interrupt (INTn instructions). */
215# define SVM_EXIT_SWINT 0x75
216/** INVD instruction. */
217# define SVM_EXIT_INVD 0x76
218/** PAUSE instruction. */
219# define SVM_EXIT_PAUSE 0x77
220/** HLT instruction. */
221# define SVM_EXIT_HLT 0x78
222/** INVLPG instructions. */
223# define SVM_EXIT_INVLPG 0x79
224/** INVLPGA instruction. */
225# define SVM_EXIT_INVLPGA 0x7A
226/** IN or OUT accessing protected port (the EXITINFO1 field provides more information). */
227# define SVM_EXIT_IOIO 0x7B
228/** RDMSR or WRMSR access to protected MSR. */
229# define SVM_EXIT_MSR 0x7C
230/** task switch. */
231# define SVM_EXIT_TASK_SWITCH 0x7D
232/** FP legacy handling enabled, and processor is frozen in an x87/mmx instruction waiting for an interrupt. */
233# define SVM_EXIT_FERR_FREEZE 0x7E
234/** Shutdown. */
235# define SVM_EXIT_SHUTDOWN 0x7F
236/** VMRUN instruction. */
237# define SVM_EXIT_VMRUN 0x80
238/** VMMCALL instruction. */
239# define SVM_EXIT_VMMCALL 0x81
240/** VMLOAD instruction. */
241# define SVM_EXIT_VMLOAD 0x82
242/** VMSAVE instruction. */
243# define SVM_EXIT_VMSAVE 0x83
244/** STGI instruction. */
245# define SVM_EXIT_STGI 0x84
246/** CLGI instruction. */
247# define SVM_EXIT_CLGI 0x85
248/** SKINIT instruction. */
249# define SVM_EXIT_SKINIT 0x86
250/** RDTSCP instruction. */
251# define SVM_EXIT_RDTSCP 0x87
252/** ICEBP instruction. */
253# define SVM_EXIT_ICEBP 0x88
254/** WBINVD instruction. */
255# define SVM_EXIT_WBINVD 0x89
256/** MONITOR instruction. */
257# define SVM_EXIT_MONITOR 0x8A
258/** MWAIT instruction. */
259# define SVM_EXIT_MWAIT 0x8B
260/** MWAIT instruction, when armed. */
261# define SVM_EXIT_MWAIT_ARMED 0x8C
262/** XSETBV instruction. */
263# define SVM_EXIT_XSETBV 0x8D
264/** Nested paging: host-level page fault occurred (EXITINFO1 contains fault errorcode; EXITINFO2 contains the guest physical address causing the fault). */
265# define SVM_EXIT_NPF 0x400
266/** AVIC: Virtual IPI delivery not completed. */
267# define SVM_EXIT_AVIC_INCOMPLETE_IPI 0x401
268/** AVIC: Attempted access by guest to a vAPIC register not handled by AVIC
269 * hardware. */
270# define SVM_EXIT_AVIC_NOACCEL 0x402
271/** The maximum possible exit value. */
272# define SVM_EXIT_MAX (SVM_EXIT_AVIC_NOACCEL)
273/** @} */
274#endif /* !IN_REM_R3*/
275
276
277/** @name SVMVMCB.u64ExitInfo2 for task switches
278 * @{
279 */
280/** Set to 1 if the task switch was caused by an IRET; else cleared to 0. */
281#define SVM_EXIT2_TASK_SWITCH_IRET RT_BIT_64(36)
282/** Set to 1 if the task switch was caused by a far jump; else cleared to 0. */
283#define SVM_EXIT2_TASK_SWITCH_JMP RT_BIT_64(38)
284/** Set to 1 if the task switch has an error code; else cleared to 0. */
285#define SVM_EXIT2_TASK_SWITCH_HAS_ERROR_CODE RT_BIT_64(44)
286/** The value of EFLAGS.RF that would be saved in the outgoing TSS if the task switch were not intercepted. */
287#define SVM_EXIT2_TASK_SWITCH_EFLAGS_RF RT_BIT_64(48)
288/** @} */
289
290/** @name SVMVMCB.u64ExitInfo1 for MSR accesses
291 * @{
292 */
293/** The access was a read MSR. */
294#define SVM_EXIT1_MSR_READ 0x0
295/** The access was a write MSR. */
296#define SVM_EXIT1_MSR_WRITE 0x1
297/** @} */
298
299/** @name SVMVMCB.u64ExitInfo1 for Mov CRx accesses.
300 * @{
301 */
302/** The mask of whether the access was via a Mov CRx instruction. */
303#define SVM_EXIT1_MOV_CRX_MASK RT_BIT_64(63)
304/** The mask for the GPR number of the Mov CRx instruction. */
305#define SVM_EXIT1_MOV_CRX_GPR_NUMBER 0xf
306/** @} */
307
308/** @name SVMVMCB.u64ExitInfo1 for Mov DRx accesses.
309 * @{
310 */
311/** The mask for the GPR number of the Mov DRx instruction. */
312#define SVM_EXIT1_MOV_DRX_GPR_NUMBER 0xf
313/** @} */
314
315/** @name SVMVMCB.ctrl.u64InterceptCtrl
316 * @{
317 */
318/** Intercept INTR (physical maskable interrupt). */
319#define SVM_CTRL_INTERCEPT_INTR RT_BIT_64(0)
320/** Intercept NMI. */
321#define SVM_CTRL_INTERCEPT_NMI RT_BIT_64(1)
322/** Intercept SMI. */
323#define SVM_CTRL_INTERCEPT_SMI RT_BIT_64(2)
324/** Intercept INIT. */
325#define SVM_CTRL_INTERCEPT_INIT RT_BIT_64(3)
326/** Intercept VINTR (virtual maskable interrupt). */
327#define SVM_CTRL_INTERCEPT_VINTR RT_BIT_64(4)
328/** Intercept CR0 writes that change bits other than CR0.TS or CR0.MP */
329#define SVM_CTRL_INTERCEPT_CR0_SEL_WRITES RT_BIT_64(5)
330/** Intercept reads of IDTR. */
331#define SVM_CTRL_INTERCEPT_IDTR_READS RT_BIT_64(6)
332/** Intercept reads of GDTR. */
333#define SVM_CTRL_INTERCEPT_GDTR_READS RT_BIT_64(7)
334/** Intercept reads of LDTR. */
335#define SVM_CTRL_INTERCEPT_LDTR_READS RT_BIT_64(8)
336/** Intercept reads of TR. */
337#define SVM_CTRL_INTERCEPT_TR_READS RT_BIT_64(9)
338/** Intercept writes of IDTR. */
339#define SVM_CTRL_INTERCEPT_IDTR_WRITES RT_BIT_64(10)
340/** Intercept writes of GDTR. */
341#define SVM_CTRL_INTERCEPT_GDTR_WRITES RT_BIT_64(11)
342/** Intercept writes of LDTR. */
343#define SVM_CTRL_INTERCEPT_LDTR_WRITES RT_BIT_64(12)
344/** Intercept writes of TR. */
345#define SVM_CTRL_INTERCEPT_TR_WRITES RT_BIT_64(13)
346/** Intercept RDTSC instruction. */
347#define SVM_CTRL_INTERCEPT_RDTSC RT_BIT_64(14)
348/** Intercept RDPMC instruction. */
349#define SVM_CTRL_INTERCEPT_RDPMC RT_BIT_64(15)
350/** Intercept PUSHF instruction. */
351#define SVM_CTRL_INTERCEPT_PUSHF RT_BIT_64(16)
352/** Intercept POPF instruction. */
353#define SVM_CTRL_INTERCEPT_POPF RT_BIT_64(17)
354/** Intercept CPUID instruction. */
355#define SVM_CTRL_INTERCEPT_CPUID RT_BIT_64(18)
356/** Intercept RSM instruction. */
357#define SVM_CTRL_INTERCEPT_RSM RT_BIT_64(19)
358/** Intercept IRET instruction. */
359#define SVM_CTRL_INTERCEPT_IRET RT_BIT_64(20)
360/** Intercept INTn instruction. */
361#define SVM_CTRL_INTERCEPT_INTN RT_BIT_64(21)
362/** Intercept INVD instruction. */
363#define SVM_CTRL_INTERCEPT_INVD RT_BIT_64(22)
364/** Intercept PAUSE instruction. */
365#define SVM_CTRL_INTERCEPT_PAUSE RT_BIT_64(23)
366/** Intercept HLT instruction. */
367#define SVM_CTRL_INTERCEPT_HLT RT_BIT_64(24)
368/** Intercept INVLPG instruction. */
369#define SVM_CTRL_INTERCEPT_INVLPG RT_BIT_64(25)
370/** Intercept INVLPGA instruction. */
371#define SVM_CTRL_INTERCEPT_INVLPGA RT_BIT_64(26)
372/** IOIO_PROT Intercept IN/OUT accesses to selected ports. */
373#define SVM_CTRL_INTERCEPT_IOIO_PROT RT_BIT_64(27)
374/** MSR_PROT Intercept RDMSR or WRMSR accesses to selected MSRs. */
375#define SVM_CTRL_INTERCEPT_MSR_PROT RT_BIT_64(28)
376/** Intercept task switches. */
377#define SVM_CTRL_INTERCEPT_TASK_SWITCH RT_BIT_64(29)
378/** FERR_FREEZE: intercept processor "freezing" during legacy FERR handling. */
379#define SVM_CTRL_INTERCEPT_FERR_FREEZE RT_BIT_64(30)
380/** Intercept shutdown events. */
381#define SVM_CTRL_INTERCEPT_SHUTDOWN RT_BIT_64(31)
382/** Intercept VMRUN instruction. */
383#define SVM_CTRL_INTERCEPT_VMRUN RT_BIT_64(32 + 0)
384/** Intercept VMMCALL instruction. */
385#define SVM_CTRL_INTERCEPT_VMMCALL RT_BIT_64(32 + 1)
386/** Intercept VMLOAD instruction. */
387#define SVM_CTRL_INTERCEPT_VMLOAD RT_BIT_64(32 + 2)
388/** Intercept VMSAVE instruction. */
389#define SVM_CTRL_INTERCEPT_VMSAVE RT_BIT_64(32 + 3)
390/** Intercept STGI instruction. */
391#define SVM_CTRL_INTERCEPT_STGI RT_BIT_64(32 + 4)
392/** Intercept CLGI instruction. */
393#define SVM_CTRL_INTERCEPT_CLGI RT_BIT_64(32 + 5)
394/** Intercept SKINIT instruction. */
395#define SVM_CTRL_INTERCEPT_SKINIT RT_BIT_64(32 + 6)
396/** Intercept RDTSCP instruction. */
397#define SVM_CTRL_INTERCEPT_RDTSCP RT_BIT_64(32 + 7)
398/** Intercept ICEBP instruction. */
399#define SVM_CTRL_INTERCEPT_ICEBP RT_BIT_64(32 + 8)
400/** Intercept WBINVD instruction. */
401#define SVM_CTRL_INTERCEPT_WBINVD RT_BIT_64(32 + 9)
402/** Intercept MONITOR instruction. */
403#define SVM_CTRL_INTERCEPT_MONITOR RT_BIT_64(32 + 10)
404/** Intercept MWAIT instruction unconditionally. */
405#define SVM_CTRL_INTERCEPT_MWAIT RT_BIT_64(32 + 11)
406/** Intercept MWAIT instruction when armed. */
407#define SVM_CTRL_INTERCEPT_MWAIT_ARMED RT_BIT_64(32 + 12)
408/** Intercept XSETBV instruction. */
409#define SVM_CTRL_INTERCEPT_XSETBV RT_BIT_64(32 + 13)
410/* Bit 14 - Reserved, SBZ. */
411/** Intercept EFER writes after guest instruction finishes. */
412#define SVM_CTRL_INTERCEPT_EFER_WRITES_TRAP RT_BIT_64(32 + 15)
413/** Intercept CR0 writes after guest instruction finishes. */
414#define SVM_CTRL_INTERCEPT_CR0_WRITES_TRAP RT_BIT_64(32 + 16)
415/** Intercept CR0 writes after guest instruction finishes. */
416#define SVM_CTRL_INTERCEPT_CR1_WRITES_TRAP RT_BIT_64(32 + 17)
417/** Intercept CR0 writes after guest instruction finishes. */
418#define SVM_CTRL_INTERCEPT_CR2_WRITES_TRAP RT_BIT_64(32 + 18)
419/** Intercept CR0 writes after guest instruction finishes. */
420#define SVM_CTRL_INTERCEPT_CR3_WRITES_TRAP RT_BIT_64(32 + 19)
421/** Intercept CR0 writes after guest instruction finishes. */
422#define SVM_CTRL_INTERCEPT_CR4_WRITES_TRAP RT_BIT_64(32 + 20)
423/** Intercept CR0 writes after guest instruction finishes. */
424#define SVM_CTRL_INTERCEPT_CR5_WRITES_TRAP RT_BIT_64(32 + 21)
425/** Intercept CR0 writes after guest instruction finishes. */
426#define SVM_CTRL_INTERCEPT_CR6_WRITES_TRAP RT_BIT_64(32 + 22)
427/** Intercept CR0 writes after guest instruction finishes. */
428#define SVM_CTRL_INTERCEPT_CR7_WRITES_TRAP RT_BIT_64(32 + 23)
429/** Intercept CR0 writes after guest instruction finishes. */
430#define SVM_CTRL_INTERCEPT_CR8_WRITES_TRAP RT_BIT_64(32 + 24)
431/** Intercept CR0 writes after guest instruction finishes. */
432#define SVM_CTRL_INTERCEPT_CR9_WRITES_TRAP RT_BIT_64(32 + 25)
433/** Intercept CR0 writes after guest instruction finishes. */
434#define SVM_CTRL_INTERCEPT_CR10_WRITES_TRAP RT_BIT_64(32 + 26)
435/** Intercept CR0 writes after guest instruction finishes. */
436#define SVM_CTRL_INTERCEPT_CR11_WRITES_TRAP RT_BIT_64(32 + 27)
437/** Intercept CR0 writes after guest instruction finishes. */
438#define SVM_CTRL_INTERCEPT_CR12_WRITES_TRAP RT_BIT_64(32 + 28)
439/** Intercept CR0 writes after guest instruction finishes. */
440#define SVM_CTRL_INTERCEPT_CR13_WRITES_TRAP RT_BIT_64(32 + 29)
441/** Intercept CR0 writes after guest instruction finishes. */
442#define SVM_CTRL_INTERCEPT_CR14_WRITES_TRAP RT_BIT_64(32 + 30)
443/** Intercept CR0 writes after guest instruction finishes. */
444#define SVM_CTRL_INTERCEPT_CR15_WRITES_TRAP RT_BIT_64(32 + 31)
445/** @} */
446
447/** @name SVMINTCTRL.u3Type
448 * @{
449 */
450/** External or virtual interrupt. */
451#define SVM_EVENT_EXTERNAL_IRQ 0
452/** Non-maskable interrupt. */
453#define SVM_EVENT_NMI 2
454/** Exception; fault or trap. */
455#define SVM_EVENT_EXCEPTION 3
456/** Software interrupt. */
457#define SVM_EVENT_SOFTWARE_INT 4
458/** @} */
459
460/** @name SVMVMCB.ctrl.TLBCtrl.n.u8TLBFlush
461 * @{
462 */
463/** Flush nothing. */
464#define SVM_TLB_FLUSH_NOTHING 0
465/** Flush entire TLB (host+guest entries) */
466#define SVM_TLB_FLUSH_ENTIRE 1
467/** Flush this guest's TLB entries (by ASID) */
468#define SVM_TLB_FLUSH_SINGLE_CONTEXT 3
469/** Flush this guest's non-global TLB entries (by ASID) */
470#define SVM_TLB_FLUSH_SINGLE_CONTEXT_RETAIN_GLOBALS 7
471/** @} */
472
473/**
474 * SVM selector/segment register type.
475 */
476typedef struct
477{
478 uint16_t u16Sel;
479 uint16_t u16Attr;
480 uint32_t u32Limit;
481 uint64_t u64Base; /**< Only lower 32 bits are implemented for CS, DS, ES & SS. */
482} SVMSELREG;
483AssertCompileSize(SVMSELREG, 16);
484/** Pointer to the SVMSELREG struct. */
485typedef SVMSELREG *PSVMSELREG;
486/** Pointer to a const SVMSELREG struct. */
487typedef const SVMSELREG *PCSVMSELREG;
488
489/**
490 * SVM GDTR/IDTR type.
491 */
492typedef struct
493{
494 uint16_t u16Reserved1;
495 uint16_t u16Reserved2;
496 uint32_t u32Limit; /**< Only lower 16 bits are implemented. */
497 uint64_t u64Base;
498} SVMXDTR;
499AssertCompileSize(SVMXDTR, 16);
500typedef SVMXDTR SVMIDTR;
501typedef SVMXDTR SVMGDTR;
502/** Pointer to the SVMXDTR struct. */
503typedef SVMXDTR *PSVMXDTR;
504/** Pointer to a const SVMXDTR struct. */
505typedef const SVMXDTR *PCSVMXDTR;
506
507/**
508 * SVM Event injection structure (EVENTINJ and EXITINTINFO).
509 */
510typedef union
511{
512 struct
513 {
514 uint32_t u8Vector : 8;
515 uint32_t u3Type : 3;
516 uint32_t u1ErrorCodeValid : 1;
517 uint32_t u19Reserved : 19;
518 uint32_t u1Valid : 1;
519 uint32_t u32ErrorCode : 32;
520 } n;
521 uint64_t u;
522} SVMEVENT;
523/** Pointer to the SVMEVENT union. */
524typedef SVMEVENT *PSVMEVENT;
525/** Pointer to a const SVMEVENT union. */
526typedef const SVMEVENT *PCSVMEVENT;
527
528/** Gets the event type given an SVMEVENT parameter. */
529#define SVM_EVENT_GET_TYPE(a_SvmEvent) (((a_SvmEvent) >> 8) & 7)
530
531/**
532 * SVM Interrupt control structure (Virtual Interrupt Control).
533 */
534typedef union
535{
536 struct
537 {
538 uint32_t u8VTPR : 8; /* V_TPR */
539 uint32_t u1VIrqPending : 1; /* V_IRQ */
540 uint32_t u1VGif : 1; /* VGIF */
541 uint32_t u6Reserved0 : 6;
542 uint32_t u4VIntrPrio : 4; /* V_INTR_PRIO */
543 uint32_t u1IgnoreTPR : 1; /* V_IGN_TPR */
544 uint32_t u3Reserved : 3;
545 uint32_t u1VIntrMasking : 1; /* V_INTR_MASKING */
546 uint32_t u6Reserved1 : 6;
547 uint32_t u1AvicEnable : 1;
548 uint32_t u8VIntrVector : 8; /* V_INTR_VECTOR */
549 uint32_t u24Reserved : 24;
550 } n;
551 uint64_t u;
552} SVMINTCTRL;
553
554/**
555 * SVM TLB control structure.
556 */
557typedef union
558{
559 struct
560 {
561 uint32_t u32ASID : 32;
562 uint32_t u8TLBFlush : 8;
563 uint32_t u24Reserved : 24;
564 } n;
565 uint64_t u;
566} SVMTLBCTRL;
567
568/**
569 * SVM IOIO exit info. structure (EXITINFO1 for IOIO intercepts).
570 */
571typedef union
572{
573 struct
574 {
575 uint32_t u1Type : 1; /**< Bit 0: 0 = out, 1 = in */
576 uint32_t u1Reserved : 1; /**< Bit 1: Reserved */
577 uint32_t u1STR : 1; /**< Bit 2: String I/O (1) or not (0). */
578 uint32_t u1REP : 1; /**< Bit 3: Repeat prefixed string I/O. */
579 uint32_t u1OP8 : 1; /**< Bit 4: 8-bit operand. */
580 uint32_t u1OP16 : 1; /**< Bit 5: 16-bit operand. */
581 uint32_t u1OP32 : 1; /**< Bit 6: 32-bit operand. */
582 uint32_t u1ADDR16 : 1; /**< Bit 7: 16-bit address size. */
583 uint32_t u1ADDR32 : 1; /**< Bit 8: 32-bit address size. */
584 uint32_t u1ADDR64 : 1; /**< Bit 9: 64-bit address size. */
585 uint32_t u3SEG : 3; /**< BITS 12:10: Effective segment number. Added w/ decode assist in APM v3.17. */
586 uint32_t u3Reserved : 3;
587 uint32_t u16Port : 16; /**< Bits 31:16: Port number. */
588 } n;
589 uint32_t u;
590} SVMIOIOEXITINFO;
591/** Pointer to an SVM IOIO exit info. structure. */
592typedef SVMIOIOEXITINFO *PSVMIOIOEXITINFO;
593/** Pointer to a const SVM IOIO exit info. structure. */
594typedef const SVMIOIOEXITINFO *PCSVMIOIOEXITINFO;
595
596/** 8-bit IO transfer. */
597#define SVM_IOIO_8_BIT_OP RT_BIT_32(4)
598/** 16-bit IO transfer. */
599#define SVM_IOIO_16_BIT_OP RT_BIT_32(5)
600/** 32-bit IO transfer. */
601#define SVM_IOIO_32_BIT_OP RT_BIT_32(6)
602/** Number of bits to shift right to get the operand sizes. */
603#define SVM_IOIO_OP_SIZE_SHIFT 4
604/** Mask of all possible IO transfer sizes. */
605#define SVM_IOIO_OP_SIZE_MASK (SVM_IOIO_8_BIT_OP | SVM_IOIO_16_BIT_OP | SVM_IOIO_32_BIT_OP)
606/** 16-bit address for the IO buffer. */
607#define SVM_IOIO_16_BIT_ADDR RT_BIT_32(7)
608/** 32-bit address for the IO buffer. */
609#define SVM_IOIO_32_BIT_ADDR RT_BIT_32(8)
610/** 64-bit address for the IO buffer. */
611#define SVM_IOIO_64_BIT_ADDR RT_BIT_32(9)
612/** Number of bits to shift right to get the address sizes. */
613#define SVM_IOIO_ADDR_SIZE_SHIFT 7
614/** Mask of all the IO address sizes. */
615#define SVM_IOIO_ADDR_SIZE_MASK (SVM_IOIO_16_BIT_ADDR | SVM_IOIO_32_BIT_ADDR | SVM_IOIO_64_BIT_ADDR)
616/** Number of bits to shift right to get the IO port number. */
617#define SVM_IOIO_PORT_SHIFT 16
618/** IO write. */
619#define SVM_IOIO_WRITE 0
620/** IO read. */
621#define SVM_IOIO_READ 1
622/**
623 * SVM IOIO transfer type.
624 */
625typedef enum
626{
627 SVMIOIOTYPE_OUT = SVM_IOIO_WRITE,
628 SVMIOIOTYPE_IN = SVM_IOIO_READ
629} SVMIOIOTYPE;
630
631/**
632 * SVM AVIC.
633 */
634typedef union
635{
636 struct
637 {
638 uint64_t u12Reserved1 : 12;
639 uint64_t u40Addr : 40;
640 uint64_t u12Reserved2 : 12;
641 } n;
642 uint64_t u;
643} SVMAVIC;
644AssertCompileSize(SVMAVIC, 8);
645
646/**
647 * SVM AVIC PHYSICAL_TABLE pointer.
648 */
649typedef union
650{
651 struct
652 {
653 uint64_t u8LastGuestCoreId : 8;
654 uint64_t u4Reserved : 4;
655 uint64_t u40Addr : 40;
656 uint64_t u12Reserved : 12;
657 } n;
658 uint64_t u;
659} SVMAVICPHYS;
660AssertCompileSize(SVMAVICPHYS, 8);
661
662/**
663 * SVM Nested Paging struct.
664 */
665typedef union
666{
667 struct
668 {
669 uint32_t u1NestedPaging : 1;
670 uint32_t u1Sev : 1;
671 uint32_t u1SevEs : 1;
672 uint32_t u29Reserved0 : 29;
673 } n;
674 uint64_t u;
675} SVMNP;
676AssertCompileSize(SVMNP, 8);
677
678/**
679 * SVM Interrupt shadow struct.
680 */
681typedef union
682{
683 struct
684 {
685 uint32_t u1IntShadow : 1;
686 uint32_t u1GuestIntMask : 1;
687 uint32_t u30Reserved0 : 30;
688 } n;
689 uint64_t u;
690} SVMINTSHADOW;
691AssertCompileSize(SVMINTSHADOW, 8);
692
693/**
694 * SVM LBR virtualization struct.
695 */
696typedef union
697{
698 struct
699 {
700 uint32_t u1LbrVirt : 1;
701 uint32_t u1VirtVmsaveVmload : 1;
702 uint32_t u30Reserved1 : 30;
703 } n;
704 uint64_t u;
705} SVMLBRVIRT;
706AssertCompileSize(SVMLBRVIRT, 8);
707
708/**
709 * SVM VMCB control area.
710 */
711#pragma pack(1)
712typedef struct
713{
714 /** Offset 0x00 - Intercept reads of CR0-CR15. */
715 uint16_t u16InterceptRdCRx;
716 /** Offset 0x02 - Intercept writes to CR0-CR15. */
717 uint16_t u16InterceptWrCRx;
718 /** Offset 0x04 - Intercept reads of DR0-DR15. */
719 uint16_t u16InterceptRdDRx;
720 /** Offset 0x06 - Intercept writes to DR0-DR15. */
721 uint16_t u16InterceptWrDRx;
722 /** Offset 0x08 - Intercept exception vectors 0-31. */
723 uint32_t u32InterceptXcpt;
724 /** Offset 0x0c - Intercept control. */
725 uint64_t u64InterceptCtrl;
726 /** Offset 0x14-0x3f - Reserved. */
727 uint8_t u8Reserved[0x3c - 0x14];
728 /** Offset 0x3c - PAUSE filter threshold. */
729 uint16_t u16PauseFilterThreshold;
730 /** Offset 0x3e - PAUSE intercept filter count. */
731 uint16_t u16PauseFilterCount;
732 /** Offset 0x40 - Physical address of IOPM. */
733 uint64_t u64IOPMPhysAddr;
734 /** Offset 0x48 - Physical address of MSRPM. */
735 uint64_t u64MSRPMPhysAddr;
736 /** Offset 0x50 - TSC Offset. */
737 uint64_t u64TSCOffset;
738 /** Offset 0x58 - TLB control field. */
739 SVMTLBCTRL TLBCtrl;
740 /** Offset 0x60 - Interrupt control field. */
741 SVMINTCTRL IntCtrl;
742 /** Offset 0x68 - Interrupt shadow. */
743 SVMINTSHADOW IntShadow;
744 /** Offset 0x70 - Exit code. */
745 uint64_t u64ExitCode;
746 /** Offset 0x78 - Exit info 1. */
747 uint64_t u64ExitInfo1;
748 /** Offset 0x80 - Exit info 2. */
749 uint64_t u64ExitInfo2;
750 /** Offset 0x88 - Exit Interrupt info. */
751 SVMEVENT ExitIntInfo;
752 /** Offset 0x90 - Nested Paging. */
753 SVMNP NestedPaging;
754 /** Offset 0x98 - AVIC APIC BAR. */
755 SVMAVIC AvicBar;
756 /** Offset 0xa0-0xa7 - Reserved. */
757 uint8_t u8Reserved2[0xA8 - 0xA0];
758 /** Offset 0xa8 - Event injection. */
759 SVMEVENT EventInject;
760 /** Offset 0xb0 - Host CR3 for nested paging. */
761 uint64_t u64NestedPagingCR3;
762 /** Offset 0xb8 - LBR Virtualization. */
763 SVMLBRVIRT LbrVirt;
764 /** Offset 0xc0 - VMCB Clean Bits. */
765 uint32_t u32VmcbCleanBits;
766 uint32_t u32Reserved0;
767 /** Offset 0xc8 - Next sequential instruction pointer. */
768 uint64_t u64NextRIP;
769 /** Offset 0xd0 - Number of bytes fetched. */
770 uint8_t cbInstrFetched;
771 /** Offset 0xd1 - Fetched bytes. */
772 uint8_t abInstr[15];
773 /** Offset 0xe0 - AVIC APIC_BACKING_PAGE pointer. */
774 SVMAVIC AvicBackingPagePtr;
775 /** Offset 0xe8-0xef - Reserved. */
776 uint8_t u8Reserved3[0xF0 - 0xE8];
777 /** Offset 0xf0 - AVIC LOGICAL_TABLE pointer. */
778 SVMAVIC AvicLogicalTablePtr;
779 /** Offset 0xf8 - AVIC PHYSICAL_TABLE pointer. */
780 SVMAVICPHYS AvicPhysicalTablePtr;
781} SVMVMCBCTRL;
782#pragma pack()
783/** Pointer to the SVMVMCBSTATESAVE structure. */
784typedef SVMVMCBCTRL *PSVMVMCBCTRL;
785/** Pointer to a const SVMVMCBSTATESAVE structure. */
786typedef const SVMVMCBCTRL *PCSVMVMCBCTRL;
787AssertCompileSize(SVMVMCBCTRL, 0x100);
788AssertCompileMemberOffset(SVMVMCBCTRL, u16InterceptRdCRx, 0x00);
789AssertCompileMemberOffset(SVMVMCBCTRL, u16InterceptWrCRx, 0x02);
790AssertCompileMemberOffset(SVMVMCBCTRL, u16InterceptRdDRx, 0x04);
791AssertCompileMemberOffset(SVMVMCBCTRL, u16InterceptWrDRx, 0x06);
792AssertCompileMemberOffset(SVMVMCBCTRL, u32InterceptXcpt, 0x08);
793AssertCompileMemberOffset(SVMVMCBCTRL, u64InterceptCtrl, 0x0c);
794AssertCompileMemberOffset(SVMVMCBCTRL, u8Reserved, 0x14);
795AssertCompileMemberOffset(SVMVMCBCTRL, u16PauseFilterThreshold, 0x3c);
796AssertCompileMemberOffset(SVMVMCBCTRL, u16PauseFilterCount, 0x3e);
797AssertCompileMemberOffset(SVMVMCBCTRL, u64IOPMPhysAddr, 0x40);
798AssertCompileMemberOffset(SVMVMCBCTRL, u64MSRPMPhysAddr, 0x48);
799AssertCompileMemberOffset(SVMVMCBCTRL, u64TSCOffset, 0x50);
800AssertCompileMemberOffset(SVMVMCBCTRL, TLBCtrl, 0x58);
801AssertCompileMemberOffset(SVMVMCBCTRL, IntCtrl, 0x60);
802AssertCompileMemberOffset(SVMVMCBCTRL, IntShadow, 0x68);
803AssertCompileMemberOffset(SVMVMCBCTRL, u64ExitCode, 0x70);
804AssertCompileMemberOffset(SVMVMCBCTRL, u64ExitInfo1, 0x78);
805AssertCompileMemberOffset(SVMVMCBCTRL, u64ExitInfo2, 0x80);
806AssertCompileMemberOffset(SVMVMCBCTRL, ExitIntInfo, 0x88);
807AssertCompileMemberOffset(SVMVMCBCTRL, NestedPaging, 0x90);
808AssertCompileMemberOffset(SVMVMCBCTRL, AvicBar, 0x98);
809AssertCompileMemberOffset(SVMVMCBCTRL, u8Reserved2, 0xa0);
810AssertCompileMemberOffset(SVMVMCBCTRL, EventInject, 0xa8);
811AssertCompileMemberOffset(SVMVMCBCTRL, u64NestedPagingCR3, 0xb0);
812AssertCompileMemberOffset(SVMVMCBCTRL, LbrVirt, 0xb8);
813AssertCompileMemberOffset(SVMVMCBCTRL, u32VmcbCleanBits, 0xc0);
814AssertCompileMemberOffset(SVMVMCBCTRL, u64NextRIP, 0xc8);
815AssertCompileMemberOffset(SVMVMCBCTRL, cbInstrFetched, 0xd0);
816AssertCompileMemberOffset(SVMVMCBCTRL, abInstr, 0xd1);
817AssertCompileMemberOffset(SVMVMCBCTRL, AvicBackingPagePtr, 0xe0);
818AssertCompileMemberOffset(SVMVMCBCTRL, u8Reserved3, 0xe8);
819AssertCompileMemberOffset(SVMVMCBCTRL, AvicLogicalTablePtr, 0xf0);
820AssertCompileMemberOffset(SVMVMCBCTRL, AvicPhysicalTablePtr, 0xf8);
821
822/**
823 * SVM VMCB state save area.
824 */
825#pragma pack(1)
826typedef struct
827{
828 /** Offset 0x400 - Guest ES register + hidden parts. */
829 SVMSELREG ES;
830 /** Offset 0x410 - Guest CS register + hidden parts. */
831 SVMSELREG CS;
832 /** Offset 0x420 - Guest SS register + hidden parts. */
833 SVMSELREG SS;
834 /** Offset 0x430 - Guest DS register + hidden parts. */
835 SVMSELREG DS;
836 /** Offset 0x440 - Guest FS register + hidden parts. */
837 SVMSELREG FS;
838 /** Offset 0x450 - Guest GS register + hidden parts. */
839 SVMSELREG GS;
840 /** Offset 0x460 - Guest GDTR register. */
841 SVMGDTR GDTR;
842 /** Offset 0x470 - Guest LDTR register + hidden parts. */
843 SVMSELREG LDTR;
844 /** Offset 0x480 - Guest IDTR register. */
845 SVMIDTR IDTR;
846 /** Offset 0x490 - Guest TR register + hidden parts. */
847 SVMSELREG TR;
848 /** Offset 0x4A0-0x4CA - Reserved. */
849 uint8_t u8Reserved4[0x4CB - 0x4A0];
850 /** Offset 0x4CB - CPL. */
851 uint8_t u8CPL;
852 /** Offset 0x4CC-0x4CF - Reserved. */
853 uint8_t u8Reserved5[0x4D0 - 0x4CC];
854 /** Offset 0x4D0 - EFER. */
855 uint64_t u64EFER;
856 /** Offset 0x4D8-0x547 - Reserved. */
857 uint8_t u8Reserved6[0x548 - 0x4D8];
858 /** Offset 0x548 - CR4. */
859 uint64_t u64CR4;
860 /** Offset 0x550 - CR3. */
861 uint64_t u64CR3;
862 /** Offset 0x558 - CR0. */
863 uint64_t u64CR0;
864 /** Offset 0x560 - DR7. */
865 uint64_t u64DR7;
866 /** Offset 0x568 - DR6. */
867 uint64_t u64DR6;
868 /** Offset 0x570 - RFLAGS. */
869 uint64_t u64RFlags;
870 /** Offset 0x578 - RIP. */
871 uint64_t u64RIP;
872 /** Offset 0x580-0x5D7 - Reserved. */
873 uint8_t u8Reserved7[0x5D8 - 0x580];
874 /** Offset 0x5D8 - RSP. */
875 uint64_t u64RSP;
876 /** Offset 0x5E0-0x5F7 - Reserved. */
877 uint8_t u8Reserved8[0x5F8 - 0x5E0];
878 /** Offset 0x5F8 - RAX. */
879 uint64_t u64RAX;
880 /** Offset 0x600 - STAR. */
881 uint64_t u64STAR;
882 /** Offset 0x608 - LSTAR. */
883 uint64_t u64LSTAR;
884 /** Offset 0x610 - CSTAR. */
885 uint64_t u64CSTAR;
886 /** Offset 0x618 - SFMASK. */
887 uint64_t u64SFMASK;
888 /** Offset 0x620 - KernelGSBase. */
889 uint64_t u64KernelGSBase;
890 /** Offset 0x628 - SYSENTER_CS. */
891 uint64_t u64SysEnterCS;
892 /** Offset 0x630 - SYSENTER_ESP. */
893 uint64_t u64SysEnterESP;
894 /** Offset 0x638 - SYSENTER_EIP. */
895 uint64_t u64SysEnterEIP;
896 /** Offset 0x640 - CR2. */
897 uint64_t u64CR2;
898 /** Offset 0x648-0x667 - Reserved. */
899 uint8_t u8Reserved9[0x668 - 0x648];
900 /** Offset 0x668 - G_PAT. */
901 uint64_t u64GPAT;
902 /** Offset 0x670 - DBGCTL. */
903 uint64_t u64DBGCTL;
904 /** Offset 0x678 - BR_FROM. */
905 uint64_t u64BR_FROM;
906 /** Offset 0x680 - BR_TO. */
907 uint64_t u64BR_TO;
908 /** Offset 0x688 - LASTEXCPFROM. */
909 uint64_t u64LASTEXCPFROM;
910 /** Offset 0x690 - LASTEXCPTO. */
911 uint64_t u64LASTEXCPTO;
912} SVMVMCBSTATESAVE;
913#pragma pack()
914/** Pointer to the SVMVMCBSTATESAVE structure. */
915typedef SVMVMCBSTATESAVE *PSVMVMCBSTATESAVE;
916/** Pointer to a const SVMVMCBSTATESAVE structure. */
917typedef const SVMVMCBSTATESAVE *PCSVMVMCBSTATESAVE;
918AssertCompileSize(SVMVMCBSTATESAVE, 0x298);
919AssertCompileMemberOffset(SVMVMCBSTATESAVE, ES, 0x400 - 0x400);
920AssertCompileMemberOffset(SVMVMCBSTATESAVE, CS, 0x410 - 0x400);
921AssertCompileMemberOffset(SVMVMCBSTATESAVE, SS, 0x420 - 0x400);
922AssertCompileMemberOffset(SVMVMCBSTATESAVE, DS, 0x430 - 0x400);
923AssertCompileMemberOffset(SVMVMCBSTATESAVE, FS, 0x440 - 0x400);
924AssertCompileMemberOffset(SVMVMCBSTATESAVE, GS, 0x450 - 0x400);
925AssertCompileMemberOffset(SVMVMCBSTATESAVE, GDTR, 0x460 - 0x400);
926AssertCompileMemberOffset(SVMVMCBSTATESAVE, LDTR, 0x470 - 0x400);
927AssertCompileMemberOffset(SVMVMCBSTATESAVE, IDTR, 0x480 - 0x400);
928AssertCompileMemberOffset(SVMVMCBSTATESAVE, TR, 0x490 - 0x400);
929AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8Reserved4, 0x4a0 - 0x400);
930AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8CPL, 0x4cb - 0x400);
931AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8Reserved5, 0x4cc - 0x400);
932AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64EFER, 0x4d0 - 0x400);
933AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8Reserved6, 0x4d8 - 0x400);
934AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64CR4, 0x548 - 0x400);
935AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64CR3, 0x550 - 0x400);
936AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64CR0, 0x558 - 0x400);
937AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64DR7, 0x560 - 0x400);
938AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64DR6, 0x568 - 0x400);
939AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64RFlags, 0x570 - 0x400);
940AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64RIP, 0x578 - 0x400);
941AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8Reserved7, 0x580 - 0x400);
942AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64RSP, 0x5d8 - 0x400);
943AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8Reserved8, 0x5e0 - 0x400);
944AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64RAX, 0x5f8 - 0x400);
945AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64STAR, 0x600 - 0x400);
946AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64LSTAR, 0x608 - 0x400);
947AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64CSTAR, 0x610 - 0x400);
948AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64SFMASK, 0x618 - 0x400);
949AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64KernelGSBase, 0x620 - 0x400);
950AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64SysEnterCS, 0x628 - 0x400);
951AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64SysEnterESP, 0x630 - 0x400);
952AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64SysEnterEIP, 0x638 - 0x400);
953AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64CR2, 0x640 - 0x400);
954AssertCompileMemberOffset(SVMVMCBSTATESAVE, u8Reserved9, 0x648 - 0x400);
955AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64GPAT, 0x668 - 0x400);
956AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64DBGCTL, 0x670 - 0x400);
957AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64BR_FROM, 0x678 - 0x400);
958AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64BR_TO, 0x680 - 0x400);
959AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64LASTEXCPFROM, 0x688 - 0x400);
960AssertCompileMemberOffset(SVMVMCBSTATESAVE, u64LASTEXCPTO, 0x690 - 0x400);
961
962/**
963 * SVM VM Control Block. (VMCB)
964 */
965#pragma pack(1)
966typedef struct SVMVMCB
967{
968 /** Offset 0x00 - Control area. */
969 SVMVMCBCTRL ctrl;
970 /** Offset 0x100-0x3FF - Reserved. */
971 uint8_t u8Reserved3[0x400 - 0x100];
972 /** Offset 0x400 - State save area. */
973 SVMVMCBSTATESAVE guest;
974 /** Offset 0x698-0xFFF- Reserved. */
975 uint8_t u8Reserved10[0x1000 - 0x698];
976} SVMVMCB;
977#pragma pack()
978/** Pointer to the SVMVMCB structure. */
979typedef SVMVMCB *PSVMVMCB;
980/** Pointer to a const SVMVMCB structure. */
981typedef const SVMVMCB *PCSVMVMCB;
982AssertCompileMemberOffset(SVMVMCB, ctrl, 0x00);
983AssertCompileMemberOffset(SVMVMCB, u8Reserved3, 0x100);
984AssertCompileMemberOffset(SVMVMCB, guest, 0x400);
985AssertCompileMemberOffset(SVMVMCB, u8Reserved10, 0x698);
986AssertCompileSize(SVMVMCB, 0x1000);
987
988/** SVM nested-guest VMCB cache.
989 *
990 * A state structure for holding information across AMD-V VMRUN/\#VMEXIT
991 * operation during execution of the nested-guest, restored on \#VMEXIT.
992 */
993#pragma pack(1)
994typedef struct SVMNESTEDVMCBCACHE
995{
996 /** @name Nested-guest VMCB controls.
997 * @{ */
998 /** Cache of CRX read intercepts. */
999 uint16_t u16InterceptRdCRx;
1000 /** Cache of CRX write intercepts. */
1001 uint16_t u16InterceptWrCRx;
1002 /** Cache of DRX read intercepts. */
1003 uint16_t u16InterceptRdDRx;
1004 /** Cache of DRX write intercepts. */
1005 uint16_t u16InterceptWrDRx;
1006 /** Cache of exception intercepts. */
1007 uint32_t u32InterceptXcpt;
1008 /** Alignment. */
1009 uint32_t u32Padding0;
1010
1011 /** Cache of control intercepts. */
1012 uint64_t u64InterceptCtrl;
1013 /** Cache of IOPM nested-guest physical address. */
1014 uint64_t u64IOPMPhysAddr;
1015 /** Cache of MSRPM nested-guest physical address. */
1016 uint64_t u64MSRPMPhysAddr;
1017 /** Cache of the TSC offset. */
1018 uint64_t u64TSCOffset;
1019 /** Cache of the VMCB clean bits. */
1020 uint32_t u32VmcbCleanBits;
1021 uint32_t u32Reserved0;
1022 /** Cache of the TLB control. */
1023 SVMTLBCTRL TLBCtrl;
1024 /** Cache of the nested-paging control. */
1025 uint32_t u1NestedPaging : 1;
1026 /** Cache of the LBR virtualization control. */
1027 uint32_t u1LbrVirt : 1;
1028 uint32_t u31Reserved0 : 30;
1029 uint32_t u32Reserved1;
1030 /** @} */
1031
1032 /** @name Nested-guest VMCB guest state.
1033 * @{ */
1034 /** Cache of CR0. */
1035 uint64_t u64CR0;
1036 /** Cache of CR3. */
1037 uint64_t u64CR3;
1038 /** Cache of CR4. */
1039 uint64_t u64CR4;
1040 /** Cache of EFER. */
1041 uint64_t u64EFER;
1042 /** Cache of DBGCTL. */
1043 uint64_t u64DBGCTL;
1044 /** @} */
1045
1046 /** @name Other miscellaneous state.
1047 * @{ */
1048 /** Cache of V_INTR_MASKING bit. */
1049 bool fVIntrMasking;
1050 /** Alignment. */
1051 bool afPadding0[7];
1052 /** @} */
1053} SVMNESTEDVMCBCACHE;
1054#pragma pack()
1055/** Pointer to the SVMNESTEDVMCBCACHE structure. */
1056typedef SVMNESTEDVMCBCACHE *PSVMNESTEDVMCBCACHE;
1057/** Pointer to a const SVMNESTEDVMCBCACHE structure. */
1058typedef const SVMNESTEDVMCBCACHE *PCSVMNESTEDVMCBCACHE;
1059AssertCompileSizeAlignment(SVMNESTEDVMCBCACHE, 8);
1060
1061#ifdef IN_RING0
1062VMMR0DECL(int) SVMR0InvalidatePage(PVM pVM, PVMCPU pVCpu, RTGCPTR GCVirt);
1063#endif /* IN_RING0 */
1064
1065/**
1066 * Segment attribute conversion between CPU and AMD-V VMCB format.
1067 *
1068 * The CPU format of the segment attribute is described in X86DESCATTRBITS
1069 * which is 16-bits (i.e. includes 4 bits of the segment limit).
1070 *
1071 * The AMD-V VMCB format the segment attribute is compact 12-bits (strictly
1072 * only the attribute bits and nothing else). Upper 4-bits are unused.
1073 */
1074#define HMSVM_CPU_2_VMCB_SEG_ATTR(a) ( ((a) & 0xff) | (((a) & 0xf000) >> 4) )
1075#define HMSVM_VMCB_2_CPU_SEG_ATTR(a) ( ((a) & 0xff) | (((a) & 0x0f00) << 4) )
1076
1077/** @def HMSVM_SEG_REG_COPY_TO_VMCB
1078 * Copies the specified segment register to a VMCB from a virtual CPU context.
1079 *
1080 * @param a_pCtx The virtual-CPU context.
1081 * @param a_pVmcbStateSave Pointer to the VMCB state-save area.
1082 * @param a_REG The segment register in the VMCB state-save
1083 * struct (ES/CS/SS/DS).
1084 * @param a_reg The segment register in the virtual CPU struct
1085 * (es/cs/ss/ds).
1086 */
1087#define HMSVM_SEG_REG_COPY_TO_VMCB(a_pCtx, a_pVmcbStateSave, a_REG, a_reg) \
1088 do \
1089 { \
1090 Assert((a_pCtx)->a_reg.fFlags & CPUMSELREG_FLAGS_VALID); \
1091 Assert((a_pCtx)->a_reg.ValidSel == (a_pCtx)->a_reg.Sel); \
1092 (a_pVmcbStateSave)->a_REG.u16Sel = (a_pCtx)->a_reg.Sel; \
1093 (a_pVmcbStateSave)->a_REG.u32Limit = (a_pCtx)->a_reg.u32Limit; \
1094 (a_pVmcbStateSave)->a_REG.u64Base = (a_pCtx)->a_reg.u64Base; \
1095 (a_pVmcbStateSave)->a_REG.u16Attr = HMSVM_CPU_2_VMCB_SEG_ATTR((a_pCtx)->a_reg.Attr.u); \
1096 } while (0)
1097
1098/** @def HMSVM_SEG_REG_COPY_TO_VMCB
1099 * Copies the specified segment register from the VMCB to a virtual CPU
1100 * context.
1101 *
1102 * @param a_pCtx The virtual-CPU context.
1103 * @param a_pVmcbStateSave Pointer to the VMCB state-save area.
1104 * @param a_REG The segment register in the VMCB state-save
1105 * struct (ES/CS/SS/DS).
1106 * @param a_reg The segment register in the virtual CPU struct
1107 * (es/ds/ss/ds).
1108 */
1109#define HMSVM_SEG_REG_COPY_FROM_VMCB(a_pCtx, a_pVmcbStateSave, a_REG, a_reg) \
1110 do \
1111 { \
1112 (a_pCtx)->a_reg.Sel = (a_pVmcbStateSave)->a_REG.u16Sel; \
1113 (a_pCtx)->a_reg.ValidSel = (a_pVmcbStateSave)->a_REG.u16Sel; \
1114 (a_pCtx)->a_reg.fFlags = CPUMSELREG_FLAGS_VALID; \
1115 (a_pCtx)->a_reg.u32Limit = (a_pVmcbStateSave)->a_REG.u32Limit; \
1116 (a_pCtx)->a_reg.u64Base = (a_pVmcbStateSave)->a_REG.u64Base; \
1117 (a_pCtx)->a_reg.Attr.u = HMSVM_VMCB_2_CPU_SEG_ATTR((a_pVmcbStateSave)->a_REG.u16Attr); \
1118 } while (0)
1119
1120VMM_INT_DECL(bool) HMIsGuestSvmCtrlInterceptSet(PVMCPU pVCpu, PCPUMCTX pCtx, uint64_t fIntercept);
1121VMM_INT_DECL(bool) HMIsGuestSvmReadCRxInterceptSet(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uCr);
1122VMM_INT_DECL(bool) HMIsGuestSvmWriteCRxInterceptSet(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uCr);
1123VMM_INT_DECL(bool) HMIsGuestSvmReadDRxInterceptSet(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uDr);
1124VMM_INT_DECL(bool) HMIsGuestSvmWriteDRxInterceptSet(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uDr);
1125VMM_INT_DECL(bool) HMIsGuestSvmXcptInterceptSet(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uVector);
1126VMM_INT_DECL(bool) HMCanSvmNstGstTakePhysIntr(PVMCPU pVCpu, PCCPUMCTX pCtx);
1127
1128/** @} */
1129
1130#endif
1131
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