VirtualBox

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

Last change on this file since 12669 was 12662, checked in by vboxsync, 16 years ago

cpum.h: removed @todos.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 29.5 KB
Line 
1/** @file
2 * CPUM - CPU Monitor(/ Manager).
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
38__BEGIN_DECLS
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 instead of relying on the exact esp & ss layout as before (prevented us from using a union with rsp). */
251 uint32_t lss_esp;
252 RTSEL ss;
253 RTSEL ssPadding;
254
255 RTSEL gs;
256 RTSEL gsPadding;
257 RTSEL fs;
258 RTSEL fsPadding;
259 RTSEL es;
260 RTSEL esPadding;
261 RTSEL ds;
262 RTSEL dsPadding;
263 RTSEL cs;
264 RTSEL csPadding[3]; /* 3 words to force 8 byte alignment for the remainder */
265
266 union
267 {
268 X86EFLAGS eflags;
269 X86RFLAGS rflags;
270 };
271 union
272 {
273 uint16_t ip;
274 uint32_t eip;
275 uint64_t rip;
276 };
277
278 uint64_t r8;
279 uint64_t r9;
280 uint64_t r10;
281 uint64_t r11;
282 uint64_t r12;
283 uint64_t r13;
284 uint64_t r14;
285 uint64_t r15;
286
287 /** Hidden selector registers.
288 * @{ */
289 CPUMSELREGHID esHid;
290 CPUMSELREGHID csHid;
291 CPUMSELREGHID ssHid;
292 CPUMSELREGHID dsHid;
293 CPUMSELREGHID fsHid;
294 CPUMSELREGHID gsHid;
295 /** @} */
296
297 /** @} */
298
299 /** Control registers.
300 * @{ */
301 uint64_t cr0;
302 uint64_t cr2;
303 uint64_t cr3;
304 uint64_t cr4;
305 /** @} */
306
307 /** Debug registers.
308 * @remarks DR4 and DR5 should not be used since they are aliases for
309 * DR6 and DR7 respectively on both AMD and Intel CPUs.
310 * @remarks DR8-15 are currently not supported by AMD or Intel, so
311 * neither do we.
312 * @{ */
313 uint64_t dr[8];
314 /** @} */
315
316 /** Global Descriptor Table register. */
317 VBOXGDTR gdtr;
318 uint16_t gdtrPadding;
319 /** Interrupt Descriptor Table register. */
320 VBOXIDTR idtr;
321 uint16_t idtrPadding;
322 /** The task register.
323 * Only the guest context uses all the members. */
324 RTSEL ldtr;
325 RTSEL ldtrPadding;
326 /** The task register.
327 * Only the guest context uses all the members. */
328 RTSEL tr;
329 RTSEL trPadding;
330
331 /** The sysenter msr registers.
332 * This member is not used by the hypervisor context. */
333 CPUMSYSENTER SysEnter;
334
335 /** System MSRs.
336 * @{ */
337 uint64_t msrEFER;
338 uint64_t msrSTAR; /* legacy syscall eip, cs & ss */
339 uint64_t msrPAT;
340 uint64_t msrLSTAR; /* 64 bits mode syscall rip */
341 uint64_t msrCSTAR; /* compatibility mode syscall rip */
342 uint64_t msrSFMASK; /* syscall flag mask */
343 uint64_t msrKERNELGSBASE;/* swapgs exchange value */
344 /** @} */
345
346 /** Hidden selector registers.
347 * @{ */
348 CPUMSELREGHID ldtrHid;
349 CPUMSELREGHID trHid;
350 /** @} */
351
352#if 0
353 /*& Padding to align the size on a 64 byte boundrary. */
354 uint32_t padding[6];
355#endif
356} CPUMCTX;
357#pragma pack()
358
359/**
360 * Gets the CPUMCTXCORE part of a CPUMCTX.
361 */
362#define CPUMCTX2CORE(pCtx) ((PCPUMCTXCORE)(void *)&(pCtx)->edi)
363
364
365/**
366 * Selector hidden registers, for version 1.6 saved state.
367 */
368typedef struct CPUMSELREGHID_VER1_6
369{
370 /** Base register. */
371 uint32_t u32Base;
372 /** Limit (expanded). */
373 uint32_t u32Limit;
374 /** Flags.
375 * This is the high 32-bit word of the descriptor entry.
376 * Only the flags, dpl and type are used. */
377 X86DESCATTR Attr;
378} CPUMSELREGHID_VER1_6;
379
380/**
381 * CPU context, for version 1.6 saved state.
382 * @remarks PATM uses this, which is why it has to be here.
383 */
384#pragma pack(1)
385typedef struct CPUMCTX_VER1_6
386{
387 /** FPU state. (16-byte alignment)
388 * @todo This doesn't have to be in X86FXSTATE on CPUs without fxsr - we need a type for the
389 * actual format or convert it (waste of time). */
390 X86FXSTATE fpu;
391
392 /** CPUMCTXCORE Part.
393 * @{ */
394 union
395 {
396 uint32_t edi;
397 uint64_t rdi;
398 };
399 union
400 {
401 uint32_t esi;
402 uint64_t rsi;
403 };
404 union
405 {
406 uint32_t ebp;
407 uint64_t rbp;
408 };
409 union
410 {
411 uint32_t eax;
412 uint64_t rax;
413 };
414 union
415 {
416 uint32_t ebx;
417 uint64_t rbx;
418 };
419 union
420 {
421 uint32_t edx;
422 uint64_t rdx;
423 };
424 union
425 {
426 uint32_t ecx;
427 uint64_t rcx;
428 };
429 /* Note: we rely on the exact layout, because we use lss esp, [] in the switcher */
430 uint32_t esp;
431 RTSEL ss;
432 RTSEL ssPadding;
433 /* Note: no overlap with esp here. */
434 uint64_t rsp_notused;
435
436 RTSEL gs;
437 RTSEL gsPadding;
438 RTSEL fs;
439 RTSEL fsPadding;
440 RTSEL es;
441 RTSEL esPadding;
442 RTSEL ds;
443 RTSEL dsPadding;
444 RTSEL cs;
445 RTSEL csPadding[3]; /* 3 words to force 8 byte alignment for the remainder */
446
447 union
448 {
449 X86EFLAGS eflags;
450 X86RFLAGS rflags;
451 };
452 union
453 {
454 uint32_t eip;
455 uint64_t rip;
456 };
457
458 uint64_t r8;
459 uint64_t r9;
460 uint64_t r10;
461 uint64_t r11;
462 uint64_t r12;
463 uint64_t r13;
464 uint64_t r14;
465 uint64_t r15;
466
467 /** Hidden selector registers.
468 * @{ */
469 CPUMSELREGHID_VER1_6 esHid;
470 CPUMSELREGHID_VER1_6 csHid;
471 CPUMSELREGHID_VER1_6 ssHid;
472 CPUMSELREGHID_VER1_6 dsHid;
473 CPUMSELREGHID_VER1_6 fsHid;
474 CPUMSELREGHID_VER1_6 gsHid;
475 /** @} */
476
477 /** @} */
478
479 /** Control registers.
480 * @{ */
481 uint64_t cr0;
482 uint64_t cr2;
483 uint64_t cr3;
484 uint64_t cr4;
485 uint64_t cr8;
486 /** @} */
487
488 /** Debug registers.
489 * @{ */
490 uint64_t dr0;
491 uint64_t dr1;
492 uint64_t dr2;
493 uint64_t dr3;
494 uint64_t dr4; /**< @todo remove dr4 and dr5. */
495 uint64_t dr5;
496 uint64_t dr6;
497 uint64_t dr7;
498 /* DR8-15 are currently not supported */
499 /** @} */
500
501 /** Global Descriptor Table register. */
502 VBOXGDTR_VER1_6 gdtr;
503 uint16_t gdtrPadding;
504 uint32_t gdtrPadding64;/** @todo fix this hack */
505 /** Interrupt Descriptor Table register. */
506 VBOXIDTR_VER1_6 idtr;
507 uint16_t idtrPadding;
508 uint32_t idtrPadding64;/** @todo fix this hack */
509 /** The task register.
510 * Only the guest context uses all the members. */
511 RTSEL ldtr;
512 RTSEL ldtrPadding;
513 /** The task register.
514 * Only the guest context uses all the members. */
515 RTSEL tr;
516 RTSEL trPadding;
517
518 /** The sysenter msr registers.
519 * This member is not used by the hypervisor context. */
520 CPUMSYSENTER SysEnter;
521
522 /** System MSRs.
523 * @{ */
524 uint64_t msrEFER;
525 uint64_t msrSTAR;
526 uint64_t msrPAT;
527 uint64_t msrLSTAR;
528 uint64_t msrCSTAR;
529 uint64_t msrSFMASK;
530 uint64_t msrFSBASE;
531 uint64_t msrGSBASE;
532 uint64_t msrKERNELGSBASE;
533 /** @} */
534
535 /** Hidden selector registers.
536 * @{ */
537 CPUMSELREGHID_VER1_6 ldtrHid;
538 CPUMSELREGHID_VER1_6 trHid;
539 /** @} */
540
541 /* padding to get 32byte aligned size */
542 uint32_t padding[2];
543} CPUMCTX_VER1_6;
544#pragma pack()
545
546
547/**
548 * The register set returned by a CPUID operation.
549 */
550typedef struct CPUMCPUID
551{
552 uint32_t eax;
553 uint32_t ebx;
554 uint32_t ecx;
555 uint32_t edx;
556} CPUMCPUID;
557/** Pointer to a CPUID leaf. */
558typedef CPUMCPUID *PCPUMCPUID;
559/** Pointer to a const CPUID leaf. */
560typedef const CPUMCPUID *PCCPUMCPUID;
561
562/**
563 * CPUID feature to set or clear.
564 */
565typedef enum CPUMCPUIDFEATURE
566{
567 CPUMCPUIDFEATURE_INVALID = 0,
568 /** The APIC feature bit. (Std+Ext) */
569 CPUMCPUIDFEATURE_APIC,
570 /** The sysenter/sysexit feature bit. (Std) */
571 CPUMCPUIDFEATURE_SEP,
572 /** The SYSCALL/SYSEXIT feature bit (64 bits mode only for Intel CPUs). (Ext) */
573 CPUMCPUIDFEATURE_SYSCALL,
574 /** The PAE feature bit. (Std+Ext) */
575 CPUMCPUIDFEATURE_PAE,
576 /** The NXE feature bit. (Ext) */
577 CPUMCPUIDFEATURE_NXE,
578 /** The LAHF/SAHF feature bit (64 bits mode only). (Ext) */
579 CPUMCPUIDFEATURE_LAHF,
580 /** The LONG MODE feature bit. (Ext) */
581 CPUMCPUIDFEATURE_LONG_MODE,
582 /** The PAT feature bit. (Std+Ext) */
583 CPUMCPUIDFEATURE_PAT,
584 /** 32bit hackishness. */
585 CPUMCPUIDFEATURE_32BIT_HACK = 0x7fffffff
586} CPUMCPUIDFEATURE;
587
588/**
589 * CPU Vendor.
590 */
591typedef enum CPUMCPUVENDOR
592{
593 CPUMCPUVENDOR_INVALID = 0,
594 CPUMCPUVENDOR_INTEL,
595 CPUMCPUVENDOR_AMD,
596 CPUMCPUVENDOR_VIA,
597 CPUMCPUVENDOR_UNKNOWN,
598 /** 32bit hackishness. */
599 CPUMCPUVENDOR_32BIT_HACK = 0x7fffffff
600} CPUMCPUVENDOR;
601
602
603/** @name Guest Register Getters.
604 * @{ */
605CPUMDECL(void) CPUMGetGuestGDTR(PVM pVM, PVBOXGDTR pGDTR);
606CPUMDECL(RTGCPTR) CPUMGetGuestIDTR(PVM pVM, uint16_t *pcbLimit);
607CPUMDECL(RTSEL) CPUMGetGuestTR(PVM pVM);
608CPUMDECL(RTSEL) CPUMGetGuestLDTR(PVM pVM);
609CPUMDECL(uint64_t) CPUMGetGuestCR0(PVM pVM);
610CPUMDECL(uint64_t) CPUMGetGuestCR2(PVM pVM);
611CPUMDECL(uint64_t) CPUMGetGuestCR3(PVM pVM);
612CPUMDECL(uint64_t) CPUMGetGuestCR4(PVM pVM);
613CPUMDECL(int) CPUMGetGuestCRx(PVM pVM, unsigned iReg, uint64_t *pValue);
614CPUMDECL(uint32_t) CPUMGetGuestEFlags(PVM pVM);
615CPUMDECL(uint32_t) CPUMGetGuestEIP(PVM pVM);
616CPUMDECL(uint64_t) CPUMGetGuestRIP(PVM pVM);
617CPUMDECL(uint32_t) CPUMGetGuestEAX(PVM pVM);
618CPUMDECL(uint32_t) CPUMGetGuestEBX(PVM pVM);
619CPUMDECL(uint32_t) CPUMGetGuestECX(PVM pVM);
620CPUMDECL(uint32_t) CPUMGetGuestEDX(PVM pVM);
621CPUMDECL(uint32_t) CPUMGetGuestESI(PVM pVM);
622CPUMDECL(uint32_t) CPUMGetGuestEDI(PVM pVM);
623CPUMDECL(uint32_t) CPUMGetGuestESP(PVM pVM);
624CPUMDECL(uint32_t) CPUMGetGuestEBP(PVM pVM);
625CPUMDECL(RTSEL) CPUMGetGuestCS(PVM pVM);
626CPUMDECL(RTSEL) CPUMGetGuestDS(PVM pVM);
627CPUMDECL(RTSEL) CPUMGetGuestES(PVM pVM);
628CPUMDECL(RTSEL) CPUMGetGuestFS(PVM pVM);
629CPUMDECL(RTSEL) CPUMGetGuestGS(PVM pVM);
630CPUMDECL(RTSEL) CPUMGetGuestSS(PVM pVM);
631CPUMDECL(uint64_t) CPUMGetGuestDR0(PVM pVM);
632CPUMDECL(uint64_t) CPUMGetGuestDR1(PVM pVM);
633CPUMDECL(uint64_t) CPUMGetGuestDR2(PVM pVM);
634CPUMDECL(uint64_t) CPUMGetGuestDR3(PVM pVM);
635CPUMDECL(uint64_t) CPUMGetGuestDR6(PVM pVM);
636CPUMDECL(uint64_t) CPUMGetGuestDR7(PVM pVM);
637CPUMDECL(int) CPUMGetGuestDRx(PVM pVM, uint32_t iReg, uint64_t *pValue);
638CPUMDECL(void) CPUMGetGuestCpuId(PVM pVM, uint32_t iLeaf, uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx);
639CPUMDECL(RCPTRTYPE(PCCPUMCPUID)) CPUMGetGuestCpuIdStdGCPtr(PVM pVM);
640CPUMDECL(RCPTRTYPE(PCCPUMCPUID)) CPUMGetGuestCpuIdExtGCPtr(PVM pVM);
641CPUMDECL(RCPTRTYPE(PCCPUMCPUID)) CPUMGetGuestCpuIdCentaurGCPtr(PVM pVM);
642CPUMDECL(RCPTRTYPE(PCCPUMCPUID)) CPUMGetGuestCpuIdDefGCPtr(PVM pVM);
643CPUMDECL(uint32_t) CPUMGetGuestCpuIdStdMax(PVM pVM);
644CPUMDECL(uint32_t) CPUMGetGuestCpuIdExtMax(PVM pVM);
645CPUMDECL(uint32_t) CPUMGetGuestCpuIdCentaurMax(PVM pVM);
646CPUMDECL(CPUMSELREGHID *) CPUMGetGuestTRHid(PVM pVM);
647CPUMDECL(uint64_t) CPUMGetGuestEFER(PVM pVM);
648CPUMDECL(uint64_t) CPUMGetGuestMsr(PVM pVM, unsigned idMsr);
649/** @} */
650
651/** @name Guest Register Setters.
652 * @{ */
653CPUMDECL(int) CPUMSetGuestGDTR(PVM pVM, uint32_t addr, uint16_t limit);
654CPUMDECL(int) CPUMSetGuestIDTR(PVM pVM, uint32_t addr, uint16_t limit);
655CPUMDECL(int) CPUMSetGuestTR(PVM pVM, uint16_t tr);
656CPUMDECL(int) CPUMSetGuestLDTR(PVM pVM, uint16_t ldtr);
657CPUMDECL(int) CPUMSetGuestCR0(PVM pVM, uint64_t cr0);
658CPUMDECL(int) CPUMSetGuestCR2(PVM pVM, uint64_t cr2);
659CPUMDECL(int) CPUMSetGuestCR3(PVM pVM, uint64_t cr3);
660CPUMDECL(int) CPUMSetGuestCR4(PVM pVM, uint64_t cr4);
661CPUMDECL(int) CPUMSetGuestDR0(PVM pVM, uint64_t uDr0);
662CPUMDECL(int) CPUMSetGuestDR1(PVM pVM, uint64_t uDr1);
663CPUMDECL(int) CPUMSetGuestDR2(PVM pVM, uint64_t uDr2);
664CPUMDECL(int) CPUMSetGuestDR3(PVM pVM, uint64_t uDr3);
665CPUMDECL(int) CPUMSetGuestDR6(PVM pVM, uint64_t uDr6);
666CPUMDECL(int) CPUMSetGuestDR7(PVM pVM, uint64_t uDr7);
667CPUMDECL(int) CPUMSetGuestDRx(PVM pVM, uint32_t iReg, uint64_t Value);
668CPUMDECL(int) CPUMSetGuestEFlags(PVM pVM, uint32_t eflags);
669CPUMDECL(int) CPUMSetGuestEIP(PVM pVM, uint32_t eip);
670CPUMDECL(int) CPUMSetGuestEAX(PVM pVM, uint32_t eax);
671CPUMDECL(int) CPUMSetGuestEBX(PVM pVM, uint32_t ebx);
672CPUMDECL(int) CPUMSetGuestECX(PVM pVM, uint32_t ecx);
673CPUMDECL(int) CPUMSetGuestEDX(PVM pVM, uint32_t edx);
674CPUMDECL(int) CPUMSetGuestESI(PVM pVM, uint32_t esi);
675CPUMDECL(int) CPUMSetGuestEDI(PVM pVM, uint32_t edi);
676CPUMDECL(int) CPUMSetGuestESP(PVM pVM, uint32_t esp);
677CPUMDECL(int) CPUMSetGuestEBP(PVM pVM, uint32_t ebp);
678CPUMDECL(int) CPUMSetGuestCS(PVM pVM, uint16_t cs);
679CPUMDECL(int) CPUMSetGuestDS(PVM pVM, uint16_t ds);
680CPUMDECL(int) CPUMSetGuestES(PVM pVM, uint16_t es);
681CPUMDECL(int) CPUMSetGuestFS(PVM pVM, uint16_t fs);
682CPUMDECL(int) CPUMSetGuestGS(PVM pVM, uint16_t gs);
683CPUMDECL(int) CPUMSetGuestSS(PVM pVM, uint16_t ss);
684CPUMDECL(void) CPUMSetGuestEFER(PVM pVM, uint64_t val);
685CPUMDECL(void) CPUMSetGuestCpuIdFeature(PVM pVM, CPUMCPUIDFEATURE enmFeature);
686CPUMDECL(void) CPUMClearGuestCpuIdFeature(PVM pVM, CPUMCPUIDFEATURE enmFeature);
687CPUMDECL(bool) CPUMGetGuestCpuIdFeature(PVM pVM, CPUMCPUIDFEATURE enmFeature);
688CPUMDECL(void) CPUMSetGuestCtx(PVM pVM, const PCPUMCTX pCtx);
689/** @} */
690
691/** @name Misc Guest Predicate Functions.
692 * @{ */
693
694
695CPUMDECL(bool) CPUMIsGuestIn16BitCode(PVM pVM);
696CPUMDECL(bool) CPUMIsGuestIn32BitCode(PVM pVM);
697CPUMDECL(CPUMCPUVENDOR) CPUMGetCPUVendor(PVM pVM);
698
699/**
700 * Tests if the guest is running in real mode or not.
701 *
702 * @returns true if in real mode, otherwise false.
703 * @param pVM The VM handle.
704 */
705DECLINLINE(bool) CPUMIsGuestInRealMode(PVM pVM)
706{
707 return !(CPUMGetGuestCR0(pVM) & X86_CR0_PE);
708}
709
710/**
711 * Tests if the guest is running in protected or not.
712 *
713 * @returns true if in protected mode, otherwise false.
714 * @param pVM The VM handle.
715 */
716DECLINLINE(bool) CPUMIsGuestInProtectedMode(PVM pVM)
717{
718 return !!(CPUMGetGuestCR0(pVM) & X86_CR0_PE);
719}
720
721/**
722 * Tests if the guest is running in paged protected or not.
723 *
724 * @returns true if in paged protected mode, otherwise false.
725 * @param pVM The VM handle.
726 */
727DECLINLINE(bool) CPUMIsGuestInPagedProtectedMode(PVM pVM)
728{
729 return (CPUMGetGuestCR0(pVM) & (X86_CR0_PE | X86_CR0_PG)) == (X86_CR0_PE | X86_CR0_PG);
730}
731
732/**
733 * Tests if the guest is running in long mode or not.
734 *
735 * @returns true if in long mode, otherwise false.
736 * @param pVM The VM handle.
737 */
738DECLINLINE(bool) CPUMIsGuestInLongMode(PVM pVM)
739{
740 return (CPUMGetGuestEFER(pVM) & MSR_K6_EFER_LMA) == MSR_K6_EFER_LMA;
741}
742
743/**
744 * Tests if the guest is running in long mode or not.
745 *
746 * @returns true if in long mode, otherwise false.
747 * @param pCtx Current CPU context
748 */
749DECLINLINE(bool) CPUMIsGuestInLongModeEx(PCPUMCTX pCtx)
750{
751 return (pCtx->msrEFER & MSR_K6_EFER_LMA) == MSR_K6_EFER_LMA;
752}
753
754/**
755 * Tests if the guest is running in 64 bits mode or not.
756 *
757 * @returns true if in 64 bits protected mode, otherwise false.
758 * @param pVM The VM handle.
759 * @param pCtx Current CPU context
760 */
761DECLINLINE(bool) CPUMIsGuestIn64BitCode(PVM pVM, PCCPUMCTXCORE pCtx)
762{
763 if (!CPUMIsGuestInLongMode(pVM))
764 return false;
765
766 return pCtx->csHid.Attr.n.u1Long;
767}
768
769/**
770 * Tests if the guest is running in 64 bits mode or not.
771 *
772 * @returns true if in 64 bits protected mode, otherwise false.
773 * @param pVM The VM handle.
774 * @param pCtx Current CPU context
775 */
776DECLINLINE(bool) CPUMIsGuestIn64BitCodeEx(PCCPUMCTX pCtx)
777{
778 if (!(pCtx->msrEFER & MSR_K6_EFER_LMA))
779 return false;
780
781 return pCtx->csHid.Attr.n.u1Long;
782}
783
784/** @} */
785
786
787/** @name Hypervisor Register Getters.
788 * @{ */
789CPUMDECL(RTSEL) CPUMGetHyperCS(PVM pVM);
790CPUMDECL(RTSEL) CPUMGetHyperDS(PVM pVM);
791CPUMDECL(RTSEL) CPUMGetHyperES(PVM pVM);
792CPUMDECL(RTSEL) CPUMGetHyperFS(PVM pVM);
793CPUMDECL(RTSEL) CPUMGetHyperGS(PVM pVM);
794CPUMDECL(RTSEL) CPUMGetHyperSS(PVM pVM);
795#if 0 /* these are not correct. */
796CPUMDECL(uint32_t) CPUMGetHyperCR0(PVM pVM);
797CPUMDECL(uint32_t) CPUMGetHyperCR2(PVM pVM);
798CPUMDECL(uint32_t) CPUMGetHyperCR3(PVM pVM);
799CPUMDECL(uint32_t) CPUMGetHyperCR4(PVM pVM);
800#endif
801/** This register is only saved on fatal traps. */
802CPUMDECL(uint32_t) CPUMGetHyperEAX(PVM pVM);
803CPUMDECL(uint32_t) CPUMGetHyperEBX(PVM pVM);
804/** This register is only saved on fatal traps. */
805CPUMDECL(uint32_t) CPUMGetHyperECX(PVM pVM);
806/** This register is only saved on fatal traps. */
807CPUMDECL(uint32_t) CPUMGetHyperEDX(PVM pVM);
808CPUMDECL(uint32_t) CPUMGetHyperESI(PVM pVM);
809CPUMDECL(uint32_t) CPUMGetHyperEDI(PVM pVM);
810CPUMDECL(uint32_t) CPUMGetHyperEBP(PVM pVM);
811CPUMDECL(uint32_t) CPUMGetHyperESP(PVM pVM);
812CPUMDECL(uint32_t) CPUMGetHyperEFlags(PVM pVM);
813CPUMDECL(uint32_t) CPUMGetHyperEIP(PVM pVM);
814CPUMDECL(uint64_t) CPUMGetHyperRIP(PVM pVM);
815CPUMDECL(uint32_t) CPUMGetHyperIDTR(PVM pVM, uint16_t *pcbLimit);
816CPUMDECL(uint32_t) CPUMGetHyperGDTR(PVM pVM, uint16_t *pcbLimit);
817CPUMDECL(RTSEL) CPUMGetHyperLDTR(PVM pVM);
818CPUMDECL(RTGCUINTREG) CPUMGetHyperDR0(PVM pVM);
819CPUMDECL(RTGCUINTREG) CPUMGetHyperDR1(PVM pVM);
820CPUMDECL(RTGCUINTREG) CPUMGetHyperDR2(PVM pVM);
821CPUMDECL(RTGCUINTREG) CPUMGetHyperDR3(PVM pVM);
822CPUMDECL(RTGCUINTREG) CPUMGetHyperDR6(PVM pVM);
823CPUMDECL(RTGCUINTREG) CPUMGetHyperDR7(PVM pVM);
824CPUMDECL(void) CPUMGetHyperCtx(PVM pVM, PCPUMCTX pCtx);
825/** @} */
826
827/** @name Hypervisor Register Setters.
828 * @{ */
829CPUMDECL(void) CPUMSetHyperGDTR(PVM pVM, uint32_t addr, uint16_t limit);
830CPUMDECL(void) CPUMSetHyperLDTR(PVM pVM, RTSEL SelLDTR);
831CPUMDECL(void) CPUMSetHyperIDTR(PVM pVM, uint32_t addr, uint16_t limit);
832CPUMDECL(void) CPUMSetHyperCR3(PVM pVM, uint32_t cr3);
833CPUMDECL(void) CPUMSetHyperTR(PVM pVM, RTSEL SelTR);
834CPUMDECL(void) CPUMSetHyperCS(PVM pVM, RTSEL SelCS);
835CPUMDECL(void) CPUMSetHyperDS(PVM pVM, RTSEL SelDS);
836CPUMDECL(void) CPUMSetHyperES(PVM pVM, RTSEL SelDS);
837CPUMDECL(void) CPUMSetHyperFS(PVM pVM, RTSEL SelDS);
838CPUMDECL(void) CPUMSetHyperGS(PVM pVM, RTSEL SelDS);
839CPUMDECL(void) CPUMSetHyperSS(PVM pVM, RTSEL SelSS);
840CPUMDECL(void) CPUMSetHyperESP(PVM pVM, uint32_t u32ESP);
841CPUMDECL(int) CPUMSetHyperEFlags(PVM pVM, uint32_t Efl);
842CPUMDECL(void) CPUMSetHyperEIP(PVM pVM, uint32_t u32EIP);
843CPUMDECL(void) CPUMSetHyperDR0(PVM pVM, RTGCUINTREG uDr0);
844CPUMDECL(void) CPUMSetHyperDR1(PVM pVM, RTGCUINTREG uDr1);
845CPUMDECL(void) CPUMSetHyperDR2(PVM pVM, RTGCUINTREG uDr2);
846CPUMDECL(void) CPUMSetHyperDR3(PVM pVM, RTGCUINTREG uDr3);
847CPUMDECL(void) CPUMSetHyperDR6(PVM pVM, RTGCUINTREG uDr6);
848CPUMDECL(void) CPUMSetHyperDR7(PVM pVM, RTGCUINTREG uDr7);
849CPUMDECL(void) CPUMSetHyperCtx(PVM pVM, const PCPUMCTX pCtx);
850CPUMDECL(int) CPUMRecalcHyperDRx(PVM pVM);
851/** @} */
852
853CPUMDECL(void) CPUMPushHyper(PVM pVM, uint32_t u32);
854CPUMDECL(void) CPUMHyperSetCtxCore(PVM pVM, PCPUMCTXCORE pCtxCore);
855CPUMDECL(int) CPUMQueryGuestCtxPtr(PVM pVM, PCPUMCTX *ppCtx);
856CPUMDECL(int) CPUMQueryHyperCtxPtr(PVM pVM, PCPUMCTX *ppCtx);
857CPUMDECL(PCCPUMCTXCORE) CPUMGetGuestCtxCore(PVM pVM);
858CPUMDECL(PCCPUMCTXCORE) CPUMGetHyperCtxCore(PVM pVM);
859CPUMDECL(void) CPUMSetGuestCtxCore(PVM pVM, PCCPUMCTXCORE pCtxCore);
860CPUMDECL(int) CPUMRawEnter(PVM pVM, PCPUMCTXCORE pCtxCore);
861CPUMDECL(int) CPUMRawLeave(PVM pVM, PCPUMCTXCORE pCtxCore, int rc);
862CPUMDECL(uint32_t) CPUMRawGetEFlags(PVM pVM, PCPUMCTXCORE pCtxCore);
863CPUMDECL(void) CPUMRawSetEFlags(PVM pVM, PCPUMCTXCORE pCtxCore, uint32_t eflags);
864CPUMDECL(int) CPUMHandleLazyFPU(PVM pVM);
865CPUMDECL(int) CPUMRestoreHostFPUState(PVM pVM);
866
867/** @name Changed flags
868 * These flags are used to keep track of which important register that
869 * have been changed since last they were reset. The only one allowed
870 * to clear them is REM!
871 * @{
872 */
873#define CPUM_CHANGED_FPU_REM RT_BIT(0)
874#define CPUM_CHANGED_CR0 RT_BIT(1)
875#define CPUM_CHANGED_CR4 RT_BIT(2)
876#define CPUM_CHANGED_GLOBAL_TLB_FLUSH RT_BIT(3)
877#define CPUM_CHANGED_CR3 RT_BIT(4)
878#define CPUM_CHANGED_GDTR RT_BIT(5)
879#define CPUM_CHANGED_IDTR RT_BIT(6)
880#define CPUM_CHANGED_LDTR RT_BIT(7)
881#define CPUM_CHANGED_TR RT_BIT(8)
882#define CPUM_CHANGED_SYSENTER_MSR RT_BIT(9)
883#define CPUM_CHANGED_HIDDEN_SEL_REGS RT_BIT(10)
884#define CPUM_CHANGED_CPUID RT_BIT(11)
885#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)
886/** @} */
887
888CPUMDECL(unsigned) CPUMGetAndClearChangedFlagsREM(PVM pVM);
889CPUMDECL(void) CPUMSetChangedFlags(PVM pVM, uint32_t fChangedFlags);
890CPUMDECL(bool) CPUMSupportsFXSR(PVM pVM);
891CPUMDECL(bool) CPUMIsHostUsingSysEnter(PVM pVM);
892CPUMDECL(bool) CPUMIsHostUsingSysCall(PVM pVM);
893CPUMDECL(bool) CPUMIsGuestFPUStateActive(PVM pVM);
894CPUMDECL(void) CPUMDeactivateGuestFPUState(PVM pVM);
895CPUMDECL(bool) CPUMIsGuestDebugStateActive(PVM pVM);
896CPUMDECL(void) CPUMDeactivateGuestDebugState(PVM pVM);
897CPUMDECL(bool) CPUMAreHiddenSelRegsValid(PVM pVM);
898CPUMDECL(void) CPUMSetHiddenSelRegsValid(PVM pVM, bool fValid);
899CPUMDECL(uint32_t) CPUMGetGuestCPL(PVM pVM, PCPUMCTXCORE pCtxCore);
900
901/**
902 * CPU modes.
903 */
904typedef enum CPUMMODE
905{
906 /** The usual invalid zero entry. */
907 CPUMMODE_INVALID = 0,
908 /** Real mode. */
909 CPUMMODE_REAL,
910 /** Protected mode (32-bit). */
911 CPUMMODE_PROTECTED,
912 /** Long mode (64-bit). */
913 CPUMMODE_LONG
914} CPUMMODE;
915
916CPUMDECL(CPUMMODE) CPUMGetGuestMode(PVM pVM);
917
918
919#ifdef IN_RING3
920/** @defgroup grp_cpum_r3 The CPU Monitor(/Manager) API
921 * @ingroup grp_cpum
922 * @{
923 */
924
925CPUMR3DECL(int) CPUMR3Init(PVM pVM);
926CPUMR3DECL(void) CPUMR3Relocate(PVM pVM);
927CPUMR3DECL(int) CPUMR3Term(PVM pVM);
928CPUMR3DECL(void) CPUMR3Reset(PVM pVM);
929CPUMR3DECL(int) CPUMR3QueryGuestCtxGCPtr(PVM pVM, RCPTRTYPE(PCPUMCTX) *ppCtx);
930# ifdef DEBUG
931CPUMR3DECL(void) CPUMR3SaveEntryCtx(PVM pVM);
932# endif
933CPUMR3DECL(int) CPUMR3SetCR4Feature(PVM pVM, RTHCUINTREG fOr, RTHCUINTREG fAnd);
934
935/** @} */
936#endif /* IN_RING3 */
937
938#ifdef IN_GC
939/** @defgroup grp_cpum_gc The CPU Monitor(/Manager) API
940 * @ingroup grp_cpum
941 * @{
942 */
943
944/**
945 * Calls a guest trap/interrupt handler directly
946 * Assumes a trap stack frame has already been setup on the guest's stack!
947 *
948 * @param pRegFrame Original trap/interrupt context
949 * @param selCS Code selector of handler
950 * @param pHandler GC virtual address of handler
951 * @param eflags Callee's EFLAGS
952 * @param selSS Stack selector for handler
953 * @param pEsp Stack address for handler
954 *
955 * This function does not return!
956 */
957DECLASM(void) CPUMGCCallGuestTrapHandler(PCPUMCTXCORE pRegFrame, uint32_t selCS, RTRCPTR pHandler, uint32_t eflags, uint32_t selSS, RTRCPTR pEsp);
958CPUMGCDECL(void) CPUMGCCallV86Code(PCPUMCTXCORE pRegFrame);
959
960/** @} */
961#endif /* IN_GC */
962
963#ifdef IN_RING0
964/** @defgroup grp_cpum_r0 The CPU Monitor(/Manager) API
965 * @ingroup grp_cpum
966 * @{
967 */
968CPUMR0DECL(int) CPUMR0Init(PVM pVM);
969CPUMR0DECL(int) CPUMR0LoadGuestFPU(PVM pVM, PCPUMCTX pCtx);
970CPUMR0DECL(int) CPUMR0SaveGuestFPU(PVM pVM, PCPUMCTX pCtx);
971CPUMR0DECL(int) CPUMR0SaveGuestDebugState(PVM pVM, PCPUMCTX pCtx, bool fDR6);
972CPUMR0DECL(int) CPUMR0LoadGuestDebugState(PVM pVM, PCPUMCTX pCtx, bool fDR6);
973
974/** @} */
975#endif /* IN_RING0 */
976
977/** @} */
978__END_DECLS
979
980
981#endif
982
983
984
985
986
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