VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR0/VMMR0.cpp@ 19773

Last change on this file since 19773 was 19734, checked in by vboxsync, 16 years ago

Must export vmmR0LoggerFlush

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 44.9 KB
Line 
1/* $Id: VMMR0.cpp 19734 2009-05-15 12:37:11Z vboxsync $ */
2/** @file
3 * VMM - Host Context Ring 0.
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_VMM
26#include <VBox/vmm.h>
27#include <VBox/sup.h>
28#include <VBox/trpm.h>
29#include <VBox/cpum.h>
30#include <VBox/pgm.h>
31#include <VBox/stam.h>
32#include <VBox/tm.h>
33#include "VMMInternal.h"
34#include <VBox/vm.h>
35#include <VBox/gvmm.h>
36#include <VBox/gmm.h>
37#include <VBox/intnet.h>
38#include <VBox/hwaccm.h>
39#include <VBox/param.h>
40
41#include <VBox/err.h>
42#include <VBox/version.h>
43#include <VBox/log.h>
44#include <iprt/assert.h>
45#include <iprt/stdarg.h>
46#include <iprt/mp.h>
47#include <iprt/string.h>
48
49#if defined(_MSC_VER) && defined(RT_ARCH_AMD64) /** @todo check this with with VC7! */
50# pragma intrinsic(_AddressOfReturnAddress)
51#endif
52
53
54/*******************************************************************************
55* Internal Functions *
56*******************************************************************************/
57__BEGIN_DECLS
58VMMR0DECL(int) ModuleInit(void);
59VMMR0DECL(void) ModuleTerm(void);
60__END_DECLS
61
62
63/*******************************************************************************
64* Global Variables *
65*******************************************************************************/
66/** Pointer to the internal networking service instance. */
67PINTNET g_pIntNet = 0;
68
69
70/**
71 * Initialize the module.
72 * This is called when we're first loaded.
73 *
74 * @returns 0 on success.
75 * @returns VBox status on failure.
76 */
77VMMR0DECL(int) ModuleInit(void)
78{
79 LogFlow(("ModuleInit:\n"));
80
81 /*
82 * Initialize the GVMM, GMM, HWACCM, PGM (Darwin) and INTNET.
83 */
84 int rc = GVMMR0Init();
85 if (RT_SUCCESS(rc))
86 {
87 rc = GMMR0Init();
88 if (RT_SUCCESS(rc))
89 {
90 rc = HWACCMR0Init();
91 if (RT_SUCCESS(rc))
92 {
93 rc = PGMRegisterStringFormatTypes();
94 if (RT_SUCCESS(rc))
95 {
96#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
97 rc = PGMR0DynMapInit();
98#endif
99 if (RT_SUCCESS(rc))
100 {
101 LogFlow(("ModuleInit: g_pIntNet=%p\n", g_pIntNet));
102 g_pIntNet = NULL;
103 LogFlow(("ModuleInit: g_pIntNet=%p should be NULL now...\n", g_pIntNet));
104 rc = INTNETR0Create(&g_pIntNet);
105 if (RT_SUCCESS(rc))
106 {
107 LogFlow(("ModuleInit: returns success. g_pIntNet=%p\n", g_pIntNet));
108 return VINF_SUCCESS;
109 }
110
111 /* bail out */
112 g_pIntNet = NULL;
113 LogFlow(("ModuleTerm: returns %Rrc\n", rc));
114#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
115 PGMR0DynMapTerm();
116#endif
117 }
118 PGMDeregisterStringFormatTypes();
119 }
120 HWACCMR0Term();
121 }
122 GMMR0Term();
123 }
124 GVMMR0Term();
125 }
126
127 LogFlow(("ModuleInit: failed %Rrc\n", rc));
128 return rc;
129}
130
131
132/**
133 * Terminate the module.
134 * This is called when we're finally unloaded.
135 */
136VMMR0DECL(void) ModuleTerm(void)
137{
138 LogFlow(("ModuleTerm:\n"));
139
140 /*
141 * Destroy the internal networking instance.
142 */
143 if (g_pIntNet)
144 {
145 INTNETR0Destroy(g_pIntNet);
146 g_pIntNet = NULL;
147 }
148
149 /*
150 * PGM (Darwin) and HWACCM global cleanup.
151 * Destroy the GMM and GVMM instances.
152 */
153#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
154 PGMR0DynMapTerm();
155#endif
156 PGMDeregisterStringFormatTypes();
157 HWACCMR0Term();
158
159 GMMR0Term();
160 GVMMR0Term();
161
162 LogFlow(("ModuleTerm: returns\n"));
163}
164
165
166/**
167 * Initaties the R0 driver for a particular VM instance.
168 *
169 * @returns VBox status code.
170 *
171 * @param pVM The VM instance in question.
172 * @param uSvnRev The SVN revision of the ring-3 part.
173 * @thread EMT.
174 */
175static int vmmR0InitVM(PVM pVM, uint32_t uSvnRev)
176{
177 /*
178 * Match the SVN revisions.
179 */
180 if (uSvnRev != VMMGetSvnRev())
181 {
182 LogRel(("VMMR0InitVM: Revision mismatch, r3=%d r0=%d\n", uSvnRev, VMMGetSvnRev()));
183 SUPR0Printf("VMMR0InitVM: Revision mismatch, r3=%d r0=%d\n", uSvnRev, VMMGetSvnRev());
184 return VERR_VERSION_MISMATCH;
185 }
186 if ( !VALID_PTR(pVM)
187 || pVM->pVMR0 != pVM)
188 return VERR_INVALID_PARAMETER;
189
190#ifdef LOG_ENABLED
191 /*
192 * Register the EMT R0 logger instance for VCPU 0.
193 */
194 PVMCPU pVCpu = &pVM->aCpus[0];
195
196 PVMMR0LOGGER pR0Logger = pVCpu->vmm.s.pR0LoggerR0;
197 if (pR0Logger)
198 {
199# if 0 /* testing of the logger. */
200 LogCom(("vmmR0InitVM: before %p\n", RTLogDefaultInstance()));
201 LogCom(("vmmR0InitVM: pfnFlush=%p actual=%p\n", pR0Logger->Logger.pfnFlush, vmmR0LoggerFlush));
202 LogCom(("vmmR0InitVM: pfnLogger=%p actual=%p\n", pR0Logger->Logger.pfnLogger, vmmR0LoggerWrapper));
203 LogCom(("vmmR0InitVM: offScratch=%d fFlags=%#x fDestFlags=%#x\n", pR0Logger->Logger.offScratch, pR0Logger->Logger.fFlags, pR0Logger->Logger.fDestFlags));
204
205 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
206 LogCom(("vmmR0InitVM: after %p reg\n", RTLogDefaultInstance()));
207 RTLogSetDefaultInstanceThread(NULL, pVM->pSession);
208 LogCom(("vmmR0InitVM: after %p dereg\n", RTLogDefaultInstance()));
209
210 pR0Logger->Logger.pfnLogger("hello ring-0 logger\n");
211 LogCom(("vmmR0InitVM: returned succesfully from direct logger call.\n"));
212 pR0Logger->Logger.pfnFlush(&pR0Logger->Logger);
213 LogCom(("vmmR0InitVM: returned succesfully from direct flush call.\n"));
214
215 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
216 LogCom(("vmmR0InitVM: after %p reg2\n", RTLogDefaultInstance()));
217 pR0Logger->Logger.pfnLogger("hello ring-0 logger\n");
218 LogCom(("vmmR0InitVM: returned succesfully from direct logger call (2). offScratch=%d\n", pR0Logger->Logger.offScratch));
219 RTLogSetDefaultInstanceThread(NULL, pVM->pSession);
220 LogCom(("vmmR0InitVM: after %p dereg2\n", RTLogDefaultInstance()));
221
222 RTLogLoggerEx(&pR0Logger->Logger, 0, ~0U, "hello ring-0 logger (RTLogLoggerEx)\n");
223 LogCom(("vmmR0InitVM: RTLogLoggerEx returned fine offScratch=%d\n", pR0Logger->Logger.offScratch));
224
225 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
226 RTLogPrintf("hello ring-0 logger (RTLogPrintf)\n");
227 LogCom(("vmmR0InitVM: RTLogPrintf returned fine offScratch=%d\n", pR0Logger->Logger.offScratch));
228# endif
229 Log(("Switching to per-thread logging instance %p (key=%p)\n", &pR0Logger->Logger, pVM->pSession));
230 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
231 pR0Logger->fRegistered = true;
232 }
233#endif /* LOG_ENABLED */
234
235 /*
236 * Initialize the per VM data for GVMM and GMM.
237 */
238 int rc = GVMMR0InitVM(pVM);
239// if (RT_SUCCESS(rc))
240// rc = GMMR0InitPerVMData(pVM);
241 if (RT_SUCCESS(rc))
242 {
243 /*
244 * Init HWACCM, CPUM and PGM (Darwin only).
245 */
246 rc = HWACCMR0InitVM(pVM);
247 if (RT_SUCCESS(rc))
248 {
249 rc = CPUMR0Init(pVM); /** @todo rename to CPUMR0InitVM */
250 if (RT_SUCCESS(rc))
251 {
252#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
253 rc = PGMR0DynMapInitVM(pVM);
254#endif
255 if (RT_SUCCESS(rc))
256 {
257 GVMMR0DoneInitVM(pVM);
258 return rc;
259 }
260
261 /* bail out */
262 }
263 HWACCMR0TermVM(pVM);
264 }
265 }
266 RTLogSetDefaultInstanceThread(NULL, (uintptr_t)pVM->pSession);
267 return rc;
268}
269
270
271/**
272 * Terminates the R0 driver for a particular VM instance.
273 *
274 * This is normally called by ring-3 as part of the VM termination process, but
275 * may alternatively be called during the support driver session cleanup when
276 * the VM object is destroyed (see GVMM).
277 *
278 * @returns VBox status code.
279 *
280 * @param pVM The VM instance in question.
281 * @param pGVM Pointer to the global VM structure. Optional.
282 * @thread EMT or session clean up thread.
283 */
284VMMR0DECL(int) VMMR0TermVM(PVM pVM, PGVM pGVM)
285{
286 /*
287 * Tell GVMM what we're up to and check that we only do this once.
288 */
289 if (GVMMR0DoingTermVM(pVM, pGVM))
290 {
291#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
292 PGMR0DynMapTermVM(pVM);
293#endif
294 HWACCMR0TermVM(pVM);
295 }
296
297 /*
298 * Deregister the logger.
299 */
300 RTLogSetDefaultInstanceThread(NULL, (uintptr_t)pVM->pSession);
301 return VINF_SUCCESS;
302}
303
304
305/**
306 * Calls the ring-3 host code.
307 *
308 * @returns VBox status code of the ring-3 call.
309 * @param pVM The VM handle.
310 * @param enmOperation The operation.
311 * @param uArg The argument to the operation.
312 */
313VMMR0DECL(int) VMMR0CallHost(PVM pVM, VMMCALLHOST enmOperation, uint64_t uArg)
314{
315 PVMCPU pVCpu = VMMGetCpu(pVM);
316
317/** @todo profile this! */
318 pVCpu->vmm.s.enmCallHostOperation = enmOperation;
319 pVCpu->vmm.s.u64CallHostArg = uArg;
320 pVCpu->vmm.s.rcCallHost = VERR_INTERNAL_ERROR;
321 int rc = vmmR0CallHostLongJmp(&pVCpu->vmm.s.CallHostR0JmpBuf, VINF_VMM_CALL_HOST);
322 if (rc == VINF_SUCCESS)
323 rc = pVCpu->vmm.s.rcCallHost;
324 return rc;
325}
326
327
328#ifdef VBOX_WITH_STATISTICS
329/**
330 * Record return code statistics
331 * @param pVM The VM handle.
332 * @param pVCpu The VMCPU handle.
333 * @param rc The status code.
334 */
335static void vmmR0RecordRC(PVM pVM, PVMCPU pVCpu, int rc)
336{
337 /*
338 * Collect statistics.
339 */
340 switch (rc)
341 {
342 case VINF_SUCCESS:
343 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetNormal);
344 break;
345 case VINF_EM_RAW_INTERRUPT:
346 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetInterrupt);
347 break;
348 case VINF_EM_RAW_INTERRUPT_HYPER:
349 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetInterruptHyper);
350 break;
351 case VINF_EM_RAW_GUEST_TRAP:
352 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetGuestTrap);
353 break;
354 case VINF_EM_RAW_RING_SWITCH:
355 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetRingSwitch);
356 break;
357 case VINF_EM_RAW_RING_SWITCH_INT:
358 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetRingSwitchInt);
359 break;
360 case VINF_EM_RAW_EXCEPTION_PRIVILEGED:
361 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetExceptionPrivilege);
362 break;
363 case VINF_EM_RAW_STALE_SELECTOR:
364 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetStaleSelector);
365 break;
366 case VINF_EM_RAW_IRET_TRAP:
367 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIRETTrap);
368 break;
369 case VINF_IOM_HC_IOPORT_READ:
370 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIORead);
371 break;
372 case VINF_IOM_HC_IOPORT_WRITE:
373 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIOWrite);
374 break;
375 case VINF_IOM_HC_MMIO_READ:
376 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIORead);
377 break;
378 case VINF_IOM_HC_MMIO_WRITE:
379 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOWrite);
380 break;
381 case VINF_IOM_HC_MMIO_READ_WRITE:
382 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOReadWrite);
383 break;
384 case VINF_PATM_HC_MMIO_PATCH_READ:
385 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOPatchRead);
386 break;
387 case VINF_PATM_HC_MMIO_PATCH_WRITE:
388 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOPatchWrite);
389 break;
390 case VINF_EM_RAW_EMULATE_INSTR:
391 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetEmulate);
392 break;
393 case VINF_EM_RAW_EMULATE_IO_BLOCK:
394 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIOBlockEmulate);
395 break;
396 case VINF_PATCH_EMULATE_INSTR:
397 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchEmulate);
398 break;
399 case VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT:
400 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetLDTFault);
401 break;
402 case VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT:
403 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetGDTFault);
404 break;
405 case VINF_EM_RAW_EMULATE_INSTR_IDT_FAULT:
406 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIDTFault);
407 break;
408 case VINF_EM_RAW_EMULATE_INSTR_TSS_FAULT:
409 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetTSSFault);
410 break;
411 case VINF_EM_RAW_EMULATE_INSTR_PD_FAULT:
412 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPDFault);
413 break;
414 case VINF_CSAM_PENDING_ACTION:
415 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetCSAMTask);
416 break;
417 case VINF_PGM_SYNC_CR3:
418 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetSyncCR3);
419 break;
420 case VINF_PATM_PATCH_INT3:
421 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchInt3);
422 break;
423 case VINF_PATM_PATCH_TRAP_PF:
424 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchPF);
425 break;
426 case VINF_PATM_PATCH_TRAP_GP:
427 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchGP);
428 break;
429 case VINF_PATM_PENDING_IRQ_AFTER_IRET:
430 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchIretIRQ);
431 break;
432 case VERR_REM_FLUSHED_PAGES_OVERFLOW:
433 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPageOverflow);
434 break;
435 case VINF_EM_RESCHEDULE_REM:
436 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetRescheduleREM);
437 break;
438 case VINF_EM_RAW_TO_R3:
439 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3);
440 break;
441 case VINF_EM_RAW_TIMER_PENDING:
442 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetTimerPending);
443 break;
444 case VINF_EM_RAW_INTERRUPT_PENDING:
445 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetInterruptPending);
446 break;
447 case VINF_VMM_CALL_HOST:
448 switch (pVCpu->vmm.s.enmCallHostOperation)
449 {
450 case VMMCALLHOST_PDM_LOCK:
451 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPDMLock);
452 break;
453 case VMMCALLHOST_PDM_QUEUE_FLUSH:
454 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPDMQueueFlush);
455 break;
456 case VMMCALLHOST_PGM_POOL_GROW:
457 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMPoolGrow);
458 break;
459 case VMMCALLHOST_PGM_LOCK:
460 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMLock);
461 break;
462 case VMMCALLHOST_PGM_MAP_CHUNK:
463 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMMapChunk);
464 break;
465 case VMMCALLHOST_PGM_ALLOCATE_HANDY_PAGES:
466 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMAllocHandy);
467 break;
468 case VMMCALLHOST_REM_REPLAY_HANDLER_NOTIFICATIONS:
469 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallRemReplay);
470 break;
471 case VMMCALLHOST_VMM_LOGGER_FLUSH:
472 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallLogFlush);
473 break;
474 case VMMCALLHOST_VM_SET_ERROR:
475 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallVMSetError);
476 break;
477 case VMMCALLHOST_VM_SET_RUNTIME_ERROR:
478 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallVMSetRuntimeError);
479 break;
480 case VMMCALLHOST_VM_R0_ASSERTION:
481 default:
482 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetCallHost);
483 break;
484 }
485 break;
486 case VINF_PATM_DUPLICATE_FUNCTION:
487 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPATMDuplicateFn);
488 break;
489 case VINF_PGM_CHANGE_MODE:
490 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPGMChangeMode);
491 break;
492 case VINF_EM_RAW_EMULATE_INSTR_HLT:
493 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetEmulHlt);
494 break;
495 case VINF_EM_PENDING_REQUEST:
496 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPendingRequest);
497 break;
498 default:
499 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMisc);
500 break;
501 }
502}
503#endif /* VBOX_WITH_STATISTICS */
504
505
506/**
507 * Unused ring-0 entry point that used to be called from the interrupt gate.
508 *
509 * Will be removed one of the next times we do a major SUPDrv version bump.
510 *
511 * @returns VBox status code.
512 * @param pVM The VM to operate on.
513 * @param enmOperation Which operation to execute.
514 * @param pvArg Argument to the operation.
515 * @remarks Assume called with interrupts disabled.
516 */
517VMMR0DECL(int) VMMR0EntryInt(PVM pVM, VMMR0OPERATION enmOperation, void *pvArg)
518{
519 /*
520 * We're returning VERR_NOT_SUPPORT here so we've got something else
521 * than -1 which the interrupt gate glue code might return.
522 */
523 Log(("operation %#x is not supported\n", enmOperation));
524 return VERR_NOT_SUPPORTED;
525}
526
527
528/**
529 * The Ring 0 entry point, called by the fast-ioctl path.
530 *
531 * @param pVM The VM to operate on.
532 * The return code is stored in pVM->vmm.s.iLastGZRc.
533 * @param idCpu The Virtual CPU ID of the calling EMT.
534 * @param enmOperation Which operation to execute.
535 * @remarks Assume called with interrupts _enabled_.
536 */
537VMMR0DECL(void) VMMR0EntryFast(PVM pVM, VMCPUID idCpu, VMMR0OPERATION enmOperation)
538{
539 if (RT_UNLIKELY(idCpu >= pVM->cCPUs))
540 return;
541 PVMCPU pVCpu = &pVM->aCpus[idCpu];
542
543 switch (enmOperation)
544 {
545 /*
546 * Switch to GC and run guest raw mode code.
547 * Disable interrupts before doing the world switch.
548 */
549 case VMMR0_DO_RAW_RUN:
550 {
551 /* Safety precaution as hwaccm disables the switcher. */
552 if (RT_LIKELY(!pVM->vmm.s.fSwitcherDisabled))
553 {
554 RTCCUINTREG uFlags = ASMIntDisableFlags();
555 int rc;
556 bool fVTxDisabled;
557
558 if (RT_UNLIKELY(pVM->cCPUs > 1))
559 {
560 pVCpu->vmm.s.iLastGZRc = VERR_RAW_MODE_INVALID_SMP;
561 return;
562 }
563
564#ifndef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
565 if (RT_UNLIKELY(!PGMGetHyperCR3(pVCpu)))
566 {
567 pVCpu->vmm.s.iLastGZRc = VERR_PGM_NO_CR3_SHADOW_ROOT;
568 return;
569 }
570#endif
571
572 /* We might need to disable VT-x if the active switcher turns off paging. */
573 rc = HWACCMR0EnterSwitcher(pVM, &fVTxDisabled);
574 if (RT_FAILURE(rc))
575 {
576 pVCpu->vmm.s.iLastGZRc = rc;
577 return;
578 }
579
580 ASMAtomicWriteU32(&pVCpu->idHostCpu, RTMpCpuId());
581 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED_EXEC);
582
583 TMNotifyStartOfExecution(pVCpu);
584 rc = pVM->vmm.s.pfnHostToGuestR0(pVM);
585 pVCpu->vmm.s.iLastGZRc = rc;
586 TMNotifyEndOfExecution(pVCpu);
587
588 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED);
589 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
590
591 /* Re-enable VT-x if previously turned off. */
592 HWACCMR0LeaveSwitcher(pVM, fVTxDisabled);
593
594 if ( rc == VINF_EM_RAW_INTERRUPT
595 || rc == VINF_EM_RAW_INTERRUPT_HYPER)
596 TRPMR0DispatchHostInterrupt(pVM);
597
598 ASMSetFlags(uFlags);
599
600#ifdef VBOX_WITH_STATISTICS
601 STAM_COUNTER_INC(&pVM->vmm.s.StatRunRC);
602 vmmR0RecordRC(pVM, pVCpu, rc);
603#endif
604 }
605 else
606 {
607 Assert(!pVM->vmm.s.fSwitcherDisabled);
608 pVCpu->vmm.s.iLastGZRc = VERR_NOT_SUPPORTED;
609 }
610 break;
611 }
612
613 /*
614 * Run guest code using the available hardware acceleration technology.
615 *
616 * Disable interrupts before we do anything interesting. On Windows we avoid
617 * this by having the support driver raise the IRQL before calling us, this way
618 * we hope to get away with page faults and later calling into the kernel.
619 */
620 case VMMR0_DO_HWACC_RUN:
621 {
622 int rc;
623
624 STAM_COUNTER_INC(&pVM->vmm.s.StatRunRC);
625
626#if !defined(RT_OS_WINDOWS) /** @todo check other hosts */
627 RTCCUINTREG uFlags = ASMIntDisableFlags();
628#endif
629 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
630
631#ifdef LOG_ENABLED
632 if (pVCpu->idCpu > 0)
633 {
634 /* Lazy registration of ring 0 loggers. */
635 PVMMR0LOGGER pR0Logger = pVCpu->vmm.s.pR0LoggerR0;
636 if ( pR0Logger
637 && !pR0Logger->fRegistered)
638 {
639 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
640 pR0Logger->fRegistered = true;
641 }
642 }
643#endif
644 if (!HWACCMR0SuspendPending())
645 {
646 rc = HWACCMR0Enter(pVM, pVCpu);
647 if (RT_SUCCESS(rc))
648 {
649 rc = vmmR0CallHostSetJmp(&pVCpu->vmm.s.CallHostR0JmpBuf, HWACCMR0RunGuestCode, pVM, pVCpu); /* this may resume code. */
650 int rc2 = HWACCMR0Leave(pVM, pVCpu);
651 AssertRC(rc2);
652 }
653 }
654 else
655 {
656 /* System is about to go into suspend mode; go back to ring 3. */
657 rc = VINF_EM_RAW_INTERRUPT;
658 }
659 pVCpu->vmm.s.iLastGZRc = rc;
660
661 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
662#if !defined(RT_OS_WINDOWS) /** @todo check other hosts */
663 ASMSetFlags(uFlags);
664#endif
665
666#ifdef VBOX_WITH_STATISTICS
667 vmmR0RecordRC(pVM, pVCpu, rc);
668#endif
669 /* No special action required for external interrupts, just return. */
670 break;
671 }
672
673 /*
674 * For profiling.
675 */
676 case VMMR0_DO_NOP:
677 pVCpu->vmm.s.iLastGZRc = VINF_SUCCESS;
678 break;
679
680 /*
681 * Impossible.
682 */
683 default:
684 AssertMsgFailed(("%#x\n", enmOperation));
685 pVCpu->vmm.s.iLastGZRc = VERR_NOT_SUPPORTED;
686 break;
687 }
688}
689
690
691/**
692 * Validates a session or VM session argument.
693 *
694 * @returns true / false accordingly.
695 * @param pVM The VM argument.
696 * @param pSession The session argument.
697 */
698DECLINLINE(bool) vmmR0IsValidSession(PVM pVM, PSUPDRVSESSION pClaimedSession, PSUPDRVSESSION pSession)
699{
700 /* This must be set! */
701 if (!pSession)
702 return false;
703
704 /* Only one out of the two. */
705 if (pVM && pClaimedSession)
706 return false;
707 if (pVM)
708 pClaimedSession = pVM->pSession;
709 return pClaimedSession == pSession;
710}
711
712
713/**
714 * VMMR0EntryEx worker function, either called directly or when ever possible
715 * called thru a longjmp so we can exit safely on failure.
716 *
717 * @returns VBox status code.
718 * @param pVM The VM to operate on.
719 * @param idCpu Virtual CPU ID argument. Must be NIL_VMCPUID if pVM
720 * is NIL_RTR0PTR, and may be NIL_VMCPUID if it isn't
721 * @param enmOperation Which operation to execute.
722 * @param pReqHdr This points to a SUPVMMR0REQHDR packet. Optional.
723 * The support driver validates this if it's present.
724 * @param u64Arg Some simple constant argument.
725 * @param pSession The session of the caller.
726 * @remarks Assume called with interrupts _enabled_.
727 */
728static int vmmR0EntryExWorker(PVM pVM, VMCPUID idCpu, VMMR0OPERATION enmOperation, PSUPVMMR0REQHDR pReqHdr, uint64_t u64Arg, PSUPDRVSESSION pSession)
729{
730 /*
731 * Common VM pointer validation.
732 */
733 if (pVM)
734 {
735 if (RT_UNLIKELY( !VALID_PTR(pVM)
736 || ((uintptr_t)pVM & PAGE_OFFSET_MASK)))
737 {
738 SUPR0Printf("vmmR0EntryExWorker: Invalid pVM=%p! (op=%d)\n", pVM, enmOperation);
739 return VERR_INVALID_POINTER;
740 }
741 if (RT_UNLIKELY( pVM->enmVMState < VMSTATE_CREATING
742 || pVM->enmVMState > VMSTATE_TERMINATED
743 || pVM->pVMR0 != pVM))
744 {
745 SUPR0Printf("vmmR0EntryExWorker: Invalid pVM=%p:{enmVMState=%d, .pVMR0=%p}! (op=%d)\n",
746 pVM, pVM->enmVMState, pVM->pVMR0, enmOperation);
747 return VERR_INVALID_POINTER;
748 }
749
750 if (RT_UNLIKELY(idCpu >= pVM->cCPUs && idCpu != NIL_VMCPUID))
751 {
752 SUPR0Printf("vmmR0EntryExWorker: Invalid idCpu (%u vs cCPUs=%u)\n", idCpu, pVM->cCPUs);
753 return VERR_INVALID_PARAMETER;
754 }
755 }
756 else if (RT_UNLIKELY(idCpu != NIL_VMCPUID))
757 {
758 SUPR0Printf("vmmR0EntryExWorker: Invalid idCpu=%u\n", idCpu);
759 return VERR_INVALID_PARAMETER;
760 }
761
762
763 switch (enmOperation)
764 {
765 /*
766 * GVM requests
767 */
768 case VMMR0_DO_GVMM_CREATE_VM:
769 if (pVM || u64Arg || idCpu != NIL_VMCPUID)
770 return VERR_INVALID_PARAMETER;
771 return GVMMR0CreateVMReq((PGVMMCREATEVMREQ)pReqHdr);
772
773 case VMMR0_DO_GVMM_DESTROY_VM:
774 if (pReqHdr || u64Arg)
775 return VERR_INVALID_PARAMETER;
776 return GVMMR0DestroyVM(pVM);
777
778 case VMMR0_DO_GVMM_REGISTER_VMCPU:
779 {
780 if (!pVM)
781 return VERR_INVALID_PARAMETER;
782 return GVMMR0RegisterVCpu(pVM, idCpu);
783 }
784
785 case VMMR0_DO_GVMM_SCHED_HALT:
786 if (pReqHdr)
787 return VERR_INVALID_PARAMETER;
788 return GVMMR0SchedHalt(pVM, idCpu, u64Arg);
789
790 case VMMR0_DO_GVMM_SCHED_WAKE_UP:
791 if (pReqHdr || u64Arg)
792 return VERR_INVALID_PARAMETER;
793 return GVMMR0SchedWakeUp(pVM, idCpu);
794
795 case VMMR0_DO_GVMM_SCHED_POKE:
796 if (pReqHdr || u64Arg)
797 return VERR_INVALID_PARAMETER;
798 return GVMMR0SchedPoke(pVM, idCpu);
799
800 case VMMR0_DO_GVMM_SCHED_WAKE_UP_AND_POKE_CPUS:
801 if (u64Arg)
802 return VERR_INVALID_PARAMETER;
803 return GVMMR0SchedWakeUpAndPokeCpusReq(pVM, (PGVMMSCHEDWAKEUPANDPOKECPUSREQ)pReqHdr);
804
805 case VMMR0_DO_GVMM_SCHED_POLL:
806 if (pReqHdr || u64Arg > 1)
807 return VERR_INVALID_PARAMETER;
808 return GVMMR0SchedPoll(pVM, idCpu, !!u64Arg);
809
810 case VMMR0_DO_GVMM_QUERY_STATISTICS:
811 if (u64Arg)
812 return VERR_INVALID_PARAMETER;
813 return GVMMR0QueryStatisticsReq(pVM, (PGVMMQUERYSTATISTICSSREQ)pReqHdr);
814
815 case VMMR0_DO_GVMM_RESET_STATISTICS:
816 if (u64Arg)
817 return VERR_INVALID_PARAMETER;
818 return GVMMR0ResetStatisticsReq(pVM, (PGVMMRESETSTATISTICSSREQ)pReqHdr);
819
820 /*
821 * Initialize the R0 part of a VM instance.
822 */
823 case VMMR0_DO_VMMR0_INIT:
824 return vmmR0InitVM(pVM, (uint32_t)u64Arg);
825
826 /*
827 * Terminate the R0 part of a VM instance.
828 */
829 case VMMR0_DO_VMMR0_TERM:
830 return VMMR0TermVM(pVM, NULL);
831
832 /*
833 * Attempt to enable hwacc mode and check the current setting.
834 *
835 */
836 case VMMR0_DO_HWACC_ENABLE:
837 return HWACCMR0EnableAllCpus(pVM);
838
839 /*
840 * Setup the hardware accelerated raw-mode session.
841 */
842 case VMMR0_DO_HWACC_SETUP_VM:
843 {
844 RTCCUINTREG fFlags = ASMIntDisableFlags();
845 int rc = HWACCMR0SetupVM(pVM);
846 ASMSetFlags(fFlags);
847 return rc;
848 }
849
850 /*
851 * Switch to RC to execute Hypervisor function.
852 */
853 case VMMR0_DO_CALL_HYPERVISOR:
854 {
855 int rc;
856 bool fVTxDisabled;
857
858 /* Safety precaution as HWACCM can disable the switcher. */
859 Assert(!pVM->vmm.s.fSwitcherDisabled);
860 if (RT_UNLIKELY(pVM->vmm.s.fSwitcherDisabled))
861 return VERR_NOT_SUPPORTED;
862
863#ifndef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
864 if (RT_UNLIKELY(!PGMGetHyperCR3(VMMGetCpu0(pVM))))
865 return VERR_PGM_NO_CR3_SHADOW_ROOT;
866#endif
867
868 RTCCUINTREG fFlags = ASMIntDisableFlags();
869
870 /* We might need to disable VT-x if the active switcher turns off paging. */
871 rc = HWACCMR0EnterSwitcher(pVM, &fVTxDisabled);
872 if (RT_FAILURE(rc))
873 return rc;
874
875 rc = pVM->vmm.s.pfnHostToGuestR0(pVM);
876
877 /* Re-enable VT-x if previously turned off. */
878 HWACCMR0LeaveSwitcher(pVM, fVTxDisabled);
879
880 /** @todo dispatch interrupts? */
881 ASMSetFlags(fFlags);
882 return rc;
883 }
884
885 /*
886 * PGM wrappers.
887 */
888 case VMMR0_DO_PGM_ALLOCATE_HANDY_PAGES:
889 if (idCpu == NIL_VMCPUID)
890 return VERR_INVALID_CPU_ID;
891 return PGMR0PhysAllocateHandyPages(pVM, &pVM->aCpus[idCpu]);
892
893 /*
894 * GMM wrappers.
895 */
896 case VMMR0_DO_GMM_INITIAL_RESERVATION:
897 if (u64Arg)
898 return VERR_INVALID_PARAMETER;
899 return GMMR0InitialReservationReq(pVM, idCpu, (PGMMINITIALRESERVATIONREQ)pReqHdr);
900
901 case VMMR0_DO_GMM_UPDATE_RESERVATION:
902 if (u64Arg)
903 return VERR_INVALID_PARAMETER;
904 return GMMR0UpdateReservationReq(pVM, idCpu, (PGMMUPDATERESERVATIONREQ)pReqHdr);
905
906 case VMMR0_DO_GMM_ALLOCATE_PAGES:
907 if (u64Arg)
908 return VERR_INVALID_PARAMETER;
909 return GMMR0AllocatePagesReq(pVM, idCpu, (PGMMALLOCATEPAGESREQ)pReqHdr);
910
911 case VMMR0_DO_GMM_FREE_PAGES:
912 if (u64Arg)
913 return VERR_INVALID_PARAMETER;
914 return GMMR0FreePagesReq(pVM, idCpu, (PGMMFREEPAGESREQ)pReqHdr);
915
916 case VMMR0_DO_GMM_BALLOONED_PAGES:
917 if (u64Arg)
918 return VERR_INVALID_PARAMETER;
919 return GMMR0BalloonedPagesReq(pVM, idCpu, (PGMMBALLOONEDPAGESREQ)pReqHdr);
920
921 case VMMR0_DO_GMM_DEFLATED_BALLOON:
922 if (pReqHdr)
923 return VERR_INVALID_PARAMETER;
924 return GMMR0DeflatedBalloon(pVM, idCpu, (uint32_t)u64Arg);
925
926 case VMMR0_DO_GMM_MAP_UNMAP_CHUNK:
927 if (u64Arg)
928 return VERR_INVALID_PARAMETER;
929 return GMMR0MapUnmapChunkReq(pVM, idCpu, (PGMMMAPUNMAPCHUNKREQ)pReqHdr);
930
931 case VMMR0_DO_GMM_SEED_CHUNK:
932 if (pReqHdr)
933 return VERR_INVALID_PARAMETER;
934 return GMMR0SeedChunk(pVM, idCpu, (RTR3PTR)u64Arg);
935
936 /*
937 * A quick GCFGM mock-up.
938 */
939 /** @todo GCFGM with proper access control, ring-3 management interface and all that. */
940 case VMMR0_DO_GCFGM_SET_VALUE:
941 case VMMR0_DO_GCFGM_QUERY_VALUE:
942 {
943 if (pVM || !pReqHdr || u64Arg || idCpu != NIL_VMCPUID)
944 return VERR_INVALID_PARAMETER;
945 PGCFGMVALUEREQ pReq = (PGCFGMVALUEREQ)pReqHdr;
946 if (pReq->Hdr.cbReq != sizeof(*pReq))
947 return VERR_INVALID_PARAMETER;
948 int rc;
949 if (enmOperation == VMMR0_DO_GCFGM_SET_VALUE)
950 {
951 rc = GVMMR0SetConfig(pReq->pSession, &pReq->szName[0], pReq->u64Value);
952 //if (rc == VERR_CFGM_VALUE_NOT_FOUND)
953 // rc = GMMR0SetConfig(pReq->pSession, &pReq->szName[0], pReq->u64Value);
954 }
955 else
956 {
957 rc = GVMMR0QueryConfig(pReq->pSession, &pReq->szName[0], &pReq->u64Value);
958 //if (rc == VERR_CFGM_VALUE_NOT_FOUND)
959 // rc = GMMR0QueryConfig(pReq->pSession, &pReq->szName[0], &pReq->u64Value);
960 }
961 return rc;
962 }
963
964
965 /*
966 * Requests to the internal networking service.
967 */
968 case VMMR0_DO_INTNET_OPEN:
969 {
970 PINTNETOPENREQ pReq = (PINTNETOPENREQ)pReqHdr;
971 if (u64Arg || !pReq || !vmmR0IsValidSession(pVM, pReq->pSession, pSession) || idCpu != NIL_VMCPUID)
972 return VERR_INVALID_PARAMETER;
973 if (!g_pIntNet)
974 return VERR_NOT_SUPPORTED;
975 return INTNETR0OpenReq(g_pIntNet, pSession, pReq);
976 }
977
978 case VMMR0_DO_INTNET_IF_CLOSE:
979 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFCLOSEREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
980 return VERR_INVALID_PARAMETER;
981 if (!g_pIntNet)
982 return VERR_NOT_SUPPORTED;
983 return INTNETR0IfCloseReq(g_pIntNet, pSession, (PINTNETIFCLOSEREQ)pReqHdr);
984
985 case VMMR0_DO_INTNET_IF_GET_RING3_BUFFER:
986 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFGETRING3BUFFERREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
987 return VERR_INVALID_PARAMETER;
988 if (!g_pIntNet)
989 return VERR_NOT_SUPPORTED;
990 return INTNETR0IfGetRing3BufferReq(g_pIntNet, pSession, (PINTNETIFGETRING3BUFFERREQ)pReqHdr);
991
992 case VMMR0_DO_INTNET_IF_SET_PROMISCUOUS_MODE:
993 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSETPROMISCUOUSMODEREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
994 return VERR_INVALID_PARAMETER;
995 if (!g_pIntNet)
996 return VERR_NOT_SUPPORTED;
997 return INTNETR0IfSetPromiscuousModeReq(g_pIntNet, pSession, (PINTNETIFSETPROMISCUOUSMODEREQ)pReqHdr);
998
999 case VMMR0_DO_INTNET_IF_SET_MAC_ADDRESS:
1000 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSETMACADDRESSREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1001 return VERR_INVALID_PARAMETER;
1002 if (!g_pIntNet)
1003 return VERR_NOT_SUPPORTED;
1004 return INTNETR0IfSetMacAddressReq(g_pIntNet, pSession, (PINTNETIFSETMACADDRESSREQ)pReqHdr);
1005
1006 case VMMR0_DO_INTNET_IF_SET_ACTIVE:
1007 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSETACTIVEREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1008 return VERR_INVALID_PARAMETER;
1009 if (!g_pIntNet)
1010 return VERR_NOT_SUPPORTED;
1011 return INTNETR0IfSetActiveReq(g_pIntNet, pSession, (PINTNETIFSETACTIVEREQ)pReqHdr);
1012
1013 case VMMR0_DO_INTNET_IF_SEND:
1014 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSENDREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1015 return VERR_INVALID_PARAMETER;
1016 if (!g_pIntNet)
1017 return VERR_NOT_SUPPORTED;
1018 return INTNETR0IfSendReq(g_pIntNet, pSession, (PINTNETIFSENDREQ)pReqHdr);
1019
1020 case VMMR0_DO_INTNET_IF_WAIT:
1021 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFWAITREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1022 return VERR_INVALID_PARAMETER;
1023 if (!g_pIntNet)
1024 return VERR_NOT_SUPPORTED;
1025 return INTNETR0IfWaitReq(g_pIntNet, pSession, (PINTNETIFWAITREQ)pReqHdr);
1026
1027 /*
1028 * For profiling.
1029 */
1030 case VMMR0_DO_NOP:
1031 case VMMR0_DO_SLOW_NOP:
1032 return VINF_SUCCESS;
1033
1034 /*
1035 * For testing Ring-0 APIs invoked in this environment.
1036 */
1037 case VMMR0_DO_TESTS:
1038 /** @todo make new test */
1039 return VINF_SUCCESS;
1040
1041
1042#if HC_ARCH_BITS == 32 && defined(VBOX_WITH_64_BITS_GUESTS) && !defined(VBOX_WITH_HYBRID_32BIT_KERNEL)
1043 case VMMR0_DO_TEST_SWITCHER3264:
1044 if (idCpu == NIL_VMCPUID)
1045 return VERR_INVALID_CPU_ID;
1046 return HWACCMR0TestSwitcher3264(pVM);
1047#endif
1048 default:
1049 /*
1050 * We're returning VERR_NOT_SUPPORT here so we've got something else
1051 * than -1 which the interrupt gate glue code might return.
1052 */
1053 Log(("operation %#x is not supported\n", enmOperation));
1054 return VERR_NOT_SUPPORTED;
1055 }
1056}
1057
1058
1059/**
1060 * Argument for vmmR0EntryExWrapper containing the arguments for VMMR0EntryEx.
1061 */
1062typedef struct VMMR0ENTRYEXARGS
1063{
1064 PVM pVM;
1065 VMCPUID idCpu;
1066 VMMR0OPERATION enmOperation;
1067 PSUPVMMR0REQHDR pReq;
1068 uint64_t u64Arg;
1069 PSUPDRVSESSION pSession;
1070} VMMR0ENTRYEXARGS;
1071/** Pointer to a vmmR0EntryExWrapper argument package. */
1072typedef VMMR0ENTRYEXARGS *PVMMR0ENTRYEXARGS;
1073
1074/**
1075 * This is just a longjmp wrapper function for VMMR0EntryEx calls.
1076 *
1077 * @returns VBox status code.
1078 * @param pvArgs The argument package
1079 */
1080static int vmmR0EntryExWrapper(void *pvArgs)
1081{
1082 return vmmR0EntryExWorker(((PVMMR0ENTRYEXARGS)pvArgs)->pVM,
1083 ((PVMMR0ENTRYEXARGS)pvArgs)->idCpu,
1084 ((PVMMR0ENTRYEXARGS)pvArgs)->enmOperation,
1085 ((PVMMR0ENTRYEXARGS)pvArgs)->pReq,
1086 ((PVMMR0ENTRYEXARGS)pvArgs)->u64Arg,
1087 ((PVMMR0ENTRYEXARGS)pvArgs)->pSession);
1088}
1089
1090
1091/**
1092 * The Ring 0 entry point, called by the support library (SUP).
1093 *
1094 * @returns VBox status code.
1095 * @param pVM The VM to operate on.
1096 * @param idCpu Virtual CPU ID argument. Must be NIL_VMCPUID if pVM
1097 * is NIL_RTR0PTR, and may be NIL_VMCPUID if it isn't
1098 * @param enmOperation Which operation to execute.
1099 * @param pReq This points to a SUPVMMR0REQHDR packet. Optional.
1100 * @param u64Arg Some simple constant argument.
1101 * @param pSession The session of the caller.
1102 * @remarks Assume called with interrupts _enabled_.
1103 */
1104VMMR0DECL(int) VMMR0EntryEx(PVM pVM, VMCPUID idCpu, VMMR0OPERATION enmOperation, PSUPVMMR0REQHDR pReq, uint64_t u64Arg, PSUPDRVSESSION pSession)
1105{
1106 /*
1107 * Requests that should only happen on the EMT thread will be
1108 * wrapped in a setjmp so we can assert without causing trouble.
1109 */
1110 if ( VALID_PTR(pVM)
1111 && pVM->pVMR0
1112 && idCpu < pVM->cCPUs)
1113 {
1114 switch (enmOperation)
1115 {
1116 /* These might/will be called before VMMR3Init. */
1117 case VMMR0_DO_GMM_INITIAL_RESERVATION:
1118 case VMMR0_DO_GMM_UPDATE_RESERVATION:
1119 case VMMR0_DO_GMM_ALLOCATE_PAGES:
1120 case VMMR0_DO_GMM_FREE_PAGES:
1121 case VMMR0_DO_GMM_BALLOONED_PAGES:
1122 case VMMR0_DO_GMM_DEFLATED_BALLOON:
1123 /* On the mac we might not have a valid jmp buf, so check these as well. */
1124 case VMMR0_DO_VMMR0_INIT:
1125 case VMMR0_DO_VMMR0_TERM:
1126 {
1127 PVMCPU pVCpu = &pVM->aCpus[idCpu];
1128
1129 if (!pVCpu->vmm.s.CallHostR0JmpBuf.pvSavedStack)
1130 break;
1131
1132 /** @todo validate this EMT claim... GVM knows. */
1133 VMMR0ENTRYEXARGS Args;
1134 Args.pVM = pVM;
1135 Args.idCpu = idCpu;
1136 Args.enmOperation = enmOperation;
1137 Args.pReq = pReq;
1138 Args.u64Arg = u64Arg;
1139 Args.pSession = pSession;
1140 return vmmR0CallHostSetJmpEx(&pVCpu->vmm.s.CallHostR0JmpBuf, vmmR0EntryExWrapper, &Args);
1141 }
1142
1143 default:
1144 break;
1145 }
1146 }
1147 return vmmR0EntryExWorker(pVM, idCpu, enmOperation, pReq, u64Arg, pSession);
1148}
1149
1150/**
1151 * Internal R0 logger worker: Flush logger.
1152 *
1153 * @param pLogger The logger instance to flush.
1154 * @remark This function must be exported!
1155 */
1156VMMR0DECL(void) vmmR0LoggerFlush(PRTLOGGER pLogger)
1157{
1158#ifdef LOG_ENABLED
1159 /*
1160 * Convert the pLogger into a VM handle and 'call' back to Ring-3.
1161 * (This is a bit paranoid code.)
1162 */
1163 PVMMR0LOGGER pR0Logger = (PVMMR0LOGGER)((uintptr_t)pLogger - RT_OFFSETOF(VMMR0LOGGER, Logger));
1164 if ( !VALID_PTR(pR0Logger)
1165 || !VALID_PTR(pR0Logger + 1)
1166 || pLogger->u32Magic != RTLOGGER_MAGIC)
1167 {
1168# ifdef DEBUG
1169 SUPR0Printf("vmmR0LoggerFlush: pLogger=%p!\n", pLogger);
1170# endif
1171 return;
1172 }
1173 if (pR0Logger->fFlushingDisabled)
1174 return; /* quietly */
1175
1176 PVM pVM = pR0Logger->pVM;
1177 if ( !VALID_PTR(pVM)
1178 || pVM->pVMR0 != pVM)
1179 {
1180# ifdef DEBUG
1181 SUPR0Printf("vmmR0LoggerFlush: pVM=%p! pVMR0=%p! pLogger=%p\n", pVM, pVM->pVMR0, pLogger);
1182# endif
1183 return;
1184 }
1185
1186 PVMCPU pVCpu = VMMGetCpu(pVM);
1187
1188 /*
1189 * Check that the jump buffer is armed.
1190 */
1191# ifdef RT_ARCH_X86
1192 if ( !pVCpu->vmm.s.CallHostR0JmpBuf.eip
1193 || pVCpu->vmm.s.CallHostR0JmpBuf.fInRing3Call)
1194# else
1195 if ( !pVCpu->vmm.s.CallHostR0JmpBuf.rip
1196 || pVCpu->vmm.s.CallHostR0JmpBuf.fInRing3Call)
1197# endif
1198 {
1199# ifdef DEBUG
1200 SUPR0Printf("vmmR0LoggerFlush: Jump buffer isn't armed!\n");
1201# endif
1202 return;
1203 }
1204 VMMR0CallHost(pVM, VMMCALLHOST_VMM_LOGGER_FLUSH, 0);
1205#endif
1206}
1207
1208#ifdef LOG_ENABLED
1209/**
1210 * Disables flushing of the ring-0 debug log.
1211 *
1212 * @param pVCpu The shared virtual cpu structure.
1213 */
1214VMMR0DECL(void) VMMR0LogFlushDisable(PVMCPU pVCpu)
1215{
1216 PVM pVM = pVCpu->pVMR0;
1217 if (pVCpu->vmm.s.pR0LoggerR0)
1218 pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled = true;
1219}
1220
1221
1222/**
1223 * Enables flushing of the ring-0 debug log.
1224 *
1225 * @param pVCpu The shared virtual cpu structure.
1226 */
1227VMMR0DECL(void) VMMR0LogFlushEnable(PVMCPU pVCpu)
1228{
1229 PVM pVM = pVCpu->pVMR0;
1230 if (pVCpu->vmm.s.pR0LoggerR0)
1231 pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled = false;
1232}
1233#endif
1234
1235/**
1236 * Jump back to ring-3 if we're the EMT and the longjmp is armed.
1237 *
1238 * @returns true if the breakpoint should be hit, false if it should be ignored.
1239 */
1240DECLEXPORT(bool) RTCALL RTAssertShouldPanic(void)
1241{
1242#if 0
1243 return true;
1244#else
1245 PVM pVM = GVMMR0GetVMByEMT(NIL_RTNATIVETHREAD);
1246 if (pVM)
1247 {
1248 PVMCPU pVCpu = VMMGetCpu(pVM);
1249
1250#ifdef RT_ARCH_X86
1251 if ( pVCpu->vmm.s.CallHostR0JmpBuf.eip
1252 && !pVCpu->vmm.s.CallHostR0JmpBuf.fInRing3Call)
1253#else
1254 if ( pVCpu->vmm.s.CallHostR0JmpBuf.rip
1255 && !pVCpu->vmm.s.CallHostR0JmpBuf.fInRing3Call)
1256#endif
1257 {
1258 int rc = VMMR0CallHost(pVM, VMMCALLHOST_VM_R0_ASSERTION, 0);
1259 return RT_FAILURE_NP(rc);
1260 }
1261 }
1262#ifdef RT_OS_LINUX
1263 return true;
1264#else
1265 return false;
1266#endif
1267#endif
1268}
1269
1270
1271/**
1272 * Override this so we can push it up to ring-3.
1273 *
1274 * @param pszExpr Expression. Can be NULL.
1275 * @param uLine Location line number.
1276 * @param pszFile Location file name.
1277 * @param pszFunction Location function name.
1278 */
1279DECLEXPORT(void) RTCALL AssertMsg1(const char *pszExpr, unsigned uLine, const char *pszFile, const char *pszFunction)
1280{
1281#if !defined(DEBUG_sandervl) && !defined(RT_OS_DARWIN)
1282 SUPR0Printf("\n!!R0-Assertion Failed!!\n"
1283 "Expression: %s\n"
1284 "Location : %s(%d) %s\n",
1285 pszExpr, pszFile, uLine, pszFunction);
1286#endif
1287 LogAlways(("\n!!R0-Assertion Failed!!\n"
1288 "Expression: %s\n"
1289 "Location : %s(%d) %s\n",
1290 pszExpr, pszFile, uLine, pszFunction));
1291
1292 PVM pVM = GVMMR0GetVMByEMT(NIL_RTNATIVETHREAD);
1293 if (pVM)
1294 RTStrPrintf(pVM->vmm.s.szRing0AssertMsg1, sizeof(pVM->vmm.s.szRing0AssertMsg1),
1295 "\n!!R0-Assertion Failed!!\n"
1296 "Expression: %s\n"
1297 "Location : %s(%d) %s\n",
1298 pszExpr, pszFile, uLine, pszFunction);
1299#ifdef RT_OS_DARWIN
1300 RTAssertMsg1(pszExpr, uLine, pszFile, pszFunction);
1301#endif
1302}
1303
1304
1305/**
1306 * Callback for RTLogFormatV which writes to the ring-3 log port.
1307 * See PFNLOGOUTPUT() for details.
1308 */
1309static DECLCALLBACK(size_t) rtLogOutput(void *pv, const char *pachChars, size_t cbChars)
1310{
1311 for (size_t i = 0; i < cbChars; i++)
1312 {
1313#if !defined(DEBUG_sandervl) && !defined(RT_OS_DARWIN)
1314 SUPR0Printf("%c", pachChars[i]);
1315#endif
1316 LogAlways(("%c", pachChars[i]));
1317 }
1318
1319 return cbChars;
1320}
1321
1322
1323DECLEXPORT(void) RTCALL AssertMsg2(const char *pszFormat, ...)
1324{
1325 va_list va;
1326
1327 PRTLOGGER pLog = RTLogDefaultInstance(); /** @todo we want this for release as well! */
1328 if (pLog)
1329 {
1330 va_start(va, pszFormat);
1331 RTLogFormatV(rtLogOutput, pLog, pszFormat, va);
1332 va_end(va);
1333
1334 PVM pVM = GVMMR0GetVMByEMT(NIL_RTNATIVETHREAD);
1335 if (pVM)
1336 {
1337 va_start(va, pszFormat);
1338 RTStrPrintfV(pVM->vmm.s.szRing0AssertMsg2, sizeof(pVM->vmm.s.szRing0AssertMsg2), pszFormat, va);
1339 va_end(va);
1340 }
1341 }
1342
1343#ifdef RT_OS_DARWIN
1344 va_start(va, pszFormat);
1345 RTAssertMsg2V(pszFormat, va);
1346 va_end(va);
1347#endif
1348}
1349
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