VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/Support/SUPLib.cpp@ 4226

Last change on this file since 4226 was 4191, checked in by vboxsync, 17 years ago

fake mode on amd64.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 44.8 KB
Line 
1/* $Id: SUPLib.cpp 4191 2007-08-16 22:49:13Z vboxsync $ */
2/** @file
3 * VirtualBox Support Library - Common code.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/** @page pg_sup SUP - The Support Library
19 *
20 * The support library is responsible for providing facilities to load
21 * VMM Host Ring-0 code, to call Host VMM Ring-0 code from Ring-3 Host
22 * code, to pin down physical memory, and more.
23 *
24 * The VMM Host Ring-0 code can be combined in the support driver if
25 * permitted by kernel module license policies. If it is not combined
26 * it will be externalized in a .r0 module that will be loaded using
27 * the IPRT loader.
28 *
29 * The Ring-0 calling is done thru a generic SUP interface which will
30 * tranfer an argument set and call a predefined entry point in the Host
31 * VMM Ring-0 code.
32 *
33 * See @ref grp_sup "SUP - Support APIs" for API details.
34 */
35
36
37/*******************************************************************************
38* Header Files *
39*******************************************************************************/
40#define LOG_GROUP LOG_GROUP_SUP
41#include <VBox/sup.h>
42#include <VBox/err.h>
43#include <VBox/param.h>
44#ifdef VBOX_WITHOUT_IDT_PATCHING
45# include <VBox/vmm.h>
46#endif
47#include <VBox/log.h>
48
49#include <iprt/assert.h>
50#include <iprt/alloc.h>
51#include <iprt/alloca.h>
52#include <iprt/ldr.h>
53#include <iprt/asm.h>
54#include <iprt/system.h>
55#include <iprt/thread.h>
56#include <iprt/process.h>
57#include <iprt/string.h>
58#include <iprt/env.h>
59
60#include "SUPLibInternal.h"
61#include "SUPDRVIOC.h"
62
63
64
65/*******************************************************************************
66* Defined Constants And Macros *
67*******************************************************************************/
68/** R0 VMM module name. */
69#define VMMR0_NAME "VMMR0"
70
71
72/*******************************************************************************
73* Structures and Typedefs *
74*******************************************************************************/
75typedef DECLCALLBACK(int) FNCALLVMMR0(PVMR0 pVMR0, unsigned uOperation, void *pvArg);
76typedef FNCALLVMMR0 *PFNCALLVMMR0;
77
78
79/*******************************************************************************
80* Global Variables *
81*******************************************************************************/
82/** Pointer to the Global Information Page.
83 *
84 * This pointer is valid as long as SUPLib has a open session. Anyone using
85 * the page must treat this pointer as higly volatile and not trust it beyond
86 * one transaction.
87 *
88 * @todo This will probably deserve it's own session or some other good solution...
89 */
90DECLEXPORT(PSUPGLOBALINFOPAGE) g_pSUPGlobalInfoPage;
91/** Address of the ring-0 mapping of the GIP. */
92static PSUPGLOBALINFOPAGE g_pSUPGlobalInfoPageR0;
93/** The physical address of the GIP. */
94static RTHCPHYS g_HCPhysSUPGlobalInfoPage = NIL_RTHCPHYS;
95
96/** The negotiated cookie. */
97uint32_t g_u32Cookie = 0;
98/** The negotiated session cookie. */
99uint32_t g_u32SessionCookie;
100/** Session handle. */
101PSUPDRVSESSION g_pSession;
102/** R0 SUP Functions used for resolving referenced to the SUPR0 module. */
103static PSUPQUERYFUNCS_OUT g_pFunctions;
104
105#ifndef VBOX_WITHOUT_IDT_PATCHING
106/** The negotiated interrupt number. */
107static uint8_t g_u8Interrupt = 3;
108/** Pointer to the generated code fore calling VMMR0. */
109static PFNCALLVMMR0 g_pfnCallVMMR0;
110#endif
111/** VMMR0 Load Address. */
112static RTR0PTR g_pvVMMR0 = NIL_RTR0PTR;
113/** Init counter. */
114static unsigned g_cInits = 0;
115/** Fake mode indicator. (~0 at first, 0 or 1 after first test) */
116static uint32_t g_u32FakeMode = ~0;
117
118
119/*******************************************************************************
120* Internal Functions *
121*******************************************************************************/
122static int supInitFake(PSUPDRVSESSION *ppSession);
123static int supLoadModule(const char *pszFilename, const char *pszModule, void **ppvImageBase);
124#ifndef VBOX_WITHOUT_IDT_PATCHING
125static int supInstallIDTE(void);
126#endif
127static DECLCALLBACK(int) supLoadModuleResolveImport(RTLDRMOD hLdrMod, const char *pszModule, const char *pszSymbol, unsigned uSymbol, RTUINTPTR *pValue, void *pvUser);
128
129
130SUPR3DECL(int) SUPInstall(void)
131{
132 return suplibOsInstall();
133}
134
135
136SUPR3DECL(int) SUPUninstall(void)
137{
138 return suplibOsUninstall();
139}
140
141
142SUPR3DECL(int) SUPInit(PSUPDRVSESSION *ppSession /* NULL */, size_t cbReserve /* 0 */)
143{
144 /*
145 * Perform some sanity checks.
146 * (Got some trouble with compile time member alignment assertions.)
147 */
148 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, u64NanoTSLastUpdateHz) & 0x7));
149 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs) & 0x1f));
150 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[1]) & 0x1f));
151 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[0].u64NanoTS) & 0x7));
152 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[0].u64TSC) & 0x7));
153 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[0].u64CpuHz) & 0x7));
154
155 /*
156 * Check if already initialized.
157 */
158 if (ppSession)
159 *ppSession = g_pSession;
160 if (g_cInits++ > 0)
161 return VINF_SUCCESS;
162
163 /*
164 * Check for fake mode.
165 *
166 * Fake mode is used when we're doing smoke testing and debugging.
167 * It's also useful on platforms where we haven't root access or which
168 * we haven't ported the support driver to.
169 */
170 if (g_u32FakeMode == ~0U)
171 {
172 const char *psz = RTEnvGet("VBOX_SUPLIB_FAKE");
173 if (psz && !strcmp(psz, "fake"))
174 ASMAtomicCmpXchgU32(&g_u32FakeMode, 1, ~0U);
175 else
176 ASMAtomicCmpXchgU32(&g_u32FakeMode, 0, ~0U);
177 }
178 if (RT_UNLIKELY(g_u32FakeMode))
179 return supInitFake(ppSession);
180
181 /**
182 * Open the support driver.
183 */
184 int rc = suplibOsInit(cbReserve);
185 if (VBOX_SUCCESS(rc))
186 {
187 /*
188 * Negotiate the cookie.
189 */
190 SUPCOOKIE_IN In;
191 SUPCOOKIE_OUT Out = {0,0,0,0,0,NIL_RTR0PTR};
192 strcpy(In.szMagic, SUPCOOKIE_MAGIC);
193 In.u32ReqVersion = SUPDRVIOC_VERSION;
194 In.u32MinVersion = SUPDRVIOC_VERSION & 0xffff0000;
195 rc = suplibOsIOCtl(SUP_IOCTL_COOKIE, &In, sizeof(In), &Out, sizeof(Out));
196 if (VBOX_SUCCESS(rc))
197 {
198 if ((Out.u32SessionVersion & 0xffff0000) == (SUPDRVIOC_VERSION & 0xffff0000))
199 {
200 /*
201 * Query the functions.
202 */
203 SUPQUERYFUNCS_IN FuncsIn;
204 FuncsIn.u32Cookie = Out.u32Cookie;
205 FuncsIn.u32SessionCookie = Out.u32SessionCookie;
206 unsigned cbFuncsOut = RT_OFFSETOF(SUPQUERYFUNCS_OUT, aFunctions[Out.cFunctions]);
207 PSUPQUERYFUNCS_OUT pFuncsOut = (PSUPQUERYFUNCS_OUT)RTMemAllocZ(cbFuncsOut);
208 if (pFuncsOut)
209 {
210 rc = suplibOsIOCtl(SUP_IOCTL_QUERY_FUNCS, &FuncsIn, sizeof(FuncsIn), pFuncsOut, cbFuncsOut);
211 if (VBOX_SUCCESS(rc))
212 {
213 g_u32Cookie = Out.u32Cookie;
214 g_u32SessionCookie = Out.u32SessionCookie;
215 g_pSession = Out.pSession;
216 g_pFunctions = pFuncsOut;
217 if (ppSession)
218 *ppSession = Out.pSession;
219
220 /*
221 * Map the GIP into userspace.
222 * This is an optional feature, so we will ignore any failures here.
223 */
224 if (!g_pSUPGlobalInfoPage)
225 {
226 SUPGIPMAP_IN GipIn = {0};
227 SUPGIPMAP_OUT GipOut = {NULL, 0};
228 GipIn.u32Cookie = Out.u32Cookie;
229 GipIn.u32SessionCookie = Out.u32SessionCookie;
230 rc = suplibOsIOCtl(SUP_IOCTL_GIP_MAP, &GipIn, sizeof(GipIn), &GipOut, sizeof(GipOut));
231 if (VBOX_SUCCESS(rc))
232 {
233 AssertRelease(GipOut.pGipR3->u32Magic == SUPGLOBALINFOPAGE_MAGIC);
234 AssertRelease(GipOut.pGipR3->u32Version >= SUPGLOBALINFOPAGE_VERSION);
235 ASMAtomicXchgSize(&g_HCPhysSUPGlobalInfoPage, GipOut.HCPhysGip);
236 ASMAtomicCmpXchgPtr((void * volatile *)&g_pSUPGlobalInfoPage, (void *)GipOut.pGipR3, NULL);
237 ASMAtomicCmpXchgPtr((void * volatile *)&g_pSUPGlobalInfoPageR0, (void *)GipOut.pGipR0, NULL);
238 }
239 else
240 rc = VINF_SUCCESS;
241 }
242 return rc;
243 }
244 RTMemFree(pFuncsOut);
245 }
246 else
247 rc = VERR_NO_MEMORY;
248 }
249 else
250 {
251 LogRel(("Support driver version mismatch: SessionVersion=%#x DriverVersion=%#x ClientVersion=%#x\n",
252 Out.u32SessionVersion, Out.u32DriverVersion, SUPDRVIOC_VERSION));
253 rc = VERR_VM_DRIVER_VERSION_MISMATCH;
254 }
255 }
256 else
257 {
258 if (rc == VERR_INVALID_PARAMETER) /* for pre 0x00040002 drivers */
259 rc = VERR_VM_DRIVER_VERSION_MISMATCH;
260 if (rc == VERR_VM_DRIVER_VERSION_MISMATCH)
261 LogRel(("Support driver version mismatch: DriverVersion=%#x ClientVersion=%#x\n",
262 Out.u32DriverVersion, SUPDRVIOC_VERSION));
263 else
264 LogRel(("Support driver version/Cookie negotiations error: rc=%Vrc\n", rc));
265 }
266
267 suplibOsTerm();
268 }
269 AssertMsgFailed(("SUPInit() failed rc=%Vrc\n", rc));
270 g_cInits--;
271
272 return rc;
273}
274
275/**
276 * Fake mode init.
277 */
278static int supInitFake(PSUPDRVSESSION *ppSession)
279{
280 Log(("SUP: Fake mode!\n"));
281 static const SUPFUNC s_aFakeFunctions[] =
282 {
283 /* name function */
284 { "SUPR0ObjRegister", 0xefef0000 },
285 { "SUPR0ObjAddRef", 0xefef0001 },
286 { "SUPR0ObjRelease", 0xefef0002 },
287 { "SUPR0ObjVerifyAccess", 0xefef0003 },
288 { "SUPR0LockMem", 0xefef0004 },
289 { "SUPR0UnlockMem", 0xefef0005 },
290 { "SUPR0ContAlloc", 0xefef0006 },
291 { "SUPR0ContFree", 0xefef0007 },
292 { "SUPR0MemAlloc", 0xefef0008 },
293 { "SUPR0MemGetPhys", 0xefef0009 },
294 { "SUPR0MemFree", 0xefef000a },
295 { "SUPR0Printf", 0xefef000b },
296 { "RTMemAlloc", 0xefef000c },
297 { "RTMemAllocZ", 0xefef000d },
298 { "RTMemFree", 0xefef000e },
299 { "RTSemFastMutexCreate", 0xefef000f },
300 { "RTSemFastMutexDestroy", 0xefef0010 },
301 { "RTSemFastMutexRequest", 0xefef0011 },
302 { "RTSemFastMutexRelease", 0xefef0012 },
303 { "RTSemEventCreate", 0xefef0013 },
304 { "RTSemEventSignal", 0xefef0014 },
305 { "RTSemEventWait", 0xefef0015 },
306 { "RTSemEventDestroy", 0xefef0016 },
307 { "RTSpinlockCreate", 0xefef0017 },
308 { "RTSpinlockDestroy", 0xefef0018 },
309 { "RTSpinlockAcquire", 0xefef0019 },
310 { "RTSpinlockRelease", 0xefef001a },
311 { "RTSpinlockAcquireNoInts", 0xefef001b },
312 { "RTSpinlockReleaseNoInts", 0xefef001c },
313 { "RTThreadNativeSelf", 0xefef001d },
314 { "RTThreadSleep", 0xefef001e },
315 { "RTThreadYield", 0xefef001f },
316 { "RTLogDefaultInstance", 0xefef0020 },
317 { "RTLogRelDefaultInstance", 0xefef0021 },
318 { "RTLogSetDefaultInstanceThread", 0xefef0022 },
319 { "RTLogLogger", 0xefef0023 },
320 { "RTLogLoggerEx", 0xefef0024 },
321 { "RTLogLoggerExV", 0xefef0025 },
322 { "AssertMsg1", 0xefef0026 },
323 { "AssertMsg2", 0xefef0027 },
324 };
325
326 /* fake r0 functions. */
327 g_pFunctions = (PSUPQUERYFUNCS_OUT)RTMemAllocZ(RT_OFFSETOF(SUPQUERYFUNCS_OUT, aFunctions[RT_ELEMENTS(s_aFakeFunctions)]));
328 if (g_pFunctions)
329 {
330 g_pFunctions->cFunctions = RT_ELEMENTS(s_aFakeFunctions);
331 memcpy(&g_pFunctions->aFunctions[0], &s_aFakeFunctions[0], sizeof(s_aFakeFunctions));
332 g_pSession = (PSUPDRVSESSION)(void *)g_pFunctions;
333 if (ppSession)
334 *ppSession = g_pSession;
335#ifndef VBOX_WITHOUT_IDT_PATCHING
336 Assert(g_u8Interrupt == 3);
337#endif
338
339 /* fake the GIP. */
340 g_pSUPGlobalInfoPage = (PSUPGLOBALINFOPAGE)RTMemPageAlloc(PAGE_SIZE);
341 if (g_pSUPGlobalInfoPage)
342 {
343 g_pSUPGlobalInfoPageR0 = g_pSUPGlobalInfoPage;
344 g_HCPhysSUPGlobalInfoPage = NIL_RTHCPHYS & ~(RTHCPHYS)PAGE_OFFSET_MASK;
345 /* the page is supposed to be invalid, so don't set the magic. */
346 return VINF_SUCCESS;
347 }
348
349 RTMemFree(g_pFunctions);
350 g_pFunctions = NULL;
351 }
352 return VERR_NO_MEMORY;
353}
354
355
356SUPR3DECL(int) SUPTerm(bool fForced)
357{
358 /*
359 * Verify state.
360 */
361 AssertMsg(g_cInits > 0, ("SUPTerm() is called before SUPInit()!\n"));
362 if (g_cInits == 0)
363 return VERR_WRONG_ORDER;
364 if (g_cInits == 1 || fForced)
365 {
366 /*
367 * NULL the GIP pointer.
368 */
369 if (g_pSUPGlobalInfoPage)
370 {
371 ASMAtomicXchgPtr((void * volatile *)&g_pSUPGlobalInfoPage, NULL);
372 ASMAtomicXchgPtr((void * volatile *)&g_pSUPGlobalInfoPageR0, NULL);
373 ASMAtomicXchgSize(&g_HCPhysSUPGlobalInfoPage, NIL_RTHCPHYS);
374 /* just a little safe guard against threads using the page. */
375 RTThreadSleep(50);
376 }
377
378 /*
379 * Close the support driver.
380 */
381 int rc = suplibOsTerm();
382 if (rc)
383 return rc;
384
385 g_u32Cookie = 0;
386 g_u32SessionCookie = 0;
387#ifndef VBOX_WITHOUT_IDT_PATCHING
388 g_u8Interrupt = 3;
389#endif
390 g_cInits = 0;
391 }
392 else
393 g_cInits--;
394
395 return 0;
396}
397
398
399SUPR3DECL(SUPPAGINGMODE) SUPGetPagingMode(void)
400{
401 /*
402 * Issue IOCtl to the SUPDRV kernel module.
403 */
404 SUPGETPAGINGMODE_IN In;
405 In.u32Cookie = g_u32Cookie;
406 In.u32SessionCookie = g_u32SessionCookie;
407 SUPGETPAGINGMODE_OUT Out = {SUPPAGINGMODE_INVALID};
408 int rc;
409 if (!g_u32FakeMode)
410 {
411 rc = suplibOsIOCtl(SUP_IOCTL_GET_PAGING_MODE, &In, sizeof(In), &Out, sizeof(Out));
412 if (VBOX_FAILURE(rc))
413 Out.enmMode = SUPPAGINGMODE_INVALID;
414 }
415 else
416#ifdef RT_ARCH_AMD64
417 Out.enmMode = SUPPAGINGMODE_AMD64_GLOBAL_NX;
418#else
419 Out.enmMode = SUPPAGINGMODE_32_BIT_GLOBAL;
420#endif
421
422 return Out.enmMode;
423}
424
425SUPR3DECL(int) SUPCallVMMR0Ex(PVMR0 pVMR0, unsigned uOperation, void *pvArg, unsigned cbArg)
426{
427 /*
428 * Issue IOCtl to the SUPDRV kernel module.
429 */
430 SUPCALLVMMR0_IN In;
431 In.u32Cookie = g_u32Cookie;
432 In.u32SessionCookie = g_u32SessionCookie;
433 In.pVMR0 = pVMR0;
434 In.uOperation = uOperation;
435 In.cbArg = cbArg;
436 In.pvArg = pvArg;
437 Assert(!g_u32FakeMode);
438 SUPCALLVMMR0_OUT Out = {VINF_SUCCESS};
439 int rc = suplibOsIOCtl(SUP_IOCTL_CALL_VMMR0, &In, sizeof(In), &Out, sizeof(Out));
440 if (VBOX_SUCCESS(rc))
441 rc = Out.rc;
442 return rc;
443}
444
445
446SUPR3DECL(int) SUPCallVMMR0(PVMR0 pVMR0, unsigned uOperation, void *pvArg)
447{
448#ifndef VBOX_WITHOUT_IDT_PATCHING
449 return g_pfnCallVMMR0(pVMR0, uOperation, pvArg);
450
451#else
452 if (RT_LIKELY(uOperation == VMMR0_DO_RAW_RUN))
453 {
454 Assert(!pvArg);
455 return suplibOSIOCtlFast(SUP_IOCTL_FAST_DO_RAW_RUN);
456 }
457 if (RT_LIKELY(uOperation == VMMR0_DO_HWACC_RUN))
458 {
459 Assert(!pvArg);
460 return suplibOSIOCtlFast(SUP_IOCTL_FAST_DO_HWACC_RUN);
461 }
462 if (uOperation == VMMR0_DO_NOP)
463 {
464 Assert(!pvArg);
465 return suplibOSIOCtlFast(SUP_IOCTL_FAST_DO_NOP);
466 }
467 return SUPCallVMMR0Ex(pVMR0, uOperation, pvArg, pvArg ? sizeof(pvArg) : 0);
468#endif
469}
470
471
472SUPR3DECL(int) SUPSetVMForFastIOCtl(PVMR0 pVMR0)
473{
474 SUPSETVMFORFAST_IN In;
475 In.u32Cookie = g_u32Cookie;
476 In.u32SessionCookie = g_u32SessionCookie;
477 In.pVMR0 = pVMR0;
478 int rc;
479 if (RT_LIKELY(!g_u32FakeMode))
480 rc = suplibOsIOCtl(SUP_IOCTL_SET_VM_FOR_FAST, &In, sizeof(In), NULL, 0);
481 else
482 rc = VINF_SUCCESS;
483 return rc;
484}
485
486
487SUPR3DECL(int) SUPPageLock(void *pvStart, size_t cPages, PSUPPAGE paPages)
488{
489 /*
490 * Validate.
491 */
492 AssertPtr(pvStart);
493 AssertMsg(RT_ALIGN_P(pvStart, PAGE_SIZE) == pvStart, ("pvStart (%p) must be page aligned\n", pvStart));
494 AssertPtr(paPages);
495
496 /*
497 * Issue IOCtl to the SUPDRV kernel module.
498 */
499 SUPPINPAGES_IN In;
500 In.u32Cookie = g_u32Cookie;
501 In.u32SessionCookie = g_u32SessionCookie;
502 In.pvR3 = pvStart;
503 In.cPages = cPages; AssertRelease(In.cPages == cPages);
504 int rc;
505 if (!g_u32FakeMode)
506 {
507 PSUPPINPAGES_OUT pOut;
508 AssertCompile(sizeof(paPages[0]) == sizeof(pOut->aPages[0]));
509
510#if 0
511 size_t cbOut = RT_OFFSETOF(SUPPINPAGES_OUT, aPages[cPages]);
512 pOut = (PSUPPINPAGES_OUT)RTMemTmpAllocZ(cbOut);
513 if (!pOut)
514 return VERR_NO_TMP_MEMORY;
515
516 rc = suplibOsIOCtl(SUP_IOCTL_PINPAGES, &In, sizeof(In), pOut, cbOut);
517 if (RT_SUCCESS(rc))
518 memcpy(paPages, &pOut->aPages[0], sizeof(paPages[0]) * cPages);
519 RTMemTmpFree(pOut);
520
521#else
522 /* a hack to save some time. */
523 pOut = (PSUPPINPAGES_OUT)(void*)paPages;
524 Assert(RT_OFFSETOF(SUPPINPAGES_OUT, aPages) == 0 && sizeof(paPages[0]) == sizeof(pOut->aPages[0]));
525 rc = suplibOsIOCtl(SUP_IOCTL_PINPAGES, &In, sizeof(In), pOut, RT_OFFSETOF(SUPPINPAGES_OUT, aPages[cPages]));
526#endif
527 }
528 else
529 {
530 /* fake a successfull result. */
531 RTHCPHYS Phys = (uintptr_t)pvStart + PAGE_SIZE * 1024;
532 unsigned iPage = cPages;
533 while (iPage-- > 0)
534 paPages[iPage].Phys = Phys + (iPage << PAGE_SHIFT);
535 rc = VINF_SUCCESS;
536 }
537
538 return rc;
539}
540
541
542SUPR3DECL(int) SUPPageUnlock(void *pvStart)
543{
544 /*
545 * Validate.
546 */
547 AssertPtr(pvStart);
548 AssertMsg(RT_ALIGN_P(pvStart, PAGE_SIZE) == pvStart, ("pvStart (%p) must be page aligned\n", pvStart));
549
550 /*
551 * Issue IOCtl to the SUPDRV kernel module.
552 */
553 SUPUNPINPAGES_IN In;
554 In.u32Cookie = g_u32Cookie;
555 In.u32SessionCookie = g_u32SessionCookie;
556 In.pvR3 = pvStart;
557 int rc;
558 if (!g_u32FakeMode)
559 rc = suplibOsIOCtl(SUP_IOCTL_UNPINPAGES, &In, sizeof(In), NULL, 0);
560 else
561 rc = VINF_SUCCESS;
562
563 return rc;
564}
565
566
567SUPR3DECL(void *) SUPContAlloc(size_t cPages, PRTHCPHYS pHCPhys)
568{
569 return SUPContAlloc2(cPages, NIL_RTR0PTR, pHCPhys);
570}
571
572
573SUPR3DECL(void *) SUPContAlloc2(size_t cPages, PRTR0PTR pR0Ptr, PRTHCPHYS pHCPhys)
574{
575 /*
576 * Validate.
577 */
578 AssertMsg(cPages > 0 && cPages < 256, ("cPages=%d must be > 0 and < 256\n", cPages));
579 AssertPtr(pHCPhys);
580 *pHCPhys = NIL_RTHCPHYS;
581 AssertPtrNull(pR0Ptr);
582 if (pR0Ptr)
583 *pR0Ptr = NIL_RTR0PTR;
584
585 /*
586 * Issue IOCtl to the SUPDRV kernel module.
587 */
588 SUPCONTALLOC_IN In;
589 In.u32Cookie = g_u32Cookie;
590 In.u32SessionCookie = g_u32SessionCookie;
591 In.cPages = cPages;
592 SUPCONTALLOC_OUT Out;
593 int rc;
594 if (!g_u32FakeMode)
595 rc = suplibOsIOCtl(SUP_IOCTL_CONT_ALLOC, &In, sizeof(In), &Out, sizeof(Out));
596 else
597 {
598 rc = SUPPageAlloc(In.cPages, &Out.pvR3);
599 Out.HCPhys = (uintptr_t)Out.pvR3 + (PAGE_SHIFT * 1024);
600 Out.pvR0 = (uintptr_t)Out.pvR3;
601 }
602 if (VBOX_SUCCESS(rc))
603 {
604 *pHCPhys = (RTHCPHYS)Out.HCPhys;
605 if (pR0Ptr)
606 *pR0Ptr = Out.pvR0;
607 return Out.pvR3;
608 }
609
610 return NULL;
611}
612
613
614SUPR3DECL(int) SUPContFree(void *pv, size_t cPages)
615{
616 /*
617 * Validate.
618 */
619 AssertPtr(pv);
620 if (!pv)
621 return VINF_SUCCESS;
622
623 /*
624 * Issue IOCtl to the SUPDRV kernel module.
625 */
626 SUPCONTFREE_IN In;
627 In.u32Cookie = g_u32Cookie;
628 In.u32SessionCookie = g_u32SessionCookie;
629 In.pvR3 = pv;
630 int rc;
631 if (!g_u32FakeMode)
632 rc = suplibOsIOCtl(SUP_IOCTL_CONT_FREE, &In, sizeof(In), NULL, 0);
633 else
634 rc = SUPPageFree(pv, cPages);
635
636 return rc;
637}
638
639
640SUPR3DECL(int) SUPLowAlloc(size_t cPages, void **ppvPages, PRTR0PTR ppvPagesR0, PSUPPAGE paPages)
641{
642 /*
643 * Validate.
644 */
645 AssertMsg(cPages > 0 && cPages < 256, ("cPages=%d must be > 0 and < 256\n", cPages));
646 AssertPtr(ppvPages);
647 *ppvPages = NULL;
648 AssertPtr(paPages);
649
650 int rc;
651 if (!g_u32FakeMode)
652 {
653 /*
654 * Issue IOCtl to the SUPDRV kernel module.
655 */
656 SUPLOWALLOC_IN In;
657 In.u32Cookie = g_u32Cookie;
658 In.u32SessionCookie = g_u32SessionCookie;
659 In.cPages = cPages;
660 size_t cbOut = RT_OFFSETOF(SUPLOWALLOC_OUT, aPages[cPages]);
661 PSUPLOWALLOC_OUT pOut = (PSUPLOWALLOC_OUT)RTMemTmpAllocZ(cbOut);
662 if (pOut)
663 {
664 rc = suplibOsIOCtl(SUP_IOCTL_LOW_ALLOC, &In, sizeof(In), pOut, cbOut);
665 if (VBOX_SUCCESS(rc))
666 {
667 *ppvPages = pOut->pvR3;
668 if (ppvPagesR0)
669 *ppvPagesR0 = pOut->pvR0;
670 AssertCompile(sizeof(paPages[0]) == sizeof(pOut->aPages[0]));
671 memcpy(paPages, &pOut->aPages[0], sizeof(paPages[0]) * cPages);
672#ifdef VBOX_STRICT
673 for (unsigned i = 0; i < cPages; i++)
674 AssertReleaseMsg( paPages[i].Phys <= 0xfffff000
675 && !(paPages[i].Phys & PAGE_OFFSET_MASK)
676 && paPages[i].Phys > 0,
677 ("[%d]=%VHp\n", paPages[i].Phys));
678#endif
679 }
680 RTMemTmpFree(pOut);
681 }
682 else
683 rc = VERR_NO_TMP_MEMORY;
684 }
685 else
686 {
687 rc = SUPPageAlloc(cPages, ppvPages);
688 if (VBOX_SUCCESS(rc))
689 {
690 /* fake physical addresses. */
691 RTHCPHYS Phys = (uintptr_t)*ppvPages + PAGE_SIZE * 1024;
692 unsigned iPage = cPages;
693 while (iPage-- > 0)
694 paPages[iPage].Phys = Phys + (iPage << PAGE_SHIFT);
695 }
696 }
697
698 return rc;
699}
700
701
702SUPR3DECL(int) SUPLowFree(void *pv, size_t cPages)
703{
704 /*
705 * Validate.
706 */
707 AssertPtr(pv);
708 if (!pv)
709 return VINF_SUCCESS;
710
711 /*
712 * Issue IOCtl to the SUPDRV kernel module.
713 */
714 SUPLOWFREE_IN In;
715 In.u32Cookie = g_u32Cookie;
716 In.u32SessionCookie = g_u32SessionCookie;
717 In.pvR3 = pv;
718 int rc;
719 if (!g_u32FakeMode)
720 rc = suplibOsIOCtl(SUP_IOCTL_LOW_FREE, &In, sizeof(In), NULL, 0);
721 else
722 rc = SUPPageFree(pv, cPages);
723
724 return rc;
725}
726
727
728SUPR3DECL(int) SUPPageAlloc(size_t cPages, void **ppvPages)
729{
730 /*
731 * Validate.
732 */
733 if (cPages == 0)
734 {
735 AssertMsgFailed(("Invalid param cPages=0, must be > 0\n"));
736 return VERR_INVALID_PARAMETER;
737 }
738 AssertPtr(ppvPages);
739 if (!ppvPages)
740 return VERR_INVALID_PARAMETER;
741 *ppvPages = NULL;
742
743 /*
744 * Call OS specific worker.
745 */
746 return suplibOsPageAlloc(cPages, ppvPages);
747}
748
749
750SUPR3DECL(int) SUPPageFree(void *pvPages, size_t cPages)
751{
752 /*
753 * Validate.
754 */
755 AssertPtr(pvPages);
756 if (!pvPages)
757 return VINF_SUCCESS;
758
759 /*
760 * Call OS specific worker.
761 */
762 return suplibOsPageFree(pvPages, cPages);
763}
764
765
766SUPR3DECL(int) SUPLoadModule(const char *pszFilename, const char *pszModule, void **ppvImageBase)
767{
768 /*
769 * Load the module.
770 * If it's VMMR0.r0 we need to install the IDTE.
771 */
772 int rc = supLoadModule(pszFilename, pszModule, ppvImageBase);
773#ifndef VBOX_WITHOUT_IDT_PATCHING
774 if ( VBOX_SUCCESS(rc)
775 && !strcmp(pszModule, "VMMR0.r0"))
776 {
777 rc = supInstallIDTE();
778 if (VBOX_FAILURE(rc))
779 SUPFreeModule(*ppvImageBase);
780 }
781#endif /* VBOX_WITHOUT_IDT_PATCHING */
782
783 return rc;
784}
785
786
787#ifndef VBOX_WITHOUT_IDT_PATCHING
788/**
789 * Generates the code for calling the interrupt gate.
790 *
791 * @returns VBox status code.
792 * g_pfnCallVMMR0 is changed on success.
793 * @param u8Interrupt The interrupt number.
794 */
795static int suplibGenerateCallVMMR0(uint8_t u8Interrupt)
796{
797 /*
798 * Allocate memory.
799 */
800 uint8_t *pb = (uint8_t *)RTMemExecAlloc(256);
801 AssertReturn(pb, VERR_NO_MEMORY);
802 memset(pb, 0xcc, 256);
803 Assert(!g_pfnCallVMMR0);
804 g_pfnCallVMMR0 = *(PFNCALLVMMR0*)&pb;
805
806 /*
807 * Generate the code.
808 */
809#ifdef RT_ARCH_AMD64
810 /*
811 * reg params:
812 * <GCC> <MSC> <argument>
813 * rdi rcx pVMR0
814 * esi edx uOperation
815 * rdx r8 pvArg
816 *
817 * eax eax [g_u32Gookie]
818 */
819 *pb++ = 0xb8; /* mov eax, <g_u32Cookie> */
820 *(uint32_t *)pb = g_u32Cookie;
821 pb += sizeof(uint32_t);
822
823 *pb++ = 0xcd; /* int <u8Interrupt> */
824 *pb++ = u8Interrupt;
825
826 *pb++ = 0xc3; /* ret */
827
828#else
829 /*
830 * x86 stack:
831 * 0 saved esi
832 * 0 4 ret
833 * 4 8 pVM
834 * 8 c uOperation
835 * c 10 pvArg
836 */
837 *pb++ = 0x56; /* push esi */
838
839 *pb++ = 0x8b; /* mov eax, [pVM] */
840 *pb++ = 0x44;
841 *pb++ = 0x24;
842 *pb++ = 0x08; /* esp+08h */
843
844 *pb++ = 0x8b; /* mov edx, [uOperation] */
845 *pb++ = 0x54;
846 *pb++ = 0x24;
847 *pb++ = 0x0c; /* esp+0ch */
848
849 *pb++ = 0x8b; /* mov ecx, [pvArg] */
850 *pb++ = 0x4c;
851 *pb++ = 0x24;
852 *pb++ = 0x10; /* esp+10h */
853
854 *pb++ = 0xbe; /* mov esi, <g_u32Cookie> */
855 *(uint32_t *)pb = g_u32Cookie;
856 pb += sizeof(uint32_t);
857
858 *pb++ = 0xcd; /* int <u8Interrupt> */
859 *pb++ = u8Interrupt;
860
861 *pb++ = 0x5e; /* pop esi */
862
863 *pb++ = 0xc3; /* ret */
864#endif
865
866 return VINF_SUCCESS;
867}
868
869
870/**
871 * Installs the IDTE patch.
872 *
873 * @return VBox status code.
874 */
875static int supInstallIDTE(void)
876{
877 /* already installed? */
878 if (g_u8Interrupt != 3 || g_u32FakeMode)
879 return VINF_SUCCESS;
880
881 int rc = VINF_SUCCESS;
882 const unsigned cCpus = RTSystemProcessorGetCount();
883 if (cCpus <= 1)
884 {
885 /* UNI */
886 SUPIDTINSTALL_IN In;
887 In.u32Cookie = g_u32Cookie;
888 In.u32SessionCookie = g_u32SessionCookie;
889 SUPIDTINSTALL_OUT Out = {3};
890
891 rc = suplibOsIOCtl(SUP_IOCTL_IDT_INSTALL, &In, sizeof(In), &Out, sizeof(Out));
892 if (VBOX_SUCCESS(rc))
893 {
894 g_u8Interrupt = Out.u8Idt;
895 rc = suplibGenerateCallVMMR0(Out.u8Idt);
896 }
897 }
898 else
899 {
900 /* SMP */
901 uint64_t u64AffMaskSaved = RTThreadGetAffinity();
902 uint64_t u64AffMaskPatched = RTSystemProcessorGetActiveMask() & u64AffMaskSaved;
903 unsigned cCpusPatched = 0;
904
905 for (int i = 0; i < 64; i++)
906 {
907 /* Skip absent and inactive processors. */
908 uint64_t u64Mask = 1ULL << i;
909 if (!(u64Mask & u64AffMaskPatched))
910 continue;
911
912 /* Change CPU */
913 int rc2 = RTThreadSetAffinity(u64Mask);
914 if (VBOX_FAILURE(rc2))
915 {
916 u64AffMaskPatched &= ~u64Mask;
917 Log(("SUPLoadVMM: Failed to set affinity to cpu no. %d, rc=%Vrc.\n", i, rc2));
918 continue;
919 }
920
921 /* Patch the CPU. */
922 SUPIDTINSTALL_IN In;
923 In.u32Cookie = g_u32Cookie;
924 In.u32SessionCookie = g_u32SessionCookie;
925 SUPIDTINSTALL_OUT Out = {3};
926
927 rc2 = suplibOsIOCtl(SUP_IOCTL_IDT_INSTALL, &In, sizeof(In), &Out, sizeof(Out));
928 if (VBOX_SUCCESS(rc2))
929 {
930 if (!cCpusPatched)
931 {
932 g_u8Interrupt = Out.u8Idt;
933 rc2 = suplibGenerateCallVMMR0(Out.u8Idt);
934 if (VBOX_FAILURE(rc))
935 rc2 = rc;
936 }
937 else
938 Assert(g_u8Interrupt == Out.u8Idt);
939 cCpusPatched++;
940 }
941 else
942 {
943
944 Log(("SUPLoadVMM: Failed to patch cpu no. %d, rc=%Vrc.\n", i, rc2));
945 if (VBOX_SUCCESS(rc))
946 rc = rc2;
947 }
948 }
949
950 /* Fail if no CPUs was patched! */
951 if (VBOX_SUCCESS(rc) && cCpusPatched <= 0)
952 rc = VERR_GENERAL_FAILURE;
953 /* Ignore failures if a CPU was patched. */
954 else if (VBOX_FAILURE(rc) && cCpusPatched > 0)
955 {
956 /** @todo add an eventlog/syslog line out this. */
957 rc = VINF_SUCCESS;
958 }
959
960 /* Set/restore the thread affinity. */
961 if (VBOX_SUCCESS(rc))
962 {
963 rc = RTThreadSetAffinity(u64AffMaskPatched);
964 AssertRC(rc);
965 }
966 else
967 {
968 int rc2 = RTThreadSetAffinity(u64AffMaskSaved);
969 AssertRC(rc2);
970 }
971 }
972 return rc;
973}
974#endif /* !VBOX_WITHOUT_IDT_PATCHING */
975
976
977/**
978 * Resolve an external symbol during RTLdrGetBits().
979 *
980 * @returns VBox status code.
981 * @param hLdrMod The loader module handle.
982 * @param pszModule Module name.
983 * @param pszSymbol Symbol name, NULL if uSymbol should be used.
984 * @param uSymbol Symbol ordinal, ~0 if pszSymbol should be used.
985 * @param pValue Where to store the symbol value (address).
986 * @param pvUser User argument.
987 */
988static DECLCALLBACK(int) supLoadModuleResolveImport(RTLDRMOD hLdrMod, const char *pszModule,
989 const char *pszSymbol, unsigned uSymbol, RTUINTPTR *pValue, void *pvUser)
990{
991 AssertPtr(pValue);
992 AssertPtr(pvUser);
993
994 /*
995 * Only SUPR0 and VMMR0.r0
996 */
997 if ( pszModule
998 && *pszModule
999 && strcmp(pszModule, "SUPR0.dll")
1000 && strcmp(pszModule, "VMMR0.r0"))
1001 {
1002 AssertMsgFailed(("%s is importing from %s! (expected 'SUPR0.dll' or 'VMMR0.r0', case-sensitiv)\n", pvUser, pszModule));
1003 return VERR_SYMBOL_NOT_FOUND;
1004 }
1005
1006 /*
1007 * No ordinals.
1008 */
1009 if (pszSymbol < (const char*)0x10000)
1010 {
1011 AssertMsgFailed(("%s is importing by ordinal (ord=%d)\n", pvUser, (int)(uintptr_t)pszSymbol));
1012 return VERR_SYMBOL_NOT_FOUND;
1013 }
1014
1015 /*
1016 * Lookup symbol.
1017 */
1018 /* skip the 64-bit ELF import prefix first. */
1019 if (!strncmp(pszSymbol, "SUPR0$", sizeof("SUPR0$") - 1))
1020 pszSymbol += sizeof("SUPR0$") - 1;
1021
1022 /*
1023 * Check the VMMR0.r0 module if loaded.
1024 */
1025 /** @todo call the SUPLoadModule caller.... */
1026 /** @todo proper reference counting and such. */
1027 if (g_pvVMMR0 != NIL_RTR0PTR)
1028 {
1029 void *pvValue;
1030 if (!SUPGetSymbolR0((void *)g_pvVMMR0, pszSymbol, &pvValue))
1031 {
1032 *pValue = (uintptr_t)pvValue;
1033 return VINF_SUCCESS;
1034 }
1035 }
1036
1037 /* iterate the function table. */
1038 int c = g_pFunctions->cFunctions;
1039 PSUPFUNC pFunc = &g_pFunctions->aFunctions[0];
1040 while (c-- > 0)
1041 {
1042 if (!strcmp(pFunc->szName, pszSymbol))
1043 {
1044 *pValue = (uintptr_t)pFunc->pfn;
1045 return VINF_SUCCESS;
1046 }
1047 pFunc++;
1048 }
1049
1050 /*
1051 * The GIP.
1052 */
1053 /** @todo R0 mapping? */
1054 if ( pszSymbol
1055 && g_pSUPGlobalInfoPage
1056 && g_pSUPGlobalInfoPageR0
1057 && !strcmp(pszSymbol, "g_SUPGlobalInfoPage"))
1058 {
1059 *pValue = (uintptr_t)g_pSUPGlobalInfoPageR0;
1060 return VINF_SUCCESS;
1061 }
1062
1063 /*
1064 * Despair.
1065 */
1066 c = g_pFunctions->cFunctions;
1067 pFunc = &g_pFunctions->aFunctions[0];
1068 while (c-- > 0)
1069 {
1070 AssertMsg2("%d: %s\n", g_pFunctions->cFunctions - c, pFunc->szName);
1071 pFunc++;
1072 }
1073
1074 AssertMsgFailed(("%s is importing %s which we couldn't find\n", pvUser, pszSymbol));
1075 return VERR_SYMBOL_NOT_FOUND;
1076}
1077
1078
1079/** Argument package for supLoadModuleCalcSizeCB. */
1080typedef struct SUPLDRCALCSIZEARGS
1081{
1082 size_t cbStrings;
1083 uint32_t cSymbols;
1084 size_t cbImage;
1085} SUPLDRCALCSIZEARGS, *PSUPLDRCALCSIZEARGS;
1086
1087/**
1088 * Callback used to calculate the image size.
1089 * @return VINF_SUCCESS
1090 */
1091static DECLCALLBACK(int) supLoadModuleCalcSizeCB(RTLDRMOD hLdrMod, const char *pszSymbol, unsigned uSymbol, RTUINTPTR Value, void *pvUser)
1092{
1093 PSUPLDRCALCSIZEARGS pArgs = (PSUPLDRCALCSIZEARGS)pvUser;
1094 if ( pszSymbol != NULL
1095 && *pszSymbol
1096 && Value <= pArgs->cbImage)
1097 {
1098 pArgs->cSymbols++;
1099 pArgs->cbStrings += strlen(pszSymbol) + 1;
1100 }
1101 return VINF_SUCCESS;
1102}
1103
1104
1105/** Argument package for supLoadModuleCreateTabsCB. */
1106typedef struct SUPLDRCREATETABSARGS
1107{
1108 size_t cbImage;
1109 PSUPLDRSYM pSym;
1110 char *pszBase;
1111 char *psz;
1112} SUPLDRCREATETABSARGS, *PSUPLDRCREATETABSARGS;
1113
1114/**
1115 * Callback used to calculate the image size.
1116 * @return VINF_SUCCESS
1117 */
1118static DECLCALLBACK(int) supLoadModuleCreateTabsCB(RTLDRMOD hLdrMod, const char *pszSymbol, unsigned uSymbol, RTUINTPTR Value, void *pvUser)
1119{
1120 PSUPLDRCREATETABSARGS pArgs = (PSUPLDRCREATETABSARGS)pvUser;
1121 if ( pszSymbol != NULL
1122 && *pszSymbol
1123 && Value <= pArgs->cbImage)
1124 {
1125 pArgs->pSym->offSymbol = (uint32_t)Value;
1126 pArgs->pSym->offName = pArgs->psz - pArgs->pszBase;
1127 pArgs->pSym++;
1128
1129 size_t cbCopy = strlen(pszSymbol) + 1;
1130 memcpy(pArgs->psz, pszSymbol, cbCopy);
1131 pArgs->psz += cbCopy;
1132 }
1133 return VINF_SUCCESS;
1134}
1135
1136
1137/**
1138 * Worker for SUPLoadModule().
1139 *
1140 * @returns VBox status code.
1141 * @param pszFilename Name of the VMMR0 image file
1142 */
1143static int supLoadModule(const char *pszFilename, const char *pszModule, void **ppvImageBase)
1144{
1145 /*
1146 * Validate input.
1147 */
1148 AssertPtrReturn(pszFilename, VERR_INVALID_PARAMETER);
1149 AssertPtrReturn(pszModule, VERR_INVALID_PARAMETER);
1150 AssertPtrReturn(ppvImageBase, VERR_INVALID_PARAMETER);
1151 AssertReturn(strlen(pszModule) < SIZEOFMEMB(SUPLDROPEN_IN, szName), VERR_FILENAME_TOO_LONG);
1152
1153 const bool fIsVMMR0 = !strcmp(pszModule, "VMMR0.r0");
1154 *ppvImageBase = NULL;
1155
1156 /*
1157 * Open image file and figure its size.
1158 */
1159 RTLDRMOD hLdrMod;
1160 int rc = RTLdrOpen(pszFilename, &hLdrMod);
1161 if (!VBOX_SUCCESS(rc))
1162 return rc;
1163
1164 SUPLDRCALCSIZEARGS CalcArgs;
1165 CalcArgs.cbStrings = 0;
1166 CalcArgs.cSymbols = 0;
1167 CalcArgs.cbImage = RTLdrSize(hLdrMod);
1168 rc = RTLdrEnumSymbols(hLdrMod, 0, NULL, 0, supLoadModuleCalcSizeCB, &CalcArgs);
1169 if (VBOX_SUCCESS(rc))
1170 {
1171 const uint32_t offSymTab = RT_ALIGN_32(CalcArgs.cbImage, 8);
1172 const uint32_t offStrTab = offSymTab + CalcArgs.cSymbols * sizeof(SUPLDRSYM);
1173 const uint32_t cbImage = RT_ALIGN_32(offStrTab + CalcArgs.cbStrings, 8);
1174
1175 /*
1176 * Open the R0 image.
1177 */
1178 SUPLDROPEN_IN OpenIn;
1179 OpenIn.u32Cookie = g_u32Cookie;
1180 OpenIn.u32SessionCookie = g_u32SessionCookie;
1181 OpenIn.cbImage = cbImage;
1182 strcpy(OpenIn.szName, pszModule);
1183 SUPLDROPEN_OUT OpenOut;
1184 if (!g_u32FakeMode)
1185 rc = suplibOsIOCtl(SUP_IOCTL_LDR_OPEN, &OpenIn, sizeof(OpenIn), &OpenOut, sizeof(OpenOut));
1186 else
1187 {
1188 OpenOut.fNeedsLoading = true;
1189 OpenOut.pvImageBase = 0xef423420;
1190 }
1191 *ppvImageBase = (void *)OpenOut.pvImageBase;
1192 if ( VBOX_SUCCESS(rc)
1193 && OpenOut.fNeedsLoading)
1194 {
1195 /*
1196 * We need to load it.
1197 * Allocate memory for the image bits.
1198 */
1199 unsigned cbIn = RT_OFFSETOF(SUPLDRLOAD_IN, achImage[cbImage]);
1200 PSUPLDRLOAD_IN pIn = (PSUPLDRLOAD_IN)RTMemTmpAlloc(cbIn);
1201 if (pIn)
1202 {
1203 /*
1204 * Get the image bits.
1205 */
1206 rc = RTLdrGetBits(hLdrMod, &pIn->achImage[0], (uintptr_t)OpenOut.pvImageBase,
1207 supLoadModuleResolveImport, (void *)pszModule);
1208
1209 if (VBOX_SUCCESS(rc))
1210 {
1211 /*
1212 * Get the entry points.
1213 */
1214 RTUINTPTR VMMR0Entry = 0;
1215 RTUINTPTR ModuleInit = 0;
1216 RTUINTPTR ModuleTerm = 0;
1217 if (fIsVMMR0)
1218 rc = RTLdrGetSymbolEx(hLdrMod, &pIn->achImage[0], (uintptr_t)OpenOut.pvImageBase, "VMMR0Entry", &VMMR0Entry);
1219 if (VBOX_SUCCESS(rc))
1220 {
1221 int rc2 = RTLdrGetSymbolEx(hLdrMod, &pIn->achImage[0], (uintptr_t)OpenOut.pvImageBase, "ModuleInit", &ModuleInit);
1222 if (VBOX_FAILURE(rc2))
1223 ModuleInit = 0;
1224
1225 rc2 = RTLdrGetSymbolEx(hLdrMod, &pIn->achImage[0], (uintptr_t)OpenOut.pvImageBase, "ModuleTerm", &ModuleTerm);
1226 if (VBOX_FAILURE(rc2))
1227 ModuleTerm = 0;
1228 }
1229 if (VBOX_SUCCESS(rc))
1230 {
1231 /*
1232 * Create the symbol and string tables.
1233 */
1234 SUPLDRCREATETABSARGS CreateArgs;
1235 CreateArgs.cbImage = CalcArgs.cbImage;
1236 CreateArgs.pSym = (PSUPLDRSYM)&pIn->achImage[offSymTab];
1237 CreateArgs.pszBase = (char *)&pIn->achImage[offStrTab];
1238 CreateArgs.psz = CreateArgs.pszBase;
1239 rc = RTLdrEnumSymbols(hLdrMod, 0, NULL, 0, supLoadModuleCreateTabsCB, &CreateArgs);
1240 if (VBOX_SUCCESS(rc))
1241 {
1242 AssertRelease((size_t)(CreateArgs.psz - CreateArgs.pszBase) <= CalcArgs.cbStrings);
1243 AssertRelease((size_t)(CreateArgs.pSym - (PSUPLDRSYM)&pIn->achImage[offSymTab]) <= CalcArgs.cSymbols);
1244
1245 /*
1246 * Upload the image.
1247 */
1248 pIn->u32Cookie = g_u32Cookie;
1249 pIn->u32SessionCookie = g_u32SessionCookie;
1250 pIn->pfnModuleInit = (RTR0PTR)ModuleInit;
1251 pIn->pfnModuleTerm = (RTR0PTR)ModuleTerm;
1252 if (fIsVMMR0)
1253 {
1254 pIn->eEPType = pIn->EP_VMMR0;
1255 pIn->EP.VMMR0.pvVMMR0 = OpenOut.pvImageBase;
1256 pIn->EP.VMMR0.pvVMMR0Entry = (RTR0PTR)VMMR0Entry;
1257 }
1258 else
1259 pIn->eEPType = pIn->EP_NOTHING;
1260 pIn->offStrTab = offStrTab;
1261 pIn->cbStrTab = (uint32_t)CalcArgs.cbStrings;
1262 AssertRelease(pIn->cbStrTab == CalcArgs.cbStrings);
1263 pIn->offSymbols = offSymTab;
1264 pIn->cSymbols = CalcArgs.cSymbols;
1265 pIn->cbImage = cbImage;
1266 pIn->pvImageBase = OpenOut.pvImageBase;
1267 if (!g_u32FakeMode)
1268 rc = suplibOsIOCtl(SUP_IOCTL_LDR_LOAD, pIn, cbIn, NULL, 0);
1269 else
1270 rc = VINF_SUCCESS;
1271 if ( VBOX_SUCCESS(rc)
1272 || rc == VERR_ALREADY_LOADED /* this is because of a competing process. */
1273 )
1274 {
1275 if (fIsVMMR0)
1276 g_pvVMMR0 = OpenOut.pvImageBase;
1277 RTMemTmpFree(pIn);
1278 RTLdrClose(hLdrMod);
1279 return VINF_SUCCESS;
1280 }
1281 }
1282 }
1283 }
1284 RTMemTmpFree(pIn);
1285 }
1286 else
1287 {
1288 AssertMsgFailed(("failed to allocated %d bytes for SUPLDRLOAD_IN structure!\n", cbIn));
1289 rc = VERR_NO_TMP_MEMORY;
1290 }
1291 }
1292 else if (VBOX_SUCCESS(rc) && fIsVMMR0)
1293 g_pvVMMR0 = OpenOut.pvImageBase;
1294 }
1295 RTLdrClose(hLdrMod);
1296 return rc;
1297}
1298
1299
1300SUPR3DECL(int) SUPFreeModule(void *pvImageBase)
1301{
1302 /*
1303 * There is one special module. When this is freed we'll
1304 * free the IDT entry that goes with it.
1305 *
1306 * Note that we don't keep count of VMMR0.r0 loads here, so the
1307 * first unload will free it.
1308 */
1309 if ((RTR0PTR)pvImageBase == g_pvVMMR0)
1310 {
1311 /*
1312 * This is the point where we remove the IDT hook. We do
1313 * that before unloading the R0 VMM part.
1314 */
1315 if (g_u32FakeMode)
1316 {
1317#ifndef VBOX_WITHOUT_IDT_PATCHING
1318 g_u8Interrupt = 3;
1319 RTMemExecFree(*(void **)&g_pfnCallVMMR0);
1320 g_pfnCallVMMR0 = NULL;
1321#endif
1322 g_pvVMMR0 = NIL_RTR0PTR;
1323 return VINF_SUCCESS;
1324 }
1325
1326#ifndef VBOX_WITHOUT_IDT_PATCHING
1327 /*
1328 * Uninstall IDT entry.
1329 */
1330 int rc = 0;
1331 if (g_u8Interrupt != 3)
1332 {
1333 SUPIDTREMOVE_IN In;
1334 In.u32Cookie = g_u32Cookie;
1335 In.u32SessionCookie = g_u32SessionCookie;
1336 rc = suplibOsIOCtl(SUP_IOCTL_IDT_REMOVE, &In, sizeof(In), NULL, 0);
1337 g_u8Interrupt = 3;
1338 RTMemExecFree(*(void **)&g_pfnCallVMMR0);
1339 g_pfnCallVMMR0 = NULL;
1340 }
1341#endif
1342 }
1343
1344 /*
1345 * Free the requested module.
1346 */
1347 SUPLDRFREE_IN In;
1348 In.u32Cookie = g_u32Cookie;
1349 In.u32SessionCookie = g_u32SessionCookie;
1350 In.pvImageBase = (RTR0PTR)pvImageBase;
1351 int rc = VINF_SUCCESS;
1352 if (!g_u32FakeMode)
1353 rc = suplibOsIOCtl(SUP_IOCTL_LDR_FREE, &In, sizeof(In), NULL, 0);
1354 if ( VBOX_SUCCESS(rc)
1355 && (RTR0PTR)pvImageBase == g_pvVMMR0)
1356 g_pvVMMR0 = NIL_RTR0PTR;
1357 return rc;
1358}
1359
1360
1361SUPR3DECL(int) SUPGetSymbolR0(void *pvImageBase, const char *pszSymbol, void **ppvValue)
1362{
1363 *ppvValue = NULL;
1364
1365 /*
1366 * Do ioctl.
1367 */
1368 size_t cchSymbol = strlen(pszSymbol);
1369 const size_t cbIn = RT_OFFSETOF(SUPLDRGETSYMBOL_IN, szSymbol[cchSymbol + 1]);
1370 SUPLDRGETSYMBOL_OUT Out = { NIL_RTR0PTR };
1371 PSUPLDRGETSYMBOL_IN pIn = (PSUPLDRGETSYMBOL_IN)alloca(cbIn);
1372 pIn->u32Cookie = g_u32Cookie;
1373 pIn->u32SessionCookie = g_u32SessionCookie;
1374 pIn->pvImageBase = (RTR0PTR)pvImageBase;
1375 memcpy(pIn->szSymbol, pszSymbol, cchSymbol + 1);
1376 int rc;
1377 if (RT_LIKELY(!g_u32FakeMode))
1378 rc = suplibOsIOCtl(SUP_IOCTL_LDR_GET_SYMBOL, pIn, cbIn, &Out, sizeof(Out));
1379 else
1380 {
1381 rc = VINF_SUCCESS;
1382 Out.pvSymbol = 0xdeadf00d;
1383 }
1384 if (VBOX_SUCCESS(rc))
1385 *ppvValue = (void *)Out.pvSymbol;
1386 return rc;
1387}
1388
1389
1390SUPR3DECL(int) SUPLoadVMM(const char *pszFilename)
1391{
1392 void *pvImageBase;
1393 return SUPLoadModule(pszFilename, "VMMR0.r0", &pvImageBase);
1394}
1395
1396
1397SUPR3DECL(int) SUPUnloadVMM(void)
1398{
1399 return SUPFreeModule((void*)g_pvVMMR0);
1400}
1401
1402
1403SUPR3DECL(int) SUPGipGetPhys(PRTHCPHYS pHCPhys)
1404{
1405 if (g_pSUPGlobalInfoPage)
1406 {
1407 *pHCPhys = g_HCPhysSUPGlobalInfoPage;
1408 return VINF_SUCCESS;
1409 }
1410 *pHCPhys = NIL_RTHCPHYS;
1411 return VERR_WRONG_ORDER;
1412}
1413
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