VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMRZ/VMMRZ.cpp@ 49409

Last change on this file since 49409 was 48473, checked in by vboxsync, 11 years ago

VMM/HMVMXR0: Try restore state properly on assertions in longjmps contexts. Reevaluate FPU state after injecting #NM. Other cosmetics.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.1 KB
Line 
1/* $Id: VMMRZ.cpp 48473 2013-09-13 14:17:58Z vboxsync $ */
2/** @file
3 * VMM - Virtual Machine Monitor, Raw-mode and ring-0 context code.
4 */
5
6/*
7 * Copyright (C) 2009-2012 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#include <VBox/vmm/vmm.h>
23#include "VMMInternal.h"
24#include <VBox/vmm/vm.h>
25#include <VBox/err.h>
26
27#include <iprt/assert.h>
28#include <iprt/string.h>
29
30
31/**
32 * Calls the ring-3 host code.
33 *
34 * @returns VBox status code of the ring-3 call.
35 * @retval VERR_VMM_RING3_CALL_DISABLED if called at the wrong time. This must
36 * be passed up the stack, or if that isn't possible then VMMRZCallRing3
37 * needs to change it into an assertion.
38 *
39 *
40 * @param pVM Pointer to the VM.
41 * @param pVCpu Pointer to the VMCPU of the calling EMT.
42 * @param enmOperation The operation.
43 * @param uArg The argument to the operation.
44 */
45VMMRZDECL(int) VMMRZCallRing3(PVM pVM, PVMCPU pVCpu, VMMCALLRING3 enmOperation, uint64_t uArg)
46{
47 VMCPU_ASSERT_EMT(pVCpu);
48
49 /*
50 * Check if calling ring-3 has been disabled and only let let fatal calls thru.
51 */
52 if (RT_UNLIKELY( pVCpu->vmm.s.cCallRing3Disabled != 0
53 && enmOperation != VMMCALLRING3_VM_R0_ASSERTION))
54 {
55#ifndef IN_RING0
56 /*
57 * In most cases, it's sufficient to return a status code which
58 * will then be propagated up the code usually encountering several
59 * AssertRC invocations along the way. Hitting one of those is more
60 * helpful than stopping here.
61 *
62 * However, some doesn't check the status code because they are called
63 * from void functions, and for these we'll turn this into a ring-0
64 * assertion host call.
65 */
66 if (enmOperation != VMMCALLRING3_REM_REPLAY_HANDLER_NOTIFICATIONS)
67 return VERR_VMM_RING3_CALL_DISABLED;
68#endif
69#ifdef IN_RC
70 RTStrPrintf(g_szRTAssertMsg1, sizeof(pVM->vmm.s.szRing0AssertMsg1),
71 "VMMRZCallRing3: enmOperation=%d uArg=%#llx idCpu=%#x\n", enmOperation, uArg, pVCpu->idCpu);
72#endif
73 RTStrPrintf(pVM->vmm.s.szRing0AssertMsg1, sizeof(pVM->vmm.s.szRing0AssertMsg1),
74 "VMMRZCallRing3: enmOperation=%d uArg=%#llx idCpu=%#x\n", enmOperation, uArg, pVCpu->idCpu);
75 enmOperation = VMMCALLRING3_VM_R0_ASSERTION;
76 }
77
78 /*
79 * The normal path.
80 */
81/** @todo profile this! */
82 pVCpu->vmm.s.enmCallRing3Operation = enmOperation;
83 pVCpu->vmm.s.u64CallRing3Arg = uArg;
84 pVCpu->vmm.s.rcCallRing3 = VERR_VMM_RING3_CALL_NO_RC;
85#ifdef IN_RC
86 pVM->vmm.s.pfnRCToHost(VINF_VMM_CALL_HOST);
87#else
88 int rc;
89 if (pVCpu->vmm.s.pfnCallRing3CallbackR0)
90 {
91 rc = pVCpu->vmm.s.pfnCallRing3CallbackR0(pVCpu, enmOperation, pVCpu->vmm.s.pvCallRing3CallbackUserR0);
92 if (RT_FAILURE(rc))
93 return rc;
94 }
95 rc = vmmR0CallRing3LongJmp(&pVCpu->vmm.s.CallRing3JmpBufR0, VINF_VMM_CALL_HOST);
96 if (RT_FAILURE(rc))
97 return rc;
98#endif
99 return pVCpu->vmm.s.rcCallRing3;
100}
101
102
103/**
104 * Simple wrapper that adds the pVCpu argument.
105 *
106 * @returns VBox status code of the ring-3 call.
107 * @retval VERR_VMM_RING3_CALL_DISABLED if called at the wrong time. This must
108 * be passed up the stack, or if that isn't possible then VMMRZCallRing3
109 * needs to change it into an assertion.
110 *
111 * @param pVM Pointer to the VM.
112 * @param enmOperation The operation.
113 * @param uArg The argument to the operation.
114 */
115VMMRZDECL(int) VMMRZCallRing3NoCpu(PVM pVM, VMMCALLRING3 enmOperation, uint64_t uArg)
116{
117 return VMMRZCallRing3(pVM, VMMGetCpu(pVM), enmOperation, uArg);
118}
119
120
121/**
122 * Disables all host calls, except certain fatal ones.
123 *
124 * @param pVCpu The CPU struct for the calling EMT.
125 * @thread EMT.
126 */
127VMMRZDECL(void) VMMRZCallRing3Disable(PVMCPU pVCpu)
128{
129 VMCPU_ASSERT_EMT(pVCpu);
130 Assert(pVCpu->vmm.s.cCallRing3Disabled < 16);
131 if (++pVCpu->vmm.s.cCallRing3Disabled == 1)
132 {
133 /** @todo it might make more sense to just disable logging here, then we
134 * won't flush away important bits... but that goes both ways really. */
135#ifdef IN_RC
136 pVCpu->pVMRC->vmm.s.fRCLoggerFlushingDisabled = true;
137#else
138# ifdef LOG_ENABLED
139 if (pVCpu->vmm.s.pR0LoggerR0)
140 pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled = true;
141# endif
142#endif
143 }
144}
145
146
147/**
148 * Counters VMMRZCallRing3Disable and re-enables host calls.
149 *
150 * @param pVCpu The CPU struct for the calling EMT.
151 * @thread EMT.
152 */
153VMMRZDECL(void) VMMRZCallRing3Enable(PVMCPU pVCpu)
154{
155 VMCPU_ASSERT_EMT(pVCpu);
156 Assert(pVCpu->vmm.s.cCallRing3Disabled > 0);
157 if (--pVCpu->vmm.s.cCallRing3Disabled == 0)
158 {
159#ifdef IN_RC
160 pVCpu->pVMRC->vmm.s.fRCLoggerFlushingDisabled = false;
161#else
162# ifdef LOG_ENABLED
163 if (pVCpu->vmm.s.pR0LoggerR0)
164 pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled = false;
165# endif
166#endif
167 }
168}
169
170
171/**
172 * Checks whether its possible to call host context or not.
173 *
174 * @returns true if it's safe, false if it isn't.
175 * @param pVCpu The CPU struct for the calling EMT.
176 */
177VMMRZDECL(bool) VMMRZCallRing3IsEnabled(PVMCPU pVCpu)
178{
179 VMCPU_ASSERT_EMT(pVCpu);
180 Assert(pVCpu->vmm.s.cCallRing3Disabled <= 16);
181 return pVCpu->vmm.s.cCallRing3Disabled == 0;
182}
183
184
185/**
186 * Sets the ring-0 callback before doing the ring-3 call.
187 *
188 * @param pVCpu Pointer to the VMCPU.
189 * @param pfnCallback Pointer to the callback.
190 * @param pvUser The user argument.
191 *
192 * @return VBox status code.
193 */
194VMMRZDECL(int) VMMRZCallRing3SetNotification(PVMCPU pVCpu, R0PTRTYPE(PFNVMMR0CALLRING3NOTIFICATION) pfnCallback, RTR0PTR pvUser)
195{
196 AssertPtrReturn(pVCpu, VERR_INVALID_POINTER);
197 AssertPtrReturn(pfnCallback, VERR_INVALID_POINTER);
198
199 if (pVCpu->vmm.s.pfnCallRing3CallbackR0)
200 return VERR_ALREADY_EXISTS;
201
202 pVCpu->vmm.s.pfnCallRing3CallbackR0 = pfnCallback;
203 pVCpu->vmm.s.pvCallRing3CallbackUserR0 = pvUser;
204 return VINF_SUCCESS;
205}
206
207
208/**
209 * Removes the ring-0 callback.
210 *
211 * @param pVCpu Pointer to the VMCPU.
212 */
213VMMRZDECL(void) VMMRZCallRing3RemoveNotification(PVMCPU pVCpu)
214{
215 pVCpu->vmm.s.pfnCallRing3CallbackR0 = NULL;
216}
217
218
219/**
220 * Checks whether there is a ring-0 callback notification active.
221 *
222 * @param pVCpu Pointer to the VMCPU.
223 * @returns true if there the notification is active, false otherwise.
224 */
225VMMRZDECL(bool) VMMRZCallRing3IsNotificationSet(PVMCPU pVCpu)
226{
227 return pVCpu->vmm.s.pfnCallRing3CallbackR0 != NULL;
228}
229
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