VirtualBox

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

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

The Giant CDDL Dual-License Header Change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 60.7 KB
Line 
1/* $Id: SUPLib.cpp 5999 2007-12-07 15:05:06Z 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 (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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27/** @page pg_sup SUP - The Support Library
28 *
29 * The support library is responsible for providing facilities to load
30 * VMM Host Ring-0 code, to call Host VMM Ring-0 code from Ring-3 Host
31 * code, to pin down physical memory, and more.
32 *
33 * The VMM Host Ring-0 code can be combined in the support driver if
34 * permitted by kernel module license policies. If it is not combined
35 * it will be externalized in a .r0 module that will be loaded using
36 * the IPRT loader.
37 *
38 * The Ring-0 calling is done thru a generic SUP interface which will
39 * tranfer an argument set and call a predefined entry point in the Host
40 * VMM Ring-0 code.
41 *
42 * See @ref grp_sup "SUP - Support APIs" for API details.
43 */
44
45
46/*******************************************************************************
47* Header Files *
48*******************************************************************************/
49#define LOG_GROUP LOG_GROUP_SUP
50#include <VBox/sup.h>
51#include <VBox/err.h>
52#include <VBox/param.h>
53#include <VBox/vmm.h>
54#include <VBox/log.h>
55
56#include <iprt/assert.h>
57#include <iprt/alloc.h>
58#include <iprt/alloca.h>
59#include <iprt/ldr.h>
60#include <iprt/asm.h>
61#include <iprt/system.h>
62#include <iprt/thread.h>
63#include <iprt/process.h>
64#include <iprt/string.h>
65#include <iprt/env.h>
66#include <iprt/rand.h>
67
68#include "SUPLibInternal.h"
69#include "SUPDRVIOC.h"
70
71
72
73/*******************************************************************************
74* Defined Constants And Macros *
75*******************************************************************************/
76/** R0 VMM module name. */
77#define VMMR0_NAME "VMMR0"
78
79
80/*******************************************************************************
81* Structures and Typedefs *
82*******************************************************************************/
83typedef DECLCALLBACK(int) FNCALLVMMR0(PVMR0 pVMR0, unsigned uOperation, void *pvArg);
84typedef FNCALLVMMR0 *PFNCALLVMMR0;
85
86
87/*******************************************************************************
88* Global Variables *
89*******************************************************************************/
90/** Pointer to the Global Information Page.
91 *
92 * This pointer is valid as long as SUPLib has a open session. Anyone using
93 * the page must treat this pointer as higly volatile and not trust it beyond
94 * one transaction.
95 *
96 * @todo This will probably deserve it's own session or some other good solution...
97 */
98DECLEXPORT(PSUPGLOBALINFOPAGE) g_pSUPGlobalInfoPage;
99/** Address of the ring-0 mapping of the GIP. */
100static PSUPGLOBALINFOPAGE g_pSUPGlobalInfoPageR0;
101/** The physical address of the GIP. */
102static RTHCPHYS g_HCPhysSUPGlobalInfoPage = NIL_RTHCPHYS;
103
104/** The negotiated cookie. */
105uint32_t g_u32Cookie = 0;
106/** The negotiated session cookie. */
107uint32_t g_u32SessionCookie;
108/** Session handle. */
109PSUPDRVSESSION g_pSession;
110/** R0 SUP Functions used for resolving referenced to the SUPR0 module. */
111static PSUPQUERYFUNCS g_pFunctions;
112
113#ifdef VBOX_WITH_IDT_PATCHING
114/** The negotiated interrupt number. */
115static uint8_t g_u8Interrupt = 3;
116/** Pointer to the generated code fore calling VMMR0. */
117static PFNCALLVMMR0 g_pfnCallVMMR0;
118#endif
119/** VMMR0 Load Address. */
120static RTR0PTR g_pvVMMR0 = NIL_RTR0PTR;
121/** Init counter. */
122static unsigned g_cInits = 0;
123/** PAGE_ALLOC support indicator. */
124static bool g_fSupportsPageAllocLocked = true;
125/** Fake mode indicator. (~0 at first, 0 or 1 after first test) */
126static uint32_t g_u32FakeMode = ~0;
127
128
129/*******************************************************************************
130* Internal Functions *
131*******************************************************************************/
132static int supInitFake(PSUPDRVSESSION *ppSession);
133static int supLoadModule(const char *pszFilename, const char *pszModule, void **ppvImageBase);
134#ifdef VBOX_WITH_IDT_PATCHING
135static int supInstallIDTE(void);
136#endif
137static DECLCALLBACK(int) supLoadModuleResolveImport(RTLDRMOD hLdrMod, const char *pszModule, const char *pszSymbol, unsigned uSymbol, RTUINTPTR *pValue, void *pvUser);
138
139
140SUPR3DECL(int) SUPInstall(void)
141{
142 return suplibOsInstall();
143}
144
145
146SUPR3DECL(int) SUPUninstall(void)
147{
148 return suplibOsUninstall();
149}
150
151
152SUPR3DECL(int) SUPInit(PSUPDRVSESSION *ppSession /* NULL */, size_t cbReserve /* 0 */)
153{
154 /*
155 * Perform some sanity checks.
156 * (Got some trouble with compile time member alignment assertions.)
157 */
158 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, u64NanoTSLastUpdateHz) & 0x7));
159 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs) & 0x1f));
160 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[1]) & 0x1f));
161 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[0].u64NanoTS) & 0x7));
162 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[0].u64TSC) & 0x7));
163 Assert(!(RT_OFFSETOF(SUPGLOBALINFOPAGE, aCPUs[0].u64CpuHz) & 0x7));
164
165 /*
166 * Check if already initialized.
167 */
168 if (ppSession)
169 *ppSession = g_pSession;
170 if (g_cInits++ > 0)
171 return VINF_SUCCESS;
172
173 /*
174 * Check for fake mode.
175 *
176 * Fake mode is used when we're doing smoke testing and debugging.
177 * It's also useful on platforms where we haven't root access or which
178 * we haven't ported the support driver to.
179 */
180 if (g_u32FakeMode == ~0U)
181 {
182 const char *psz = RTEnvGet("VBOX_SUPLIB_FAKE");
183 if (psz && !strcmp(psz, "fake"))
184 ASMAtomicCmpXchgU32(&g_u32FakeMode, 1, ~0U);
185 else
186 ASMAtomicCmpXchgU32(&g_u32FakeMode, 0, ~0U);
187 }
188 if (RT_UNLIKELY(g_u32FakeMode))
189 return supInitFake(ppSession);
190
191 /**
192 * Open the support driver.
193 */
194 int rc = suplibOsInit(cbReserve);
195 if (RT_SUCCESS(rc))
196 {
197 /*
198 * Negotiate the cookie.
199 */
200 SUPCOOKIE CookieReq;
201 memset(&CookieReq, 0xff, sizeof(CookieReq));
202 CookieReq.Hdr.u32Cookie = SUPCOOKIE_INITIAL_COOKIE;
203 CookieReq.Hdr.u32SessionCookie = RTRandU32();
204 CookieReq.Hdr.cbIn = SUP_IOCTL_COOKIE_SIZE_IN;
205 CookieReq.Hdr.cbOut = SUP_IOCTL_COOKIE_SIZE_OUT;
206 CookieReq.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
207 CookieReq.Hdr.rc = VERR_INTERNAL_ERROR;
208 strcpy(CookieReq.u.In.szMagic, SUPCOOKIE_MAGIC);
209 CookieReq.u.In.u32ReqVersion = SUPDRVIOC_VERSION;
210 CookieReq.u.In.u32MinVersion = SUPDRVIOC_VERSION & 0xffff0000;
211 rc = suplibOsIOCtl(SUP_IOCTL_COOKIE, &CookieReq, SUP_IOCTL_COOKIE_SIZE);
212 if ( RT_SUCCESS(rc)
213 && RT_SUCCESS(CookieReq.Hdr.rc))
214 {
215 if ((CookieReq.u.Out.u32SessionVersion & 0xffff0000) == (SUPDRVIOC_VERSION & 0xffff0000))
216 {
217 /*
218 * Query the functions.
219 */
220 PSUPQUERYFUNCS pFuncsReq = (PSUPQUERYFUNCS)RTMemAllocZ(SUP_IOCTL_QUERY_FUNCS_SIZE(CookieReq.u.Out.cFunctions));
221 if (pFuncsReq)
222 {
223 pFuncsReq->Hdr.u32Cookie = CookieReq.u.Out.u32Cookie;
224 pFuncsReq->Hdr.u32SessionCookie = CookieReq.u.Out.u32SessionCookie;
225 pFuncsReq->Hdr.cbIn = SUP_IOCTL_QUERY_FUNCS_SIZE_IN;
226 pFuncsReq->Hdr.cbOut = SUP_IOCTL_QUERY_FUNCS_SIZE_OUT(CookieReq.u.Out.cFunctions);
227 pFuncsReq->Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
228 pFuncsReq->Hdr.rc = VERR_INTERNAL_ERROR;
229 rc = suplibOsIOCtl(SUP_IOCTL_QUERY_FUNCS(CookieReq.u.Out.cFunctions), pFuncsReq, SUP_IOCTL_QUERY_FUNCS_SIZE(CookieReq.u.Out.cFunctions));
230 if (RT_SUCCESS(rc))
231 rc = pFuncsReq->Hdr.rc;
232 if (RT_SUCCESS(rc))
233 {
234 g_u32Cookie = CookieReq.u.Out.u32Cookie;
235 g_u32SessionCookie = CookieReq.u.Out.u32SessionCookie;
236 g_pSession = CookieReq.u.Out.pSession;
237 g_pFunctions = pFuncsReq;
238 if (ppSession)
239 *ppSession = CookieReq.u.Out.pSession;
240
241 /*
242 * Map the GIP into userspace.
243 * This is an optional feature, so we will ignore any failures here.
244 */
245 if (!g_pSUPGlobalInfoPage)
246 {
247 SUPGIPMAP GipMapReq;
248 GipMapReq.Hdr.u32Cookie = g_u32Cookie;
249 GipMapReq.Hdr.u32SessionCookie = g_u32SessionCookie;
250 GipMapReq.Hdr.cbIn = SUP_IOCTL_GIP_MAP_SIZE_IN;
251 GipMapReq.Hdr.cbOut = SUP_IOCTL_GIP_MAP_SIZE_OUT;
252 GipMapReq.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
253 GipMapReq.Hdr.rc = VERR_INTERNAL_ERROR;
254 GipMapReq.u.Out.HCPhysGip = NIL_RTHCPHYS;
255 GipMapReq.u.Out.pGipR0 = NIL_RTR0PTR;
256 GipMapReq.u.Out.pGipR3 = NULL;
257 rc = suplibOsIOCtl(SUP_IOCTL_GIP_MAP, &GipMapReq, SUP_IOCTL_GIP_MAP_SIZE);
258 if (RT_SUCCESS(rc))
259 rc = GipMapReq.Hdr.rc;
260 if (RT_SUCCESS(rc))
261 {
262 AssertRelease(GipMapReq.u.Out.pGipR3->u32Magic == SUPGLOBALINFOPAGE_MAGIC);
263 AssertRelease(GipMapReq.u.Out.pGipR3->u32Version >= SUPGLOBALINFOPAGE_VERSION);
264 ASMAtomicXchgSize(&g_HCPhysSUPGlobalInfoPage, GipMapReq.u.Out.HCPhysGip);
265 ASMAtomicCmpXchgPtr((void * volatile *)&g_pSUPGlobalInfoPage, GipMapReq.u.Out.pGipR3, NULL);
266 ASMAtomicCmpXchgPtr((void * volatile *)&g_pSUPGlobalInfoPageR0, (void *)GipMapReq.u.Out.pGipR0, NULL);
267 }
268 }
269 return VINF_SUCCESS;
270 }
271
272 /* bailout */
273 RTMemFree(pFuncsReq);
274 }
275 else
276 rc = VERR_NO_MEMORY;
277 }
278 else
279 {
280 LogRel(("Support driver version mismatch: SessionVersion=%#x DriverVersion=%#x ClientVersion=%#x\n",
281 CookieReq.u.Out.u32SessionVersion, CookieReq.u.Out.u32DriverVersion, SUPDRVIOC_VERSION));
282 rc = VERR_VM_DRIVER_VERSION_MISMATCH;
283 }
284 }
285 else
286 {
287 if (RT_SUCCESS(rc))
288 {
289 rc = CookieReq.Hdr.rc;
290 LogRel(("Support driver version mismatch: DriverVersion=%#x ClientVersion=%#x rc=%Rrc\n",
291 CookieReq.u.Out.u32DriverVersion, SUPDRVIOC_VERSION, rc));
292 if (rc != VERR_VM_DRIVER_VERSION_MISMATCH)
293 rc = VERR_VM_DRIVER_VERSION_MISMATCH;
294 }
295 else
296 {
297 /* for pre 0x00060000 drivers */
298 LogRel(("Support driver version mismatch: DriverVersion=too-old ClientVersion=%#x\n", SUPDRVIOC_VERSION));
299 rc = VERR_VM_DRIVER_VERSION_MISMATCH;
300 }
301 }
302
303 suplibOsTerm();
304 }
305 AssertMsgFailed(("SUPInit() failed rc=%Vrc\n", rc));
306 g_cInits--;
307
308 return rc;
309}
310
311/**
312 * Fake mode init.
313 */
314static int supInitFake(PSUPDRVSESSION *ppSession)
315{
316 Log(("SUP: Fake mode!\n"));
317 static const SUPFUNC s_aFakeFunctions[] =
318 {
319 /* name function */
320 { "SUPR0ObjRegister", 0xefef0000 },
321 { "SUPR0ObjAddRef", 0xefef0001 },
322 { "SUPR0ObjRelease", 0xefef0002 },
323 { "SUPR0ObjVerifyAccess", 0xefef0003 },
324 { "SUPR0LockMem", 0xefef0004 },
325 { "SUPR0UnlockMem", 0xefef0005 },
326 { "SUPR0ContAlloc", 0xefef0006 },
327 { "SUPR0ContFree", 0xefef0007 },
328 { "SUPR0MemAlloc", 0xefef0008 },
329 { "SUPR0MemGetPhys", 0xefef0009 },
330 { "SUPR0MemFree", 0xefef000a },
331 { "SUPR0Printf", 0xefef000b },
332 { "RTMemAlloc", 0xefef000c },
333 { "RTMemAllocZ", 0xefef000d },
334 { "RTMemFree", 0xefef000e },
335 { "RTR0MemObjAddress", 0xefef0010 },
336 { "RTR0MemObjAddressR3", 0xefef0011 },
337 { "RTR0MemObjAllocPage", 0xefef0012 },
338 { "RTR0MemObjAllocPhysNC", 0xefef0013 },
339 { "RTR0MemObjAllocLow", 0xefef0014 },
340 { "RTR0MemObjFree", 0xefef0015 },
341 { "RTR0MemObjGetPagePhysAddr", 0xefef0016 },
342 { "RTR0MemObjMapUser", 0xefef0017 },
343 { "RTSemEventCreate", 0xefef0018 },
344 { "RTSemEventSignal", 0xefef0019 },
345 { "RTSemEventWait", 0xefef001a },
346 { "RTSemEventWaitNoResume", 0xefef001b },
347 { "RTSemEventDestroy", 0xefef001c },
348 { "RTSemEventMultiCreate", 0xefef001d },
349 { "RTSemEventMultiSignal", 0xefef001e },
350 { "RTSemEventMultiReset", 0xefef001f },
351 { "RTSemEventMultiWait", 0xefef0020 },
352 { "RTSemEventMultiWaitNoResume", 0xefef0021 },
353 { "RTSemEventMultiDestroy", 0xefef0022 },
354 { "RTSemFastMutexCreate", 0xefef0023 },
355 { "RTSemFastMutexDestroy", 0xefef0024 },
356 { "RTSemFastMutexRequest", 0xefef0025 },
357 { "RTSemFastMutexRelease", 0xefef0026 },
358 { "RTSpinlockCreate", 0xefef0027 },
359 { "RTSpinlockDestroy", 0xefef0028 },
360 { "RTSpinlockAcquire", 0xefef0029 },
361 { "RTSpinlockRelease", 0xefef002a },
362 { "RTSpinlockAcquireNoInts", 0xefef002b },
363 { "RTSpinlockReleaseNoInts", 0xefef002c },
364 { "RTThreadNativeSelf", 0xefef002d },
365 { "RTThreadSleep", 0xefef002e },
366 { "RTThreadYield", 0xefef002f },
367 { "RTLogDefaultInstance", 0xefef0030 },
368 { "RTLogRelDefaultInstance", 0xefef0031 },
369 { "RTLogSetDefaultInstanceThread", 0xefef0032 },
370 { "RTLogLogger", 0xefef0033 },
371 { "RTLogLoggerEx", 0xefef0034 },
372 { "RTLogLoggerExV", 0xefef0035 },
373 { "AssertMsg1", 0xefef0036 },
374 { "AssertMsg2", 0xefef0037 },
375 };
376
377 /* fake r0 functions. */
378 g_pFunctions = (PSUPQUERYFUNCS)RTMemAllocZ(SUP_IOCTL_QUERY_FUNCS_SIZE(RT_ELEMENTS(s_aFakeFunctions)));
379 if (g_pFunctions)
380 {
381 g_pFunctions->u.Out.cFunctions = RT_ELEMENTS(s_aFakeFunctions);
382 memcpy(&g_pFunctions->u.Out.aFunctions[0], &s_aFakeFunctions[0], sizeof(s_aFakeFunctions));
383 g_pSession = (PSUPDRVSESSION)(void *)g_pFunctions;
384 if (ppSession)
385 *ppSession = g_pSession;
386#ifdef VBOX_WITH_IDT_PATCHING
387 Assert(g_u8Interrupt == 3);
388#endif
389
390 /* fake the GIP. */
391 g_pSUPGlobalInfoPage = (PSUPGLOBALINFOPAGE)RTMemPageAllocZ(PAGE_SIZE);
392 if (g_pSUPGlobalInfoPage)
393 {
394 g_pSUPGlobalInfoPageR0 = g_pSUPGlobalInfoPage;
395 g_HCPhysSUPGlobalInfoPage = NIL_RTHCPHYS & ~(RTHCPHYS)PAGE_OFFSET_MASK;
396 /* the page is supposed to be invalid, so don't set the magic. */
397 return VINF_SUCCESS;
398 }
399
400 RTMemFree(g_pFunctions);
401 g_pFunctions = NULL;
402 }
403 return VERR_NO_MEMORY;
404}
405
406
407SUPR3DECL(int) SUPTerm(bool fForced)
408{
409 /*
410 * Verify state.
411 */
412 AssertMsg(g_cInits > 0, ("SUPTerm() is called before SUPInit()!\n"));
413 if (g_cInits == 0)
414 return VERR_WRONG_ORDER;
415 if (g_cInits == 1 || fForced)
416 {
417 /*
418 * NULL the GIP pointer.
419 */
420 if (g_pSUPGlobalInfoPage)
421 {
422 ASMAtomicXchgPtr((void * volatile *)&g_pSUPGlobalInfoPage, NULL);
423 ASMAtomicXchgPtr((void * volatile *)&g_pSUPGlobalInfoPageR0, NULL);
424 ASMAtomicXchgSize(&g_HCPhysSUPGlobalInfoPage, NIL_RTHCPHYS);
425 /* just a little safe guard against threads using the page. */
426 RTThreadSleep(50);
427 }
428
429 /*
430 * Close the support driver.
431 */
432 int rc = suplibOsTerm();
433 if (rc)
434 return rc;
435
436 g_u32Cookie = 0;
437 g_u32SessionCookie = 0;
438#ifdef VBOX_WITH_IDT_PATCHING
439 g_u8Interrupt = 3;
440#endif
441 g_cInits = 0;
442 }
443 else
444 g_cInits--;
445
446 return 0;
447}
448
449
450SUPR3DECL(SUPPAGINGMODE) SUPGetPagingMode(void)
451{
452 /* fake */
453 if (RT_UNLIKELY(g_u32FakeMode))
454#ifdef RT_ARCH_AMD64
455 return SUPPAGINGMODE_AMD64_GLOBAL_NX;
456#else
457 return SUPPAGINGMODE_32_BIT_GLOBAL;
458#endif
459
460 /*
461 * Issue IOCtl to the SUPDRV kernel module.
462 */
463 SUPGETPAGINGMODE Req;
464 Req.Hdr.u32Cookie = g_u32Cookie;
465 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
466 Req.Hdr.cbIn = SUP_IOCTL_GET_PAGING_MODE_SIZE_IN;
467 Req.Hdr.cbOut = SUP_IOCTL_GET_PAGING_MODE_SIZE_OUT;
468 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
469 Req.Hdr.rc = VERR_INTERNAL_ERROR;
470 int rc = suplibOsIOCtl(SUP_IOCTL_GET_PAGING_MODE, &Req, SUP_IOCTL_GET_PAGING_MODE_SIZE);
471 if ( RT_FAILURE(rc)
472 || RT_FAILURE(Req.Hdr.rc))
473 {
474 LogRel(("SUPGetPagingMode: %Rrc %Rrc\n", rc, Req.Hdr.rc));
475 Req.u.Out.enmMode = SUPPAGINGMODE_INVALID;
476 }
477
478 return Req.u.Out.enmMode;
479}
480
481
482/**
483 * For later.
484 */
485static int supCallVMMR0ExFake(PVMR0 pVMR0, unsigned uOperation, uint64_t u64Arg, PSUPVMMR0REQHDR pReqHdr)
486{
487 AssertMsgFailed(("%d\n", uOperation));
488 return VERR_NOT_SUPPORTED;
489}
490
491
492SUPR3DECL(int) SUPCallVMMR0Fast(PVMR0 pVMR0, unsigned uOperation)
493{
494 if (RT_LIKELY(uOperation == SUP_VMMR0_DO_RAW_RUN))
495 return suplibOsIOCtlFast(SUP_IOCTL_FAST_DO_RAW_RUN);
496 if (RT_LIKELY(uOperation == SUP_VMMR0_DO_HWACC_RUN))
497 return suplibOsIOCtlFast(SUP_IOCTL_FAST_DO_HWACC_RUN);
498 if (RT_LIKELY(uOperation == SUP_VMMR0_DO_NOP))
499 return suplibOsIOCtlFast(SUP_IOCTL_FAST_DO_NOP);
500
501 AssertMsgFailed(("%#x\n", uOperation));
502 return VERR_INTERNAL_ERROR;
503}
504
505
506SUPR3DECL(int) SUPCallVMMR0Ex(PVMR0 pVMR0, unsigned uOperation, uint64_t u64Arg, PSUPVMMR0REQHDR pReqHdr)
507{
508#if 0 /* temp hack. */
509 /*
510 * The following operations don't belong here.
511 */
512 AssertMsgReturn( uOperation != SUP_VMMR0_DO_RAW_RUN
513 && uOperation != SUP_VMMR0_DO_HWACC_RUN
514 && uOperation != SUP_VMMR0_DO_NOP,
515 ("%#x\n", uOperation),
516 VERR_INTERNAL_ERROR);
517#else
518 if ( ( uOperation == SUP_VMMR0_DO_RAW_RUN
519 || uOperation == SUP_VMMR0_DO_HWACC_RUN
520 || uOperation == SUP_VMMR0_DO_NOP)
521 && !pReqHdr
522 && !u64Arg)
523 return (int) SUPCallVMMR0Fast(pVMR0, uOperation);
524#endif
525
526 /* fake */
527 if (RT_UNLIKELY(g_u32FakeMode))
528 return supCallVMMR0ExFake(pVMR0, uOperation, u64Arg, pReqHdr);
529
530 int rc;
531 if (!pReqHdr)
532 {
533 /* no data. */
534 SUPCALLVMMR0 Req;
535 Req.Hdr.u32Cookie = g_u32Cookie;
536 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
537 Req.Hdr.cbIn = SUP_IOCTL_CALL_VMMR0_SIZE_IN(0);
538 Req.Hdr.cbOut = SUP_IOCTL_CALL_VMMR0_SIZE_OUT(0);
539 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
540 Req.Hdr.rc = VERR_INTERNAL_ERROR;
541 Req.u.In.pVMR0 = pVMR0;
542 Req.u.In.uOperation = uOperation;
543 Req.u.In.u64Arg = u64Arg;
544 rc = suplibOsIOCtl(SUP_IOCTL_CALL_VMMR0(0), &Req, SUP_IOCTL_CALL_VMMR0_SIZE(0));
545 if (RT_SUCCESS(rc))
546 rc = Req.Hdr.rc;
547 }
548 else if (SUP_IOCTL_CALL_VMMR0_SIZE(pReqHdr->cbReq) < _4K) /* FreeBSD won't copy more than 4K. */
549 {
550 AssertPtrReturn(pReqHdr, VERR_INVALID_POINTER);
551 AssertReturn(pReqHdr->u32Magic == SUPVMMR0REQHDR_MAGIC, VERR_INVALID_MAGIC);
552 const size_t cbReq = pReqHdr->cbReq;
553
554 PSUPCALLVMMR0 pReq = (PSUPCALLVMMR0)alloca(SUP_IOCTL_CALL_VMMR0_SIZE(cbReq));
555 pReq->Hdr.u32Cookie = g_u32Cookie;
556 pReq->Hdr.u32SessionCookie = g_u32SessionCookie;
557 pReq->Hdr.cbIn = SUP_IOCTL_CALL_VMMR0_SIZE_IN(cbReq);
558 pReq->Hdr.cbOut = SUP_IOCTL_CALL_VMMR0_SIZE_OUT(cbReq);
559 pReq->Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
560 pReq->Hdr.rc = VERR_INTERNAL_ERROR;
561 pReq->u.In.pVMR0 = pVMR0;
562 pReq->u.In.uOperation = uOperation;
563 pReq->u.In.u64Arg = u64Arg;
564 memcpy(&pReq->abReqPkt[0], pReqHdr, cbReq);
565 rc = suplibOsIOCtl(SUP_IOCTL_CALL_VMMR0(cbReq), pReq, SUP_IOCTL_CALL_VMMR0_SIZE(cbReq));
566 if (RT_SUCCESS(rc))
567 rc = pReq->Hdr.rc;
568 memcpy(pReqHdr, &pReq->abReqPkt[0], cbReq);
569 }
570 else /** @todo may have to remove the size limits one this request... */
571 AssertMsgFailedReturn(("cbReq=%#x\n", pReqHdr->cbReq), VERR_INTERNAL_ERROR);
572 return rc;
573}
574
575
576SUPR3DECL(int) SUPCallVMMR0(PVMR0 pVMR0, unsigned uOperation, void *pvArg)
577{
578#if defined(VBOX_WITH_IDT_PATCHING)
579 return g_pfnCallVMMR0(pVMR0, uOperation, pvArg);
580
581#else
582 if (RT_LIKELY(uOperation == SUP_VMMR0_DO_RAW_RUN))
583 {
584 Assert(!pvArg);
585 return suplibOsIOCtlFast(SUP_IOCTL_FAST_DO_RAW_RUN);
586 }
587 if (RT_LIKELY(uOperation == SUP_VMMR0_DO_HWACC_RUN))
588 {
589 Assert(!pvArg);
590 return suplibOsIOCtlFast(SUP_IOCTL_FAST_DO_HWACC_RUN);
591 }
592 if (RT_LIKELY(uOperation == SUP_VMMR0_DO_NOP))
593 {
594 Assert(!pvArg);
595 return suplibOsIOCtlFast(SUP_IOCTL_FAST_DO_NOP);
596 }
597 return SUPCallVMMR0Ex(pVMR0, uOperation, (uintptr_t)pvArg, NULL);
598#endif
599}
600
601
602SUPR3DECL(int) SUPSetVMForFastIOCtl(PVMR0 pVMR0)
603{
604 if (RT_UNLIKELY(g_u32FakeMode))
605 return VINF_SUCCESS;
606
607 SUPSETVMFORFAST Req;
608 Req.Hdr.u32Cookie = g_u32Cookie;
609 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
610 Req.Hdr.cbIn = SUP_IOCTL_SET_VM_FOR_FAST_SIZE_IN;
611 Req.Hdr.cbOut = SUP_IOCTL_SET_VM_FOR_FAST_SIZE_OUT;
612 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
613 Req.Hdr.rc = VERR_INTERNAL_ERROR;
614 Req.u.In.pVMR0 = pVMR0;
615 int rc = suplibOsIOCtl(SUP_IOCTL_SET_VM_FOR_FAST, &Req, SUP_IOCTL_SET_VM_FOR_FAST_SIZE);
616 if (RT_SUCCESS(rc))
617 rc = Req.Hdr.rc;
618 return rc;
619}
620
621
622SUPR3DECL(int) SUPPageAlloc(size_t cPages, void **ppvPages)
623{
624 /*
625 * Validate.
626 */
627 AssertPtrReturn(ppvPages, VERR_INVALID_POINTER);
628 *ppvPages = NULL;
629 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
630
631#ifdef RT_OS_WINDOWS
632 /*
633 * Temporary hack for windows until we've sorted out the
634 * locked memory that doesn't need to be accessible from kernel space.
635 */
636 return SUPPageAllocLockedEx(cPages, ppvPages, NULL);
637#else
638 /*
639 * Call OS specific worker.
640 */
641 return suplibOsPageAlloc(cPages, ppvPages);
642#endif
643}
644
645
646SUPR3DECL(int) SUPPageFree(void *pvPages, size_t cPages)
647{
648 /*
649 * Validate.
650 */
651 AssertPtrReturn(pvPages, VERR_INVALID_POINTER);
652 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
653
654#ifdef RT_OS_WINDOWS
655 /*
656 * Temporary hack for windows, see above.
657 */
658 return SUPPageFreeLocked(pvPages, cPages);
659#else
660 /*
661 * Call OS specific worker.
662 */
663 return suplibOsPageFree(pvPages, cPages);
664#endif
665}
666
667
668SUPR3DECL(int) SUPPageLock(void *pvStart, size_t cPages, PSUPPAGE paPages)
669{
670 /*
671 * Validate.
672 */
673 AssertPtr(pvStart);
674 AssertMsg(RT_ALIGN_P(pvStart, PAGE_SIZE) == pvStart, ("pvStart (%p) must be page aligned\n", pvStart));
675 AssertPtr(paPages);
676
677 /* fake */
678 if (RT_UNLIKELY(g_u32FakeMode))
679 {
680 RTHCPHYS Phys = (uintptr_t)pvStart + PAGE_SIZE * 1024;
681 unsigned iPage = cPages;
682 while (iPage-- > 0)
683 paPages[iPage].Phys = Phys + (iPage << PAGE_SHIFT);
684 return VINF_SUCCESS;
685 }
686
687 /*
688 * Issue IOCtl to the SUPDRV kernel module.
689 */
690 int rc;
691 PSUPPAGELOCK pReq = (PSUPPAGELOCK)RTMemTmpAllocZ(SUP_IOCTL_PAGE_LOCK_SIZE(cPages));
692 if (RT_LIKELY(pReq))
693 {
694 pReq->Hdr.u32Cookie = g_u32Cookie;
695 pReq->Hdr.u32SessionCookie = g_u32SessionCookie;
696 pReq->Hdr.cbIn = SUP_IOCTL_PAGE_LOCK_SIZE_IN;
697 pReq->Hdr.cbOut = SUP_IOCTL_PAGE_LOCK_SIZE_OUT(cPages);
698 pReq->Hdr.fFlags = SUPREQHDR_FLAGS_MAGIC | SUPREQHDR_FLAGS_EXTRA_OUT;
699 pReq->Hdr.rc = VERR_INTERNAL_ERROR;
700 pReq->u.In.pvR3 = pvStart;
701 pReq->u.In.cPages = cPages; AssertRelease(pReq->u.In.cPages == cPages);
702 rc = suplibOsIOCtl(SUP_IOCTL_PAGE_LOCK, pReq, SUP_IOCTL_PAGE_LOCK_SIZE(cPages));
703 if (RT_SUCCESS(rc))
704 rc = pReq->Hdr.rc;
705 if (RT_SUCCESS(rc))
706 {
707 for (uint32_t iPage = 0; iPage < cPages; iPage++)
708 {
709 paPages[iPage].uReserved = 0;
710 paPages[iPage].Phys = pReq->u.Out.aPages[iPage];
711 Assert(!(paPages[iPage].Phys & ~X86_PTE_PAE_PG_MASK));
712 }
713 }
714 RTMemTmpFree(pReq);
715 }
716 else
717 rc = VERR_NO_TMP_MEMORY;
718
719 return rc;
720}
721
722
723SUPR3DECL(int) SUPPageUnlock(void *pvStart)
724{
725 /*
726 * Validate.
727 */
728 AssertPtr(pvStart);
729 AssertMsg(RT_ALIGN_P(pvStart, PAGE_SIZE) == pvStart, ("pvStart (%p) must be page aligned\n", pvStart));
730
731 /* fake */
732 if (RT_UNLIKELY(g_u32FakeMode))
733 return VINF_SUCCESS;
734
735 /*
736 * Issue IOCtl to the SUPDRV kernel module.
737 */
738 SUPPAGEUNLOCK Req;
739 Req.Hdr.u32Cookie = g_u32Cookie;
740 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
741 Req.Hdr.cbIn = SUP_IOCTL_PAGE_UNLOCK_SIZE_IN;
742 Req.Hdr.cbOut = SUP_IOCTL_PAGE_UNLOCK_SIZE_OUT;
743 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
744 Req.Hdr.rc = VERR_INTERNAL_ERROR;
745 Req.u.In.pvR3 = pvStart;
746 int rc = suplibOsIOCtl(SUP_IOCTL_PAGE_UNLOCK, &Req, SUP_IOCTL_PAGE_UNLOCK_SIZE);
747 if (RT_SUCCESS(rc))
748 rc = Req.Hdr.rc;
749 return rc;
750}
751
752
753SUPR3DECL(int) SUPPageAllocLocked(size_t cPages, void **ppvPages)
754{
755 return SUPPageAllocLockedEx(cPages, ppvPages, NULL);
756}
757
758
759/**
760 * Fallback for SUPPageAllocLockedEx on systems where RTR0MemObjPhysAllocNC isn't supported.
761 */
762static int supPageAllocLockedFallback(size_t cPages, void **ppvPages, PSUPPAGE paPages)
763{
764 int rc = suplibOsPageAlloc(cPages, ppvPages);
765 if (RT_SUCCESS(rc))
766 {
767 if (!paPages)
768 paPages = (PSUPPAGE)alloca(sizeof(paPages[0]) * cPages);
769 rc = SUPPageLock(*ppvPages, cPages, paPages);
770 if (RT_FAILURE(rc))
771 suplibOsPageFree(*ppvPages, cPages);
772 }
773 return rc;
774}
775
776
777SUPR3DECL(int) SUPPageAllocLockedEx(size_t cPages, void **ppvPages, PSUPPAGE paPages)
778{
779 /*
780 * Validate.
781 */
782 AssertPtrReturn(ppvPages, VERR_INVALID_POINTER);
783 *ppvPages = NULL;
784 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
785
786 /* fake */
787 if (RT_UNLIKELY(g_u32FakeMode))
788 {
789 *ppvPages = RTMemPageAllocZ((size_t)cPages * PAGE_SIZE);
790 if (!*ppvPages)
791 return VERR_NO_MEMORY;
792 if (paPages)
793 for (size_t iPage = 0; iPage < cPages; iPage++)
794 {
795 paPages[iPage].uReserved = 0;
796 paPages[iPage].Phys = (iPage + 1234) << PAGE_SHIFT;
797 Assert(!(paPages[iPage].Phys & ~X86_PTE_PAE_PG_MASK));
798 }
799 return VINF_SUCCESS;
800 }
801
802 /* use fallback? */
803 if (!g_fSupportsPageAllocLocked)
804 return supPageAllocLockedFallback(cPages, ppvPages, paPages);
805
806 /*
807 * Issue IOCtl to the SUPDRV kernel module.
808 */
809 int rc;
810 PSUPPAGEALLOC pReq = (PSUPPAGEALLOC)RTMemTmpAllocZ(SUP_IOCTL_PAGE_ALLOC_SIZE(cPages));
811 if (pReq)
812 {
813 pReq->Hdr.u32Cookie = g_u32Cookie;
814 pReq->Hdr.u32SessionCookie = g_u32SessionCookie;
815 pReq->Hdr.cbIn = SUP_IOCTL_PAGE_ALLOC_SIZE_IN;
816 pReq->Hdr.cbOut = SUP_IOCTL_PAGE_ALLOC_SIZE_OUT(cPages);
817 pReq->Hdr.fFlags = SUPREQHDR_FLAGS_MAGIC | SUPREQHDR_FLAGS_EXTRA_OUT;
818 pReq->Hdr.rc = VERR_INTERNAL_ERROR;
819 pReq->u.In.cPages = cPages; AssertRelease(pReq->u.In.cPages == cPages);
820 rc = suplibOsIOCtl(SUP_IOCTL_PAGE_ALLOC, pReq, SUP_IOCTL_PAGE_ALLOC_SIZE(cPages));
821 if (RT_SUCCESS(rc))
822 {
823 rc = pReq->Hdr.rc;
824 if (RT_SUCCESS(rc))
825 {
826 *ppvPages = pReq->u.Out.pvR3;
827 if (paPages)
828 for (size_t iPage = 0; iPage < cPages; iPage++)
829 {
830 paPages[iPage].uReserved = 0;
831 paPages[iPage].Phys = pReq->u.Out.aPages[iPage];
832 Assert(!(paPages[iPage].Phys & ~X86_PTE_PAE_PG_MASK));
833 }
834 }
835 else if (rc == VERR_NOT_SUPPORTED)
836 {
837 g_fSupportsPageAllocLocked = false;
838 rc = supPageAllocLockedFallback(cPages, ppvPages, paPages);
839 }
840 }
841
842 RTMemTmpFree(pReq);
843 }
844 else
845 rc = VERR_NO_TMP_MEMORY;
846 return rc;
847}
848
849
850SUPR3DECL(int) SUPPageFreeLocked(void *pvPages, size_t cPages)
851{
852 /*
853 * Validate.
854 */
855 AssertPtrReturn(pvPages, VERR_INVALID_POINTER);
856 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
857
858 /* fake */
859 if (RT_UNLIKELY(g_u32FakeMode))
860 {
861 RTMemPageFree(pvPages);
862 return VINF_SUCCESS;
863 }
864
865 /*
866 * Issue IOCtl to the SUPDRV kernel module.
867 */
868 int rc;
869 if (g_fSupportsPageAllocLocked)
870 {
871 SUPPAGEFREE Req;
872 Req.Hdr.u32Cookie = g_u32Cookie;
873 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
874 Req.Hdr.cbIn = SUP_IOCTL_PAGE_FREE_SIZE_IN;
875 Req.Hdr.cbOut = SUP_IOCTL_PAGE_FREE_SIZE_OUT;
876 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
877 Req.Hdr.rc = VERR_INTERNAL_ERROR;
878 Req.u.In.pvR3 = pvPages;
879 rc = suplibOsIOCtl(SUP_IOCTL_PAGE_FREE, &Req, SUP_IOCTL_PAGE_FREE_SIZE);
880 if (RT_SUCCESS(rc))
881 rc = Req.Hdr.rc;
882 }
883 else
884 {
885 /* fallback */
886 rc = SUPPageUnlock(pvPages);
887 if (RT_SUCCESS(rc))
888 rc = suplibOsPageFree(pvPages, cPages);
889 }
890 return rc;
891}
892
893
894SUPR3DECL(void *) SUPContAlloc(size_t cPages, PRTHCPHYS pHCPhys)
895{
896 return SUPContAlloc2(cPages, NIL_RTR0PTR, pHCPhys);
897}
898
899
900SUPR3DECL(void *) SUPContAlloc2(size_t cPages, PRTR0PTR pR0Ptr, PRTHCPHYS pHCPhys)
901{
902 /*
903 * Validate.
904 */
905 AssertPtrReturn(pHCPhys, NULL);
906 *pHCPhys = NIL_RTHCPHYS;
907 AssertPtrNullReturn(pR0Ptr, NULL);
908 if (pR0Ptr)
909 *pR0Ptr = NIL_RTR0PTR;
910 AssertPtrNullReturn(pHCPhys, NULL);
911 AssertMsgReturn(cPages > 0 && cPages < 256, ("cPages=%d must be > 0 and < 256\n", cPages), NULL);
912
913 /* fake */
914 if (RT_UNLIKELY(g_u32FakeMode))
915 {
916 void *pv = RTMemPageAllocZ(cPages * PAGE_SIZE);
917 if (pR0Ptr)
918 *pR0Ptr = (RTR0PTR)pv;
919 if (pHCPhys)
920 *pHCPhys = (uintptr_t)pv + (PAGE_SHIFT * 1024);
921 return pv;
922 }
923
924 /*
925 * Issue IOCtl to the SUPDRV kernel module.
926 */
927 SUPCONTALLOC Req;
928 Req.Hdr.u32Cookie = g_u32Cookie;
929 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
930 Req.Hdr.cbIn = SUP_IOCTL_CONT_ALLOC_SIZE_IN;
931 Req.Hdr.cbOut = SUP_IOCTL_CONT_ALLOC_SIZE_OUT;
932 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
933 Req.Hdr.rc = VERR_INTERNAL_ERROR;
934 Req.u.In.cPages = cPages;
935 int rc = suplibOsIOCtl(SUP_IOCTL_CONT_ALLOC, &Req, SUP_IOCTL_CONT_ALLOC_SIZE);
936 if ( RT_SUCCESS(rc)
937 && RT_SUCCESS(Req.Hdr.rc))
938 {
939 *pHCPhys = Req.u.Out.HCPhys;
940 if (pR0Ptr)
941 *pR0Ptr = Req.u.Out.pvR0;
942 return Req.u.Out.pvR3;
943 }
944
945 return NULL;
946}
947
948
949SUPR3DECL(int) SUPContFree(void *pv, size_t cPages)
950{
951 /*
952 * Validate.
953 */
954 if (!pv)
955 return VINF_SUCCESS;
956 AssertPtrReturn(pv, VERR_INVALID_POINTER);
957 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
958
959 /* fake */
960 if (RT_UNLIKELY(g_u32FakeMode))
961 {
962 RTMemPageFree(pv);
963 return VINF_SUCCESS;
964 }
965
966 /*
967 * Issue IOCtl to the SUPDRV kernel module.
968 */
969 SUPCONTFREE Req;
970 Req.Hdr.u32Cookie = g_u32Cookie;
971 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
972 Req.Hdr.cbIn = SUP_IOCTL_CONT_FREE_SIZE_IN;
973 Req.Hdr.cbOut = SUP_IOCTL_CONT_FREE_SIZE_OUT;
974 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
975 Req.Hdr.rc = VERR_INTERNAL_ERROR;
976 Req.u.In.pvR3 = pv;
977 int rc = suplibOsIOCtl(SUP_IOCTL_CONT_FREE, &Req, SUP_IOCTL_CONT_FREE_SIZE);
978 if (RT_SUCCESS(rc))
979 rc = Req.Hdr.rc;
980 return rc;
981}
982
983
984SUPR3DECL(int) SUPLowAlloc(size_t cPages, void **ppvPages, PRTR0PTR ppvPagesR0, PSUPPAGE paPages)
985{
986 /*
987 * Validate.
988 */
989 AssertPtrReturn(ppvPages, VERR_INVALID_POINTER);
990 *ppvPages = NULL;
991 AssertPtrReturn(paPages, VERR_INVALID_POINTER);
992 AssertMsgReturn(cPages > 0 && cPages < 256, ("cPages=%d must be > 0 and < 256\n", cPages), VERR_INVALID_PARAMETER);
993
994 /* fake */
995 if (RT_UNLIKELY(g_u32FakeMode))
996 {
997 *ppvPages = RTMemPageAllocZ((size_t)cPages * PAGE_SIZE);
998 if (!*ppvPages)
999 return VERR_NO_LOW_MEMORY;
1000
1001 /* fake physical addresses. */
1002 RTHCPHYS Phys = (uintptr_t)*ppvPages + PAGE_SIZE * 1024;
1003 unsigned iPage = cPages;
1004 while (iPage-- > 0)
1005 paPages[iPage].Phys = Phys + (iPage << PAGE_SHIFT);
1006 return VINF_SUCCESS;
1007 }
1008
1009 /*
1010 * Issue IOCtl to the SUPDRV kernel module.
1011 */
1012 int rc;
1013 PSUPLOWALLOC pReq = (PSUPLOWALLOC)RTMemTmpAllocZ(SUP_IOCTL_LOW_ALLOC_SIZE(cPages));
1014 if (pReq)
1015 {
1016 pReq->Hdr.u32Cookie = g_u32Cookie;
1017 pReq->Hdr.u32SessionCookie = g_u32SessionCookie;
1018 pReq->Hdr.cbIn = SUP_IOCTL_LOW_ALLOC_SIZE_IN;
1019 pReq->Hdr.cbOut = SUP_IOCTL_LOW_ALLOC_SIZE_OUT(cPages);
1020 pReq->Hdr.fFlags = SUPREQHDR_FLAGS_MAGIC | SUPREQHDR_FLAGS_EXTRA_OUT;
1021 pReq->Hdr.rc = VERR_INTERNAL_ERROR;
1022 pReq->u.In.cPages = cPages; AssertRelease(pReq->u.In.cPages == cPages);
1023 rc = suplibOsIOCtl(SUP_IOCTL_LOW_ALLOC, pReq, SUP_IOCTL_LOW_ALLOC_SIZE(cPages));
1024 if (RT_SUCCESS(rc))
1025 rc = pReq->Hdr.rc;
1026 if (RT_SUCCESS(rc))
1027 {
1028 *ppvPages = pReq->u.Out.pvR3;
1029 if (ppvPagesR0)
1030 *ppvPagesR0 = pReq->u.Out.pvR0;
1031 if (paPages)
1032 for (size_t iPage = 0; iPage < cPages; iPage++)
1033 {
1034 paPages[iPage].uReserved = 0;
1035 paPages[iPage].Phys = pReq->u.Out.aPages[iPage];
1036 Assert(!(paPages[iPage].Phys & ~X86_PTE_PAE_PG_MASK));
1037 Assert(paPages[iPage].Phys <= UINT32_C(0xfffff000));
1038 }
1039 }
1040 RTMemTmpFree(pReq);
1041 }
1042 else
1043 rc = VERR_NO_TMP_MEMORY;
1044
1045 return rc;
1046}
1047
1048
1049SUPR3DECL(int) SUPLowFree(void *pv, size_t cPages)
1050{
1051 /*
1052 * Validate.
1053 */
1054 if (!pv)
1055 return VINF_SUCCESS;
1056 AssertPtrReturn(pv, VERR_INVALID_POINTER);
1057 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
1058
1059 /* fake */
1060 if (RT_UNLIKELY(g_u32FakeMode))
1061 {
1062 RTMemPageFree(pv);
1063 return VINF_SUCCESS;
1064 }
1065
1066 /*
1067 * Issue IOCtl to the SUPDRV kernel module.
1068 */
1069 SUPCONTFREE Req;
1070 Req.Hdr.u32Cookie = g_u32Cookie;
1071 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
1072 Req.Hdr.cbIn = SUP_IOCTL_LOW_FREE_SIZE_IN;
1073 Req.Hdr.cbOut = SUP_IOCTL_LOW_FREE_SIZE_OUT;
1074 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1075 Req.Hdr.rc = VERR_INTERNAL_ERROR;
1076 Req.u.In.pvR3 = pv;
1077 int rc = suplibOsIOCtl(SUP_IOCTL_LOW_FREE, &Req, SUP_IOCTL_LOW_FREE_SIZE);
1078 if (RT_SUCCESS(rc))
1079 rc = Req.Hdr.rc;
1080 return rc;
1081}
1082
1083
1084SUPR3DECL(int) SUPLoadModule(const char *pszFilename, const char *pszModule, void **ppvImageBase)
1085{
1086 /*
1087 * Load the module.
1088 * If it's VMMR0.r0 we need to install the IDTE.
1089 */
1090 int rc = supLoadModule(pszFilename, pszModule, ppvImageBase);
1091#ifdef VBOX_WITH_IDT_PATCHING
1092 if ( RT_SUCCESS(rc)
1093 && !strcmp(pszModule, "VMMR0.r0"))
1094 {
1095 rc = supInstallIDTE();
1096 if (RT_FAILURE(rc))
1097 SUPFreeModule(*ppvImageBase);
1098 }
1099#endif /* VBOX_WITH_IDT_PATCHING */
1100
1101 return rc;
1102}
1103
1104
1105#ifdef VBOX_WITH_IDT_PATCHING
1106/**
1107 * Generates the code for calling the interrupt gate.
1108 *
1109 * @returns VBox status code.
1110 * g_pfnCallVMMR0 is changed on success.
1111 * @param u8Interrupt The interrupt number.
1112 */
1113static int suplibGenerateCallVMMR0(uint8_t u8Interrupt)
1114{
1115 /*
1116 * Allocate memory.
1117 */
1118 uint8_t *pb = (uint8_t *)RTMemExecAlloc(256);
1119 AssertReturn(pb, VERR_NO_MEMORY);
1120 memset(pb, 0xcc, 256);
1121 Assert(!g_pfnCallVMMR0);
1122 g_pfnCallVMMR0 = *(PFNCALLVMMR0*)&pb;
1123
1124 /*
1125 * Generate the code.
1126 */
1127#ifdef RT_ARCH_AMD64
1128 /*
1129 * reg params:
1130 * <GCC> <MSC> <argument>
1131 * rdi rcx pVMR0
1132 * esi edx uOperation
1133 * rdx r8 pvArg
1134 *
1135 * eax eax [g_u32Gookie]
1136 */
1137 *pb++ = 0xb8; /* mov eax, <g_u32Cookie> */
1138 *(uint32_t *)pb = g_u32Cookie;
1139 pb += sizeof(uint32_t);
1140
1141 *pb++ = 0xcd; /* int <u8Interrupt> */
1142 *pb++ = u8Interrupt;
1143
1144 *pb++ = 0xc3; /* ret */
1145
1146#else
1147 /*
1148 * x86 stack:
1149 * 0 saved esi
1150 * 0 4 ret
1151 * 4 8 pVM
1152 * 8 c uOperation
1153 * c 10 pvArg
1154 */
1155 *pb++ = 0x56; /* push esi */
1156
1157 *pb++ = 0x8b; /* mov eax, [pVM] */
1158 *pb++ = 0x44;
1159 *pb++ = 0x24;
1160 *pb++ = 0x08; /* esp+08h */
1161
1162 *pb++ = 0x8b; /* mov edx, [uOperation] */
1163 *pb++ = 0x54;
1164 *pb++ = 0x24;
1165 *pb++ = 0x0c; /* esp+0ch */
1166
1167 *pb++ = 0x8b; /* mov ecx, [pvArg] */
1168 *pb++ = 0x4c;
1169 *pb++ = 0x24;
1170 *pb++ = 0x10; /* esp+10h */
1171
1172 *pb++ = 0xbe; /* mov esi, <g_u32Cookie> */
1173 *(uint32_t *)pb = g_u32Cookie;
1174 pb += sizeof(uint32_t);
1175
1176 *pb++ = 0xcd; /* int <u8Interrupt> */
1177 *pb++ = u8Interrupt;
1178
1179 *pb++ = 0x5e; /* pop esi */
1180
1181 *pb++ = 0xc3; /* ret */
1182#endif
1183
1184 return VINF_SUCCESS;
1185}
1186
1187
1188/**
1189 * Installs the IDTE patch.
1190 *
1191 * @return VBox status code.
1192 */
1193static int supInstallIDTE(void)
1194{
1195 /* already installed? */
1196 if (g_u8Interrupt != 3 || g_u32FakeMode)
1197 return VINF_SUCCESS;
1198
1199 int rc = VINF_SUCCESS;
1200 const unsigned cCpus = RTSystemProcessorGetCount();
1201 if (cCpus <= 1)
1202 {
1203 /* UNI */
1204 SUPIDTINSTALL Req;
1205 Req.Hdr.u32Cookie = g_u32Cookie;
1206 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
1207 Req.Hdr.cbIn = SUP_IOCTL_IDT_INSTALL_SIZE_IN;
1208 Req.Hdr.cbOut = SUP_IOCTL_IDT_INSTALL_SIZE_OUT;
1209 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1210 Req.Hdr.rc = VERR_INTERNAL_ERROR;
1211 rc = suplibOsIOCtl(SUP_IOCTL_IDT_INSTALL, &Req, SUP_IOCTL_IDT_INSTALL_SIZE);
1212 if (RT_SUCCESS(rc))
1213 rc = Req.Hdr.rc;
1214 if (RT_SUCCESS(rc))
1215 {
1216 g_u8Interrupt = Req.u.Out.u8Idt;
1217 rc = suplibGenerateCallVMMR0(Req.u.Out.u8Idt);
1218 }
1219 }
1220 else
1221 {
1222 /* SMP */
1223 uint64_t u64AffMaskSaved = RTThreadGetAffinity();
1224 uint64_t u64AffMaskPatched = RTSystemProcessorGetActiveMask() & u64AffMaskSaved;
1225 unsigned cCpusPatched = 0;
1226
1227 for (int i = 0; i < 64; i++)
1228 {
1229 /* Skip absent and inactive processors. */
1230 uint64_t u64Mask = 1ULL << i;
1231 if (!(u64Mask & u64AffMaskPatched))
1232 continue;
1233
1234 /* Change CPU */
1235 int rc2 = RTThreadSetAffinity(u64Mask);
1236 if (RT_FAILURE(rc2))
1237 {
1238 u64AffMaskPatched &= ~u64Mask;
1239 LogRel(("SUPLoadVMM: Failed to set affinity to cpu no. %d, rc=%Vrc.\n", i, rc2));
1240 continue;
1241 }
1242
1243 /* Patch the CPU. */
1244 SUPIDTINSTALL Req;
1245 Req.Hdr.u32Cookie = g_u32Cookie;
1246 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
1247 Req.Hdr.cbIn = SUP_IOCTL_IDT_INSTALL_SIZE_IN;
1248 Req.Hdr.cbOut = SUP_IOCTL_IDT_INSTALL_SIZE_OUT;
1249 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1250 Req.Hdr.rc = VERR_INTERNAL_ERROR;
1251 rc2 = suplibOsIOCtl(SUP_IOCTL_IDT_INSTALL, &Req, SUP_IOCTL_IDT_INSTALL_SIZE);
1252 if (RT_SUCCESS(rc2))
1253 rc2 = Req.Hdr.rc;
1254 if (RT_SUCCESS(rc2))
1255 {
1256 if (!cCpusPatched)
1257 {
1258 g_u8Interrupt = Req.u.Out.u8Idt;
1259 rc2 = suplibGenerateCallVMMR0(Req.u.Out.u8Idt);
1260 if (RT_FAILURE(rc2))
1261 {
1262 LogRel(("suplibGenerateCallVMMR0 failed with rc=%Vrc.\n", i, rc2));
1263 rc = rc2;
1264 }
1265 }
1266 else
1267 Assert(g_u8Interrupt == Req.u.Out.u8Idt);
1268 cCpusPatched++;
1269 }
1270 else
1271 {
1272
1273 LogRel(("SUPLoadVMM: Failed to patch cpu no. %d, rc=%Vrc.\n", i, rc2));
1274 if (RT_SUCCESS(rc))
1275 rc = rc2;
1276 }
1277 }
1278
1279 /* Fail if no CPUs was patched! */
1280 if (RT_SUCCESS(rc) && cCpusPatched <= 0)
1281 rc = VERR_GENERAL_FAILURE;
1282 /* Ignore failures if a CPU was patched. */
1283 else if (RT_FAILURE(rc) && cCpusPatched > 0)
1284 rc = VINF_SUCCESS;
1285
1286 /* Set/restore the thread affinity. */
1287 if (RT_SUCCESS(rc))
1288 {
1289 rc = RTThreadSetAffinity(u64AffMaskPatched);
1290 AssertRC(rc);
1291 }
1292 else
1293 {
1294 int rc2 = RTThreadSetAffinity(u64AffMaskSaved);
1295 AssertRC(rc2);
1296 }
1297 }
1298 return rc;
1299}
1300#endif /* VBOX_WITH_IDT_PATCHING */
1301
1302
1303/**
1304 * Resolve an external symbol during RTLdrGetBits().
1305 *
1306 * @returns VBox status code.
1307 * @param hLdrMod The loader module handle.
1308 * @param pszModule Module name.
1309 * @param pszSymbol Symbol name, NULL if uSymbol should be used.
1310 * @param uSymbol Symbol ordinal, ~0 if pszSymbol should be used.
1311 * @param pValue Where to store the symbol value (address).
1312 * @param pvUser User argument.
1313 */
1314static DECLCALLBACK(int) supLoadModuleResolveImport(RTLDRMOD hLdrMod, const char *pszModule,
1315 const char *pszSymbol, unsigned uSymbol, RTUINTPTR *pValue, void *pvUser)
1316{
1317 AssertPtr(pValue);
1318 AssertPtr(pvUser);
1319
1320 /*
1321 * Only SUPR0 and VMMR0.r0
1322 */
1323 if ( pszModule
1324 && *pszModule
1325 && strcmp(pszModule, "SUPR0.dll")
1326 && strcmp(pszModule, "VMMR0.r0"))
1327 {
1328 AssertMsgFailed(("%s is importing from %s! (expected 'SUPR0.dll' or 'VMMR0.r0', case-sensitiv)\n", pvUser, pszModule));
1329 return VERR_SYMBOL_NOT_FOUND;
1330 }
1331
1332 /*
1333 * No ordinals.
1334 */
1335 if (pszSymbol < (const char*)0x10000)
1336 {
1337 AssertMsgFailed(("%s is importing by ordinal (ord=%d)\n", pvUser, (int)(uintptr_t)pszSymbol));
1338 return VERR_SYMBOL_NOT_FOUND;
1339 }
1340
1341 /*
1342 * Lookup symbol.
1343 */
1344 /* skip the 64-bit ELF import prefix first. */
1345 if (!strncmp(pszSymbol, "SUPR0$", sizeof("SUPR0$") - 1))
1346 pszSymbol += sizeof("SUPR0$") - 1;
1347
1348 /*
1349 * Check the VMMR0.r0 module if loaded.
1350 */
1351 /** @todo call the SUPLoadModule caller.... */
1352 /** @todo proper reference counting and such. */
1353 if (g_pvVMMR0 != NIL_RTR0PTR)
1354 {
1355 void *pvValue;
1356 if (!SUPGetSymbolR0((void *)g_pvVMMR0, pszSymbol, &pvValue))
1357 {
1358 *pValue = (uintptr_t)pvValue;
1359 return VINF_SUCCESS;
1360 }
1361 }
1362
1363 /* iterate the function table. */
1364 int c = g_pFunctions->u.Out.cFunctions;
1365 PSUPFUNC pFunc = &g_pFunctions->u.Out.aFunctions[0];
1366 while (c-- > 0)
1367 {
1368 if (!strcmp(pFunc->szName, pszSymbol))
1369 {
1370 *pValue = (uintptr_t)pFunc->pfn;
1371 return VINF_SUCCESS;
1372 }
1373 pFunc++;
1374 }
1375
1376 /*
1377 * The GIP.
1378 */
1379 /** @todo R0 mapping? */
1380 if ( pszSymbol
1381 && g_pSUPGlobalInfoPage
1382 && g_pSUPGlobalInfoPageR0
1383 && !strcmp(pszSymbol, "g_SUPGlobalInfoPage"))
1384 {
1385 *pValue = (uintptr_t)g_pSUPGlobalInfoPageR0;
1386 return VINF_SUCCESS;
1387 }
1388
1389 /*
1390 * Despair.
1391 */
1392 c = g_pFunctions->u.Out.cFunctions;
1393 pFunc = &g_pFunctions->u.Out.aFunctions[0];
1394 while (c-- > 0)
1395 {
1396 AssertMsg2("%d: %s\n", g_pFunctions->u.Out.cFunctions - c, pFunc->szName);
1397 pFunc++;
1398 }
1399
1400 AssertMsg2("%s is importing %s which we couldn't find\n", pvUser, pszSymbol);
1401 AssertMsgFailed(("%s is importing %s which we couldn't find\n", pvUser, pszSymbol));
1402 if (g_u32FakeMode)
1403 {
1404 *pValue = 0xdeadbeef;
1405 return VINF_SUCCESS;
1406 }
1407 return VERR_SYMBOL_NOT_FOUND;
1408}
1409
1410
1411/** Argument package for supLoadModuleCalcSizeCB. */
1412typedef struct SUPLDRCALCSIZEARGS
1413{
1414 size_t cbStrings;
1415 uint32_t cSymbols;
1416 size_t cbImage;
1417} SUPLDRCALCSIZEARGS, *PSUPLDRCALCSIZEARGS;
1418
1419/**
1420 * Callback used to calculate the image size.
1421 * @return VINF_SUCCESS
1422 */
1423static DECLCALLBACK(int) supLoadModuleCalcSizeCB(RTLDRMOD hLdrMod, const char *pszSymbol, unsigned uSymbol, RTUINTPTR Value, void *pvUser)
1424{
1425 PSUPLDRCALCSIZEARGS pArgs = (PSUPLDRCALCSIZEARGS)pvUser;
1426 if ( pszSymbol != NULL
1427 && *pszSymbol
1428 && Value <= pArgs->cbImage)
1429 {
1430 pArgs->cSymbols++;
1431 pArgs->cbStrings += strlen(pszSymbol) + 1;
1432 }
1433 return VINF_SUCCESS;
1434}
1435
1436
1437/** Argument package for supLoadModuleCreateTabsCB. */
1438typedef struct SUPLDRCREATETABSARGS
1439{
1440 size_t cbImage;
1441 PSUPLDRSYM pSym;
1442 char *pszBase;
1443 char *psz;
1444} SUPLDRCREATETABSARGS, *PSUPLDRCREATETABSARGS;
1445
1446/**
1447 * Callback used to calculate the image size.
1448 * @return VINF_SUCCESS
1449 */
1450static DECLCALLBACK(int) supLoadModuleCreateTabsCB(RTLDRMOD hLdrMod, const char *pszSymbol, unsigned uSymbol, RTUINTPTR Value, void *pvUser)
1451{
1452 PSUPLDRCREATETABSARGS pArgs = (PSUPLDRCREATETABSARGS)pvUser;
1453 if ( pszSymbol != NULL
1454 && *pszSymbol
1455 && Value <= pArgs->cbImage)
1456 {
1457 pArgs->pSym->offSymbol = (uint32_t)Value;
1458 pArgs->pSym->offName = pArgs->psz - pArgs->pszBase;
1459 pArgs->pSym++;
1460
1461 size_t cbCopy = strlen(pszSymbol) + 1;
1462 memcpy(pArgs->psz, pszSymbol, cbCopy);
1463 pArgs->psz += cbCopy;
1464 }
1465 return VINF_SUCCESS;
1466}
1467
1468
1469/**
1470 * Worker for SUPLoadModule().
1471 *
1472 * @returns VBox status code.
1473 * @param pszFilename Name of the VMMR0 image file
1474 */
1475static int supLoadModule(const char *pszFilename, const char *pszModule, void **ppvImageBase)
1476{
1477 /*
1478 * Validate input.
1479 */
1480 AssertPtrReturn(pszFilename, VERR_INVALID_PARAMETER);
1481 AssertPtrReturn(pszModule, VERR_INVALID_PARAMETER);
1482 AssertPtrReturn(ppvImageBase, VERR_INVALID_PARAMETER);
1483 AssertReturn(strlen(pszModule) < RT_SIZEOFMEMB(SUPLDROPEN, u.In.szName), VERR_FILENAME_TOO_LONG);
1484
1485 const bool fIsVMMR0 = !strcmp(pszModule, "VMMR0.r0");
1486 *ppvImageBase = NULL;
1487
1488 /*
1489 * Open image file and figure its size.
1490 */
1491 RTLDRMOD hLdrMod;
1492 int rc = RTLdrOpen(pszFilename, &hLdrMod);
1493 if (!RT_SUCCESS(rc))
1494 return rc;
1495
1496 SUPLDRCALCSIZEARGS CalcArgs;
1497 CalcArgs.cbStrings = 0;
1498 CalcArgs.cSymbols = 0;
1499 CalcArgs.cbImage = RTLdrSize(hLdrMod);
1500 rc = RTLdrEnumSymbols(hLdrMod, 0, NULL, 0, supLoadModuleCalcSizeCB, &CalcArgs);
1501 if (RT_SUCCESS(rc))
1502 {
1503 const uint32_t offSymTab = RT_ALIGN_32(CalcArgs.cbImage, 8);
1504 const uint32_t offStrTab = offSymTab + CalcArgs.cSymbols * sizeof(SUPLDRSYM);
1505 const uint32_t cbImage = RT_ALIGN_32(offStrTab + CalcArgs.cbStrings, 8);
1506
1507 /*
1508 * Open the R0 image.
1509 */
1510 SUPLDROPEN OpenReq;
1511 OpenReq.Hdr.u32Cookie = g_u32Cookie;
1512 OpenReq.Hdr.u32SessionCookie = g_u32SessionCookie;
1513 OpenReq.Hdr.cbIn = SUP_IOCTL_LDR_OPEN_SIZE_IN;
1514 OpenReq.Hdr.cbOut = SUP_IOCTL_LDR_OPEN_SIZE_OUT;
1515 OpenReq.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1516 OpenReq.Hdr.rc = VERR_INTERNAL_ERROR;
1517 OpenReq.u.In.cbImage = cbImage;
1518 strcpy(OpenReq.u.In.szName, pszModule);
1519 if (!g_u32FakeMode)
1520 {
1521 rc = suplibOsIOCtl(SUP_IOCTL_LDR_OPEN, &OpenReq, SUP_IOCTL_LDR_OPEN_SIZE);
1522 if (RT_SUCCESS(rc))
1523 rc = OpenReq.Hdr.rc;
1524 }
1525 else
1526 {
1527 OpenReq.u.Out.fNeedsLoading = true;
1528 OpenReq.u.Out.pvImageBase = 0xef423420;
1529 }
1530 *ppvImageBase = (void *)OpenReq.u.Out.pvImageBase;
1531 if ( RT_SUCCESS(rc)
1532 && OpenReq.u.Out.fNeedsLoading)
1533 {
1534 /*
1535 * We need to load it.
1536 * Allocate memory for the image bits.
1537 */
1538 PSUPLDRLOAD pLoadReq = (PSUPLDRLOAD)RTMemTmpAlloc(SUP_IOCTL_LDR_LOAD_SIZE(cbImage));
1539 if (pLoadReq)
1540 {
1541 /*
1542 * Get the image bits.
1543 */
1544 rc = RTLdrGetBits(hLdrMod, &pLoadReq->u.In.achImage[0], (uintptr_t)OpenReq.u.Out.pvImageBase,
1545 supLoadModuleResolveImport, (void *)pszModule);
1546
1547 if (RT_SUCCESS(rc))
1548 {
1549 /*
1550 * Get the entry points.
1551 */
1552 RTUINTPTR VMMR0EntryInt = 0;
1553 RTUINTPTR VMMR0EntryFast = 0;
1554 RTUINTPTR VMMR0EntryEx = 0;
1555 RTUINTPTR ModuleInit = 0;
1556 RTUINTPTR ModuleTerm = 0;
1557 if (fIsVMMR0)
1558 {
1559 rc = RTLdrGetSymbolEx(hLdrMod, &pLoadReq->u.In.achImage[0], (uintptr_t)OpenReq.u.Out.pvImageBase, "VMMR0EntryInt", &VMMR0EntryInt);
1560 if (RT_SUCCESS(rc))
1561 rc = RTLdrGetSymbolEx(hLdrMod, &pLoadReq->u.In.achImage[0], (uintptr_t)OpenReq.u.Out.pvImageBase, "VMMR0EntryFast", &VMMR0EntryFast);
1562 if (RT_SUCCESS(rc))
1563 rc = RTLdrGetSymbolEx(hLdrMod, &pLoadReq->u.In.achImage[0], (uintptr_t)OpenReq.u.Out.pvImageBase, "VMMR0EntryEx", &VMMR0EntryEx);
1564 }
1565 if (RT_SUCCESS(rc))
1566 {
1567 int rc2 = RTLdrGetSymbolEx(hLdrMod, &pLoadReq->u.In.achImage[0], (uintptr_t)OpenReq.u.Out.pvImageBase, "ModuleInit", &ModuleInit);
1568 if (RT_FAILURE(rc2))
1569 ModuleInit = 0;
1570
1571 rc2 = RTLdrGetSymbolEx(hLdrMod, &pLoadReq->u.In.achImage[0], (uintptr_t)OpenReq.u.Out.pvImageBase, "ModuleTerm", &ModuleTerm);
1572 if (RT_FAILURE(rc2))
1573 ModuleTerm = 0;
1574 }
1575 if (RT_SUCCESS(rc))
1576 {
1577 /*
1578 * Create the symbol and string tables.
1579 */
1580 SUPLDRCREATETABSARGS CreateArgs;
1581 CreateArgs.cbImage = CalcArgs.cbImage;
1582 CreateArgs.pSym = (PSUPLDRSYM)&pLoadReq->u.In.achImage[offSymTab];
1583 CreateArgs.pszBase = (char *)&pLoadReq->u.In.achImage[offStrTab];
1584 CreateArgs.psz = CreateArgs.pszBase;
1585 rc = RTLdrEnumSymbols(hLdrMod, 0, NULL, 0, supLoadModuleCreateTabsCB, &CreateArgs);
1586 if (RT_SUCCESS(rc))
1587 {
1588 AssertRelease((size_t)(CreateArgs.psz - CreateArgs.pszBase) <= CalcArgs.cbStrings);
1589 AssertRelease((size_t)(CreateArgs.pSym - (PSUPLDRSYM)&pLoadReq->u.In.achImage[offSymTab]) <= CalcArgs.cSymbols);
1590
1591 /*
1592 * Upload the image.
1593 */
1594 pLoadReq->Hdr.u32Cookie = g_u32Cookie;
1595 pLoadReq->Hdr.u32SessionCookie = g_u32SessionCookie;
1596 pLoadReq->Hdr.cbIn = SUP_IOCTL_LDR_LOAD_SIZE_IN(cbImage);
1597 pLoadReq->Hdr.cbOut = SUP_IOCTL_LDR_LOAD_SIZE_OUT;
1598 pLoadReq->Hdr.fFlags = SUPREQHDR_FLAGS_MAGIC | SUPREQHDR_FLAGS_EXTRA_IN;
1599 pLoadReq->Hdr.rc = VERR_INTERNAL_ERROR;
1600
1601 pLoadReq->u.In.pfnModuleInit = (RTR0PTR)ModuleInit;
1602 pLoadReq->u.In.pfnModuleTerm = (RTR0PTR)ModuleTerm;
1603 if (fIsVMMR0)
1604 {
1605 pLoadReq->u.In.eEPType = SUPLDRLOADEP_VMMR0;
1606 pLoadReq->u.In.EP.VMMR0.pvVMMR0 = OpenReq.u.Out.pvImageBase;
1607 pLoadReq->u.In.EP.VMMR0.pvVMMR0EntryInt = (RTR0PTR)VMMR0EntryInt;
1608 pLoadReq->u.In.EP.VMMR0.pvVMMR0EntryFast= (RTR0PTR)VMMR0EntryFast;
1609 pLoadReq->u.In.EP.VMMR0.pvVMMR0EntryEx = (RTR0PTR)VMMR0EntryEx;
1610 }
1611 else
1612 pLoadReq->u.In.eEPType = SUPLDRLOADEP_NOTHING;
1613 pLoadReq->u.In.offStrTab = offStrTab;
1614 pLoadReq->u.In.cbStrTab = (uint32_t)CalcArgs.cbStrings;
1615 AssertRelease(pLoadReq->u.In.cbStrTab == CalcArgs.cbStrings);
1616 pLoadReq->u.In.offSymbols = offSymTab;
1617 pLoadReq->u.In.cSymbols = CalcArgs.cSymbols;
1618 pLoadReq->u.In.cbImage = cbImage;
1619 pLoadReq->u.In.pvImageBase = OpenReq.u.Out.pvImageBase;
1620 if (!g_u32FakeMode)
1621 {
1622 rc = suplibOsIOCtl(SUP_IOCTL_LDR_LOAD, pLoadReq, SUP_IOCTL_LDR_LOAD_SIZE(cbImage));
1623 if (RT_SUCCESS(rc))
1624 rc = pLoadReq->Hdr.rc;
1625 }
1626 else
1627 rc = VINF_SUCCESS;
1628 if ( RT_SUCCESS(rc)
1629 || rc == VERR_ALREADY_LOADED /* A competing process. */
1630 )
1631 {
1632 LogRel(("SUP: Loaded %s (%s) at %#p - ModuleInit at %RTptr and ModuleTerm at %RTptr\n", pszModule, pszFilename,
1633 OpenReq.u.Out.pvImageBase, ModuleInit, ModuleTerm));
1634 if (fIsVMMR0)
1635 {
1636 g_pvVMMR0 = OpenReq.u.Out.pvImageBase;
1637 LogRel(("SUP: VMMR0EntryEx located at %RTptr, VMMR0EntryFast at %RTptr and VMMR0EntryInt at %RTptr\n",
1638 VMMR0EntryEx, VMMR0EntryFast, VMMR0EntryInt));
1639 }
1640#ifdef RT_OS_WINDOWS
1641 LogRel(("SUP: windbg> .reload /f %s=%#p\n", pszFilename, OpenReq.u.Out.pvImageBase));
1642#endif
1643
1644 RTMemTmpFree(pLoadReq);
1645 RTLdrClose(hLdrMod);
1646 return VINF_SUCCESS;
1647 }
1648 }
1649 }
1650 }
1651 RTMemTmpFree(pLoadReq);
1652 }
1653 else
1654 {
1655 AssertMsgFailed(("failed to allocated %d bytes for SUPLDRLOAD_IN structure!\n", SUP_IOCTL_LDR_LOAD_SIZE(cbImage)));
1656 rc = VERR_NO_TMP_MEMORY;
1657 }
1658 }
1659 else if (RT_SUCCESS(rc))
1660 {
1661 if (fIsVMMR0)
1662 g_pvVMMR0 = OpenReq.u.Out.pvImageBase;
1663 LogRel(("SUP: Opened %s (%s) at %#p.\n", pszModule, pszFilename, OpenReq.u.Out.pvImageBase));
1664#ifdef RT_OS_WINDOWS
1665 LogRel(("SUP: windbg> .reload /f %s=%#p\n", pszFilename, OpenReq.u.Out.pvImageBase));
1666#endif
1667 }
1668 }
1669 RTLdrClose(hLdrMod);
1670 return rc;
1671}
1672
1673
1674SUPR3DECL(int) SUPFreeModule(void *pvImageBase)
1675{
1676 /* fake */
1677 if (RT_UNLIKELY(g_u32FakeMode))
1678 {
1679#ifdef VBOX_WITH_IDT_PATCHING
1680 g_u8Interrupt = 3;
1681 RTMemExecFree(*(void **)&g_pfnCallVMMR0);
1682 g_pfnCallVMMR0 = NULL;
1683#endif
1684 g_pvVMMR0 = NIL_RTR0PTR;
1685 return VINF_SUCCESS;
1686 }
1687
1688#ifdef VBOX_WITH_IDT_PATCHING
1689 /*
1690 * There is one special module. When this is freed we'll
1691 * free the IDT entry that goes with it.
1692 *
1693 * Note that we don't keep count of VMMR0.r0 loads here, so the
1694 * first unload will free it.
1695 */
1696 if ( (RTR0PTR)pvImageBase == g_pvVMMR0
1697 && g_u8Interrupt != 3)
1698 {
1699 SUPIDTREMOVE Req;
1700 Req.Hdr.u32Cookie = g_u32Cookie;
1701 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
1702 Req.Hdr.cbIn = SUP_IOCTL_IDT_REMOVE_SIZE_IN;
1703 Req.Hdr.cbOut = SUP_IOCTL_IDT_REMOVE_SIZE_OUT;
1704 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1705 Req.Hdr.rc = VERR_INTERNAL_ERROR;
1706 int rc = suplibOsIOCtl(SUP_IOCTL_IDT_REMOVE, &Req, SUP_IOCTL_IDT_REMOVE_SIZE);
1707 if (RT_SUCCESS(rc))
1708 rc = Req.Hdr.rc;
1709 AssertRC(rc);
1710 g_u8Interrupt = 3;
1711 RTMemExecFree(*(void **)&g_pfnCallVMMR0);
1712 g_pfnCallVMMR0 = NULL;
1713 }
1714#endif /* VBOX_WITH_IDT_PATCHING */
1715
1716 /*
1717 * Free the requested module.
1718 */
1719 SUPLDRFREE Req;
1720 Req.Hdr.u32Cookie = g_u32Cookie;
1721 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
1722 Req.Hdr.cbIn = SUP_IOCTL_LDR_FREE_SIZE_IN;
1723 Req.Hdr.cbOut = SUP_IOCTL_LDR_FREE_SIZE_OUT;
1724 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1725 Req.Hdr.rc = VERR_INTERNAL_ERROR;
1726 Req.u.In.pvImageBase = (RTR0PTR)pvImageBase;
1727 int rc = suplibOsIOCtl(SUP_IOCTL_LDR_FREE, &Req, SUP_IOCTL_LDR_FREE_SIZE);
1728 if (RT_SUCCESS(rc))
1729 rc = Req.Hdr.rc;
1730 if ( RT_SUCCESS(rc)
1731 && (RTR0PTR)pvImageBase == g_pvVMMR0)
1732 g_pvVMMR0 = NIL_RTR0PTR;
1733 return rc;
1734}
1735
1736
1737SUPR3DECL(int) SUPGetSymbolR0(void *pvImageBase, const char *pszSymbol, void **ppvValue)
1738{
1739 *ppvValue = NULL;
1740
1741 /* fake */
1742 if (RT_UNLIKELY(g_u32FakeMode))
1743 {
1744 *ppvValue = (void *)(uintptr_t)0xdeadf00d;
1745 return VINF_SUCCESS;
1746 }
1747
1748 /*
1749 * Do ioctl.
1750 */
1751 SUPLDRGETSYMBOL Req;
1752 Req.Hdr.u32Cookie = g_u32Cookie;
1753 Req.Hdr.u32SessionCookie = g_u32SessionCookie;
1754 Req.Hdr.cbIn = SUP_IOCTL_LDR_GET_SYMBOL_SIZE_IN;
1755 Req.Hdr.cbOut = SUP_IOCTL_LDR_GET_SYMBOL_SIZE_OUT;
1756 Req.Hdr.fFlags = SUPREQHDR_FLAGS_DEFAULT;
1757 Req.Hdr.rc = VERR_INTERNAL_ERROR;
1758 Req.u.In.pvImageBase = (RTR0PTR)pvImageBase;
1759 size_t cchSymbol = strlen(pszSymbol);
1760 if (cchSymbol >= sizeof(Req.u.In.szSymbol))
1761 return VERR_SYMBOL_NOT_FOUND;
1762 memcpy(Req.u.In.szSymbol, pszSymbol, cchSymbol + 1);
1763 int rc = suplibOsIOCtl(SUP_IOCTL_LDR_GET_SYMBOL, &Req, SUP_IOCTL_LDR_GET_SYMBOL_SIZE);
1764 if (RT_SUCCESS(rc))
1765 rc = Req.Hdr.rc;
1766 if (RT_SUCCESS(rc))
1767 *ppvValue = (void *)Req.u.Out.pvSymbol;
1768 return rc;
1769}
1770
1771
1772SUPR3DECL(int) SUPLoadVMM(const char *pszFilename)
1773{
1774 void *pvImageBase;
1775 return SUPLoadModule(pszFilename, "VMMR0.r0", &pvImageBase);
1776}
1777
1778
1779SUPR3DECL(int) SUPUnloadVMM(void)
1780{
1781 return SUPFreeModule((void*)g_pvVMMR0);
1782}
1783
1784
1785SUPR3DECL(int) SUPGipGetPhys(PRTHCPHYS pHCPhys)
1786{
1787 if (g_pSUPGlobalInfoPage)
1788 {
1789 *pHCPhys = g_HCPhysSUPGlobalInfoPage;
1790 return VINF_SUCCESS;
1791 }
1792 *pHCPhys = NIL_RTHCPHYS;
1793 return VERR_WRONG_ORDER;
1794}
1795
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