VirtualBox

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

Last change on this file since 61126 was 60847, checked in by vboxsync, 9 years ago

IOM: New way of defer RC+R0 I/O port writes, prepping for MMIO writes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 81.5 KB
Line 
1/* $Id: VMMR0.cpp 60847 2016-05-05 15:24:46Z vboxsync $ */
2/** @file
3 * VMM - Host Context Ring 0.
4 */
5
6/*
7 * Copyright (C) 2006-2015 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_VMM
23#include <VBox/vmm/vmm.h>
24#include <VBox/sup.h>
25#include <VBox/vmm/trpm.h>
26#include <VBox/vmm/cpum.h>
27#include <VBox/vmm/pdmapi.h>
28#include <VBox/vmm/pgm.h>
29#include <VBox/vmm/stam.h>
30#include <VBox/vmm/tm.h>
31#include "VMMInternal.h"
32#include <VBox/vmm/vm.h>
33#ifdef VBOX_WITH_PCI_PASSTHROUGH
34# include <VBox/vmm/pdmpci.h>
35#endif
36#ifdef VBOX_WITH_NEW_APIC
37# include <VBox/vmm/apic.h>
38#endif
39
40#include <VBox/vmm/gvmm.h>
41#include <VBox/vmm/gmm.h>
42#include <VBox/vmm/gim.h>
43#include <VBox/intnet.h>
44#include <VBox/vmm/hm.h>
45#include <VBox/param.h>
46#include <VBox/err.h>
47#include <VBox/version.h>
48#include <VBox/log.h>
49
50#include <iprt/asm-amd64-x86.h>
51#include <iprt/assert.h>
52#include <iprt/crc.h>
53#include <iprt/mp.h>
54#include <iprt/once.h>
55#include <iprt/stdarg.h>
56#include <iprt/string.h>
57#include <iprt/thread.h>
58#include <iprt/timer.h>
59
60#include "dtrace/VBoxVMM.h"
61
62
63#if defined(_MSC_VER) && defined(RT_ARCH_AMD64) /** @todo check this with with VC7! */
64# pragma intrinsic(_AddressOfReturnAddress)
65#endif
66
67#if defined(RT_OS_DARWIN) && ARCH_BITS == 32
68# error "32-bit darwin is no longer supported. Go back to 4.3 or earlier!"
69#endif
70
71
72
73/*********************************************************************************************************************************
74* Defined Constants And Macros *
75*********************************************************************************************************************************/
76/** @def VMM_CHECK_SMAP_SETUP
77 * SMAP check setup. */
78/** @def VMM_CHECK_SMAP_CHECK
79 * Checks that the AC flag is set if SMAP is enabled. If AC is not set,
80 * it will be logged and @a a_BadExpr is executed. */
81/** @def VMM_CHECK_SMAP_CHECK2
82 * Checks that the AC flag is set if SMAP is enabled. If AC is not set, it will
83 * be logged, written to the VMs assertion text buffer, and @a a_BadExpr is
84 * executed. */
85#if defined(VBOX_STRICT) || 1
86# define VMM_CHECK_SMAP_SETUP() uint32_t const fKernelFeatures = SUPR0GetKernelFeatures()
87# define VMM_CHECK_SMAP_CHECK(a_BadExpr) \
88 do { \
89 if (fKernelFeatures & SUPKERNELFEATURES_SMAP) \
90 { \
91 RTCCUINTREG fEflCheck = ASMGetFlags(); \
92 if (RT_LIKELY(fEflCheck & X86_EFL_AC)) \
93 { /* likely */ } \
94 else \
95 { \
96 SUPR0Printf("%s, line %d: EFLAGS.AC is clear! (%#x)\n", __FUNCTION__, __LINE__, (uint32_t)fEflCheck); \
97 a_BadExpr; \
98 } \
99 } \
100 } while (0)
101# define VMM_CHECK_SMAP_CHECK2(a_pVM, a_BadExpr) \
102 do { \
103 if (fKernelFeatures & SUPKERNELFEATURES_SMAP) \
104 { \
105 RTCCUINTREG fEflCheck = ASMGetFlags(); \
106 if (RT_LIKELY(fEflCheck & X86_EFL_AC)) \
107 { /* likely */ } \
108 else \
109 { \
110 SUPR0BadContext((a_pVM) ? (a_pVM)->pSession : NULL, __FILE__, __LINE__, "EFLAGS.AC is zero!"); \
111 RTStrPrintf(pVM->vmm.s.szRing0AssertMsg1, sizeof(pVM->vmm.s.szRing0AssertMsg1), \
112 "%s, line %d: EFLAGS.AC is clear! (%#x)\n", __FUNCTION__, __LINE__, (uint32_t)fEflCheck); \
113 a_BadExpr; \
114 } \
115 } \
116 } while (0)
117#else
118# define VMM_CHECK_SMAP_SETUP() uint32_t const fKernelFeatures = 0
119# define VMM_CHECK_SMAP_CHECK(a_BadExpr) NOREF(fKernelFeatures)
120# define VMM_CHECK_SMAP_CHECK2(a_pVM, a_BadExpr) NOREF(fKernelFeatures)
121#endif
122
123
124/*********************************************************************************************************************************
125* Internal Functions *
126*********************************************************************************************************************************/
127RT_C_DECLS_BEGIN
128#if defined(RT_ARCH_X86) && (defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD))
129extern uint64_t __udivdi3(uint64_t, uint64_t);
130extern uint64_t __umoddi3(uint64_t, uint64_t);
131#endif
132RT_C_DECLS_END
133
134
135/*********************************************************************************************************************************
136* Global Variables *
137*********************************************************************************************************************************/
138/** Drag in necessary library bits.
139 * The runtime lives here (in VMMR0.r0) and VBoxDD*R0.r0 links against us. */
140PFNRT g_VMMR0Deps[] =
141{
142 (PFNRT)RTCrc32,
143 (PFNRT)RTOnce,
144#if defined(RT_ARCH_X86) && (defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD))
145 (PFNRT)__udivdi3,
146 (PFNRT)__umoddi3,
147#endif
148 NULL
149};
150
151#ifdef RT_OS_SOLARIS
152/* Dependency information for the native solaris loader. */
153extern "C" { char _depends_on[] = "vboxdrv"; }
154#endif
155
156
157
158/**
159 * Initialize the module.
160 * This is called when we're first loaded.
161 *
162 * @returns 0 on success.
163 * @returns VBox status on failure.
164 * @param hMod Image handle for use in APIs.
165 */
166DECLEXPORT(int) ModuleInit(void *hMod)
167{
168 VMM_CHECK_SMAP_SETUP();
169 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
170
171#ifdef VBOX_WITH_DTRACE_R0
172 /*
173 * The first thing to do is register the static tracepoints.
174 * (Deregistration is automatic.)
175 */
176 int rc2 = SUPR0TracerRegisterModule(hMod, &g_VTGObjHeader);
177 if (RT_FAILURE(rc2))
178 return rc2;
179#endif
180 LogFlow(("ModuleInit:\n"));
181
182#ifdef VBOX_WITH_64ON32_CMOS_DEBUG
183 /*
184 * Display the CMOS debug code.
185 */
186 ASMOutU8(0x72, 0x03);
187 uint8_t bDebugCode = ASMInU8(0x73);
188 LogRel(("CMOS Debug Code: %#x (%d)\n", bDebugCode, bDebugCode));
189 RTLogComPrintf("CMOS Debug Code: %#x (%d)\n", bDebugCode, bDebugCode);
190#endif
191
192 /*
193 * Initialize the VMM, GVMM, GMM, HM, PGM (Darwin) and INTNET.
194 */
195 int rc = vmmInitFormatTypes();
196 if (RT_SUCCESS(rc))
197 {
198 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
199 rc = GVMMR0Init();
200 if (RT_SUCCESS(rc))
201 {
202 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
203 rc = GMMR0Init();
204 if (RT_SUCCESS(rc))
205 {
206 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
207 rc = HMR0Init();
208 if (RT_SUCCESS(rc))
209 {
210 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
211 rc = PGMRegisterStringFormatTypes();
212 if (RT_SUCCESS(rc))
213 {
214 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
215#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
216 rc = PGMR0DynMapInit();
217#endif
218 if (RT_SUCCESS(rc))
219 {
220 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
221 rc = IntNetR0Init();
222 if (RT_SUCCESS(rc))
223 {
224#ifdef VBOX_WITH_PCI_PASSTHROUGH
225 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
226 rc = PciRawR0Init();
227#endif
228 if (RT_SUCCESS(rc))
229 {
230 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
231 rc = CPUMR0ModuleInit();
232 if (RT_SUCCESS(rc))
233 {
234#ifdef VBOX_WITH_TRIPLE_FAULT_HACK
235 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
236 rc = vmmR0TripleFaultHackInit();
237 if (RT_SUCCESS(rc))
238#endif
239 {
240 VMM_CHECK_SMAP_CHECK(rc = VERR_VMM_SMAP_BUT_AC_CLEAR);
241 if (RT_SUCCESS(rc))
242 {
243 LogFlow(("ModuleInit: returns success.\n"));
244 return VINF_SUCCESS;
245 }
246 }
247
248 /*
249 * Bail out.
250 */
251#ifdef VBOX_WITH_TRIPLE_FAULT_HACK
252 vmmR0TripleFaultHackTerm();
253#endif
254 }
255 else
256 LogRel(("ModuleInit: CPUMR0ModuleInit -> %Rrc\n", rc));
257#ifdef VBOX_WITH_PCI_PASSTHROUGH
258 PciRawR0Term();
259#endif
260 }
261 else
262 LogRel(("ModuleInit: PciRawR0Init -> %Rrc\n", rc));
263 IntNetR0Term();
264 }
265 else
266 LogRel(("ModuleInit: IntNetR0Init -> %Rrc\n", rc));
267#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
268 PGMR0DynMapTerm();
269#endif
270 }
271 else
272 LogRel(("ModuleInit: PGMR0DynMapInit -> %Rrc\n", rc));
273 PGMDeregisterStringFormatTypes();
274 }
275 else
276 LogRel(("ModuleInit: PGMRegisterStringFormatTypes -> %Rrc\n", rc));
277 HMR0Term();
278 }
279 else
280 LogRel(("ModuleInit: HMR0Init -> %Rrc\n", rc));
281 GMMR0Term();
282 }
283 else
284 LogRel(("ModuleInit: GMMR0Init -> %Rrc\n", rc));
285 GVMMR0Term();
286 }
287 else
288 LogRel(("ModuleInit: GVMMR0Init -> %Rrc\n", rc));
289 vmmTermFormatTypes();
290 }
291 else
292 LogRel(("ModuleInit: vmmInitFormatTypes -> %Rrc\n", rc));
293
294 LogFlow(("ModuleInit: failed %Rrc\n", rc));
295 return rc;
296}
297
298
299/**
300 * Terminate the module.
301 * This is called when we're finally unloaded.
302 *
303 * @param hMod Image handle for use in APIs.
304 */
305DECLEXPORT(void) ModuleTerm(void *hMod)
306{
307 NOREF(hMod);
308 LogFlow(("ModuleTerm:\n"));
309
310 /*
311 * Terminate the CPUM module (Local APIC cleanup).
312 */
313 CPUMR0ModuleTerm();
314
315 /*
316 * Terminate the internal network service.
317 */
318 IntNetR0Term();
319
320 /*
321 * PGM (Darwin), HM and PciRaw global cleanup.
322 */
323#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
324 PGMR0DynMapTerm();
325#endif
326#ifdef VBOX_WITH_PCI_PASSTHROUGH
327 PciRawR0Term();
328#endif
329 PGMDeregisterStringFormatTypes();
330 HMR0Term();
331#ifdef VBOX_WITH_TRIPLE_FAULT_HACK
332 vmmR0TripleFaultHackTerm();
333#endif
334
335 /*
336 * Destroy the GMM and GVMM instances.
337 */
338 GMMR0Term();
339 GVMMR0Term();
340
341 vmmTermFormatTypes();
342
343 LogFlow(("ModuleTerm: returns\n"));
344}
345
346
347/**
348 * Initiates the R0 driver for a particular VM instance.
349 *
350 * @returns VBox status code.
351 *
352 * @param pVM The cross context VM structure.
353 * @param uSvnRev The SVN revision of the ring-3 part.
354 * @param uBuildType Build type indicator.
355 * @thread EMT.
356 */
357static int vmmR0InitVM(PVM pVM, uint32_t uSvnRev, uint32_t uBuildType)
358{
359 VMM_CHECK_SMAP_SETUP();
360 VMM_CHECK_SMAP_CHECK(return VERR_VMM_SMAP_BUT_AC_CLEAR);
361
362 /*
363 * Match the SVN revisions and build type.
364 */
365 if (uSvnRev != VMMGetSvnRev())
366 {
367 LogRel(("VMMR0InitVM: Revision mismatch, r3=%d r0=%d\n", uSvnRev, VMMGetSvnRev()));
368 SUPR0Printf("VMMR0InitVM: Revision mismatch, r3=%d r0=%d\n", uSvnRev, VMMGetSvnRev());
369 return VERR_VMM_R0_VERSION_MISMATCH;
370 }
371 if (uBuildType != vmmGetBuildType())
372 {
373 LogRel(("VMMR0InitVM: Build type mismatch, r3=%#x r0=%#x\n", uBuildType, vmmGetBuildType()));
374 SUPR0Printf("VMMR0InitVM: Build type mismatch, r3=%#x r0=%#x\n", uBuildType, vmmGetBuildType());
375 return VERR_VMM_R0_VERSION_MISMATCH;
376 }
377 if ( !VALID_PTR(pVM)
378 || pVM->pVMR0 != pVM)
379 return VERR_INVALID_PARAMETER;
380
381
382#ifdef LOG_ENABLED
383 /*
384 * Register the EMT R0 logger instance for VCPU 0.
385 */
386 PVMCPU pVCpu = &pVM->aCpus[0];
387
388 PVMMR0LOGGER pR0Logger = pVCpu->vmm.s.pR0LoggerR0;
389 if (pR0Logger)
390 {
391# if 0 /* testing of the logger. */
392 LogCom(("vmmR0InitVM: before %p\n", RTLogDefaultInstance()));
393 LogCom(("vmmR0InitVM: pfnFlush=%p actual=%p\n", pR0Logger->Logger.pfnFlush, vmmR0LoggerFlush));
394 LogCom(("vmmR0InitVM: pfnLogger=%p actual=%p\n", pR0Logger->Logger.pfnLogger, vmmR0LoggerWrapper));
395 LogCom(("vmmR0InitVM: offScratch=%d fFlags=%#x fDestFlags=%#x\n", pR0Logger->Logger.offScratch, pR0Logger->Logger.fFlags, pR0Logger->Logger.fDestFlags));
396
397 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
398 LogCom(("vmmR0InitVM: after %p reg\n", RTLogDefaultInstance()));
399 RTLogSetDefaultInstanceThread(NULL, pVM->pSession);
400 LogCom(("vmmR0InitVM: after %p dereg\n", RTLogDefaultInstance()));
401
402 pR0Logger->Logger.pfnLogger("hello ring-0 logger\n");
403 LogCom(("vmmR0InitVM: returned successfully from direct logger call.\n"));
404 pR0Logger->Logger.pfnFlush(&pR0Logger->Logger);
405 LogCom(("vmmR0InitVM: returned successfully from direct flush call.\n"));
406
407 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
408 LogCom(("vmmR0InitVM: after %p reg2\n", RTLogDefaultInstance()));
409 pR0Logger->Logger.pfnLogger("hello ring-0 logger\n");
410 LogCom(("vmmR0InitVM: returned successfully from direct logger call (2). offScratch=%d\n", pR0Logger->Logger.offScratch));
411 RTLogSetDefaultInstanceThread(NULL, pVM->pSession);
412 LogCom(("vmmR0InitVM: after %p dereg2\n", RTLogDefaultInstance()));
413
414 RTLogLoggerEx(&pR0Logger->Logger, 0, ~0U, "hello ring-0 logger (RTLogLoggerEx)\n");
415 LogCom(("vmmR0InitVM: RTLogLoggerEx returned fine offScratch=%d\n", pR0Logger->Logger.offScratch));
416
417 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
418 RTLogPrintf("hello ring-0 logger (RTLogPrintf)\n");
419 LogCom(("vmmR0InitVM: RTLogPrintf returned fine offScratch=%d\n", pR0Logger->Logger.offScratch));
420# endif
421 Log(("Switching to per-thread logging instance %p (key=%p)\n", &pR0Logger->Logger, pVM->pSession));
422 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
423 pR0Logger->fRegistered = true;
424 }
425#endif /* LOG_ENABLED */
426
427 /*
428 * Check if the host supports high resolution timers or not.
429 */
430 if ( pVM->vmm.s.fUsePeriodicPreemptionTimers
431 && !RTTimerCanDoHighResolution())
432 pVM->vmm.s.fUsePeriodicPreemptionTimers = false;
433
434 /*
435 * Initialize the per VM data for GVMM and GMM.
436 */
437 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
438 int rc = GVMMR0InitVM(pVM);
439// if (RT_SUCCESS(rc))
440// rc = GMMR0InitPerVMData(pVM);
441 if (RT_SUCCESS(rc))
442 {
443 /*
444 * Init HM, CPUM and PGM (Darwin only).
445 */
446 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
447 rc = HMR0InitVM(pVM);
448 if (RT_SUCCESS(rc))
449 VMM_CHECK_SMAP_CHECK2(pVM, rc = VERR_VMM_RING0_ASSERTION); /* CPUR0InitVM will otherwise panic the host */
450 if (RT_SUCCESS(rc))
451 {
452 rc = CPUMR0InitVM(pVM);
453 if (RT_SUCCESS(rc))
454 {
455 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
456#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
457 rc = PGMR0DynMapInitVM(pVM);
458#endif
459 if (RT_SUCCESS(rc))
460 {
461 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
462#ifdef VBOX_WITH_PCI_PASSTHROUGH
463 rc = PciRawR0InitVM(pVM);
464#endif
465 if (RT_SUCCESS(rc))
466 {
467 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
468 rc = GIMR0InitVM(pVM);
469 if (RT_SUCCESS(rc))
470 {
471 VMM_CHECK_SMAP_CHECK2(pVM, rc = VERR_VMM_RING0_ASSERTION);
472 if (RT_SUCCESS(rc))
473 {
474 GVMMR0DoneInitVM(pVM);
475 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
476 return rc;
477 }
478
479 /* bail out*/
480 GIMR0TermVM(pVM);
481 }
482#ifdef VBOX_WITH_PCI_PASSTHROUGH
483 PciRawR0TermVM(pVM);
484#endif
485 }
486 }
487 }
488 HMR0TermVM(pVM);
489 }
490 }
491
492 RTLogSetDefaultInstanceThread(NULL, (uintptr_t)pVM->pSession);
493 return rc;
494}
495
496
497/**
498 * Terminates the R0 bits for a particular VM instance.
499 *
500 * This is normally called by ring-3 as part of the VM termination process, but
501 * may alternatively be called during the support driver session cleanup when
502 * the VM object is destroyed (see GVMM).
503 *
504 * @returns VBox status code.
505 *
506 * @param pVM The cross context VM structure.
507 * @param pGVM Pointer to the global VM structure. Optional.
508 * @thread EMT or session clean up thread.
509 */
510VMMR0_INT_DECL(int) VMMR0TermVM(PVM pVM, PGVM pGVM)
511{
512#ifdef VBOX_WITH_PCI_PASSTHROUGH
513 PciRawR0TermVM(pVM);
514#endif
515
516 /*
517 * Tell GVMM what we're up to and check that we only do this once.
518 */
519 if (GVMMR0DoingTermVM(pVM, pGVM))
520 {
521 GIMR0TermVM(pVM);
522
523 /** @todo I wish to call PGMR0PhysFlushHandyPages(pVM, &pVM->aCpus[idCpu])
524 * here to make sure we don't leak any shared pages if we crash... */
525#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
526 PGMR0DynMapTermVM(pVM);
527#endif
528 HMR0TermVM(pVM);
529 }
530
531 /*
532 * Deregister the logger.
533 */
534 RTLogSetDefaultInstanceThread(NULL, (uintptr_t)pVM->pSession);
535 return VINF_SUCCESS;
536}
537
538
539/**
540 * VMM ring-0 thread-context callback.
541 *
542 * This does common HM state updating and calls the HM-specific thread-context
543 * callback.
544 *
545 * @param enmEvent The thread-context event.
546 * @param pvUser Opaque pointer to the VMCPU.
547 *
548 * @thread EMT(pvUser)
549 */
550static DECLCALLBACK(void) vmmR0ThreadCtxCallback(RTTHREADCTXEVENT enmEvent, void *pvUser)
551{
552 PVMCPU pVCpu = (PVMCPU)pvUser;
553
554 switch (enmEvent)
555 {
556 case RTTHREADCTXEVENT_IN:
557 {
558 /*
559 * Linux may call us with preemption enabled (really!) but technically we
560 * cannot get preempted here, otherwise we end up in an infinite recursion
561 * scenario (i.e. preempted in resume hook -> preempt hook -> resume hook...
562 * ad infinitum). Let's just disable preemption for now...
563 */
564 /** @todo r=bird: I don't believe the above. The linux code is clearly enabling
565 * preemption after doing the callout (one or two functions up the
566 * call chain). */
567 /** @todo r=ramshankar: See @bugref{5313#c30}. */
568 RTTHREADPREEMPTSTATE ParanoidPreemptState = RTTHREADPREEMPTSTATE_INITIALIZER;
569 RTThreadPreemptDisable(&ParanoidPreemptState);
570
571 /* We need to update the VCPU <-> host CPU mapping. */
572 RTCPUID idHostCpu;
573 uint32_t iHostCpuSet = RTMpCurSetIndexAndId(&idHostCpu);
574 pVCpu->iHostCpuSet = iHostCpuSet;
575 ASMAtomicWriteU32(&pVCpu->idHostCpu, idHostCpu);
576
577 /* In the very unlikely event that the GIP delta for the CPU we're
578 rescheduled needs calculating, try force a return to ring-3.
579 We unfortunately cannot do the measurements right here. */
580 if (RT_UNLIKELY(SUPIsTscDeltaAvailableForCpuSetIndex(iHostCpuSet)))
581 VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
582
583 /* Invoke the HM-specific thread-context callback. */
584 HMR0ThreadCtxCallback(enmEvent, pvUser);
585
586 /* Restore preemption. */
587 RTThreadPreemptRestore(&ParanoidPreemptState);
588 break;
589 }
590
591 case RTTHREADCTXEVENT_OUT:
592 {
593 /* Invoke the HM-specific thread-context callback. */
594 HMR0ThreadCtxCallback(enmEvent, pvUser);
595
596 /*
597 * Sigh. See VMMGetCpu() used by VMCPU_ASSERT_EMT(). We cannot let several VCPUs
598 * have the same host CPU associated with it.
599 */
600 pVCpu->iHostCpuSet = UINT32_MAX;
601 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
602 break;
603 }
604
605 default:
606 /* Invoke the HM-specific thread-context callback. */
607 HMR0ThreadCtxCallback(enmEvent, pvUser);
608 break;
609 }
610}
611
612
613/**
614 * Creates thread switching hook for the current EMT thread.
615 *
616 * This is called by GVMMR0CreateVM and GVMMR0RegisterVCpu. If the host
617 * platform does not implement switcher hooks, no hooks will be create and the
618 * member set to NIL_RTTHREADCTXHOOK.
619 *
620 * @returns VBox status code.
621 * @param pVCpu The cross context virtual CPU structure.
622 * @thread EMT(pVCpu)
623 */
624VMMR0_INT_DECL(int) VMMR0ThreadCtxHookCreateForEmt(PVMCPU pVCpu)
625{
626 VMCPU_ASSERT_EMT(pVCpu);
627 Assert(pVCpu->vmm.s.hCtxHook == NIL_RTTHREADCTXHOOK);
628
629 int rc = RTThreadCtxHookCreate(&pVCpu->vmm.s.hCtxHook, 0, vmmR0ThreadCtxCallback, pVCpu);
630 if (RT_SUCCESS(rc))
631 return rc;
632
633 pVCpu->vmm.s.hCtxHook = NIL_RTTHREADCTXHOOK;
634 if (rc == VERR_NOT_SUPPORTED)
635 return VINF_SUCCESS;
636
637 LogRelMax(32, ("RTThreadCtxHookCreate failed! rc=%Rrc pVCpu=%p idCpu=%RU32\n", rc, pVCpu, pVCpu->idCpu));
638 return VINF_SUCCESS; /* Just ignore it, we can live without context hooks. */
639}
640
641
642/**
643 * Destroys the thread switching hook for the specified VCPU.
644 *
645 * @param pVCpu The cross context virtual CPU structure.
646 * @remarks Can be called from any thread.
647 */
648VMMR0_INT_DECL(void) VMMR0ThreadCtxHookDestroyForEmt(PVMCPU pVCpu)
649{
650 int rc = RTThreadCtxHookDestroy(pVCpu->vmm.s.hCtxHook);
651 AssertRC(rc);
652}
653
654
655/**
656 * Disables the thread switching hook for this VCPU (if we got one).
657 *
658 * @param pVCpu The cross context virtual CPU structure.
659 * @thread EMT(pVCpu)
660 *
661 * @remarks This also clears VMCPU::idHostCpu, so the mapping is invalid after
662 * this call. This means you have to be careful with what you do!
663 */
664VMMR0_INT_DECL(void) VMMR0ThreadCtxHookDisable(PVMCPU pVCpu)
665{
666 /*
667 * Clear the VCPU <-> host CPU mapping as we've left HM context.
668 * @bugref{7726#c19} explains the need for this trick:
669 *
670 * hmR0VmxCallRing3Callback/hmR0SvmCallRing3Callback &
671 * hmR0VmxLeaveSession/hmR0SvmLeaveSession disables context hooks during
672 * longjmp & normal return to ring-3, which opens a window where we may be
673 * rescheduled without changing VMCPUID::idHostCpu and cause confusion if
674 * the CPU starts executing a different EMT. Both functions first disables
675 * preemption and then calls HMR0LeaveCpu which invalids idHostCpu, leaving
676 * an opening for getting preempted.
677 */
678 /** @todo Make HM not need this API! Then we could leave the hooks enabled
679 * all the time. */
680 /** @todo move this into the context hook disabling if(). */
681 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
682
683 /*
684 * Disable the context hook, if we got one.
685 */
686 if (pVCpu->vmm.s.hCtxHook != NIL_RTTHREADCTXHOOK)
687 {
688 Assert(!RTThreadPreemptIsEnabled(NIL_RTTHREAD));
689 int rc = RTThreadCtxHookDisable(pVCpu->vmm.s.hCtxHook);
690 AssertRC(rc);
691 }
692}
693
694
695/**
696 * Internal version of VMMR0ThreadCtxHooksAreRegistered.
697 *
698 * @returns true if registered, false otherwise.
699 * @param pVCpu The cross context virtual CPU structure.
700 */
701DECLINLINE(bool) vmmR0ThreadCtxHookIsEnabled(PVMCPU pVCpu)
702{
703 return RTThreadCtxHookIsEnabled(pVCpu->vmm.s.hCtxHook);
704}
705
706
707/**
708 * Whether thread-context hooks are registered for this VCPU.
709 *
710 * @returns true if registered, false otherwise.
711 * @param pVCpu The cross context virtual CPU structure.
712 */
713VMMR0_INT_DECL(bool) VMMR0ThreadCtxHookIsEnabled(PVMCPU pVCpu)
714{
715 return vmmR0ThreadCtxHookIsEnabled(pVCpu);
716}
717
718
719#ifdef VBOX_WITH_STATISTICS
720/**
721 * Record return code statistics
722 * @param pVM The cross context VM structure.
723 * @param pVCpu The cross context virtual CPU structure.
724 * @param rc The status code.
725 */
726static void vmmR0RecordRC(PVM pVM, PVMCPU pVCpu, int rc)
727{
728 /*
729 * Collect statistics.
730 */
731 switch (rc)
732 {
733 case VINF_SUCCESS:
734 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetNormal);
735 break;
736 case VINF_EM_RAW_INTERRUPT:
737 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetInterrupt);
738 break;
739 case VINF_EM_RAW_INTERRUPT_HYPER:
740 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetInterruptHyper);
741 break;
742 case VINF_EM_RAW_GUEST_TRAP:
743 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetGuestTrap);
744 break;
745 case VINF_EM_RAW_RING_SWITCH:
746 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetRingSwitch);
747 break;
748 case VINF_EM_RAW_RING_SWITCH_INT:
749 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetRingSwitchInt);
750 break;
751 case VINF_EM_RAW_STALE_SELECTOR:
752 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetStaleSelector);
753 break;
754 case VINF_EM_RAW_IRET_TRAP:
755 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIRETTrap);
756 break;
757 case VINF_IOM_R3_IOPORT_READ:
758 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIORead);
759 break;
760 case VINF_IOM_R3_IOPORT_WRITE:
761 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIOWrite);
762 break;
763 case VINF_IOM_R3_IOPORT_COMMIT_WRITE:
764 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIOCommitWrite);
765 break;
766 case VINF_IOM_R3_MMIO_READ:
767 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIORead);
768 break;
769 case VINF_IOM_R3_MMIO_WRITE:
770 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOWrite);
771 break;
772 case VINF_IOM_R3_MMIO_COMMIT_WRITE:
773 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOCommitWrite);
774 break;
775 case VINF_IOM_R3_MMIO_READ_WRITE:
776 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOReadWrite);
777 break;
778 case VINF_PATM_HC_MMIO_PATCH_READ:
779 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOPatchRead);
780 break;
781 case VINF_PATM_HC_MMIO_PATCH_WRITE:
782 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMMIOPatchWrite);
783 break;
784 case VINF_CPUM_R3_MSR_READ:
785 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMSRRead);
786 break;
787 case VINF_CPUM_R3_MSR_WRITE:
788 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMSRWrite);
789 break;
790 case VINF_EM_RAW_EMULATE_INSTR:
791 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetEmulate);
792 break;
793 case VINF_EM_RAW_EMULATE_IO_BLOCK:
794 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIOBlockEmulate);
795 break;
796 case VINF_PATCH_EMULATE_INSTR:
797 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchEmulate);
798 break;
799 case VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT:
800 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetLDTFault);
801 break;
802 case VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT:
803 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetGDTFault);
804 break;
805 case VINF_EM_RAW_EMULATE_INSTR_IDT_FAULT:
806 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetIDTFault);
807 break;
808 case VINF_EM_RAW_EMULATE_INSTR_TSS_FAULT:
809 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetTSSFault);
810 break;
811 case VINF_CSAM_PENDING_ACTION:
812 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetCSAMTask);
813 break;
814 case VINF_PGM_SYNC_CR3:
815 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetSyncCR3);
816 break;
817 case VINF_PATM_PATCH_INT3:
818 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchInt3);
819 break;
820 case VINF_PATM_PATCH_TRAP_PF:
821 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchPF);
822 break;
823 case VINF_PATM_PATCH_TRAP_GP:
824 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchGP);
825 break;
826 case VINF_PATM_PENDING_IRQ_AFTER_IRET:
827 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchIretIRQ);
828 break;
829 case VINF_EM_RESCHEDULE_REM:
830 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetRescheduleREM);
831 break;
832 case VINF_EM_RAW_TO_R3:
833 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3Total);
834 if (VM_FF_IS_PENDING(pVM, VM_FF_TM_VIRTUAL_SYNC))
835 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3TMVirt);
836 else if (VM_FF_IS_PENDING(pVM, VM_FF_PGM_NEED_HANDY_PAGES))
837 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3HandyPages);
838 else if (VM_FF_IS_PENDING(pVM, VM_FF_PDM_QUEUES))
839 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3PDMQueues);
840 else if (VM_FF_IS_PENDING(pVM, VM_FF_EMT_RENDEZVOUS))
841 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3Rendezvous);
842 else if (VM_FF_IS_PENDING(pVM, VM_FF_PDM_DMA))
843 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3DMA);
844 else if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_TIMER))
845 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3Timer);
846 else if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_PDM_CRITSECT))
847 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3CritSect);
848 else if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_TO_R3))
849 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3FF);
850 else if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_IEM))
851 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3Iem);
852 else if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_IOM))
853 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3Iom);
854 else
855 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetToR3Unknown);
856 break;
857
858 case VINF_EM_RAW_TIMER_PENDING:
859 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetTimerPending);
860 break;
861 case VINF_EM_RAW_INTERRUPT_PENDING:
862 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetInterruptPending);
863 break;
864 case VINF_VMM_CALL_HOST:
865 switch (pVCpu->vmm.s.enmCallRing3Operation)
866 {
867 case VMMCALLRING3_PDM_CRIT_SECT_ENTER:
868 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPDMCritSectEnter);
869 break;
870 case VMMCALLRING3_PDM_LOCK:
871 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPDMLock);
872 break;
873 case VMMCALLRING3_PGM_POOL_GROW:
874 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMPoolGrow);
875 break;
876 case VMMCALLRING3_PGM_LOCK:
877 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMLock);
878 break;
879 case VMMCALLRING3_PGM_MAP_CHUNK:
880 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMMapChunk);
881 break;
882 case VMMCALLRING3_PGM_ALLOCATE_HANDY_PAGES:
883 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallPGMAllocHandy);
884 break;
885 case VMMCALLRING3_REM_REPLAY_HANDLER_NOTIFICATIONS:
886 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallRemReplay);
887 break;
888 case VMMCALLRING3_VMM_LOGGER_FLUSH:
889 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallLogFlush);
890 break;
891 case VMMCALLRING3_VM_SET_ERROR:
892 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallVMSetError);
893 break;
894 case VMMCALLRING3_VM_SET_RUNTIME_ERROR:
895 STAM_COUNTER_INC(&pVM->vmm.s.StatRZCallVMSetRuntimeError);
896 break;
897 case VMMCALLRING3_VM_R0_ASSERTION:
898 default:
899 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetCallRing3);
900 break;
901 }
902 break;
903 case VINF_PATM_DUPLICATE_FUNCTION:
904 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPATMDuplicateFn);
905 break;
906 case VINF_PGM_CHANGE_MODE:
907 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPGMChangeMode);
908 break;
909 case VINF_PGM_POOL_FLUSH_PENDING:
910 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPGMFlushPending);
911 break;
912 case VINF_EM_PENDING_REQUEST:
913 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPendingRequest);
914 break;
915 case VINF_EM_HM_PATCH_TPR_INSTR:
916 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetPatchTPR);
917 break;
918 default:
919 STAM_COUNTER_INC(&pVM->vmm.s.StatRZRetMisc);
920 break;
921 }
922}
923#endif /* VBOX_WITH_STATISTICS */
924
925
926/**
927 * The Ring 0 entry point, called by the fast-ioctl path.
928 *
929 * @param pVM The cross context VM structure.
930 * The return code is stored in pVM->vmm.s.iLastGZRc.
931 * @param idCpu The Virtual CPU ID of the calling EMT.
932 * @param enmOperation Which operation to execute.
933 * @remarks Assume called with interrupts _enabled_.
934 */
935VMMR0DECL(void) VMMR0EntryFast(PVM pVM, VMCPUID idCpu, VMMR0OPERATION enmOperation)
936{
937 /*
938 * Validation.
939 */
940 if (RT_UNLIKELY(idCpu >= pVM->cCpus))
941 return;
942 PVMCPU pVCpu = &pVM->aCpus[idCpu];
943 if (RT_UNLIKELY(pVCpu->hNativeThreadR0 != RTThreadNativeSelf()))
944 return;
945 VMM_CHECK_SMAP_SETUP();
946 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
947
948 /*
949 * Perform requested operation.
950 */
951 switch (enmOperation)
952 {
953 /*
954 * Switch to GC and run guest raw mode code.
955 * Disable interrupts before doing the world switch.
956 */
957 case VMMR0_DO_RAW_RUN:
958 {
959#ifdef VBOX_WITH_RAW_MODE
960# ifndef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
961 /* Some safety precautions first. */
962 if (RT_UNLIKELY(!PGMGetHyperCR3(pVCpu)))
963 {
964 pVCpu->vmm.s.iLastGZRc = VERR_PGM_NO_CR3_SHADOW_ROOT;
965 break;
966 }
967# endif
968
969 /*
970 * Disable preemption.
971 */
972 RTTHREADPREEMPTSTATE PreemptState = RTTHREADPREEMPTSTATE_INITIALIZER;
973 RTThreadPreemptDisable(&PreemptState);
974
975 /*
976 * Get the host CPU identifiers, make sure they are valid and that
977 * we've got a TSC delta for the CPU.
978 */
979 RTCPUID idHostCpu;
980 uint32_t iHostCpuSet = RTMpCurSetIndexAndId(&idHostCpu);
981 if (RT_LIKELY( iHostCpuSet < RTCPUSET_MAX_CPUS
982 && SUPIsTscDeltaAvailableForCpuSetIndex(iHostCpuSet)))
983 {
984 /*
985 * Commit the CPU identifiers and update the periodict preemption timer if it's active.
986 */
987# ifdef VBOX_WITH_VMMR0_DISABLE_LAPIC_NMI
988 CPUMR0SetLApic(pVCpu, iHostCpuSet);
989# endif
990 pVCpu->iHostCpuSet = iHostCpuSet;
991 ASMAtomicWriteU32(&pVCpu->idHostCpu, idHostCpu);
992
993 if (pVM->vmm.s.fUsePeriodicPreemptionTimers)
994 GVMMR0SchedUpdatePeriodicPreemptionTimer(pVM, pVCpu->idHostCpu, TMCalcHostTimerFrequency(pVM, pVCpu));
995
996 /*
997 * We might need to disable VT-x if the active switcher turns off paging.
998 */
999 bool fVTxDisabled;
1000 int rc = HMR0EnterSwitcher(pVM, pVM->vmm.s.enmSwitcher, &fVTxDisabled);
1001 if (RT_SUCCESS(rc))
1002 {
1003 /*
1004 * Disable interrupts and run raw-mode code. The loop is for efficiently
1005 * dispatching tracepoints that fired in raw-mode context.
1006 */
1007 RTCCUINTREG uFlags = ASMIntDisableFlags();
1008
1009 for (;;)
1010 {
1011 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED_EXEC);
1012 TMNotifyStartOfExecution(pVCpu);
1013
1014 rc = pVM->vmm.s.pfnR0ToRawMode(pVM);
1015 pVCpu->vmm.s.iLastGZRc = rc;
1016
1017 TMNotifyEndOfExecution(pVCpu);
1018 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED);
1019
1020 if (rc != VINF_VMM_CALL_TRACER)
1021 break;
1022 SUPR0TracerUmodProbeFire(pVM->pSession, &pVCpu->vmm.s.TracerCtx);
1023 }
1024
1025 /*
1026 * Re-enable VT-x before we dispatch any pending host interrupts and
1027 * re-enables interrupts.
1028 */
1029 HMR0LeaveSwitcher(pVM, fVTxDisabled);
1030
1031 if ( rc == VINF_EM_RAW_INTERRUPT
1032 || rc == VINF_EM_RAW_INTERRUPT_HYPER)
1033 TRPMR0DispatchHostInterrupt(pVM);
1034
1035 ASMSetFlags(uFlags);
1036
1037 /* Fire dtrace probe and collect statistics. */
1038 VBOXVMM_R0_VMM_RETURN_TO_RING3_RC(pVCpu, CPUMQueryGuestCtxPtr(pVCpu), rc);
1039# ifdef VBOX_WITH_STATISTICS
1040 STAM_COUNTER_INC(&pVM->vmm.s.StatRunRC);
1041 vmmR0RecordRC(pVM, pVCpu, rc);
1042# endif
1043 }
1044 else
1045 pVCpu->vmm.s.iLastGZRc = rc;
1046
1047 /*
1048 * Invalidate the host CPU identifiers as we restore preemption.
1049 */
1050 pVCpu->iHostCpuSet = UINT32_MAX;
1051 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
1052
1053 RTThreadPreemptRestore(&PreemptState);
1054 }
1055 /*
1056 * Invalid CPU set index or TSC delta in need of measuring.
1057 */
1058 else
1059 {
1060 RTThreadPreemptRestore(&PreemptState);
1061 if (iHostCpuSet < RTCPUSET_MAX_CPUS)
1062 {
1063 int rc = SUPR0TscDeltaMeasureBySetIndex(pVM->pSession, iHostCpuSet, 0 /*fFlags*/,
1064 2 /*cMsWaitRetry*/, 5*RT_MS_1SEC /*cMsWaitThread*/,
1065 0 /*default cTries*/);
1066 if (RT_SUCCESS(rc) || rc == VERR_CPU_OFFLINE)
1067 pVCpu->vmm.s.iLastGZRc = VINF_EM_RAW_TO_R3;
1068 else
1069 pVCpu->vmm.s.iLastGZRc = rc;
1070 }
1071 else
1072 pVCpu->vmm.s.iLastGZRc = VERR_INVALID_CPU_INDEX;
1073 }
1074
1075#else /* !VBOX_WITH_RAW_MODE */
1076 pVCpu->vmm.s.iLastGZRc = VERR_RAW_MODE_NOT_SUPPORTED;
1077#endif
1078 break;
1079 }
1080
1081 /*
1082 * Run guest code using the available hardware acceleration technology.
1083 */
1084 case VMMR0_DO_HM_RUN:
1085 {
1086 /*
1087 * Disable preemption.
1088 */
1089 Assert(!vmmR0ThreadCtxHookIsEnabled(pVCpu));
1090 RTTHREADPREEMPTSTATE PreemptState = RTTHREADPREEMPTSTATE_INITIALIZER;
1091 RTThreadPreemptDisable(&PreemptState);
1092
1093 /*
1094 * Get the host CPU identifiers, make sure they are valid and that
1095 * we've got a TSC delta for the CPU.
1096 */
1097 RTCPUID idHostCpu;
1098 uint32_t iHostCpuSet = RTMpCurSetIndexAndId(&idHostCpu);
1099 if (RT_LIKELY( iHostCpuSet < RTCPUSET_MAX_CPUS
1100 && SUPIsTscDeltaAvailableForCpuSetIndex(iHostCpuSet)))
1101 {
1102 pVCpu->iHostCpuSet = iHostCpuSet;
1103 ASMAtomicWriteU32(&pVCpu->idHostCpu, idHostCpu);
1104
1105 /*
1106 * Update the periodic preemption timer if it's active.
1107 */
1108 if (pVM->vmm.s.fUsePeriodicPreemptionTimers)
1109 GVMMR0SchedUpdatePeriodicPreemptionTimer(pVM, pVCpu->idHostCpu, TMCalcHostTimerFrequency(pVM, pVCpu));
1110 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1111
1112#ifdef LOG_ENABLED
1113 /*
1114 * Ugly: Lazy registration of ring 0 loggers.
1115 */
1116 if (pVCpu->idCpu > 0)
1117 {
1118 PVMMR0LOGGER pR0Logger = pVCpu->vmm.s.pR0LoggerR0;
1119 if ( pR0Logger
1120 && RT_UNLIKELY(!pR0Logger->fRegistered))
1121 {
1122 RTLogSetDefaultInstanceThread(&pR0Logger->Logger, (uintptr_t)pVM->pSession);
1123 pR0Logger->fRegistered = true;
1124 }
1125 }
1126#endif
1127
1128 int rc;
1129 bool fPreemptRestored = false;
1130 if (!HMR0SuspendPending())
1131 {
1132 /*
1133 * Enable the context switching hook.
1134 */
1135 if (pVCpu->vmm.s.hCtxHook != NIL_RTTHREADCTXHOOK)
1136 {
1137 Assert(!RTThreadCtxHookIsEnabled(pVCpu->vmm.s.hCtxHook));
1138 int rc2 = RTThreadCtxHookEnable(pVCpu->vmm.s.hCtxHook); AssertRC(rc2);
1139 }
1140
1141 /*
1142 * Enter HM context.
1143 */
1144 rc = HMR0Enter(pVM, pVCpu);
1145 if (RT_SUCCESS(rc))
1146 {
1147 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED_HM);
1148
1149 /*
1150 * When preemption hooks are in place, enable preemption now that
1151 * we're in HM context.
1152 */
1153 if (vmmR0ThreadCtxHookIsEnabled(pVCpu))
1154 {
1155 fPreemptRestored = true;
1156 RTThreadPreemptRestore(&PreemptState);
1157 }
1158
1159 /*
1160 * Setup the longjmp machinery and execute guest code (calls HMR0RunGuestCode).
1161 */
1162 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1163 rc = vmmR0CallRing3SetJmp(&pVCpu->vmm.s.CallRing3JmpBufR0, HMR0RunGuestCode, pVM, pVCpu);
1164 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1165
1166 /*
1167 * Assert sanity on the way out. Using manual assertions code here as normal
1168 * assertions are going to panic the host since we're outside the setjmp/longjmp zone.
1169 */
1170 if (RT_UNLIKELY( VMCPU_GET_STATE(pVCpu) != VMCPUSTATE_STARTED_HM
1171 && RT_SUCCESS_NP(rc) && rc != VINF_VMM_CALL_HOST ))
1172 {
1173 pVM->vmm.s.szRing0AssertMsg1[0] = '\0';
1174 RTStrPrintf(pVM->vmm.s.szRing0AssertMsg2, sizeof(pVM->vmm.s.szRing0AssertMsg2),
1175 "Got VMCPU state %d expected %d.\n", VMCPU_GET_STATE(pVCpu), VMCPUSTATE_STARTED_HM);
1176 rc = VERR_VMM_WRONG_HM_VMCPU_STATE;
1177 }
1178 /** @todo Get rid of this. HM shouldn't disable the context hook. */
1179 else if (RT_UNLIKELY(vmmR0ThreadCtxHookIsEnabled(pVCpu)))
1180 {
1181 pVM->vmm.s.szRing0AssertMsg1[0] = '\0';
1182 RTStrPrintf(pVM->vmm.s.szRing0AssertMsg2, sizeof(pVM->vmm.s.szRing0AssertMsg2),
1183 "Thread-context hooks still enabled! VCPU=%p Id=%u rc=%d.\n", pVCpu, pVCpu->idCpu, rc);
1184 rc = VERR_INVALID_STATE;
1185 }
1186
1187 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED);
1188 }
1189 STAM_COUNTER_INC(&pVM->vmm.s.StatRunRC);
1190
1191 /*
1192 * Invalidate the host CPU identifiers before we disable the context
1193 * hook / restore preemption.
1194 */
1195 pVCpu->iHostCpuSet = UINT32_MAX;
1196 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
1197
1198 /*
1199 * Disable context hooks. Due to unresolved cleanup issues, we
1200 * cannot leave the hooks enabled when we return to ring-3.
1201 *
1202 * Note! At the moment HM may also have disabled the hook
1203 * when we get here, but the IPRT API handles that.
1204 */
1205 if (pVCpu->vmm.s.hCtxHook != NIL_RTTHREADCTXHOOK)
1206 {
1207 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
1208 RTThreadCtxHookDisable(pVCpu->vmm.s.hCtxHook);
1209 }
1210 }
1211 /*
1212 * The system is about to go into suspend mode; go back to ring 3.
1213 */
1214 else
1215 {
1216 rc = VINF_EM_RAW_INTERRUPT;
1217 pVCpu->iHostCpuSet = UINT32_MAX;
1218 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
1219 }
1220
1221 /** @todo When HM stops messing with the context hook state, we'll disable
1222 * preemption again before the RTThreadCtxHookDisable call. */
1223 if (!fPreemptRestored)
1224 RTThreadPreemptRestore(&PreemptState);
1225
1226 pVCpu->vmm.s.iLastGZRc = rc;
1227
1228 /* Fire dtrace probe and collect statistics. */
1229 VBOXVMM_R0_VMM_RETURN_TO_RING3_HM(pVCpu, CPUMQueryGuestCtxPtr(pVCpu), rc);
1230#ifdef VBOX_WITH_STATISTICS
1231 vmmR0RecordRC(pVM, pVCpu, rc);
1232#endif
1233 }
1234 /*
1235 * Invalid CPU set index or TSC delta in need of measuring.
1236 */
1237 else
1238 {
1239 pVCpu->iHostCpuSet = UINT32_MAX;
1240 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
1241 RTThreadPreemptRestore(&PreemptState);
1242 if (iHostCpuSet < RTCPUSET_MAX_CPUS)
1243 {
1244 int rc = SUPR0TscDeltaMeasureBySetIndex(pVM->pSession, iHostCpuSet, 0 /*fFlags*/,
1245 2 /*cMsWaitRetry*/, 5*RT_MS_1SEC /*cMsWaitThread*/,
1246 0 /*default cTries*/);
1247 if (RT_SUCCESS(rc) || rc == VERR_CPU_OFFLINE)
1248 pVCpu->vmm.s.iLastGZRc = VINF_EM_RAW_TO_R3;
1249 else
1250 pVCpu->vmm.s.iLastGZRc = rc;
1251 }
1252 else
1253 pVCpu->vmm.s.iLastGZRc = VERR_INVALID_CPU_INDEX;
1254 }
1255 break;
1256 }
1257
1258 /*
1259 * For profiling.
1260 */
1261 case VMMR0_DO_NOP:
1262 pVCpu->vmm.s.iLastGZRc = VINF_SUCCESS;
1263 break;
1264
1265 /*
1266 * Impossible.
1267 */
1268 default:
1269 AssertMsgFailed(("%#x\n", enmOperation));
1270 pVCpu->vmm.s.iLastGZRc = VERR_NOT_SUPPORTED;
1271 break;
1272 }
1273 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1274}
1275
1276
1277/**
1278 * Validates a session or VM session argument.
1279 *
1280 * @returns true / false accordingly.
1281 * @param pVM The cross context VM structure.
1282 * @param pClaimedSession The session claim to validate.
1283 * @param pSession The session argument.
1284 */
1285DECLINLINE(bool) vmmR0IsValidSession(PVM pVM, PSUPDRVSESSION pClaimedSession, PSUPDRVSESSION pSession)
1286{
1287 /* This must be set! */
1288 if (!pSession)
1289 return false;
1290
1291 /* Only one out of the two. */
1292 if (pVM && pClaimedSession)
1293 return false;
1294 if (pVM)
1295 pClaimedSession = pVM->pSession;
1296 return pClaimedSession == pSession;
1297}
1298
1299
1300/**
1301 * VMMR0EntryEx worker function, either called directly or when ever possible
1302 * called thru a longjmp so we can exit safely on failure.
1303 *
1304 * @returns VBox status code.
1305 * @param pVM The cross context VM structure.
1306 * @param idCpu Virtual CPU ID argument. Must be NIL_VMCPUID if pVM
1307 * is NIL_RTR0PTR, and may be NIL_VMCPUID if it isn't
1308 * @param enmOperation Which operation to execute.
1309 * @param pReqHdr This points to a SUPVMMR0REQHDR packet. Optional.
1310 * The support driver validates this if it's present.
1311 * @param u64Arg Some simple constant argument.
1312 * @param pSession The session of the caller.
1313 * @remarks Assume called with interrupts _enabled_.
1314 */
1315static int vmmR0EntryExWorker(PVM pVM, VMCPUID idCpu, VMMR0OPERATION enmOperation, PSUPVMMR0REQHDR pReqHdr, uint64_t u64Arg, PSUPDRVSESSION pSession)
1316{
1317 /*
1318 * Common VM pointer validation.
1319 */
1320 if (pVM)
1321 {
1322 if (RT_UNLIKELY( !VALID_PTR(pVM)
1323 || ((uintptr_t)pVM & PAGE_OFFSET_MASK)))
1324 {
1325 SUPR0Printf("vmmR0EntryExWorker: Invalid pVM=%p! (op=%d)\n", pVM, enmOperation);
1326 return VERR_INVALID_POINTER;
1327 }
1328 if (RT_UNLIKELY( pVM->enmVMState < VMSTATE_CREATING
1329 || pVM->enmVMState > VMSTATE_TERMINATED
1330 || pVM->pVMR0 != pVM))
1331 {
1332 SUPR0Printf("vmmR0EntryExWorker: Invalid pVM=%p:{enmVMState=%d, .pVMR0=%p}! (op=%d)\n",
1333 pVM, pVM->enmVMState, pVM->pVMR0, enmOperation);
1334 return VERR_INVALID_POINTER;
1335 }
1336
1337 if (RT_UNLIKELY(idCpu >= pVM->cCpus && idCpu != NIL_VMCPUID))
1338 {
1339 SUPR0Printf("vmmR0EntryExWorker: Invalid idCpu (%u vs cCpus=%u)\n", idCpu, pVM->cCpus);
1340 return VERR_INVALID_PARAMETER;
1341 }
1342 }
1343 else if (RT_UNLIKELY(idCpu != NIL_VMCPUID))
1344 {
1345 SUPR0Printf("vmmR0EntryExWorker: Invalid idCpu=%u\n", idCpu);
1346 return VERR_INVALID_PARAMETER;
1347 }
1348 VMM_CHECK_SMAP_SETUP();
1349 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
1350 int rc;
1351
1352 switch (enmOperation)
1353 {
1354 /*
1355 * GVM requests
1356 */
1357 case VMMR0_DO_GVMM_CREATE_VM:
1358 if (pVM || u64Arg || idCpu != NIL_VMCPUID)
1359 return VERR_INVALID_PARAMETER;
1360 rc = GVMMR0CreateVMReq((PGVMMCREATEVMREQ)pReqHdr);
1361 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
1362 break;
1363
1364 case VMMR0_DO_GVMM_DESTROY_VM:
1365 if (pReqHdr || u64Arg)
1366 return VERR_INVALID_PARAMETER;
1367 rc = GVMMR0DestroyVM(pVM);
1368 VMM_CHECK_SMAP_CHECK(RT_NOTHING);
1369 break;
1370
1371 case VMMR0_DO_GVMM_REGISTER_VMCPU:
1372 {
1373 if (!pVM)
1374 return VERR_INVALID_PARAMETER;
1375 rc = GVMMR0RegisterVCpu(pVM, idCpu);
1376 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1377 break;
1378 }
1379
1380 case VMMR0_DO_GVMM_SCHED_HALT:
1381 if (pReqHdr)
1382 return VERR_INVALID_PARAMETER;
1383 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1384 rc = GVMMR0SchedHalt(pVM, idCpu, u64Arg);
1385 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1386 break;
1387
1388 case VMMR0_DO_GVMM_SCHED_WAKE_UP:
1389 if (pReqHdr || u64Arg)
1390 return VERR_INVALID_PARAMETER;
1391 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1392 rc = GVMMR0SchedWakeUp(pVM, idCpu);
1393 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1394 break;
1395
1396 case VMMR0_DO_GVMM_SCHED_POKE:
1397 if (pReqHdr || u64Arg)
1398 return VERR_INVALID_PARAMETER;
1399 rc = GVMMR0SchedPoke(pVM, idCpu);
1400 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1401 break;
1402
1403 case VMMR0_DO_GVMM_SCHED_WAKE_UP_AND_POKE_CPUS:
1404 if (u64Arg)
1405 return VERR_INVALID_PARAMETER;
1406 rc = GVMMR0SchedWakeUpAndPokeCpusReq(pVM, (PGVMMSCHEDWAKEUPANDPOKECPUSREQ)pReqHdr);
1407 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1408 break;
1409
1410 case VMMR0_DO_GVMM_SCHED_POLL:
1411 if (pReqHdr || u64Arg > 1)
1412 return VERR_INVALID_PARAMETER;
1413 rc = GVMMR0SchedPoll(pVM, idCpu, !!u64Arg);
1414 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1415 break;
1416
1417 case VMMR0_DO_GVMM_QUERY_STATISTICS:
1418 if (u64Arg)
1419 return VERR_INVALID_PARAMETER;
1420 rc = GVMMR0QueryStatisticsReq(pVM, (PGVMMQUERYSTATISTICSSREQ)pReqHdr);
1421 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1422 break;
1423
1424 case VMMR0_DO_GVMM_RESET_STATISTICS:
1425 if (u64Arg)
1426 return VERR_INVALID_PARAMETER;
1427 rc = GVMMR0ResetStatisticsReq(pVM, (PGVMMRESETSTATISTICSSREQ)pReqHdr);
1428 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1429 break;
1430
1431 /*
1432 * Initialize the R0 part of a VM instance.
1433 */
1434 case VMMR0_DO_VMMR0_INIT:
1435 rc = vmmR0InitVM(pVM, RT_LODWORD(u64Arg), RT_HIDWORD(u64Arg));
1436 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1437 break;
1438
1439 /*
1440 * Terminate the R0 part of a VM instance.
1441 */
1442 case VMMR0_DO_VMMR0_TERM:
1443 rc = VMMR0TermVM(pVM, NULL);
1444 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1445 break;
1446
1447 /*
1448 * Attempt to enable hm mode and check the current setting.
1449 */
1450 case VMMR0_DO_HM_ENABLE:
1451 rc = HMR0EnableAllCpus(pVM);
1452 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1453 break;
1454
1455 /*
1456 * Setup the hardware accelerated session.
1457 */
1458 case VMMR0_DO_HM_SETUP_VM:
1459 rc = HMR0SetupVM(pVM);
1460 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1461 break;
1462
1463 /*
1464 * Switch to RC to execute Hypervisor function.
1465 */
1466 case VMMR0_DO_CALL_HYPERVISOR:
1467 {
1468#ifdef VBOX_WITH_RAW_MODE
1469 /*
1470 * Validate input / context.
1471 */
1472 if (RT_UNLIKELY(idCpu != 0))
1473 return VERR_INVALID_CPU_ID;
1474 if (RT_UNLIKELY(pVM->cCpus != 1))
1475 return VERR_INVALID_PARAMETER;
1476 PVMCPU pVCpu = &pVM->aCpus[idCpu];
1477# ifndef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1478 if (RT_UNLIKELY(!PGMGetHyperCR3(pVCpu)))
1479 return VERR_PGM_NO_CR3_SHADOW_ROOT;
1480# endif
1481
1482 /*
1483 * Disable interrupts.
1484 */
1485 RTCCUINTREG fFlags = ASMIntDisableFlags();
1486
1487 /*
1488 * Get the host CPU identifiers, make sure they are valid and that
1489 * we've got a TSC delta for the CPU.
1490 */
1491 RTCPUID idHostCpu;
1492 uint32_t iHostCpuSet = RTMpCurSetIndexAndId(&idHostCpu);
1493 if (RT_UNLIKELY(iHostCpuSet >= RTCPUSET_MAX_CPUS))
1494 {
1495 ASMSetFlags(fFlags);
1496 return VERR_INVALID_CPU_INDEX;
1497 }
1498 if (RT_UNLIKELY(!SUPIsTscDeltaAvailableForCpuSetIndex(iHostCpuSet)))
1499 {
1500 ASMSetFlags(fFlags);
1501 rc = SUPR0TscDeltaMeasureBySetIndex(pVM->pSession, iHostCpuSet, 0 /*fFlags*/,
1502 2 /*cMsWaitRetry*/, 5*RT_MS_1SEC /*cMsWaitThread*/,
1503 0 /*default cTries*/);
1504 if (RT_FAILURE(rc) && rc != VERR_CPU_OFFLINE)
1505 {
1506 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1507 return rc;
1508 }
1509 }
1510
1511 /*
1512 * Commit the CPU identifiers.
1513 */
1514# ifdef VBOX_WITH_VMMR0_DISABLE_LAPIC_NMI
1515 CPUMR0SetLApic(pVCpu, iHostCpuSet);
1516# endif
1517 pVCpu->iHostCpuSet = iHostCpuSet;
1518 ASMAtomicWriteU32(&pVCpu->idHostCpu, idHostCpu);
1519
1520 /*
1521 * We might need to disable VT-x if the active switcher turns off paging.
1522 */
1523 bool fVTxDisabled;
1524 rc = HMR0EnterSwitcher(pVM, pVM->vmm.s.enmSwitcher, &fVTxDisabled);
1525 if (RT_SUCCESS(rc))
1526 {
1527 /*
1528 * Go through the wormhole...
1529 */
1530 rc = pVM->vmm.s.pfnR0ToRawMode(pVM);
1531
1532 /*
1533 * Re-enable VT-x before we dispatch any pending host interrupts.
1534 */
1535 HMR0LeaveSwitcher(pVM, fVTxDisabled);
1536
1537 if ( rc == VINF_EM_RAW_INTERRUPT
1538 || rc == VINF_EM_RAW_INTERRUPT_HYPER)
1539 TRPMR0DispatchHostInterrupt(pVM);
1540 }
1541
1542 /*
1543 * Invalidate the host CPU identifiers as we restore interrupts.
1544 */
1545 pVCpu->iHostCpuSet = UINT32_MAX;
1546 ASMAtomicWriteU32(&pVCpu->idHostCpu, NIL_RTCPUID);
1547 ASMSetFlags(fFlags);
1548
1549#else /* !VBOX_WITH_RAW_MODE */
1550 rc = VERR_RAW_MODE_NOT_SUPPORTED;
1551#endif
1552 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1553 break;
1554 }
1555
1556 /*
1557 * PGM wrappers.
1558 */
1559 case VMMR0_DO_PGM_ALLOCATE_HANDY_PAGES:
1560 if (idCpu == NIL_VMCPUID)
1561 return VERR_INVALID_CPU_ID;
1562 rc = PGMR0PhysAllocateHandyPages(pVM, &pVM->aCpus[idCpu]);
1563 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1564 break;
1565
1566 case VMMR0_DO_PGM_FLUSH_HANDY_PAGES:
1567 if (idCpu == NIL_VMCPUID)
1568 return VERR_INVALID_CPU_ID;
1569 rc = PGMR0PhysFlushHandyPages(pVM, &pVM->aCpus[idCpu]);
1570 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1571 break;
1572
1573 case VMMR0_DO_PGM_ALLOCATE_LARGE_HANDY_PAGE:
1574 if (idCpu == NIL_VMCPUID)
1575 return VERR_INVALID_CPU_ID;
1576 rc = PGMR0PhysAllocateLargeHandyPage(pVM, &pVM->aCpus[idCpu]);
1577 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1578 break;
1579
1580 case VMMR0_DO_PGM_PHYS_SETUP_IOMMU:
1581 if (idCpu != 0)
1582 return VERR_INVALID_CPU_ID;
1583 rc = PGMR0PhysSetupIommu(pVM);
1584 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1585 break;
1586
1587 /*
1588 * GMM wrappers.
1589 */
1590 case VMMR0_DO_GMM_INITIAL_RESERVATION:
1591 if (u64Arg)
1592 return VERR_INVALID_PARAMETER;
1593 rc = GMMR0InitialReservationReq(pVM, idCpu, (PGMMINITIALRESERVATIONREQ)pReqHdr);
1594 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1595 break;
1596
1597 case VMMR0_DO_GMM_UPDATE_RESERVATION:
1598 if (u64Arg)
1599 return VERR_INVALID_PARAMETER;
1600 rc = GMMR0UpdateReservationReq(pVM, idCpu, (PGMMUPDATERESERVATIONREQ)pReqHdr);
1601 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1602 break;
1603
1604 case VMMR0_DO_GMM_ALLOCATE_PAGES:
1605 if (u64Arg)
1606 return VERR_INVALID_PARAMETER;
1607 rc = GMMR0AllocatePagesReq(pVM, idCpu, (PGMMALLOCATEPAGESREQ)pReqHdr);
1608 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1609 break;
1610
1611 case VMMR0_DO_GMM_FREE_PAGES:
1612 if (u64Arg)
1613 return VERR_INVALID_PARAMETER;
1614 rc = GMMR0FreePagesReq(pVM, idCpu, (PGMMFREEPAGESREQ)pReqHdr);
1615 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1616 break;
1617
1618 case VMMR0_DO_GMM_FREE_LARGE_PAGE:
1619 if (u64Arg)
1620 return VERR_INVALID_PARAMETER;
1621 rc = GMMR0FreeLargePageReq(pVM, idCpu, (PGMMFREELARGEPAGEREQ)pReqHdr);
1622 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1623 break;
1624
1625 case VMMR0_DO_GMM_QUERY_HYPERVISOR_MEM_STATS:
1626 if (u64Arg)
1627 return VERR_INVALID_PARAMETER;
1628 rc = GMMR0QueryHypervisorMemoryStatsReq(pVM, (PGMMMEMSTATSREQ)pReqHdr);
1629 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1630 break;
1631
1632 case VMMR0_DO_GMM_QUERY_MEM_STATS:
1633 if (idCpu == NIL_VMCPUID)
1634 return VERR_INVALID_CPU_ID;
1635 if (u64Arg)
1636 return VERR_INVALID_PARAMETER;
1637 rc = GMMR0QueryMemoryStatsReq(pVM, idCpu, (PGMMMEMSTATSREQ)pReqHdr);
1638 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1639 break;
1640
1641 case VMMR0_DO_GMM_BALLOONED_PAGES:
1642 if (u64Arg)
1643 return VERR_INVALID_PARAMETER;
1644 rc = GMMR0BalloonedPagesReq(pVM, idCpu, (PGMMBALLOONEDPAGESREQ)pReqHdr);
1645 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1646 break;
1647
1648 case VMMR0_DO_GMM_MAP_UNMAP_CHUNK:
1649 if (u64Arg)
1650 return VERR_INVALID_PARAMETER;
1651 rc = GMMR0MapUnmapChunkReq(pVM, (PGMMMAPUNMAPCHUNKREQ)pReqHdr);
1652 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1653 break;
1654
1655 case VMMR0_DO_GMM_SEED_CHUNK:
1656 if (pReqHdr)
1657 return VERR_INVALID_PARAMETER;
1658 rc = GMMR0SeedChunk(pVM, idCpu, (RTR3PTR)u64Arg);
1659 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1660 break;
1661
1662 case VMMR0_DO_GMM_REGISTER_SHARED_MODULE:
1663 if (idCpu == NIL_VMCPUID)
1664 return VERR_INVALID_CPU_ID;
1665 if (u64Arg)
1666 return VERR_INVALID_PARAMETER;
1667 rc = GMMR0RegisterSharedModuleReq(pVM, idCpu, (PGMMREGISTERSHAREDMODULEREQ)pReqHdr);
1668 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1669 break;
1670
1671 case VMMR0_DO_GMM_UNREGISTER_SHARED_MODULE:
1672 if (idCpu == NIL_VMCPUID)
1673 return VERR_INVALID_CPU_ID;
1674 if (u64Arg)
1675 return VERR_INVALID_PARAMETER;
1676 rc = GMMR0UnregisterSharedModuleReq(pVM, idCpu, (PGMMUNREGISTERSHAREDMODULEREQ)pReqHdr);
1677 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1678 break;
1679
1680 case VMMR0_DO_GMM_RESET_SHARED_MODULES:
1681 if (idCpu == NIL_VMCPUID)
1682 return VERR_INVALID_CPU_ID;
1683 if ( u64Arg
1684 || pReqHdr)
1685 return VERR_INVALID_PARAMETER;
1686 rc = GMMR0ResetSharedModules(pVM, idCpu);
1687 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1688 break;
1689
1690#ifdef VBOX_WITH_PAGE_SHARING
1691 case VMMR0_DO_GMM_CHECK_SHARED_MODULES:
1692 {
1693 if (idCpu == NIL_VMCPUID)
1694 return VERR_INVALID_CPU_ID;
1695 if ( u64Arg
1696 || pReqHdr)
1697 return VERR_INVALID_PARAMETER;
1698
1699 PVMCPU pVCpu = &pVM->aCpus[idCpu];
1700 Assert(pVCpu->hNativeThreadR0 == RTThreadNativeSelf());
1701
1702# ifdef DEBUG_sandervl
1703 /* Make sure that log flushes can jump back to ring-3; annoying to get an incomplete log (this is risky though as the code doesn't take this into account). */
1704 /* Todo: this can have bad side effects for unexpected jumps back to r3. */
1705 rc = GMMR0CheckSharedModulesStart(pVM);
1706 if (rc == VINF_SUCCESS)
1707 {
1708 rc = vmmR0CallRing3SetJmp(&pVCpu->vmm.s.CallRing3JmpBufR0, GMMR0CheckSharedModules, pVM, pVCpu); /* this may resume code. */
1709 Assert( rc == VINF_SUCCESS
1710 || (rc == VINF_VMM_CALL_HOST && pVCpu->vmm.s.enmCallRing3Operation == VMMCALLRING3_VMM_LOGGER_FLUSH));
1711 GMMR0CheckSharedModulesEnd(pVM);
1712 }
1713# else
1714 rc = GMMR0CheckSharedModules(pVM, pVCpu);
1715# endif
1716 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1717 break;
1718 }
1719#endif
1720
1721#if defined(VBOX_STRICT) && HC_ARCH_BITS == 64
1722 case VMMR0_DO_GMM_FIND_DUPLICATE_PAGE:
1723 if (u64Arg)
1724 return VERR_INVALID_PARAMETER;
1725 rc = GMMR0FindDuplicatePageReq(pVM, (PGMMFINDDUPLICATEPAGEREQ)pReqHdr);
1726 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1727 break;
1728#endif
1729
1730 case VMMR0_DO_GMM_QUERY_STATISTICS:
1731 if (u64Arg)
1732 return VERR_INVALID_PARAMETER;
1733 rc = GMMR0QueryStatisticsReq(pVM, (PGMMQUERYSTATISTICSSREQ)pReqHdr);
1734 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1735 break;
1736
1737 case VMMR0_DO_GMM_RESET_STATISTICS:
1738 if (u64Arg)
1739 return VERR_INVALID_PARAMETER;
1740 rc = GMMR0ResetStatisticsReq(pVM, (PGMMRESETSTATISTICSSREQ)pReqHdr);
1741 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1742 break;
1743
1744 /*
1745 * A quick GCFGM mock-up.
1746 */
1747 /** @todo GCFGM with proper access control, ring-3 management interface and all that. */
1748 case VMMR0_DO_GCFGM_SET_VALUE:
1749 case VMMR0_DO_GCFGM_QUERY_VALUE:
1750 {
1751 if (pVM || !pReqHdr || u64Arg || idCpu != NIL_VMCPUID)
1752 return VERR_INVALID_PARAMETER;
1753 PGCFGMVALUEREQ pReq = (PGCFGMVALUEREQ)pReqHdr;
1754 if (pReq->Hdr.cbReq != sizeof(*pReq))
1755 return VERR_INVALID_PARAMETER;
1756 if (enmOperation == VMMR0_DO_GCFGM_SET_VALUE)
1757 {
1758 rc = GVMMR0SetConfig(pReq->pSession, &pReq->szName[0], pReq->u64Value);
1759 //if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1760 // rc = GMMR0SetConfig(pReq->pSession, &pReq->szName[0], pReq->u64Value);
1761 }
1762 else
1763 {
1764 rc = GVMMR0QueryConfig(pReq->pSession, &pReq->szName[0], &pReq->u64Value);
1765 //if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1766 // rc = GMMR0QueryConfig(pReq->pSession, &pReq->szName[0], &pReq->u64Value);
1767 }
1768 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1769 break;
1770 }
1771
1772 /*
1773 * PDM Wrappers.
1774 */
1775 case VMMR0_DO_PDM_DRIVER_CALL_REQ_HANDLER:
1776 {
1777 if (!pVM || !pReqHdr || u64Arg || idCpu != NIL_VMCPUID)
1778 return VERR_INVALID_PARAMETER;
1779 rc = PDMR0DriverCallReqHandler(pVM, (PPDMDRIVERCALLREQHANDLERREQ)pReqHdr);
1780 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1781 break;
1782 }
1783
1784 case VMMR0_DO_PDM_DEVICE_CALL_REQ_HANDLER:
1785 {
1786 if (!pVM || !pReqHdr || u64Arg || idCpu != NIL_VMCPUID)
1787 return VERR_INVALID_PARAMETER;
1788 rc = PDMR0DeviceCallReqHandler(pVM, (PPDMDEVICECALLREQHANDLERREQ)pReqHdr);
1789 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1790 break;
1791 }
1792
1793 /*
1794 * Requests to the internal networking service.
1795 */
1796 case VMMR0_DO_INTNET_OPEN:
1797 {
1798 PINTNETOPENREQ pReq = (PINTNETOPENREQ)pReqHdr;
1799 if (u64Arg || !pReq || !vmmR0IsValidSession(pVM, pReq->pSession, pSession) || idCpu != NIL_VMCPUID)
1800 return VERR_INVALID_PARAMETER;
1801 rc = IntNetR0OpenReq(pSession, pReq);
1802 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1803 break;
1804 }
1805
1806 case VMMR0_DO_INTNET_IF_CLOSE:
1807 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFCLOSEREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1808 return VERR_INVALID_PARAMETER;
1809 rc = IntNetR0IfCloseReq(pSession, (PINTNETIFCLOSEREQ)pReqHdr);
1810 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1811 break;
1812
1813
1814 case VMMR0_DO_INTNET_IF_GET_BUFFER_PTRS:
1815 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFGETBUFFERPTRSREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1816 return VERR_INVALID_PARAMETER;
1817 rc = IntNetR0IfGetBufferPtrsReq(pSession, (PINTNETIFGETBUFFERPTRSREQ)pReqHdr);
1818 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1819 break;
1820
1821 case VMMR0_DO_INTNET_IF_SET_PROMISCUOUS_MODE:
1822 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSETPROMISCUOUSMODEREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1823 return VERR_INVALID_PARAMETER;
1824 rc = IntNetR0IfSetPromiscuousModeReq(pSession, (PINTNETIFSETPROMISCUOUSMODEREQ)pReqHdr);
1825 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1826 break;
1827
1828 case VMMR0_DO_INTNET_IF_SET_MAC_ADDRESS:
1829 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSETMACADDRESSREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1830 return VERR_INVALID_PARAMETER;
1831 rc = IntNetR0IfSetMacAddressReq(pSession, (PINTNETIFSETMACADDRESSREQ)pReqHdr);
1832 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1833 break;
1834
1835 case VMMR0_DO_INTNET_IF_SET_ACTIVE:
1836 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSETACTIVEREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1837 return VERR_INVALID_PARAMETER;
1838 rc = IntNetR0IfSetActiveReq(pSession, (PINTNETIFSETACTIVEREQ)pReqHdr);
1839 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1840 break;
1841
1842 case VMMR0_DO_INTNET_IF_SEND:
1843 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFSENDREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1844 return VERR_INVALID_PARAMETER;
1845 rc = IntNetR0IfSendReq(pSession, (PINTNETIFSENDREQ)pReqHdr);
1846 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1847 break;
1848
1849 case VMMR0_DO_INTNET_IF_WAIT:
1850 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFWAITREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1851 return VERR_INVALID_PARAMETER;
1852 rc = IntNetR0IfWaitReq(pSession, (PINTNETIFWAITREQ)pReqHdr);
1853 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1854 break;
1855
1856 case VMMR0_DO_INTNET_IF_ABORT_WAIT:
1857 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PINTNETIFWAITREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1858 return VERR_INVALID_PARAMETER;
1859 rc = IntNetR0IfAbortWaitReq(pSession, (PINTNETIFABORTWAITREQ)pReqHdr);
1860 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1861 break;
1862
1863#ifdef VBOX_WITH_PCI_PASSTHROUGH
1864 /*
1865 * Requests to host PCI driver service.
1866 */
1867 case VMMR0_DO_PCIRAW_REQ:
1868 if (u64Arg || !pReqHdr || !vmmR0IsValidSession(pVM, ((PPCIRAWSENDREQ)pReqHdr)->pSession, pSession) || idCpu != NIL_VMCPUID)
1869 return VERR_INVALID_PARAMETER;
1870 rc = PciRawR0ProcessReq(pSession, pVM, (PPCIRAWSENDREQ)pReqHdr);
1871 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1872 break;
1873#endif
1874 /*
1875 * For profiling.
1876 */
1877 case VMMR0_DO_NOP:
1878 case VMMR0_DO_SLOW_NOP:
1879 return VINF_SUCCESS;
1880
1881 /*
1882 * For testing Ring-0 APIs invoked in this environment.
1883 */
1884 case VMMR0_DO_TESTS:
1885 /** @todo make new test */
1886 return VINF_SUCCESS;
1887
1888
1889#if HC_ARCH_BITS == 32 && defined(VBOX_WITH_64_BITS_GUESTS)
1890 case VMMR0_DO_TEST_SWITCHER3264:
1891 if (idCpu == NIL_VMCPUID)
1892 return VERR_INVALID_CPU_ID;
1893 rc = HMR0TestSwitcher3264(pVM);
1894 VMM_CHECK_SMAP_CHECK2(pVM, RT_NOTHING);
1895 break;
1896#endif
1897 default:
1898 /*
1899 * We're returning VERR_NOT_SUPPORT here so we've got something else
1900 * than -1 which the interrupt gate glue code might return.
1901 */
1902 Log(("operation %#x is not supported\n", enmOperation));
1903 return VERR_NOT_SUPPORTED;
1904 }
1905 return rc;
1906}
1907
1908
1909/**
1910 * Argument for vmmR0EntryExWrapper containing the arguments for VMMR0EntryEx.
1911 */
1912typedef struct VMMR0ENTRYEXARGS
1913{
1914 PVM pVM;
1915 VMCPUID idCpu;
1916 VMMR0OPERATION enmOperation;
1917 PSUPVMMR0REQHDR pReq;
1918 uint64_t u64Arg;
1919 PSUPDRVSESSION pSession;
1920} VMMR0ENTRYEXARGS;
1921/** Pointer to a vmmR0EntryExWrapper argument package. */
1922typedef VMMR0ENTRYEXARGS *PVMMR0ENTRYEXARGS;
1923
1924/**
1925 * This is just a longjmp wrapper function for VMMR0EntryEx calls.
1926 *
1927 * @returns VBox status code.
1928 * @param pvArgs The argument package
1929 */
1930static DECLCALLBACK(int) vmmR0EntryExWrapper(void *pvArgs)
1931{
1932 return vmmR0EntryExWorker(((PVMMR0ENTRYEXARGS)pvArgs)->pVM,
1933 ((PVMMR0ENTRYEXARGS)pvArgs)->idCpu,
1934 ((PVMMR0ENTRYEXARGS)pvArgs)->enmOperation,
1935 ((PVMMR0ENTRYEXARGS)pvArgs)->pReq,
1936 ((PVMMR0ENTRYEXARGS)pvArgs)->u64Arg,
1937 ((PVMMR0ENTRYEXARGS)pvArgs)->pSession);
1938}
1939
1940
1941/**
1942 * The Ring 0 entry point, called by the support library (SUP).
1943 *
1944 * @returns VBox status code.
1945 * @param pVM The cross context VM structure.
1946 * @param idCpu Virtual CPU ID argument. Must be NIL_VMCPUID if pVM
1947 * is NIL_RTR0PTR, and may be NIL_VMCPUID if it isn't
1948 * @param enmOperation Which operation to execute.
1949 * @param pReq Pointer to the SUPVMMR0REQHDR packet. Optional.
1950 * @param u64Arg Some simple constant argument.
1951 * @param pSession The session of the caller.
1952 * @remarks Assume called with interrupts _enabled_.
1953 */
1954VMMR0DECL(int) VMMR0EntryEx(PVM pVM, VMCPUID idCpu, VMMR0OPERATION enmOperation, PSUPVMMR0REQHDR pReq, uint64_t u64Arg, PSUPDRVSESSION pSession)
1955{
1956 /*
1957 * Requests that should only happen on the EMT thread will be
1958 * wrapped in a setjmp so we can assert without causing trouble.
1959 */
1960 if ( VALID_PTR(pVM)
1961 && pVM->pVMR0
1962 && idCpu < pVM->cCpus)
1963 {
1964 switch (enmOperation)
1965 {
1966 /* These might/will be called before VMMR3Init. */
1967 case VMMR0_DO_GMM_INITIAL_RESERVATION:
1968 case VMMR0_DO_GMM_UPDATE_RESERVATION:
1969 case VMMR0_DO_GMM_ALLOCATE_PAGES:
1970 case VMMR0_DO_GMM_FREE_PAGES:
1971 case VMMR0_DO_GMM_BALLOONED_PAGES:
1972 /* On the mac we might not have a valid jmp buf, so check these as well. */
1973 case VMMR0_DO_VMMR0_INIT:
1974 case VMMR0_DO_VMMR0_TERM:
1975 {
1976 PVMCPU pVCpu = &pVM->aCpus[idCpu];
1977
1978 if (!pVCpu->vmm.s.CallRing3JmpBufR0.pvSavedStack)
1979 break;
1980
1981 /** @todo validate this EMT claim... GVM knows. */
1982 VMMR0ENTRYEXARGS Args;
1983 Args.pVM = pVM;
1984 Args.idCpu = idCpu;
1985 Args.enmOperation = enmOperation;
1986 Args.pReq = pReq;
1987 Args.u64Arg = u64Arg;
1988 Args.pSession = pSession;
1989 return vmmR0CallRing3SetJmpEx(&pVCpu->vmm.s.CallRing3JmpBufR0, vmmR0EntryExWrapper, &Args);
1990 }
1991
1992 default:
1993 break;
1994 }
1995 }
1996 return vmmR0EntryExWorker(pVM, idCpu, enmOperation, pReq, u64Arg, pSession);
1997}
1998
1999
2000/**
2001 * Checks whether we've armed the ring-0 long jump machinery.
2002 *
2003 * @returns @c true / @c false
2004 * @param pVCpu The cross context virtual CPU structure.
2005 * @thread EMT
2006 * @sa VMMIsLongJumpArmed
2007 */
2008VMMR0_INT_DECL(bool) VMMR0IsLongJumpArmed(PVMCPU pVCpu)
2009{
2010#ifdef RT_ARCH_X86
2011 return pVCpu->vmm.s.CallRing3JmpBufR0.eip
2012 && !pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call;
2013#else
2014 return pVCpu->vmm.s.CallRing3JmpBufR0.rip
2015 && !pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call;
2016#endif
2017}
2018
2019
2020/**
2021 * Checks whether we've done a ring-3 long jump.
2022 *
2023 * @returns @c true / @c false
2024 * @param pVCpu The cross context virtual CPU structure.
2025 * @thread EMT
2026 */
2027VMMR0_INT_DECL(bool) VMMR0IsInRing3LongJump(PVMCPU pVCpu)
2028{
2029 return pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call;
2030}
2031
2032
2033/**
2034 * Internal R0 logger worker: Flush logger.
2035 *
2036 * @param pLogger The logger instance to flush.
2037 * @remark This function must be exported!
2038 */
2039VMMR0DECL(void) vmmR0LoggerFlush(PRTLOGGER pLogger)
2040{
2041#ifdef LOG_ENABLED
2042 /*
2043 * Convert the pLogger into a VM handle and 'call' back to Ring-3.
2044 * (This is a bit paranoid code.)
2045 */
2046 PVMMR0LOGGER pR0Logger = (PVMMR0LOGGER)((uintptr_t)pLogger - RT_OFFSETOF(VMMR0LOGGER, Logger));
2047 if ( !VALID_PTR(pR0Logger)
2048 || !VALID_PTR(pR0Logger + 1)
2049 || pLogger->u32Magic != RTLOGGER_MAGIC)
2050 {
2051# ifdef DEBUG
2052 SUPR0Printf("vmmR0LoggerFlush: pLogger=%p!\n", pLogger);
2053# endif
2054 return;
2055 }
2056 if (pR0Logger->fFlushingDisabled)
2057 return; /* quietly */
2058
2059 PVM pVM = pR0Logger->pVM;
2060 if ( !VALID_PTR(pVM)
2061 || pVM->pVMR0 != pVM)
2062 {
2063# ifdef DEBUG
2064 SUPR0Printf("vmmR0LoggerFlush: pVM=%p! pVMR0=%p! pLogger=%p\n", pVM, pVM->pVMR0, pLogger);
2065# endif
2066 return;
2067 }
2068
2069 PVMCPU pVCpu = VMMGetCpu(pVM);
2070 if (pVCpu)
2071 {
2072 /*
2073 * Check that the jump buffer is armed.
2074 */
2075# ifdef RT_ARCH_X86
2076 if ( !pVCpu->vmm.s.CallRing3JmpBufR0.eip
2077 || pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call)
2078# else
2079 if ( !pVCpu->vmm.s.CallRing3JmpBufR0.rip
2080 || pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call)
2081# endif
2082 {
2083# ifdef DEBUG
2084 SUPR0Printf("vmmR0LoggerFlush: Jump buffer isn't armed!\n");
2085# endif
2086 return;
2087 }
2088 VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_VMM_LOGGER_FLUSH, 0);
2089 }
2090# ifdef DEBUG
2091 else
2092 SUPR0Printf("vmmR0LoggerFlush: invalid VCPU context!\n");
2093# endif
2094#else
2095 NOREF(pLogger);
2096#endif /* LOG_ENABLED */
2097}
2098
2099/**
2100 * Internal R0 logger worker: Custom prefix.
2101 *
2102 * @returns Number of chars written.
2103 *
2104 * @param pLogger The logger instance.
2105 * @param pchBuf The output buffer.
2106 * @param cchBuf The size of the buffer.
2107 * @param pvUser User argument (ignored).
2108 */
2109VMMR0DECL(size_t) vmmR0LoggerPrefix(PRTLOGGER pLogger, char *pchBuf, size_t cchBuf, void *pvUser)
2110{
2111 NOREF(pvUser);
2112#ifdef LOG_ENABLED
2113 PVMMR0LOGGER pR0Logger = (PVMMR0LOGGER)((uintptr_t)pLogger - RT_OFFSETOF(VMMR0LOGGER, Logger));
2114 if ( !VALID_PTR(pR0Logger)
2115 || !VALID_PTR(pR0Logger + 1)
2116 || pLogger->u32Magic != RTLOGGER_MAGIC
2117 || cchBuf < 2)
2118 return 0;
2119
2120 static const char s_szHex[17] = "0123456789abcdef";
2121 VMCPUID const idCpu = pR0Logger->idCpu;
2122 pchBuf[1] = s_szHex[ idCpu & 15];
2123 pchBuf[0] = s_szHex[(idCpu >> 4) & 15];
2124
2125 return 2;
2126#else
2127 NOREF(pLogger); NOREF(pchBuf); NOREF(cchBuf);
2128 return 0;
2129#endif
2130}
2131
2132#ifdef LOG_ENABLED
2133
2134/**
2135 * Disables flushing of the ring-0 debug log.
2136 *
2137 * @param pVCpu The cross context virtual CPU structure.
2138 */
2139VMMR0_INT_DECL(void) VMMR0LogFlushDisable(PVMCPU pVCpu)
2140{
2141 if (pVCpu->vmm.s.pR0LoggerR0)
2142 pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled = true;
2143}
2144
2145
2146/**
2147 * Enables flushing of the ring-0 debug log.
2148 *
2149 * @param pVCpu The cross context virtual CPU structure.
2150 */
2151VMMR0_INT_DECL(void) VMMR0LogFlushEnable(PVMCPU pVCpu)
2152{
2153 if (pVCpu->vmm.s.pR0LoggerR0)
2154 pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled = false;
2155}
2156
2157
2158/**
2159 * Checks if log flushing is disabled or not.
2160 *
2161 * @param pVCpu The cross context virtual CPU structure.
2162 */
2163VMMR0_INT_DECL(bool) VMMR0IsLogFlushDisabled(PVMCPU pVCpu)
2164{
2165 if (pVCpu->vmm.s.pR0LoggerR0)
2166 return pVCpu->vmm.s.pR0LoggerR0->fFlushingDisabled;
2167 return true;
2168}
2169#endif /* LOG_ENABLED */
2170
2171/**
2172 * Jump back to ring-3 if we're the EMT and the longjmp is armed.
2173 *
2174 * @returns true if the breakpoint should be hit, false if it should be ignored.
2175 */
2176DECLEXPORT(bool) RTCALL RTAssertShouldPanic(void)
2177{
2178#if 0
2179 return true;
2180#else
2181 PVM pVM = GVMMR0GetVMByEMT(NIL_RTNATIVETHREAD);
2182 if (pVM)
2183 {
2184 PVMCPU pVCpu = VMMGetCpu(pVM);
2185
2186 if (pVCpu)
2187 {
2188#ifdef RT_ARCH_X86
2189 if ( pVCpu->vmm.s.CallRing3JmpBufR0.eip
2190 && !pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call)
2191#else
2192 if ( pVCpu->vmm.s.CallRing3JmpBufR0.rip
2193 && !pVCpu->vmm.s.CallRing3JmpBufR0.fInRing3Call)
2194#endif
2195 {
2196 int rc = VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_VM_R0_ASSERTION, 0);
2197 return RT_FAILURE_NP(rc);
2198 }
2199 }
2200 }
2201#ifdef RT_OS_LINUX
2202 return true;
2203#else
2204 return false;
2205#endif
2206#endif
2207}
2208
2209
2210/**
2211 * Override this so we can push it up to ring-3.
2212 *
2213 * @param pszExpr Expression. Can be NULL.
2214 * @param uLine Location line number.
2215 * @param pszFile Location file name.
2216 * @param pszFunction Location function name.
2217 */
2218DECLEXPORT(void) RTCALL RTAssertMsg1Weak(const char *pszExpr, unsigned uLine, const char *pszFile, const char *pszFunction)
2219{
2220 /*
2221 * To the log.
2222 */
2223 LogAlways(("\n!!R0-Assertion Failed!!\n"
2224 "Expression: %s\n"
2225 "Location : %s(%d) %s\n",
2226 pszExpr, pszFile, uLine, pszFunction));
2227
2228 /*
2229 * To the global VMM buffer.
2230 */
2231 PVM pVM = GVMMR0GetVMByEMT(NIL_RTNATIVETHREAD);
2232 if (pVM)
2233 RTStrPrintf(pVM->vmm.s.szRing0AssertMsg1, sizeof(pVM->vmm.s.szRing0AssertMsg1),
2234 "\n!!R0-Assertion Failed!!\n"
2235 "Expression: %s\n"
2236 "Location : %s(%d) %s\n",
2237 pszExpr, pszFile, uLine, pszFunction);
2238
2239 /*
2240 * Continue the normal way.
2241 */
2242 RTAssertMsg1(pszExpr, uLine, pszFile, pszFunction);
2243}
2244
2245
2246/**
2247 * Callback for RTLogFormatV which writes to the ring-3 log port.
2248 * See PFNLOGOUTPUT() for details.
2249 */
2250static DECLCALLBACK(size_t) rtLogOutput(void *pv, const char *pachChars, size_t cbChars)
2251{
2252 for (size_t i = 0; i < cbChars; i++)
2253 {
2254 LogAlways(("%c", pachChars[i])); NOREF(pachChars);
2255 }
2256
2257 NOREF(pv);
2258 return cbChars;
2259}
2260
2261
2262/**
2263 * Override this so we can push it up to ring-3.
2264 *
2265 * @param pszFormat The format string.
2266 * @param va Arguments.
2267 */
2268DECLEXPORT(void) RTCALL RTAssertMsg2WeakV(const char *pszFormat, va_list va)
2269{
2270 va_list vaCopy;
2271
2272 /*
2273 * Push the message to the loggers.
2274 */
2275 PRTLOGGER pLog = RTLogGetDefaultInstance(); /* Don't initialize it here... */
2276 if (pLog)
2277 {
2278 va_copy(vaCopy, va);
2279 RTLogFormatV(rtLogOutput, pLog, pszFormat, vaCopy);
2280 va_end(vaCopy);
2281 }
2282 pLog = RTLogRelGetDefaultInstance();
2283 if (pLog)
2284 {
2285 va_copy(vaCopy, va);
2286 RTLogFormatV(rtLogOutput, pLog, pszFormat, vaCopy);
2287 va_end(vaCopy);
2288 }
2289
2290 /*
2291 * Push it to the global VMM buffer.
2292 */
2293 PVM pVM = GVMMR0GetVMByEMT(NIL_RTNATIVETHREAD);
2294 if (pVM)
2295 {
2296 va_copy(vaCopy, va);
2297 RTStrPrintfV(pVM->vmm.s.szRing0AssertMsg2, sizeof(pVM->vmm.s.szRing0AssertMsg2), pszFormat, vaCopy);
2298 va_end(vaCopy);
2299 }
2300
2301 /*
2302 * Continue the normal way.
2303 */
2304 RTAssertMsg2V(pszFormat, va);
2305}
2306
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