VirtualBox

source: vbox/trunk/include/VBox/cpum.h@ 28158

Last change on this file since 28158 was 27572, checked in by vboxsync, 15 years ago

cpum.h: doxygen/javadoc style comments.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.9 KB
Line 
1/** @file
2 * CPUM - CPU Monitor(/ Manager). (VMM)
3 */
4
5/*
6 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
26 * Clara, CA 95054 USA or visit http://www.sun.com if you need
27 * additional information or have any questions.
28 */
29
30#ifndef ___VBox_cpum_h
31#define ___VBox_cpum_h
32
33#include <VBox/cdefs.h>
34#include <VBox/types.h>
35#include <VBox/x86.h>
36
37
38RT_C_DECLS_BEGIN
39
40/** @defgroup grp_cpum The CPU Monitor / Manager API
41 * @{
42 */
43
44/**
45 * Selector hidden registers.
46 */
47typedef struct CPUMSELREGHID
48{
49 /** Base register.
50 *
51 * Long mode remarks:
52 * - Unused in long mode for CS, DS, ES, SS
53 * - 32 bits for FS & GS; FS(GS)_BASE msr used for the base address
54 * - 64 bits for TR & LDTR
55 */
56 uint64_t u64Base;
57 /** Limit (expanded). */
58 uint32_t u32Limit;
59 /** Flags.
60 * This is the high 32-bit word of the descriptor entry.
61 * Only the flags, dpl and type are used. */
62 X86DESCATTR Attr;
63} CPUMSELREGHID;
64
65
66/**
67 * The sysenter register set.
68 */
69typedef struct CPUMSYSENTER
70{
71 /** Ring 0 cs.
72 * This value + 8 is the Ring 0 ss.
73 * This value + 16 is the Ring 3 cs.
74 * This value + 24 is the Ring 3 ss.
75 */
76 uint64_t cs;
77 /** Ring 0 eip. */
78 uint64_t eip;
79 /** Ring 0 esp. */
80 uint64_t esp;
81} CPUMSYSENTER;
82
83
84/**
85 * CPU context core.
86 */
87#pragma pack(1)
88typedef struct CPUMCTXCORE
89{
90 union
91 {
92 uint16_t di;
93 uint32_t edi;
94 uint64_t rdi;
95 };
96 union
97 {
98 uint16_t si;
99 uint32_t esi;
100 uint64_t rsi;
101 };
102 union
103 {
104 uint16_t bp;
105 uint32_t ebp;
106 uint64_t rbp;
107 };
108 union
109 {
110 uint16_t ax;
111 uint32_t eax;
112 uint64_t rax;
113 };
114 union
115 {
116 uint16_t bx;
117 uint32_t ebx;
118 uint64_t rbx;
119 };
120 union
121 {
122 uint16_t dx;
123 uint32_t edx;
124 uint64_t rdx;
125 };
126 union
127 {
128 uint16_t cx;
129 uint32_t ecx;
130 uint64_t rcx;
131 };
132 union
133 {
134 uint16_t sp;
135 uint32_t esp;
136 uint64_t rsp;
137 };
138 /* Note: lss esp, [] in the switcher needs some space, so we reserve it here instead of relying on the exact esp & ss layout as before. */
139 uint32_t lss_esp;
140 RTSEL ss;
141 RTSEL ssPadding;
142
143 RTSEL gs;
144 RTSEL gsPadding;
145 RTSEL fs;
146 RTSEL fsPadding;
147 RTSEL es;
148 RTSEL esPadding;
149 RTSEL ds;
150 RTSEL dsPadding;
151 RTSEL cs;
152 RTSEL csPadding[3]; /* 3 words to force 8 byte alignment for the remainder */
153
154 union
155 {
156 X86EFLAGS eflags;
157 X86RFLAGS rflags;
158 };
159 union
160 {
161 uint16_t ip;
162 uint32_t eip;
163 uint64_t rip;
164 };
165
166 uint64_t r8;
167 uint64_t r9;
168 uint64_t r10;
169 uint64_t r11;
170 uint64_t r12;
171 uint64_t r13;
172 uint64_t r14;
173 uint64_t r15;
174
175 /** Hidden selector registers.
176 * @{ */
177 CPUMSELREGHID esHid;
178 CPUMSELREGHID csHid;
179 CPUMSELREGHID ssHid;
180 CPUMSELREGHID dsHid;
181 CPUMSELREGHID fsHid;
182 CPUMSELREGHID gsHid;
183 /** @} */
184
185} CPUMCTXCORE;
186#pragma pack()
187
188
189/**
190 * CPU context.
191 */
192#pragma pack(1)
193typedef struct CPUMCTX
194{
195 /** FPU state. (16-byte alignment)
196 * @todo This doesn't have to be in X86FXSTATE on CPUs without fxsr - we need a type for the
197 * actual format or convert it (waste of time). */
198 X86FXSTATE fpu;
199
200 /** CPUMCTXCORE Part.
201 * @{ */
202 union
203 {
204 uint16_t di;
205 uint32_t edi;
206 uint64_t rdi;
207 };
208 union
209 {
210 uint16_t si;
211 uint32_t esi;
212 uint64_t rsi;
213 };
214 union
215 {
216 uint16_t bp;
217 uint32_t ebp;
218 uint64_t rbp;
219 };
220 union
221 {
222 uint16_t ax;
223 uint32_t eax;
224 uint64_t rax;
225 };
226 union
227 {
228 uint16_t bx;
229 uint32_t ebx;
230 uint64_t rbx;
231 };
232 union
233 {
234 uint16_t dx;
235 uint32_t edx;
236 uint64_t rdx;
237 };
238 union
239 {
240 uint16_t cx;
241 uint32_t ecx;
242 uint64_t rcx;
243 };
244 union
245 {
246 uint16_t sp;
247 uint32_t esp;
248 uint64_t rsp;
249 };
250 /** @note lss esp, [] in the switcher needs some space, so we reserve it here
251 * instead of relying on the exact esp & ss layout as before (prevented
252 * us from using a union with rsp). */
253 uint32_t lss_esp;
254 RTSEL ss;
255 RTSEL ssPadding;
256
257 RTSEL gs;
258 RTSEL gsPadding;
259 RTSEL fs;
260 RTSEL fsPadding;
261 RTSEL es;
262 RTSEL esPadding;
263 RTSEL ds;
264 RTSEL dsPadding;
265 RTSEL cs;
266 RTSEL csPadding[3]; /* 3 words to force 8 byte alignment for the remainder */
267
268 union
269 {
270 X86EFLAGS eflags;
271 X86RFLAGS rflags;
272 };
273 union
274 {
275 uint16_t ip;
276 uint32_t eip;
277 uint64_t rip;
278 };
279
280 uint64_t r8;
281 uint64_t r9;
282 uint64_t r10;
283 uint64_t r11;
284 uint64_t r12;
285 uint64_t r13;
286 uint64_t r14;
287 uint64_t r15;
288
289 /** Hidden selector registers.
290 * @{ */
291 CPUMSELREGHID esHid;
292 CPUMSELREGHID csHid;
293 CPUMSELREGHID ssHid;
294 CPUMSELREGHID dsHid;
295 CPUMSELREGHID fsHid;
296 CPUMSELREGHID gsHid;
297 /** @} */
298
299 /** @} */
300
301 /** Control registers.
302 * @{ */
303 uint64_t cr0;
304 uint64_t cr2;
305 uint64_t cr3;
306 uint64_t cr4;
307 /** @} */
308
309 /** Debug registers.
310 * @remarks DR4 and DR5 should not be used since they are aliases for
311 * DR6 and DR7 respectively on both AMD and Intel CPUs.
312 * @remarks DR8-15 are currently not supported by AMD or Intel, so
313 * neither do we.
314 * @{ */
315 uint64_t dr[8];
316 /** @} */
317
318 /** Global Descriptor Table register. */
319 VBOXGDTR gdtr;
320 uint16_t gdtrPadding;
321 /** Interrupt Descriptor Table register. */
322 VBOXIDTR idtr;
323 uint16_t idtrPadding;
324 /** The task register.
325 * Only the guest context uses all the members. */
326 RTSEL ldtr;
327 RTSEL ldtrPadding;
328 /** The task register.
329 * Only the guest context uses all the members. */
330 RTSEL tr;
331 RTSEL trPadding;
332
333 /** The sysenter msr registers.
334 * This member is not used by the hypervisor context. */
335 CPUMSYSENTER SysEnter;
336
337 /** System MSRs.
338 * @{ */
339 uint64_t msrEFER;
340 uint64_t msrSTAR; /**< Legacy syscall eip, cs & ss. */
341 uint64_t msrPAT;
342 uint64_t msrLSTAR; /**< 64 bits mode syscall rip. */
343 uint64_t msrCSTAR; /**< Compatibility mode syscall rip. */
344 uint64_t msrSFMASK; /**< syscall flag mask. */
345 uint64_t msrKERNELGSBASE; /**< swapgs exchange value. */
346 /** @} */
347
348 /** Hidden selector registers.
349 * @{ */
350 CPUMSELREGHID ldtrHid;
351 CPUMSELREGHID trHid;
352 /** @} */
353
354#if 0
355 /** Padding to align the size on a 64 byte boundrary. */
356 uint32_t padding[6];
357#endif
358} CPUMCTX;
359#pragma pack()
360
361/**
362 * Gets the CPUMCTXCORE part of a CPUMCTX.
363 */
364#define CPUMCTX2CORE(pCtx) ((PCPUMCTXCORE)(void *)&(pCtx)->edi)
365
366/**
367 * Selector hidden registers, for version 1.6 saved state.
368 */
369typedef struct CPUMSELREGHID_VER1_6
370{
371 /** Base register. */
372 uint32_t u32Base;
373 /** Limit (expanded). */
374 uint32_t u32Limit;
375 /** Flags.
376 * This is the high 32-bit word of the descriptor entry.
377 * Only the flags, dpl and type are used. */
378 X86DESCATTR Attr;
379} CPUMSELREGHID_VER1_6;
380
381/**
382 * CPU context, for version 1.6 saved state.
383 * @remarks PATM uses this, which is why it has to be here.
384 */
385#pragma pack(1)
386typedef struct CPUMCTX_VER1_6
387{
388 /** FPU state. (16-byte alignment)
389 * @todo This doesn't have to be in X86FXSTATE on CPUs without fxsr - we need a type for the
390 * actual format or convert it (waste of time). */
391 X86FXSTATE fpu;
392
393 /** CPUMCTXCORE Part.
394 * @{ */
395 union
396 {
397 uint32_t edi;
398 uint64_t rdi;
399 };
400 union
401 {
402 uint32_t esi;
403 uint64_t rsi;
404 };
405 union
406 {
407 uint32_t ebp;
408 uint64_t rbp;
409 };
410 union
411 {
412 uint32_t eax;
413 uint64_t rax;
414 };
415 union
416 {
417 uint32_t ebx;
418 uint64_t rbx;
419 };
420 union
421 {
422 uint32_t edx;
423 uint64_t rdx;
424 };
425 union
426 {
427 uint32_t ecx;
428 uint64_t rcx;
429 };
430 /** @note We rely on the exact layout, because we use lss esp, [] in the
431 * switcher. */
432 uint32_t esp;
433 RTSEL ss;
434 RTSEL ssPadding;
435 /* Note: no overlap with esp here. */
436 uint64_t rsp_notused;
437
438 RTSEL gs;
439 RTSEL gsPadding;
440 RTSEL fs;
441 RTSEL fsPadding;
442 RTSEL es;
443 RTSEL esPadding;
444 RTSEL ds;
445 RTSEL dsPadding;
446 RTSEL cs;
447 RTSEL csPadding[3]; /**< 3 words to force 8 byte alignment for the remainder. */
448
449 union
450 {
451 X86EFLAGS eflags;
452 X86RFLAGS rflags;
453 };
454 union
455 {
456 uint32_t eip;
457 uint64_t rip;
458 };
459
460 uint64_t r8;
461 uint64_t r9;
462 uint64_t r10;
463 uint64_t r11;
464 uint64_t r12;
465 uint64_t r13;
466 uint64_t r14;
467 uint64_t r15;
468
469 /** Hidden selector registers.
470 * @{ */
471 CPUMSELREGHID_VER1_6 esHid;
472 CPUMSELREGHID_VER1_6 csHid;
473 CPUMSELREGHID_VER1_6 ssHid;
474 CPUMSELREGHID_VER1_6 dsHid;
475 CPUMSELREGHID_VER1_6 fsHid;
476 CPUMSELREGHID_VER1_6 gsHid;
477 /** @} */
478
479 /** @} */
480
481 /** Control registers.
482 * @{ */
483 uint64_t cr0;
484 uint64_t cr2;
485 uint64_t cr3;
486 uint64_t cr4;
487 uint64_t cr8;
488 /** @} */
489
490 /** Debug registers.
491 * @{ */
492 uint64_t dr0;
493 uint64_t dr1;
494 uint64_t dr2;
495 uint64_t dr3;
496 uint64_t dr4; /**< @todo remove dr4 and dr5. */
497 uint64_t dr5;
498 uint64_t dr6;
499 uint64_t dr7;
500 /* DR8-15 are currently not supported */
501 /** @} */
502
503 /** Global Descriptor Table register. */
504 VBOXGDTR_VER1_6 gdtr;
505 uint16_t gdtrPadding;
506 uint32_t gdtrPadding64;/** @todo fix this hack */
507 /** Interrupt Descriptor Table register. */
508 VBOXIDTR_VER1_6 idtr;
509 uint16_t idtrPadding;
510 uint32_t idtrPadding64;/** @todo fix this hack */
511 /** The task register.
512 * Only the guest context uses all the members. */
513 RTSEL ldtr;
514 RTSEL ldtrPadding;
515 /** The task register.
516 * Only the guest context uses all the members. */
517 RTSEL tr;
518 RTSEL trPadding;
519
520 /** The sysenter msr registers.
521 * This member is not used by the hypervisor context. */
522 CPUMSYSENTER SysEnter;
523
524 /** System MSRs.
525 * @{ */
526 uint64_t msrEFER;
527 uint64_t msrSTAR;
528 uint64_t msrPAT;
529 uint64_t msrLSTAR;
530 uint64_t msrCSTAR;
531 uint64_t msrSFMASK;
532 uint64_t msrFSBASE;
533 uint64_t msrGSBASE;
534 uint64_t msrKERNELGSBASE;
535 /** @} */
536
537 /** Hidden selector registers.
538 * @{ */
539 CPUMSELREGHID_VER1_6 ldtrHid;
540 CPUMSELREGHID_VER1_6 trHid;
541 /** @} */
542
543 /** padding to get 32byte aligned size. */
544 uint32_t padding[2];
545} CPUMCTX_VER1_6;
546#pragma pack()
547
548/**
549 * Guest MSR state.
550 *
551 * @note Never change the order here because of saved stated!
552 */
553typedef union CPUMCTXMSR
554{
555 struct
556 {
557 uint64_t tscAux; /**< MSR_K8_TSC_AUX */
558 uint64_t miscEnable; /**< MSR_IA32_MISC_ENABLE */
559 } msr;
560 uint64_t au64[64];
561} CPUMCTXMSR;
562/** Pointer to the guest MSR state. */
563typedef CPUMCTXMSR *PCPUMCTXMSR;
564/** Pointer to the const guest MSR state. */
565typedef const CPUMCTXMSR *PCCPUMCTXMSR;
566
567
568/**
569 * The register set returned by a CPUID operation.
570 */
571typedef struct CPUMCPUID
572{
573 uint32_t eax;
574 uint32_t ebx;
575 uint32_t ecx;
576 uint32_t edx;
577} CPUMCPUID;
578/** Pointer to a CPUID leaf. */
579typedef CPUMCPUID *PCPUMCPUID;
580/** Pointer to a const CPUID leaf. */
581typedef const CPUMCPUID *PCCPUMCPUID;
582
583/**
584 * CPUID feature to set or clear.
585 */
586typedef enum CPUMCPUIDFEATURE
587{
588 CPUMCPUIDFEATURE_INVALID = 0,
589 /** The APIC feature bit. (Std+Ext) */
590 CPUMCPUIDFEATURE_APIC,
591 /** The sysenter/sysexit feature bit. (Std) */
592 CPUMCPUIDFEATURE_SEP,
593 /** The SYSCALL/SYSEXIT feature bit (64 bits mode only for Intel CPUs). (Ext) */
594 CPUMCPUIDFEATURE_SYSCALL,
595 /** The PAE feature bit. (Std+Ext) */
596 CPUMCPUIDFEATURE_PAE,
597 /** The NXE feature bit. (Ext) */
598 CPUMCPUIDFEATURE_NXE,
599 /** The LAHF/SAHF feature bit (64 bits mode only). (Ext) */
600 CPUMCPUIDFEATURE_LAHF,
601 /** The LONG MODE feature bit. (Ext) */
602 CPUMCPUIDFEATURE_LONG_MODE,
603 /** The PAT feature bit. (Std+Ext) */
604 CPUMCPUIDFEATURE_PAT,
605 /** The x2APIC feature bit. (Std) */
606 CPUMCPUIDFEATURE_X2APIC,
607 /** The RDTSCP feature bit. (Ext) */
608 CPUMCPUIDFEATURE_RDTSCP,
609 /** 32bit hackishness. */
610 CPUMCPUIDFEATURE_32BIT_HACK = 0x7fffffff
611} CPUMCPUIDFEATURE;
612
613/**
614 * CPU Vendor.
615 */
616typedef enum CPUMCPUVENDOR
617{
618 CPUMCPUVENDOR_INVALID = 0,
619 CPUMCPUVENDOR_INTEL,
620 CPUMCPUVENDOR_AMD,
621 CPUMCPUVENDOR_VIA,
622 CPUMCPUVENDOR_UNKNOWN,
623 CPUMCPUVENDOR_SYNTHETIC,
624 /** 32bit hackishness. */
625 CPUMCPUVENDOR_32BIT_HACK = 0x7fffffff
626} CPUMCPUVENDOR;
627
628
629/** @name Guest Register Getters.
630 * @{ */
631VMMDECL(void) CPUMGetGuestGDTR(PVMCPU pVCpu, PVBOXGDTR pGDTR);
632VMMDECL(RTGCPTR) CPUMGetGuestIDTR(PVMCPU pVCpu, uint16_t *pcbLimit);
633VMMDECL(RTSEL) CPUMGetGuestTR(PVMCPU pVCpu, PCPUMSELREGHID pHidden);
634VMMDECL(RTSEL) CPUMGetGuestLDTR(PVMCPU pVCpu);
635VMMDECL(uint64_t) CPUMGetGuestCR0(PVMCPU pVCpu);
636VMMDECL(uint64_t) CPUMGetGuestCR2(PVMCPU pVCpu);
637VMMDECL(uint64_t) CPUMGetGuestCR3(PVMCPU pVCpu);
638VMMDECL(uint64_t) CPUMGetGuestCR4(PVMCPU pVCpu);
639VMMDECL(int) CPUMGetGuestCRx(PVMCPU pVCpu, unsigned iReg, uint64_t *pValue);
640VMMDECL(uint32_t) CPUMGetGuestEFlags(PVMCPU pVCpu);
641VMMDECL(uint32_t) CPUMGetGuestEIP(PVMCPU pVCpu);
642VMMDECL(uint64_t) CPUMGetGuestRIP(PVMCPU pVCpu);
643VMMDECL(uint32_t) CPUMGetGuestEAX(PVMCPU pVCpu);
644VMMDECL(uint32_t) CPUMGetGuestEBX(PVMCPU pVCpu);
645VMMDECL(uint32_t) CPUMGetGuestECX(PVMCPU pVCpu);
646VMMDECL(uint32_t) CPUMGetGuestEDX(PVMCPU pVCpu);
647VMMDECL(uint32_t) CPUMGetGuestESI(PVMCPU pVCpu);
648VMMDECL(uint32_t) CPUMGetGuestEDI(PVMCPU pVCpu);
649VMMDECL(uint32_t) CPUMGetGuestESP(PVMCPU pVCpu);
650VMMDECL(uint32_t) CPUMGetGuestEBP(PVMCPU pVCpu);
651VMMDECL(RTSEL) CPUMGetGuestCS(PVMCPU pVCpu);
652VMMDECL(RTSEL) CPUMGetGuestDS(PVMCPU pVCpu);
653VMMDECL(RTSEL) CPUMGetGuestES(PVMCPU pVCpu);
654VMMDECL(RTSEL) CPUMGetGuestFS(PVMCPU pVCpu);
655VMMDECL(RTSEL) CPUMGetGuestGS(PVMCPU pVCpu);
656VMMDECL(RTSEL) CPUMGetGuestSS(PVMCPU pVCpu);
657VMMDECL(uint64_t) CPUMGetGuestDR0(PVMCPU pVCpu);
658VMMDECL(uint64_t) CPUMGetGuestDR1(PVMCPU pVCpu);
659VMMDECL(uint64_t) CPUMGetGuestDR2(PVMCPU pVCpu);
660VMMDECL(uint64_t) CPUMGetGuestDR3(PVMCPU pVCpu);
661VMMDECL(uint64_t) CPUMGetGuestDR6(PVMCPU pVCpu);
662VMMDECL(uint64_t) CPUMGetGuestDR7(PVMCPU pVCpu);
663VMMDECL(int) CPUMGetGuestDRx(PVMCPU pVCpu, uint32_t iReg, uint64_t *pValue);
664VMMDECL(void) CPUMGetGuestCpuId(PVMCPU pVCpu, uint32_t iLeaf, uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx);
665VMMDECL(uint32_t) CPUMGetGuestCpuIdStdMax(PVM pVM);
666VMMDECL(uint32_t) CPUMGetGuestCpuIdExtMax(PVM pVM);
667VMMDECL(uint32_t) CPUMGetGuestCpuIdCentaurMax(PVM pVM);
668VMMDECL(uint64_t) CPUMGetGuestEFER(PVMCPU pVCpu);
669VMMDECL(uint64_t) CPUMGetGuestMsr(PVMCPU pVCpu, unsigned idMsr);
670VMMDECL(void) CPUMSetGuestMsr(PVMCPU pVCpu, unsigned idMsr, uint64_t valMsr);
671VMMDECL(CPUMCPUVENDOR) CPUMGetGuestCpuVendor(PVM pVM);
672VMMDECL(CPUMCPUVENDOR) CPUMGetHostCpuVendor(PVM pVM);
673/** @} */
674
675/** @name Guest Register Setters.
676 * @{ */
677VMMDECL(int) CPUMSetGuestGDTR(PVMCPU pVCpu, uint32_t addr, uint16_t limit);
678VMMDECL(int) CPUMSetGuestIDTR(PVMCPU pVCpu, uint32_t addr, uint16_t limit);
679VMMDECL(int) CPUMSetGuestTR(PVMCPU pVCpu, uint16_t tr);
680VMMDECL(int) CPUMSetGuestLDTR(PVMCPU pVCpu, uint16_t ldtr);
681VMMDECL(int) CPUMSetGuestCR0(PVMCPU pVCpu, uint64_t cr0);
682VMMDECL(int) CPUMSetGuestCR2(PVMCPU pVCpu, uint64_t cr2);
683VMMDECL(int) CPUMSetGuestCR3(PVMCPU pVCpu, uint64_t cr3);
684VMMDECL(int) CPUMSetGuestCR4(PVMCPU pVCpu, uint64_t cr4);
685VMMDECL(int) CPUMSetGuestDR0(PVMCPU pVCpu, uint64_t uDr0);
686VMMDECL(int) CPUMSetGuestDR1(PVMCPU pVCpu, uint64_t uDr1);
687VMMDECL(int) CPUMSetGuestDR2(PVMCPU pVCpu, uint64_t uDr2);
688VMMDECL(int) CPUMSetGuestDR3(PVMCPU pVCpu, uint64_t uDr3);
689VMMDECL(int) CPUMSetGuestDR6(PVMCPU pVCpu, uint64_t uDr6);
690VMMDECL(int) CPUMSetGuestDR7(PVMCPU pVCpu, uint64_t uDr7);
691VMMDECL(int) CPUMSetGuestDRx(PVMCPU pVCpu, uint32_t iReg, uint64_t Value);
692VMMDECL(int) CPUMSetGuestEFlags(PVMCPU pVCpu, uint32_t eflags);
693VMMDECL(int) CPUMSetGuestEIP(PVMCPU pVCpu, uint32_t eip);
694VMMDECL(int) CPUMSetGuestEAX(PVMCPU pVCpu, uint32_t eax);
695VMMDECL(int) CPUMSetGuestEBX(PVMCPU pVCpu, uint32_t ebx);
696VMMDECL(int) CPUMSetGuestECX(PVMCPU pVCpu, uint32_t ecx);
697VMMDECL(int) CPUMSetGuestEDX(PVMCPU pVCpu, uint32_t edx);
698VMMDECL(int) CPUMSetGuestESI(PVMCPU pVCpu, uint32_t esi);
699VMMDECL(int) CPUMSetGuestEDI(PVMCPU pVCpu, uint32_t edi);
700VMMDECL(int) CPUMSetGuestESP(PVMCPU pVCpu, uint32_t esp);
701VMMDECL(int) CPUMSetGuestEBP(PVMCPU pVCpu, uint32_t ebp);
702VMMDECL(int) CPUMSetGuestCS(PVMCPU pVCpu, uint16_t cs);
703VMMDECL(int) CPUMSetGuestDS(PVMCPU pVCpu, uint16_t ds);
704VMMDECL(int) CPUMSetGuestES(PVMCPU pVCpu, uint16_t es);
705VMMDECL(int) CPUMSetGuestFS(PVMCPU pVCpu, uint16_t fs);
706VMMDECL(int) CPUMSetGuestGS(PVMCPU pVCpu, uint16_t gs);
707VMMDECL(int) CPUMSetGuestSS(PVMCPU pVCpu, uint16_t ss);
708VMMDECL(void) CPUMSetGuestEFER(PVMCPU pVCpu, uint64_t val);
709VMMDECL(void) CPUMSetGuestCpuIdFeature(PVM pVM, CPUMCPUIDFEATURE enmFeature);
710VMMDECL(void) CPUMClearGuestCpuIdFeature(PVM pVM, CPUMCPUIDFEATURE enmFeature);
711VMMDECL(bool) CPUMGetGuestCpuIdFeature(PVM pVM, CPUMCPUIDFEATURE enmFeature);
712VMMDECL(void) CPUMSetGuestCtx(PVMCPU pVCpu, const PCPUMCTX pCtx);
713/** @} */
714
715
716/** @name Misc Guest Predicate Functions.
717 * @{ */
718
719VMMDECL(bool) CPUMIsGuestIn16BitCode(PVMCPU pVCpu);
720VMMDECL(bool) CPUMIsGuestIn32BitCode(PVMCPU pVCpu);
721VMMDECL(bool) CPUMIsGuestNXEnabled(PVMCPU pVCpu);
722VMMDECL(bool) CPUMIsGuestPageSizeExtEnabled(PVMCPU pVCpu);
723VMMDECL(bool) CPUMIsGuestPagingEnabled(PVMCPU pVCpu);
724VMMDECL(bool) CPUMIsGuestR0WriteProtEnabled(PVMCPU pVCpu);
725VMMDECL(bool) CPUMIsGuestInRealMode(PVMCPU pVCpu);
726VMMDECL(bool) CPUMIsGuestInProtectedMode(PVMCPU pVCpu);
727VMMDECL(bool) CPUMIsGuestInPagedProtectedMode(PVMCPU pVCpu);
728VMMDECL(bool) CPUMIsGuestInLongMode(PVMCPU pVCpu);
729VMMDECL(bool) CPUMIsGuestInPAEMode(PVMCPU pVCpu);
730
731/**
732 * Tests if the guest is running in real mode or not.
733 *
734 * @returns true if in real mode, otherwise false.
735 * @param pCtx Current CPU context
736 */
737DECLINLINE(bool) CPUMIsGuestInRealModeEx(PCPUMCTX pCtx)
738{
739 return !(pCtx->cr0 & X86_CR0_PE);
740}
741
742/**
743 * Tests if the guest is running in paged protected or not.
744 *
745 * @returns true if in paged protected mode, otherwise false.
746 * @param pVM The VM handle.
747 */
748DECLINLINE(bool) CPUMIsGuestInPagedProtectedModeEx(PCPUMCTX pCtx)
749{
750 return (pCtx->cr0 & (X86_CR0_PE | X86_CR0_PG)) == (X86_CR0_PE | X86_CR0_PG);
751}
752
753/**
754 * Tests if the guest is running in long mode or not.
755 *
756 * @returns true if in long mode, otherwise false.
757 * @param pCtx Current CPU context
758 */
759DECLINLINE(bool) CPUMIsGuestInLongModeEx(PCPUMCTX pCtx)
760{
761 return (pCtx->msrEFER & MSR_K6_EFER_LMA) == MSR_K6_EFER_LMA;
762}
763
764/**
765 * Tests if the guest is running in 64 bits mode or not.
766 *
767 * @returns true if in 64 bits protected mode, otherwise false.
768 * @param pVM The VM handle.
769 * @param pCtx Current CPU context
770 */
771DECLINLINE(bool) CPUMIsGuestIn64BitCode(PVMCPU pVCpu, PCCPUMCTXCORE pCtx)
772{
773 if (!CPUMIsGuestInLongMode(pVCpu))
774 return false;
775
776 return pCtx->csHid.Attr.n.u1Long;
777}
778
779/**
780 * Tests if the guest is running in 64 bits mode or not.
781 *
782 * @returns true if in 64 bits protected mode, otherwise false.
783 * @param pVM The VM handle.
784 * @param pCtx Current CPU context
785 */
786DECLINLINE(bool) CPUMIsGuestIn64BitCodeEx(PCCPUMCTX pCtx)
787{
788 if (!(pCtx->msrEFER & MSR_K6_EFER_LMA))
789 return false;
790
791 return pCtx->csHid.Attr.n.u1Long;
792}
793
794/**
795 * Tests if the guest is running in PAE mode or not.
796 *
797 * @returns true if in PAE mode, otherwise false.
798 * @param pCtx Current CPU context
799 */
800DECLINLINE(bool) CPUMIsGuestInPAEModeEx(PCPUMCTX pCtx)
801{
802 return ( (pCtx->cr4 & X86_CR4_PAE)
803 && CPUMIsGuestInPagedProtectedModeEx(pCtx)
804 && !CPUMIsGuestInLongModeEx(pCtx));
805}
806
807/** @} */
808
809
810/** @name Hypervisor Register Getters.
811 * @{ */
812VMMDECL(RTSEL) CPUMGetHyperCS(PVMCPU pVCpu);
813VMMDECL(RTSEL) CPUMGetHyperDS(PVMCPU pVCpu);
814VMMDECL(RTSEL) CPUMGetHyperES(PVMCPU pVCpu);
815VMMDECL(RTSEL) CPUMGetHyperFS(PVMCPU pVCpu);
816VMMDECL(RTSEL) CPUMGetHyperGS(PVMCPU pVCpu);
817VMMDECL(RTSEL) CPUMGetHyperSS(PVMCPU pVCpu);
818#if 0 /* these are not correct. */
819VMMDECL(uint32_t) CPUMGetHyperCR0(PVMCPU pVCpu);
820VMMDECL(uint32_t) CPUMGetHyperCR2(PVMCPU pVCpu);
821VMMDECL(uint32_t) CPUMGetHyperCR3(PVMCPU pVCpu);
822VMMDECL(uint32_t) CPUMGetHyperCR4(PVMCPU pVCpu);
823#endif
824/** This register is only saved on fatal traps. */
825VMMDECL(uint32_t) CPUMGetHyperEAX(PVMCPU pVCpu);
826VMMDECL(uint32_t) CPUMGetHyperEBX(PVMCPU pVCpu);
827/** This register is only saved on fatal traps. */
828VMMDECL(uint32_t) CPUMGetHyperECX(PVMCPU pVCpu);
829/** This register is only saved on fatal traps. */
830VMMDECL(uint32_t) CPUMGetHyperEDX(PVMCPU pVCpu);
831VMMDECL(uint32_t) CPUMGetHyperESI(PVMCPU pVCpu);
832VMMDECL(uint32_t) CPUMGetHyperEDI(PVMCPU pVCpu);
833VMMDECL(uint32_t) CPUMGetHyperEBP(PVMCPU pVCpu);
834VMMDECL(uint32_t) CPUMGetHyperESP(PVMCPU pVCpu);
835VMMDECL(uint32_t) CPUMGetHyperEFlags(PVMCPU pVCpu);
836VMMDECL(uint32_t) CPUMGetHyperEIP(PVMCPU pVCpu);
837VMMDECL(uint64_t) CPUMGetHyperRIP(PVMCPU pVCpu);
838VMMDECL(uint32_t) CPUMGetHyperIDTR(PVMCPU pVCpu, uint16_t *pcbLimit);
839VMMDECL(uint32_t) CPUMGetHyperGDTR(PVMCPU pVCpu, uint16_t *pcbLimit);
840VMMDECL(RTSEL) CPUMGetHyperLDTR(PVMCPU pVCpu);
841VMMDECL(RTGCUINTREG) CPUMGetHyperDR0(PVMCPU pVCpu);
842VMMDECL(RTGCUINTREG) CPUMGetHyperDR1(PVMCPU pVCpu);
843VMMDECL(RTGCUINTREG) CPUMGetHyperDR2(PVMCPU pVCpu);
844VMMDECL(RTGCUINTREG) CPUMGetHyperDR3(PVMCPU pVCpu);
845VMMDECL(RTGCUINTREG) CPUMGetHyperDR6(PVMCPU pVCpu);
846VMMDECL(RTGCUINTREG) CPUMGetHyperDR7(PVMCPU pVCpu);
847VMMDECL(void) CPUMGetHyperCtx(PVMCPU pVCpu, PCPUMCTX pCtx);
848VMMDECL(uint32_t) CPUMGetHyperCR3(PVMCPU pVCpu);
849/** @} */
850
851/** @name Hypervisor Register Setters.
852 * @{ */
853VMMDECL(void) CPUMSetHyperGDTR(PVMCPU pVCpu, uint32_t addr, uint16_t limit);
854VMMDECL(void) CPUMSetHyperLDTR(PVMCPU pVCpu, RTSEL SelLDTR);
855VMMDECL(void) CPUMSetHyperIDTR(PVMCPU pVCpu, uint32_t addr, uint16_t limit);
856VMMDECL(void) CPUMSetHyperCR3(PVMCPU pVCpu, uint32_t cr3);
857VMMDECL(void) CPUMSetHyperTR(PVMCPU pVCpu, RTSEL SelTR);
858VMMDECL(void) CPUMSetHyperCS(PVMCPU pVCpu, RTSEL SelCS);
859VMMDECL(void) CPUMSetHyperDS(PVMCPU pVCpu, RTSEL SelDS);
860VMMDECL(void) CPUMSetHyperES(PVMCPU pVCpu, RTSEL SelDS);
861VMMDECL(void) CPUMSetHyperFS(PVMCPU pVCpu, RTSEL SelDS);
862VMMDECL(void) CPUMSetHyperGS(PVMCPU pVCpu, RTSEL SelDS);
863VMMDECL(void) CPUMSetHyperSS(PVMCPU pVCpu, RTSEL SelSS);
864VMMDECL(void) CPUMSetHyperESP(PVMCPU pVCpu, uint32_t u32ESP);
865VMMDECL(int) CPUMSetHyperEFlags(PVMCPU pVCpu, uint32_t Efl);
866VMMDECL(void) CPUMSetHyperEIP(PVMCPU pVCpu, uint32_t u32EIP);
867VMMDECL(void) CPUMSetHyperDR0(PVMCPU pVCpu, RTGCUINTREG uDr0);
868VMMDECL(void) CPUMSetHyperDR1(PVMCPU pVCpu, RTGCUINTREG uDr1);
869VMMDECL(void) CPUMSetHyperDR2(PVMCPU pVCpu, RTGCUINTREG uDr2);
870VMMDECL(void) CPUMSetHyperDR3(PVMCPU pVCpu, RTGCUINTREG uDr3);
871VMMDECL(void) CPUMSetHyperDR6(PVMCPU pVCpu, RTGCUINTREG uDr6);
872VMMDECL(void) CPUMSetHyperDR7(PVMCPU pVCpu, RTGCUINTREG uDr7);
873VMMDECL(void) CPUMSetHyperCtx(PVMCPU pVCpu, const PCPUMCTX pCtx);
874VMMDECL(int) CPUMRecalcHyperDRx(PVMCPU pVCpu);
875/** @} */
876
877VMMDECL(void) CPUMPushHyper(PVMCPU pVCpu, uint32_t u32);
878VMMDECL(void) CPUMHyperSetCtxCore(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore);
879VMMDECL(int) CPUMQueryHyperCtxPtr(PVMCPU pVCpu, PCPUMCTX *ppCtx);
880VMMDECL(PCCPUMCTXCORE) CPUMGetHyperCtxCore(PVMCPU pVCpu);
881VMMDECL(PCPUMCTX) CPUMQueryGuestCtxPtr(PVMCPU pVCpu);
882VMMDECL(PCCPUMCTXCORE) CPUMGetGuestCtxCore(PVMCPU pVCpu);
883VMMDECL(void) CPUMSetGuestCtxCore(PVMCPU pVCpu, PCCPUMCTXCORE pCtxCore);
884VMMDECL(int) CPUMRawEnter(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore);
885VMMDECL(int) CPUMRawLeave(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, int rc);
886VMMDECL(uint32_t) CPUMRawGetEFlags(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore);
887VMMDECL(void) CPUMRawSetEFlags(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, uint32_t eflags);
888VMMDECL(int) CPUMHandleLazyFPU(PVMCPU pVCpu);
889
890/** @name Changed flags
891 * These flags are used to keep track of which important register that
892 * have been changed since last they were reset. The only one allowed
893 * to clear them is REM!
894 * @{
895 */
896#define CPUM_CHANGED_FPU_REM RT_BIT(0)
897#define CPUM_CHANGED_CR0 RT_BIT(1)
898#define CPUM_CHANGED_CR4 RT_BIT(2)
899#define CPUM_CHANGED_GLOBAL_TLB_FLUSH RT_BIT(3)
900#define CPUM_CHANGED_CR3 RT_BIT(4)
901#define CPUM_CHANGED_GDTR RT_BIT(5)
902#define CPUM_CHANGED_IDTR RT_BIT(6)
903#define CPUM_CHANGED_LDTR RT_BIT(7)
904#define CPUM_CHANGED_TR RT_BIT(8)
905#define CPUM_CHANGED_SYSENTER_MSR RT_BIT(9)
906#define CPUM_CHANGED_HIDDEN_SEL_REGS RT_BIT(10)
907#define CPUM_CHANGED_CPUID RT_BIT(11)
908#define CPUM_CHANGED_ALL (CPUM_CHANGED_FPU_REM|CPUM_CHANGED_CR0|CPUM_CHANGED_CR3|CPUM_CHANGED_CR4|CPUM_CHANGED_GDTR|CPUM_CHANGED_IDTR|CPUM_CHANGED_LDTR|CPUM_CHANGED_TR|CPUM_CHANGED_SYSENTER_MSR|CPUM_CHANGED_HIDDEN_SEL_REGS|CPUM_CHANGED_CPUID)
909/** @} */
910
911VMMDECL(unsigned) CPUMGetAndClearChangedFlagsREM(PVMCPU pVCpu);
912VMMDECL(void) CPUMSetChangedFlags(PVMCPU pVCpu, uint32_t fChangedFlags);
913VMMDECL(bool) CPUMSupportsFXSR(PVM pVM);
914VMMDECL(bool) CPUMIsHostUsingSysEnter(PVM pVM);
915VMMDECL(bool) CPUMIsHostUsingSysCall(PVM pVM);
916VMMDECL(bool) CPUMIsGuestFPUStateActive(PVMCPU pVCPU);
917VMMDECL(void) CPUMDeactivateGuestFPUState(PVMCPU pVCpu);
918VMMDECL(bool) CPUMIsGuestDebugStateActive(PVMCPU pVCpu);
919VMMDECL(void) CPUMDeactivateGuestDebugState(PVMCPU pVCpu);
920VMMDECL(bool) CPUMIsHyperDebugStateActive(PVMCPU pVCpu);
921VMMDECL(void) CPUMDeactivateHyperDebugState(PVMCPU pVCpu);
922VMMDECL(uint32_t) CPUMGetGuestCPL(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore);
923VMMDECL(bool) CPUMAreHiddenSelRegsValid(PVM pVM);
924VMMDECL(CPUMMODE) CPUMGetGuestMode(PVMCPU pVCpu);
925
926
927#ifdef IN_RING3
928/** @defgroup grp_cpum_r3 The CPU Monitor(/Manager) API
929 * @ingroup grp_cpum
930 * @{
931 */
932
933VMMR3DECL(int) CPUMR3Init(PVM pVM);
934VMMR3DECL(int) CPUMR3InitCPU(PVM pVM);
935VMMR3DECL(void) CPUMR3Relocate(PVM pVM);
936VMMR3DECL(int) CPUMR3Term(PVM pVM);
937VMMR3DECL(int) CPUMR3TermCPU(PVM pVM);
938VMMR3DECL(void) CPUMR3Reset(PVM pVM);
939VMMR3DECL(void) CPUMR3ResetCpu(PVMCPU pVCpu);
940VMMDECL(bool) CPUMR3IsStateRestorePending(PVM pVM);
941VMMR3DECL(void) CPUMR3SetHWVirtEx(PVM pVM, bool fHWVirtExEnabled);
942# ifdef DEBUG
943VMMR3DECL(void) CPUMR3SaveEntryCtx(PVM pVM);
944# endif
945VMMR3DECL(int) CPUMR3SetCR4Feature(PVM pVM, RTHCUINTREG fOr, RTHCUINTREG fAnd);
946VMMR3DECL(RCPTRTYPE(PCCPUMCPUID)) CPUMR3GetGuestCpuIdStdRCPtr(PVM pVM);
947VMMR3DECL(RCPTRTYPE(PCCPUMCPUID)) CPUMR3GetGuestCpuIdExtRCPtr(PVM pVM);
948VMMR3DECL(RCPTRTYPE(PCCPUMCPUID)) CPUMR3GetGuestCpuIdCentaurRCPtr(PVM pVM);
949VMMR3DECL(RCPTRTYPE(PCCPUMCPUID)) CPUMR3GetGuestCpuIdDefRCPtr(PVM pVM);
950
951/** @} */
952#endif /* IN_RING3 */
953
954#ifdef IN_RC
955/** @defgroup grp_cpum_gc The CPU Monitor(/Manager) API
956 * @ingroup grp_cpum
957 * @{
958 */
959
960/**
961 * Calls a guest trap/interrupt handler directly
962 * Assumes a trap stack frame has already been setup on the guest's stack!
963 *
964 * @param pRegFrame Original trap/interrupt context
965 * @param selCS Code selector of handler
966 * @param pHandler GC virtual address of handler
967 * @param eflags Callee's EFLAGS
968 * @param selSS Stack selector for handler
969 * @param pEsp Stack address for handler
970 *
971 * This function does not return!
972 */
973DECLASM(void) CPUMGCCallGuestTrapHandler(PCPUMCTXCORE pRegFrame, uint32_t selCS, RTRCPTR pHandler, uint32_t eflags, uint32_t selSS, RTRCPTR pEsp);
974VMMRCDECL(void) CPUMGCCallV86Code(PCPUMCTXCORE pRegFrame);
975
976/** @} */
977#endif /* IN_RC */
978
979#ifdef IN_RING0
980/** @defgroup grp_cpum_r0 The CPU Monitor(/Manager) API
981 * @ingroup grp_cpum
982 * @{
983 */
984VMMR0DECL(int) CPUMR0Init(PVM pVM);
985VMMR0DECL(int) CPUMR0LoadGuestFPU(PVM pVM, PVMCPU pVCpu, PCPUMCTX pCtx);
986VMMR0DECL(int) CPUMR0SaveGuestFPU(PVM pVM, PVMCPU pVCpu, PCPUMCTX pCtx);
987VMMR0DECL(int) CPUMR0SaveGuestDebugState(PVM pVM, PVMCPU pVCpu, PCPUMCTX pCtx, bool fDR6);
988VMMR0DECL(int) CPUMR0LoadGuestDebugState(PVM pVM, PVMCPU pVCpu, PCPUMCTX pCtx, bool fDR6);
989VMMR0DECL(int) CPUMR0LoadHostDebugState(PVM pVM, PVMCPU pVCpu);
990VMMR0DECL(int) CPUMR0SaveHostDebugState(PVM pVM, PVMCPU pVCpu);
991VMMR0DECL(int) CPUMR0LoadHyperDebugState(PVM pVM, PVMCPU pVCpu, PCPUMCTX pCtx, bool fDR6);
992
993/** @} */
994#endif /* IN_RING0 */
995
996/** @} */
997RT_C_DECLS_END
998
999
1000#endif
1001
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