VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMGC/SELMGC.cpp@ 17390

Last change on this file since 17390 was 17144, checked in by vboxsync, 16 years ago

SELMGC: logging.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 19.5 KB
Line 
1/* $Id: SELMGC.cpp 17144 2009-02-25 18:28:36Z vboxsync $ */
2/** @file
3 * SELM - The Selector Manager, Guest Context.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25#define LOG_GROUP LOG_GROUP_SELM
26#include <VBox/selm.h>
27#include <VBox/mm.h>
28#include <VBox/em.h>
29#include <VBox/trpm.h>
30#include "SELMInternal.h"
31#include <VBox/vm.h>
32#include <VBox/pgm.h>
33
34#include <VBox/param.h>
35#include <VBox/err.h>
36#include <VBox/log.h>
37#include <iprt/assert.h>
38#include <iprt/asm.h>
39
40
41/**
42 * Synchronizes one GDT entry (guest -> shadow).
43 *
44 * @returns VBox status code (appropriate for trap handling and GC return).
45 * @param pVM VM Handle.
46 * @param pRegFrame Trap register frame.
47 * @param iGDTEntry The GDT entry to sync.
48 */
49static int selmGCSyncGDTEntry(PVM pVM, PCPUMCTXCORE pRegFrame, unsigned iGDTEntry)
50{
51 Log2(("GDT %04X LDTR=%04X\n", iGDTEntry, CPUMGetGuestLDTR(pVM)));
52
53 /*
54 * Validate the offset.
55 */
56 VBOXGDTR GdtrGuest;
57 CPUMGetGuestGDTR(pVM, &GdtrGuest);
58 unsigned offEntry = iGDTEntry * sizeof(X86DESC);
59 if ( iGDTEntry >= SELM_GDT_ELEMENTS
60 || offEntry > GdtrGuest.cbGdt)
61 return VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT;
62
63 /*
64 * Read the guest descriptor.
65 */
66 X86DESC Desc;
67 int rc = MMGCRamRead(pVM, &Desc, (uint8_t *)GdtrGuest.pGdt + offEntry, sizeof(X86DESC));
68 if (RT_FAILURE(rc))
69 return VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT;
70
71 /*
72 * Check for conflicts.
73 */
74 RTSEL Sel = iGDTEntry << X86_SEL_SHIFT;
75 Assert( !(pVM->selm.s.aHyperSel[SELM_HYPER_SEL_CS] & ~X86_SEL_MASK)
76 && !(pVM->selm.s.aHyperSel[SELM_HYPER_SEL_DS] & ~X86_SEL_MASK)
77 && !(pVM->selm.s.aHyperSel[SELM_HYPER_SEL_CS64] & ~X86_SEL_MASK)
78 && !(pVM->selm.s.aHyperSel[SELM_HYPER_SEL_TSS] & ~X86_SEL_MASK)
79 && !(pVM->selm.s.aHyperSel[SELM_HYPER_SEL_TSS_TRAP08] & ~X86_SEL_MASK));
80 if ( pVM->selm.s.aHyperSel[SELM_HYPER_SEL_CS] == Sel
81 || pVM->selm.s.aHyperSel[SELM_HYPER_SEL_DS] == Sel
82 || pVM->selm.s.aHyperSel[SELM_HYPER_SEL_CS64] == Sel
83 || pVM->selm.s.aHyperSel[SELM_HYPER_SEL_TSS] == Sel
84 || pVM->selm.s.aHyperSel[SELM_HYPER_SEL_TSS_TRAP08] == Sel)
85 {
86 if (Desc.Gen.u1Present)
87 {
88 Log(("selmGCSyncGDTEntry: Sel=%d Desc=%.8Rhxs: detected conflict!!\n", Sel, &Desc));
89 return VINF_SELM_SYNC_GDT;
90 }
91 Log(("selmGCSyncGDTEntry: Sel=%d Desc=%.8Rhxs: potential conflict (still not present)!\n", Sel, &Desc));
92
93 /* Note: we can't continue below or else we'll change the shadow descriptor!! */
94 /* When the guest makes the selector present, then we'll do a GDT sync. */
95 return VINF_SUCCESS;
96 }
97
98 /*
99 * Code and data selectors are generally 1:1, with the
100 * 'little' adjustment we do for DPL 0 selectors.
101 */
102 PX86DESC pShadowDescr = &pVM->selm.s.paGdtRC[iGDTEntry];
103 if (Desc.Gen.u1DescType)
104 {
105 /*
106 * Hack for A-bit against Trap E on read-only GDT.
107 */
108 /** @todo Fix this by loading ds and cs before turning off WP. */
109 Desc.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
110
111 /*
112 * All DPL 0 code and data segments are squeezed into DPL 1.
113 *
114 * We're skipping conforming segments here because those
115 * cannot give us any trouble.
116 */
117 if ( Desc.Gen.u2Dpl == 0
118 && (Desc.Gen.u4Type & (X86_SEL_TYPE_CODE | X86_SEL_TYPE_CONF))
119 != (X86_SEL_TYPE_CODE | X86_SEL_TYPE_CONF) )
120 Desc.Gen.u2Dpl = 1;
121 }
122 else
123 {
124 /*
125 * System type selectors are marked not present.
126 * Recompiler or special handling is required for these.
127 */
128 /** @todo what about interrupt gates and rawr0? */
129 Desc.Gen.u1Present = 0;
130 }
131 //Log(("O: base=%08X limit=%08X attr=%04X\n", X86DESC_BASE(*pShadowDescr)), X86DESC_LIMIT(*pShadowDescr), (pShadowDescr->au32[1] >> 8) & 0xFFFF ));
132 //Log(("N: base=%08X limit=%08X attr=%04X\n", X86DESC_BASE(Desc)), X86DESC_LIMIT(Desc), (Desc.au32[1] >> 8) & 0xFFFF ));
133 *pShadowDescr = Desc;
134
135 /* Check if we change the LDT selector */
136 if (Sel == CPUMGetGuestLDTR(pVM)) /** @todo this isn't correct in two(+) ways! 1. It shouldn't be done until the LDTR is reloaded. 2. It caused the next instruction to be emulated. */
137 {
138 VM_FF_SET(pVM, VM_FF_SELM_SYNC_LDT);
139 return VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT;
140 }
141
142#ifdef LOG_ENABLED
143 if (Sel == (pRegFrame->cs & X86_SEL_MASK))
144 Log(("GDT write to selector in CS register %04X\n", pRegFrame->cs));
145 else if (Sel == (pRegFrame->ds & X86_SEL_MASK))
146 Log(("GDT write to selector in DS register %04X\n", pRegFrame->ds));
147 else if (Sel == (pRegFrame->es & X86_SEL_MASK))
148 Log(("GDT write to selector in ES register %04X\n", pRegFrame->es));
149 else if (Sel == (pRegFrame->fs & X86_SEL_MASK))
150 Log(("GDT write to selector in FS register %04X\n", pRegFrame->fs));
151 else if (Sel == (pRegFrame->gs & X86_SEL_MASK))
152 Log(("GDT write to selector in GS register %04X\n", pRegFrame->gs));
153 else if (Sel == (pRegFrame->ss & X86_SEL_MASK))
154 Log(("GDT write to selector in SS register %04X\n", pRegFrame->ss));
155#endif
156 return VINF_SUCCESS;
157}
158
159
160/**
161 * \#PF Virtual Handler callback for Guest write access to the Guest's own GDT.
162 *
163 * @returns VBox status code (appropriate for trap handling and GC return).
164 * @param pVM VM Handle.
165 * @param uErrorCode CPU Error code.
166 * @param pRegFrame Trap register frame.
167 * @param pvFault The fault address (cr2).
168 * @param pvRange The base address of the handled virtual range.
169 * @param offRange The offset of the access into this range.
170 * (If it's a EIP range this's the EIP, if not it's pvFault.)
171 */
172VMMRCDECL(int) selmRCGuestGDTWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange)
173{
174 LogFlow(("selmRCGuestGDTWriteHandler errcode=%x fault=%RGv offRange=%08x\n", (uint32_t)uErrorCode, pvFault, offRange));
175
176 /*
177 * First check if this is the LDT entry.
178 * LDT updates are problemous since an invalid LDT entry will cause trouble during worldswitch.
179 */
180 int rc;
181 if (CPUMGetGuestLDTR(pVM) / sizeof(X86DESC) == offRange / sizeof(X86DESC))
182 {
183 Log(("LDTR selector change -> fall back to HC!!\n"));
184 rc = VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT;
185 /** @todo We're not handling changed to the selectors in LDTR and TR correctly at all.
186 * We should ignore any changes to those and sync them only when they are loaded by the guest! */
187 }
188 else
189 {
190 /*
191 * Attempt to emulate the instruction and sync the affected entries.
192 */
193 /** @todo should check if any affected selectors are loaded. */
194 uint32_t cb;
195 rc = EMInterpretInstruction(pVM, pRegFrame, (RTGCPTR)(RTRCUINTPTR)pvFault, &cb);
196 if (RT_SUCCESS(rc) && cb)
197 {
198 unsigned iGDTE1 = offRange / sizeof(X86DESC);
199 int rc2 = selmGCSyncGDTEntry(pVM, pRegFrame, iGDTE1);
200 if (rc2 == VINF_SUCCESS)
201 {
202 Assert(cb);
203 unsigned iGDTE2 = (offRange + cb - 1) / sizeof(X86DESC);
204 if (iGDTE1 != iGDTE2)
205 rc2 = selmGCSyncGDTEntry(pVM, pRegFrame, iGDTE2);
206 if (rc2 == VINF_SUCCESS)
207 {
208 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestGDTHandled);
209 return rc;
210 }
211 }
212 if (rc == VINF_SUCCESS || RT_FAILURE(rc2))
213 rc = rc2;
214 }
215 else
216 {
217 Assert(RT_FAILURE(rc));
218 if (rc == VERR_EM_INTERPRETER)
219 rc = VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT;
220 }
221 }
222 if ( rc != VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT
223 && rc != VINF_EM_RAW_EMULATE_INSTR_TSS_FAULT)
224 {
225 /* Not necessary when we need to go back to the host context to sync the LDT or TSS. */
226 VM_FF_SET(pVM, VM_FF_SELM_SYNC_GDT);
227 }
228 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestGDTUnhandled);
229 return rc;
230}
231
232
233/**
234 * \#PF Virtual Handler callback for Guest write access to the Guest's own LDT.
235 *
236 * @returns VBox status code (appropriate for trap handling and GC return).
237 * @param pVM VM Handle.
238 * @param uErrorCode CPU Error code.
239 * @param pRegFrame Trap register frame.
240 * @param pvFault The fault address (cr2).
241 * @param pvRange The base address of the handled virtual range.
242 * @param offRange The offset of the access into this range.
243 * (If it's a EIP range this's the EIP, if not it's pvFault.)
244 */
245VMMRCDECL(int) selmRCGuestLDTWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange)
246{
247 /** @todo To be implemented. */
248 ////LogCom(("selmRCGuestLDTWriteHandler: eip=%08X pvFault=%RGv pvRange=%RGv\r\n", pRegFrame->eip, pvFault, pvRange));
249
250 VM_FF_SET(pVM, VM_FF_SELM_SYNC_LDT);
251 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestLDT);
252 return VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT;
253}
254
255
256/**
257 * Read wrapper used by selmRCGuestTSSWriteHandler.
258 * @returns VBox status code (appropriate for trap handling and GC return).
259 * @param pVM The VM handle
260 * @param pvDst Where to put the bits we read.
261 * @param pvSrc Guest address to read from.
262 * @param cb The number of bytes to read.
263 */
264DECLINLINE(int) selmRCReadTssBits(PVM pVM, void *pvDst, void const *pvSrc, size_t cb)
265{
266 int rc = MMGCRamRead(pVM, pvDst, (void *)pvSrc, cb);
267 if (RT_SUCCESS(rc))
268 return VINF_SUCCESS;
269
270 /** @todo use different fallback? */
271 rc = PGMPrefetchPage(pVM, (uintptr_t)pvSrc);
272 AssertMsg(rc == VINF_SUCCESS, ("PGMPrefetchPage %p failed with %Rrc\n", &pvSrc, rc));
273 if (rc == VINF_SUCCESS)
274 {
275 rc = MMGCRamRead(pVM, pvDst, (void *)pvSrc, cb);
276 AssertMsg(rc == VINF_SUCCESS, ("MMGCRamRead %p failed with %Rrc\n", &pvSrc, rc));
277 }
278 return rc;
279}
280
281/**
282 * \#PF Virtual Handler callback for Guest write access to the Guest's own current TSS.
283 *
284 * @returns VBox status code (appropriate for trap handling and GC return).
285 * @param pVM VM Handle.
286 * @param uErrorCode CPU Error code.
287 * @param pRegFrame Trap register frame.
288 * @param pvFault The fault address (cr2).
289 * @param pvRange The base address of the handled virtual range.
290 * @param offRange The offset of the access into this range.
291 * (If it's a EIP range this's the EIP, if not it's pvFault.)
292 */
293VMMRCDECL(int) selmRCGuestTSSWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange)
294{
295 LogFlow(("selmRCGuestTSSWriteHandler errcode=%x fault=%RGv offRange=%08x\n", (uint32_t)uErrorCode, pvFault, offRange));
296
297 /*
298 * Try emulate the access.
299 */
300 uint32_t cb;
301 int rc = EMInterpretInstruction(pVM, pRegFrame, (RTGCPTR)(RTRCUINTPTR)pvFault, &cb);
302 if (RT_SUCCESS(rc) && cb)
303 {
304 rc = VINF_SUCCESS;
305
306 /*
307 * If it's on the same page as the esp0 and ss0 fields or actually one of them,
308 * then check if any of these has changed.
309 */
310 PCVBOXTSS pGuestTss = (PVBOXTSS)pVM->selm.s.GCPtrGuestTss;
311 if ( PAGE_ADDRESS(&pGuestTss->esp0) == PAGE_ADDRESS(&pGuestTss->padding_ss0)
312 && PAGE_ADDRESS(&pGuestTss->esp0) == PAGE_ADDRESS((uint8_t *)pGuestTss + offRange)
313 && ( pGuestTss->esp0 != pVM->selm.s.Tss.esp1
314 || pGuestTss->ss0 != (pVM->selm.s.Tss.ss1 & ~1)) /* undo raw-r0 */
315 )
316 {
317 Log(("selmRCGuestTSSWriteHandler: R0 stack: %RTsel:%RGv -> %RTsel:%RGv\n",
318 (RTSEL)(pVM->selm.s.Tss.ss1 & ~1), (RTGCPTR)pVM->selm.s.Tss.esp1, (RTSEL)pGuestTss->ss0, (RTGCPTR)pGuestTss->esp0));
319 pVM->selm.s.Tss.esp1 = pGuestTss->esp0;
320 pVM->selm.s.Tss.ss1 = pGuestTss->ss0 | 1;
321 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestTSSHandledChanged);
322 }
323 /* Handle misaligned TSS in a safe manner (just in case). */
324 else if ( offRange >= RT_UOFFSETOF(VBOXTSS, esp0)
325 && offRange < RT_UOFFSETOF(VBOXTSS, padding_ss0))
326 {
327 struct
328 {
329 uint32_t esp0;
330 uint16_t ss0;
331 uint16_t padding_ss0;
332 } s;
333 AssertCompileSize(s, 8);
334 rc = selmRCReadTssBits(pVM, &s, &pGuestTss->esp0, sizeof(s));
335 if ( rc == VINF_SUCCESS
336 && ( s.esp0 != pVM->selm.s.Tss.esp1
337 || s.ss0 != (pVM->selm.s.Tss.ss1 & ~1)) /* undo raw-r0 */
338 )
339 {
340 Log(("selmRCGuestTSSWriteHandler: R0 stack: %RTsel:%RGv -> %RTsel:%RGv [x-page]\n",
341 (RTSEL)(pVM->selm.s.Tss.ss1 & ~1), (RTGCPTR)pVM->selm.s.Tss.esp1, (RTSEL)s.ss0, (RTGCPTR)s.esp0));
342 pVM->selm.s.Tss.esp1 = s.esp0;
343 pVM->selm.s.Tss.ss1 = s.ss0 | 1;
344 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestTSSHandledChanged);
345 }
346 }
347
348 /*
349 * If VME is enabled we need to check if the interrupt redirection bitmap
350 * needs updating.
351 */
352 if ( offRange >= RT_UOFFSETOF(VBOXTSS, offIoBitmap)
353 && (CPUMGetGuestCR4(pVM) & X86_CR4_VME))
354 {
355 if (offRange - RT_UOFFSETOF(VBOXTSS, offIoBitmap) < sizeof(pGuestTss->offIoBitmap))
356 {
357 uint16_t offIoBitmap = pGuestTss->offIoBitmap;
358 if (offIoBitmap != pVM->selm.s.offGuestIoBitmap)
359 {
360 Log(("TSS offIoBitmap changed: old=%#x new=%#x -> resync in ring-3\n", pVM->selm.s.offGuestIoBitmap, offIoBitmap));
361 VM_FF_SET(pVM, VM_FF_SELM_SYNC_TSS);
362 VM_FF_SET(pVM, VM_FF_TO_R3);
363 }
364 else
365 Log(("TSS offIoBitmap: old=%#x new=%#x [unchanged]\n", pVM->selm.s.offGuestIoBitmap, offIoBitmap));
366 }
367 else
368 {
369 /** @todo not sure how the partial case is handled; probably not allowed */
370 uint32_t offIntRedirBitmap = pVM->selm.s.offGuestIoBitmap - sizeof(pVM->selm.s.Tss.IntRedirBitmap);
371 if ( offIntRedirBitmap <= offRange
372 && offIntRedirBitmap + sizeof(pVM->selm.s.Tss.IntRedirBitmap) >= offRange + cb
373 && offIntRedirBitmap + sizeof(pVM->selm.s.Tss.IntRedirBitmap) <= pVM->selm.s.cbGuestTss)
374 {
375 Log(("TSS IntRedirBitmap Changed: offIoBitmap=%x offIntRedirBitmap=%x cbTSS=%x offRange=%x cb=%x\n",
376 pVM->selm.s.offGuestIoBitmap, offIntRedirBitmap, pVM->selm.s.cbGuestTss, offRange, cb));
377
378 /** @todo only update the changed part. */
379 for (uint32_t i = 0; i < sizeof(pVM->selm.s.Tss.IntRedirBitmap) / 8; i++)
380 {
381 rc = selmRCReadTssBits(pVM, &pVM->selm.s.Tss.IntRedirBitmap[i * 8],
382 (uint8_t *)pGuestTss + offIntRedirBitmap + i * 8, 8);
383 if (rc != VINF_SUCCESS)
384 break;
385 }
386 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestTSSRedir);
387 }
388 }
389 }
390
391 /* Return to ring-3 for a full resync if any of the above fails... (?) */
392 if (rc != VINF_SUCCESS)
393 {
394 VM_FF_SET(pVM, VM_FF_SELM_SYNC_TSS);
395 VM_FF_SET(pVM, VM_FF_TO_R3);
396 if (RT_SUCCESS(rc))
397 rc = VINF_SUCCESS;
398 }
399
400 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestTSSHandled);
401 }
402 else
403 {
404 Assert(RT_FAILURE(rc));
405 VM_FF_SET(pVM, VM_FF_SELM_SYNC_TSS);
406 STAM_COUNTER_INC(&pVM->selm.s.StatRCWriteGuestTSSUnhandled);
407 if (rc == VERR_EM_INTERPRETER)
408 rc = VINF_EM_RAW_EMULATE_INSTR_TSS_FAULT;
409 }
410 return rc;
411}
412
413
414/**
415 * \#PF Virtual Handler callback for Guest write access to the VBox shadow GDT.
416 *
417 * @returns VBox status code (appropriate for trap handling and GC return).
418 * @param pVM VM Handle.
419 * @param uErrorCode CPU Error code.
420 * @param pRegFrame Trap register frame.
421 * @param pvFault The fault address (cr2).
422 * @param pvRange The base address of the handled virtual range.
423 * @param offRange The offset of the access into this range.
424 * (If it's a EIP range this's the EIP, if not it's pvFault.)
425 */
426VMMRCDECL(int) selmRCShadowGDTWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange)
427{
428 LogRel(("FATAL ERROR: selmRCShadowGDTWriteHandler: eip=%08X pvFault=%RGv pvRange=%RGv\r\n", pRegFrame->eip, pvFault, pvRange));
429 return VERR_SELM_SHADOW_GDT_WRITE;
430}
431
432
433/**
434 * \#PF Virtual Handler callback for Guest write access to the VBox shadow LDT.
435 *
436 * @returns VBox status code (appropriate for trap handling and GC return).
437 * @param pVM VM Handle.
438 * @param uErrorCode CPU Error code.
439 * @param pRegFrame Trap register frame.
440 * @param pvFault The fault address (cr2).
441 * @param pvRange The base address of the handled virtual range.
442 * @param offRange The offset of the access into this range.
443 * (If it's a EIP range this's the EIP, if not it's pvFault.)
444 */
445VMMRCDECL(int) selmRCShadowLDTWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange)
446{
447 LogRel(("FATAL ERROR: selmRCShadowLDTWriteHandler: eip=%08X pvFault=%RGv pvRange=%RGv\r\n", pRegFrame->eip, pvFault, pvRange));
448 Assert((RTRCPTR)pvFault >= pVM->selm.s.pvLdtRC && (RTRCUINTPTR)pvFault < (RTRCUINTPTR)pVM->selm.s.pvLdtRC + 65536 + PAGE_SIZE);
449 return VERR_SELM_SHADOW_LDT_WRITE;
450}
451
452
453/**
454 * \#PF Virtual Handler callback for Guest write access to the VBox shadow TSS.
455 *
456 * @returns VBox status code (appropriate for trap handling and GC return).
457 * @param pVM VM Handle.
458 * @param uErrorCode CPU Error code.
459 * @param pRegFrame Trap register frame.
460 * @param pvFault The fault address (cr2).
461 * @param pvRange The base address of the handled virtual range.
462 * @param offRange The offset of the access into this range.
463 * (If it's a EIP range this's the EIP, if not it's pvFault.)
464 */
465VMMRCDECL(int) selmRCShadowTSSWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange)
466{
467 LogRel(("FATAL ERROR: selmRCShadowTSSWriteHandler: eip=%08X pvFault=%RGv pvRange=%RGv\r\n", pRegFrame->eip, pvFault, pvRange));
468 return VERR_SELM_SHADOW_TSS_WRITE;
469}
470
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