VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/TRPMAll.cpp@ 80348

Last change on this file since 80348 was 80333, checked in by vboxsync, 5 years ago

VMM: Eliminating the VBOX_BUGREF_9217_PART_I preprocessor macro. bugref:9217

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 17.8 KB
Line 
1/* $Id: TRPMAll.cpp 80333 2019-08-16 20:28:38Z vboxsync $ */
2/** @file
3 * TRPM - Trap Monitor - Any Context.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_TRPM
23#include <VBox/vmm/trpm.h>
24#include <VBox/vmm/pgm.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/hm.h>
27#include <VBox/vmm/selm.h>
28#include <VBox/vmm/stam.h>
29#include <VBox/vmm/dbgf.h>
30#include "TRPMInternal.h"
31#include <VBox/vmm/vmcc.h>
32#include <VBox/err.h>
33#include <VBox/vmm/em.h>
34#include <VBox/log.h>
35#include <iprt/assert.h>
36#include <iprt/asm.h>
37#include <iprt/asm-amd64-x86.h>
38#include <iprt/param.h>
39#include <iprt/x86.h>
40
41
42
43/**
44 * Query info about the current active trap/interrupt.
45 * If no trap is active active an error code is returned.
46 *
47 * @returns VBox status code.
48 * @param pVCpu The cross context virtual CPU structure.
49 * @param pu8TrapNo Where to store the trap number.
50 * @param penmType Where to store the trap type
51 */
52VMMDECL(int) TRPMQueryTrap(PVMCPU pVCpu, uint8_t *pu8TrapNo, TRPMEVENT *penmType)
53{
54 /*
55 * Check if we have a trap at present.
56 */
57 if (pVCpu->trpm.s.uActiveVector != ~0U)
58 {
59 if (pu8TrapNo)
60 *pu8TrapNo = (uint8_t)pVCpu->trpm.s.uActiveVector;
61 if (penmType)
62 *penmType = pVCpu->trpm.s.enmActiveType;
63 return VINF_SUCCESS;
64 }
65
66 return VERR_TRPM_NO_ACTIVE_TRAP;
67}
68
69
70/**
71 * Gets the trap number for the current trap.
72 *
73 * The caller is responsible for making sure there is an active trap which
74 * takes an error code when making this request.
75 *
76 * @returns The current trap number.
77 * @param pVCpu The cross context virtual CPU structure.
78 */
79VMMDECL(uint8_t) TRPMGetTrapNo(PVMCPU pVCpu)
80{
81 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
82 return (uint8_t)pVCpu->trpm.s.uActiveVector;
83}
84
85
86/**
87 * Gets the error code for the current trap.
88 *
89 * The caller is responsible for making sure there is an active trap which
90 * takes an error code when making this request.
91 *
92 * @returns Error code.
93 * @param pVCpu The cross context virtual CPU structure.
94 */
95VMMDECL(RTGCUINT) TRPMGetErrorCode(PVMCPU pVCpu)
96{
97 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
98#ifdef VBOX_STRICT
99 switch (pVCpu->trpm.s.uActiveVector)
100 {
101 case X86_XCPT_TS:
102 case X86_XCPT_NP:
103 case X86_XCPT_SS:
104 case X86_XCPT_GP:
105 case X86_XCPT_PF:
106 case X86_XCPT_AC:
107 case X86_XCPT_DF:
108 break;
109 default:
110 AssertMsgFailed(("This trap (%#x) doesn't have any error code\n", pVCpu->trpm.s.uActiveVector));
111 break;
112 }
113#endif
114 return pVCpu->trpm.s.uActiveErrorCode;
115}
116
117
118/**
119 * Gets the fault address for the current trap.
120 *
121 * The caller is responsible for making sure there is an active trap 0x0e when
122 * making this request.
123 *
124 * @returns Fault address associated with the trap.
125 * @param pVCpu The cross context virtual CPU structure.
126 */
127VMMDECL(RTGCUINTPTR) TRPMGetFaultAddress(PVMCPU pVCpu)
128{
129 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
130 AssertMsg(pVCpu->trpm.s.uActiveVector == X86_XCPT_PF, ("Not page-fault trap!\n"));
131 return pVCpu->trpm.s.uActiveCR2;
132}
133
134
135/**
136 * Gets the instruction-length for the current trap (only relevant for software
137 * interrupts and software exceptions \#BP and \#OF).
138 *
139 * The caller is responsible for making sure there is an active trap 0x0e when
140 * making this request.
141 *
142 * @returns Fault address associated with the trap.
143 * @param pVCpu The cross context virtual CPU structure.
144 */
145VMMDECL(uint8_t) TRPMGetInstrLength(PVMCPU pVCpu)
146{
147 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
148 return pVCpu->trpm.s.cbInstr;
149}
150
151
152/**
153 * Clears the current active trap/exception/interrupt.
154 *
155 * The caller is responsible for making sure there is an active trap
156 * when making this request.
157 *
158 * @returns VBox status code.
159 * @param pVCpu The cross context virtual CPU structure.
160 */
161VMMDECL(int) TRPMResetTrap(PVMCPU pVCpu)
162{
163 /*
164 * Cannot reset non-existing trap!
165 */
166 if (pVCpu->trpm.s.uActiveVector == ~0U)
167 {
168 AssertMsgFailed(("No active trap!\n"));
169 return VERR_TRPM_NO_ACTIVE_TRAP;
170 }
171
172 /*
173 * Reset it.
174 */
175 pVCpu->trpm.s.uActiveVector = ~0U;
176 return VINF_SUCCESS;
177}
178
179
180/**
181 * Assert trap/exception/interrupt.
182 *
183 * The caller is responsible for making sure there is no active trap
184 * when making this request.
185 *
186 * @returns VBox status code.
187 * @param pVCpu The cross context virtual CPU structure.
188 * @param u8TrapNo The trap vector to assert.
189 * @param enmType Trap type.
190 */
191VMMDECL(int) TRPMAssertTrap(PVMCPUCC pVCpu, uint8_t u8TrapNo, TRPMEVENT enmType)
192{
193 Log2(("TRPMAssertTrap: u8TrapNo=%02x type=%d\n", u8TrapNo, enmType));
194
195 /*
196 * Cannot assert a trap when one is already active.
197 */
198 if (pVCpu->trpm.s.uActiveVector != ~0U)
199 {
200 AssertMsgFailed(("CPU%d: Active trap %#x\n", pVCpu->idCpu, pVCpu->trpm.s.uActiveVector));
201 return VERR_TRPM_ACTIVE_TRAP;
202 }
203
204 pVCpu->trpm.s.uActiveVector = u8TrapNo;
205 pVCpu->trpm.s.enmActiveType = enmType;
206 pVCpu->trpm.s.uActiveErrorCode = ~(RTGCUINT)0;
207 pVCpu->trpm.s.uActiveCR2 = 0xdeadface;
208 pVCpu->trpm.s.cbInstr = UINT8_MAX;
209 return VINF_SUCCESS;
210}
211
212
213/**
214 * Assert a page-fault exception.
215 *
216 * The caller is responsible for making sure there is no active trap
217 * when making this request.
218 *
219 * @returns VBox status code.
220 * @param pVCpu The cross context virtual CPU structure.
221 * @param uCR2 The new fault address.
222 * @param uErrorCode The error code for the page-fault.
223 */
224VMMDECL(int) TRPMAssertXcptPF(PVMCPUCC pVCpu, RTGCUINTPTR uCR2, RTGCUINT uErrorCode)
225{
226 Log2(("TRPMAssertXcptPF: uCR2=%RGv uErrorCode=%RGv\n", uCR2, uErrorCode)); /** @todo RTGCUINT to be fixed. */
227
228 /*
229 * Cannot assert a trap when one is already active.
230 */
231 if (pVCpu->trpm.s.uActiveVector != ~0U)
232 {
233 AssertMsgFailed(("CPU%d: Active trap %#x\n", pVCpu->idCpu, pVCpu->trpm.s.uActiveVector));
234 return VERR_TRPM_ACTIVE_TRAP;
235 }
236
237 pVCpu->trpm.s.uActiveVector = X86_XCPT_PF;
238 pVCpu->trpm.s.enmActiveType = TRPM_TRAP;
239 pVCpu->trpm.s.uActiveErrorCode = uErrorCode;
240 pVCpu->trpm.s.uActiveCR2 = uCR2;
241 pVCpu->trpm.s.cbInstr = UINT8_MAX;
242 return VINF_SUCCESS;
243}
244
245
246/**
247 * Sets the error code of the current trap.
248 * (This function is for use in trap handlers and such.)
249 *
250 * The caller is responsible for making sure there is an active trap
251 * which takes an errorcode when making this request.
252 *
253 * @param pVCpu The cross context virtual CPU structure.
254 * @param uErrorCode The new error code.
255 */
256VMMDECL(void) TRPMSetErrorCode(PVMCPU pVCpu, RTGCUINT uErrorCode)
257{
258 Log2(("TRPMSetErrorCode: uErrorCode=%RGv\n", uErrorCode)); /** @todo RTGCUINT mess! */
259 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
260 pVCpu->trpm.s.uActiveErrorCode = uErrorCode;
261#ifdef VBOX_STRICT
262 switch (pVCpu->trpm.s.uActiveVector)
263 {
264 case X86_XCPT_TS: case X86_XCPT_NP: case X86_XCPT_SS: case X86_XCPT_GP: case X86_XCPT_PF:
265 AssertMsg(uErrorCode != ~(RTGCUINT)0, ("Invalid uErrorCode=%#x u8TrapNo=%d\n", uErrorCode, pVCpu->trpm.s.uActiveVector));
266 break;
267 case X86_XCPT_AC: case X86_XCPT_DF:
268 AssertMsg(uErrorCode == 0, ("Invalid uErrorCode=%#x u8TrapNo=%d\n", uErrorCode, pVCpu->trpm.s.uActiveVector));
269 break;
270 default:
271 AssertMsg(uErrorCode == ~(RTGCUINT)0, ("Invalid uErrorCode=%#x u8TrapNo=%d\n", uErrorCode, pVCpu->trpm.s.uActiveVector));
272 break;
273 }
274#endif
275}
276
277
278/**
279 * Sets the fault address of the current \#PF trap. (This function is for use in
280 * trap handlers and such.)
281 *
282 * The caller is responsible for making sure there is an active trap 0e
283 * when making this request.
284 *
285 * @param pVCpu The cross context virtual CPU structure.
286 * @param uCR2 The new fault address (cr2 register).
287 */
288VMMDECL(void) TRPMSetFaultAddress(PVMCPU pVCpu, RTGCUINTPTR uCR2)
289{
290 Log2(("TRPMSetFaultAddress: uCR2=%RGv\n", uCR2));
291 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
292 AssertMsg(pVCpu->trpm.s.enmActiveType == TRPM_TRAP, ("Not hardware exception!\n"));
293 AssertMsg(pVCpu->trpm.s.uActiveVector == X86_XCPT_PF, ("Not trap 0e!\n"));
294 pVCpu->trpm.s.uActiveCR2 = uCR2;
295}
296
297
298/**
299 * Sets the instruction-length of the current trap (relevant for software
300 * interrupts and software exceptions like \#BP, \#OF).
301 *
302 * The caller is responsible for making sure there is an active trap when making
303 * this request.
304 *
305 * @param pVCpu The cross context virtual CPU structure.
306 * @param cbInstr The instruction length.
307 */
308VMMDECL(void) TRPMSetInstrLength(PVMCPU pVCpu, uint8_t cbInstr)
309{
310 Log2(("TRPMSetInstrLength: cbInstr=%u\n", cbInstr));
311 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
312 /** @todo We should be able to set the instruction length for a \#DB raised by
313 * INT1/ICEBP as well. However, TRPM currently has no way to distinguish
314 * INT1/ICEBP from regular \#DB. */
315 AssertMsg( pVCpu->trpm.s.enmActiveType == TRPM_SOFTWARE_INT
316 || ( pVCpu->trpm.s.enmActiveType == TRPM_TRAP
317 && ( pVCpu->trpm.s.uActiveVector == X86_XCPT_BP
318 || pVCpu->trpm.s.uActiveVector == X86_XCPT_OF)),
319 ("Invalid trap type %#x\n", pVCpu->trpm.s.enmActiveType));
320 pVCpu->trpm.s.cbInstr = cbInstr;
321}
322
323
324/**
325 * Checks if the current active trap/interrupt/exception/fault/whatever is a software
326 * interrupt or not.
327 *
328 * The caller is responsible for making sure there is an active trap
329 * when making this request.
330 *
331 * @returns true if software interrupt, false if not.
332 *
333 * @param pVCpu The cross context virtual CPU structure.
334 */
335VMMDECL(bool) TRPMIsSoftwareInterrupt(PVMCPU pVCpu)
336{
337 AssertMsg(pVCpu->trpm.s.uActiveVector != ~0U, ("No active trap!\n"));
338 return (pVCpu->trpm.s.enmActiveType == TRPM_SOFTWARE_INT);
339}
340
341
342/**
343 * Check if there is an active trap.
344 *
345 * @returns true if trap active, false if not.
346 * @param pVCpu The cross context virtual CPU structure.
347 */
348VMMDECL(bool) TRPMHasTrap(PVMCPU pVCpu)
349{
350 return pVCpu->trpm.s.uActiveVector != ~0U;
351}
352
353
354/**
355 * Query all info about the current active trap/interrupt.
356 * If no trap is active active an error code is returned.
357 *
358 * @returns VBox status code.
359 * @param pVCpu The cross context virtual CPU structure.
360 * @param pu8TrapNo Where to store the trap number.
361 * @param pEnmType Where to store the trap type
362 * @param puErrorCode Where to store the error code associated with some traps.
363 * ~0U is stored if the trap has no error code.
364 * @param puCR2 Where to store the CR2 associated with a trap 0E.
365 * @param pcbInstr Where to store the instruction-length
366 * associated with some traps.
367 */
368VMMDECL(int) TRPMQueryTrapAll(PVMCPU pVCpu, uint8_t *pu8TrapNo, TRPMEVENT *pEnmType, PRTGCUINT puErrorCode, PRTGCUINTPTR puCR2,
369 uint8_t *pcbInstr)
370{
371 /*
372 * Check if we have a trap at present.
373 */
374 if (pVCpu->trpm.s.uActiveVector == ~0U)
375 return VERR_TRPM_NO_ACTIVE_TRAP;
376
377 if (pu8TrapNo)
378 *pu8TrapNo = (uint8_t)pVCpu->trpm.s.uActiveVector;
379 if (pEnmType)
380 *pEnmType = pVCpu->trpm.s.enmActiveType;
381 if (puErrorCode)
382 *puErrorCode = pVCpu->trpm.s.uActiveErrorCode;
383 if (puCR2)
384 *puCR2 = pVCpu->trpm.s.uActiveCR2;
385 if (pcbInstr)
386 *pcbInstr = pVCpu->trpm.s.cbInstr;
387 return VINF_SUCCESS;
388}
389
390
391/**
392 * Save the active trap.
393 *
394 * This routine useful when doing try/catch in the hypervisor.
395 * Any function which uses temporary trap handlers should
396 * probably also use this facility to save the original trap.
397 *
398 * @param pVCpu The cross context virtual CPU structure.
399 */
400VMMDECL(void) TRPMSaveTrap(PVMCPU pVCpu)
401{
402 pVCpu->trpm.s.uSavedVector = pVCpu->trpm.s.uActiveVector;
403 pVCpu->trpm.s.enmSavedType = pVCpu->trpm.s.enmActiveType;
404 pVCpu->trpm.s.uSavedErrorCode = pVCpu->trpm.s.uActiveErrorCode;
405 pVCpu->trpm.s.uSavedCR2 = pVCpu->trpm.s.uActiveCR2;
406 pVCpu->trpm.s.cbSavedInstr = pVCpu->trpm.s.cbInstr;
407}
408
409
410/**
411 * Restore a saved trap.
412 *
413 * Multiple restores of a saved trap is possible.
414 *
415 * @param pVCpu The cross context virtual CPU structure.
416 */
417VMMDECL(void) TRPMRestoreTrap(PVMCPU pVCpu)
418{
419 pVCpu->trpm.s.uActiveVector = pVCpu->trpm.s.uSavedVector;
420 pVCpu->trpm.s.enmActiveType = pVCpu->trpm.s.enmSavedType;
421 pVCpu->trpm.s.uActiveErrorCode = pVCpu->trpm.s.uSavedErrorCode;
422 pVCpu->trpm.s.uActiveCR2 = pVCpu->trpm.s.uSavedCR2;
423 pVCpu->trpm.s.cbInstr = pVCpu->trpm.s.cbSavedInstr;
424}
425
426
427/**
428 * Raises a cpu exception which doesn't take an error code.
429 *
430 * This function may or may not dispatch the exception before returning.
431 *
432 * @returns VBox status code fit for scheduling.
433 * @retval VINF_EM_RAW_GUEST_TRAP if the exception was left pending.
434 * @retval VINF_TRPM_XCPT_DISPATCHED if the exception was raised and dispatched for raw-mode execution.
435 * @retval VINF_EM_RESCHEDULE_REM if the exception was dispatched and cannot be executed in raw-mode.
436 *
437 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
438 * @param pCtxCore The CPU context core.
439 * @param enmXcpt The exception.
440 */
441VMMDECL(int) TRPMRaiseXcpt(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, X86XCPT enmXcpt)
442{
443 LogFlow(("TRPMRaiseXcptErr: cs:eip=%RTsel:%RX32 enmXcpt=%#x\n", pCtxCore->cs.Sel, pCtxCore->eip, enmXcpt));
444 NOREF(pCtxCore);
445/** @todo dispatch the trap. */
446 pVCpu->trpm.s.uActiveVector = enmXcpt;
447 pVCpu->trpm.s.enmActiveType = TRPM_TRAP;
448 pVCpu->trpm.s.uActiveErrorCode = 0xdeadbeef;
449 pVCpu->trpm.s.uActiveCR2 = 0xdeadface;
450 pVCpu->trpm.s.cbInstr = UINT8_MAX;
451 return VINF_EM_RAW_GUEST_TRAP;
452}
453
454
455/**
456 * Raises a cpu exception with an errorcode.
457 *
458 * This function may or may not dispatch the exception before returning.
459 *
460 * @returns VBox status code fit for scheduling.
461 * @retval VINF_EM_RAW_GUEST_TRAP if the exception was left pending.
462 * @retval VINF_TRPM_XCPT_DISPATCHED if the exception was raised and dispatched for raw-mode execution.
463 * @retval VINF_EM_RESCHEDULE_REM if the exception was dispatched and cannot be executed in raw-mode.
464 *
465 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
466 * @param pCtxCore The CPU context core.
467 * @param enmXcpt The exception.
468 * @param uErr The error code.
469 */
470VMMDECL(int) TRPMRaiseXcptErr(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, X86XCPT enmXcpt, uint32_t uErr)
471{
472 LogFlow(("TRPMRaiseXcptErr: cs:eip=%RTsel:%RX32 enmXcpt=%#x uErr=%RX32\n", pCtxCore->cs.Sel, pCtxCore->eip, enmXcpt, uErr));
473 NOREF(pCtxCore);
474/** @todo dispatch the trap. */
475 pVCpu->trpm.s.uActiveVector = enmXcpt;
476 pVCpu->trpm.s.enmActiveType = TRPM_TRAP;
477 pVCpu->trpm.s.uActiveErrorCode = uErr;
478 pVCpu->trpm.s.uActiveCR2 = 0xdeadface;
479 pVCpu->trpm.s.cbInstr = UINT8_MAX;
480 return VINF_EM_RAW_GUEST_TRAP;
481}
482
483
484/**
485 * Raises a cpu exception with an errorcode and CR2.
486 *
487 * This function may or may not dispatch the exception before returning.
488 *
489 * @returns VBox status code fit for scheduling.
490 * @retval VINF_EM_RAW_GUEST_TRAP if the exception was left pending.
491 * @retval VINF_TRPM_XCPT_DISPATCHED if the exception was raised and dispatched for raw-mode execution.
492 * @retval VINF_EM_RESCHEDULE_REM if the exception was dispatched and cannot be executed in raw-mode.
493 *
494 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
495 * @param pCtxCore The CPU context core.
496 * @param enmXcpt The exception.
497 * @param uErr The error code.
498 * @param uCR2 The CR2 value.
499 */
500VMMDECL(int) TRPMRaiseXcptErrCR2(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, X86XCPT enmXcpt, uint32_t uErr, RTGCUINTPTR uCR2)
501{
502 LogFlow(("TRPMRaiseXcptErr: cs:eip=%RTsel:%RX32 enmXcpt=%#x uErr=%RX32 uCR2=%RGv\n", pCtxCore->cs.Sel, pCtxCore->eip, enmXcpt, uErr, uCR2));
503 NOREF(pCtxCore);
504/** @todo dispatch the trap. */
505 pVCpu->trpm.s.uActiveVector = enmXcpt;
506 pVCpu->trpm.s.enmActiveType = TRPM_TRAP;
507 pVCpu->trpm.s.uActiveErrorCode = uErr;
508 pVCpu->trpm.s.uActiveCR2 = uCR2;
509 pVCpu->trpm.s.cbInstr = UINT8_MAX;
510 return VINF_EM_RAW_GUEST_TRAP;
511}
512
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