VirtualBox

source: vbox/trunk/src/recompiler/VBoxREMWrapper.cpp@ 12651

Last change on this file since 12651 was 12549, checked in by vboxsync, 16 years ago

VMM: Implemented a TSC mode where it's tied to execution and halt (optionally). Fixes #3182.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 109.1 KB
Line 
1/* $Id: VBoxREMWrapper.cpp 12549 2008-09-17 18:02:02Z vboxsync $ */
2/** @file
3 *
4 * VBoxREM Win64 DLL Wrapper.
5 */
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/** @page pg_vboxrem_amd64 VBoxREM Hacks on AMD64
24 *
25 * There are problems with building BoxREM both on WIN64 and 64-bit linux.
26 *
27 * On linux binutils refuses to link shared objects without -fPIC compiled code
28 * (bitches about some fixup types). But when trying to build with -fPIC dyngen
29 * doesn't like the code anymore. Sweet. The current solution is to build the
30 * VBoxREM code as a relocatable module and use our ELF loader to load it.
31 *
32 * On WIN64 we're not aware of any GCC port which can emit code using the MSC
33 * calling convention. So, we're in for some real fun here. The choice is between
34 * porting GCC to AMD64 WIN64 and comming up with some kind of wrapper around
35 * either the win32 build or the 64-bit linux build.
36 *
37 * -# Porting GCC will be a lot of work. For one thing the calling convention differs
38 * and messing with such stuff can easily create ugly bugs. We would also have to
39 * do some binutils changes, but I think those are rather small compared to GCC.
40 * (That said, the MSC calling convention is far simpler than the linux one, it
41 * reminds me of _Optlink which we have working already.)
42 * -# Wrapping win32 code will work, but addresses outside the first 4GB are
43 * inaccessible and we will have to create 32-64 thunks for all imported functions.
44 * (To switch between 32-bit and 64-bit is load the right CS using far jmps (32->64)
45 * or far returns (both).)
46 * -# Wrapping 64-bit linux code might be the easier solution. The requirements here
47 * are:
48 * - Remove all CRT references we possibly, either by using intrinsics or using
49 * IPRT. Part of IPRT will be linked into VBoxREM2.rel, this will be yet another
50 * IPRT mode which I've dubbed 'no-crt'. The no-crt mode provide basic non-system
51 * dependent stuff.
52 * - Compile and link it into a relocatable object (include the gcc intrinsics
53 * in libgcc). Call this VBoxREM2.rel.
54 * - Write a wrapper dll, VBoxREM.dll, for which during REMR3Init() will load
55 * VBoxREM2.rel (using IPRT) and generate calling convention wrappers
56 * for all IPRT functions and VBoxVMM functions that it uses. All exports
57 * will be wrapped vice versa.
58 * - For building on windows hosts, we will use a mingw32 hosted cross compiler.
59 * and add a 'no-crt' mode to IPRT where it provides the necessary CRT headers
60 * and function implementations.
61 *
62 * The 3rd solution will be tried out first since it requires the least effort and
63 * will let us make use of the full 64-bit register set.
64 *
65 *
66 *
67 * @section sec_vboxrem_amd64_compare Comparing the GCC and MSC calling conventions
68 *
69 * GCC expects the following (cut & past from page 20 in the ABI draft 0.96):
70 *
71 * %rax temporary register; with variable arguments passes information about the
72 * number of SSE registers used; 1st return register.
73 * [Not preserved]
74 * %rbx callee-saved register; optionally used as base pointer.
75 * [Preserved]
76 * %rcx used to pass 4th integer argument to functions.
77 * [Not preserved]
78 * %rdx used to pass 3rd argument to functions; 2nd return register
79 * [Not preserved]
80 * %rsp stack pointer
81 * [Preserved]
82 * %rbp callee-saved register; optionally used as frame pointer
83 * [Preserved]
84 * %rsi used to pass 2nd argument to functions
85 * [Not preserved]
86 * %rdi used to pass 1st argument to functions
87 * [Not preserved]
88 * %r8 used to pass 5th argument to functions
89 * [Not preserved]
90 * %r9 used to pass 6th argument to functions
91 * [Not preserved]
92 * %r10 temporary register, used for passing a function’s static chain pointer
93 * [Not preserved]
94 * %r11 temporary register
95 * [Not preserved]
96 * %r12-r15 callee-saved registers
97 * [Preserved]
98 * %xmm0-%xmm1 used to pass and return floating point arguments
99 * [Not preserved]
100 * %xmm2-%xmm7 used to pass floating point arguments
101 * [Not preserved]
102 * %xmm8-%xmm15 temporary registers
103 * [Not preserved]
104 * %mmx0-%mmx7 temporary registers
105 * [Not preserved]
106 * %st0 temporary register; used to return long double arguments
107 * [Not preserved]
108 * %st1 temporary registers; used to return long double arguments
109 * [Not preserved]
110 * %st2-%st7 temporary registers
111 * [Not preserved]
112 * %fs Reserved for system use (as thread specific data register)
113 * [Not preserved]
114 *
115 * Direction flag is preserved as cleared.
116 * The stack must be aligned on a 16-byte boundrary before the 'call/jmp' instruction.
117 *
118 *
119 *
120 * MSC expects the following:
121 * rax return value, not preserved.
122 * rbx preserved.
123 * rcx 1st argument, integer, not preserved.
124 * rdx 2nd argument, integer, not preserved.
125 * rbp preserved.
126 * rsp preserved.
127 * rsi preserved.
128 * rdi preserved.
129 * r8 3rd argument, integer, not preserved.
130 * r9 4th argument, integer, not preserved.
131 * r10 scratch register, not preserved.
132 * r11 scratch register, not preserved.
133 * r12-r15 preserved.
134 * xmm0 1st argument, fp, return value, not preserved.
135 * xmm1 2st argument, fp, not preserved.
136 * xmm2 3st argument, fp, not preserved.
137 * xmm3 4st argument, fp, not preserved.
138 * xmm4-xmm5 scratch, not preserved.
139 * xmm6-xmm15 preserved.
140 *
141 * Dunno what the direction flag is...
142 * The stack must be aligned on a 16-byte boundrary before the 'call/jmp' instruction.
143 *
144 *
145 * Thus, When GCC code is calling MSC code we don't really have to preserve
146 * anything. But but MSC code is calling GCC code, we'll have to save esi and edi.
147 *
148 */
149
150
151/*******************************************************************************
152* Defined Constants And Macros *
153*******************************************************************************/
154/** @def USE_REM_STUBS
155 * Define USE_REM_STUBS to stub the entire REM stuff. This is useful during
156 * early porting (before we start running stuff).
157 */
158#if defined(__DOXYGEN__)
159# define USE_REM_STUBS
160#endif
161
162/** @def USE_REM_CALLING_CONVENTION_GLUE
163 * Define USE_REM_CALLING_CONVENTION_GLUE for platforms where it's necessary to
164 * use calling convention wrappers.
165 */
166#if (defined(RT_ARCH_AMD64) && defined(RT_OS_WINDOWS)) || defined(__DOXYGEN__)
167# define USE_REM_CALLING_CONVENTION_GLUE
168#endif
169
170/** @def USE_REM_IMPORT_JUMP_GLUE
171 * Define USE_REM_IMPORT_JUMP_GLUE for platforms where we need to
172 * emit some jump glue to deal with big addresses.
173 */
174#if (defined(RT_ARCH_AMD64) && !defined(USE_REM_CALLING_CONVENTION_GLUE) && !defined(RT_OS_DARWIN)) || defined(__DOXYGEN__)
175# define USE_REM_IMPORT_JUMP_GLUE
176#endif
177
178
179/*******************************************************************************
180* Header Files *
181*******************************************************************************/
182#define LOG_GROUP LOG_GROUP_REM
183#include <VBox/rem.h>
184#include <VBox/vmm.h>
185#include <VBox/dbgf.h>
186#include <VBox/dbg.h>
187#include <VBox/csam.h>
188#include <VBox/mm.h>
189#include <VBox/em.h>
190#include <VBox/ssm.h>
191#include <VBox/hwaccm.h>
192#include <VBox/patm.h>
193#include <VBox/pdm.h>
194#include <VBox/pgm.h>
195#include <VBox/iom.h>
196#include <VBox/vm.h>
197#include <VBox/err.h>
198#include <VBox/log.h>
199#include <VBox/dis.h>
200
201#include <iprt/alloc.h>
202#include <iprt/assert.h>
203#include <iprt/ldr.h>
204#include <iprt/param.h>
205#include <iprt/path.h>
206#include <iprt/string.h>
207#include <iprt/stream.h>
208
209
210/*******************************************************************************
211* Structures and Typedefs *
212*******************************************************************************/
213/**
214 * Parameter descriptor.
215 */
216typedef struct REMPARMDESC
217{
218 /** Parameter flags (REMPARMDESC_FLAGS_*). */
219 uint8_t fFlags;
220 /** The parameter size if REMPARMDESC_FLAGS_SIZE is set. */
221 uint8_t cb;
222 /** Pointer to additional data.
223 * For REMPARMDESC_FLAGS_PFN this is a PREMFNDESC. */
224 void *pvExtra;
225
226} REMPARMDESC, *PREMPARMDESC;
227/** Pointer to a constant parameter descriptor. */
228typedef const REMPARMDESC *PCREMPARMDESC;
229
230/** @name Parameter descriptor flags.
231 * @{ */
232/** The parameter type is a kind of integer which could fit in a register. This includes pointers. */
233#define REMPARMDESC_FLAGS_INT 0
234/** The parameter is a GC pointer. */
235#define REMPARMDESC_FLAGS_GCPTR 1
236/** The parameter is a GC physical address. */
237#define REMPARMDESC_FLAGS_GCPHYS 2
238/** The parameter is a HC physical address. */
239#define REMPARMDESC_FLAGS_HCPHYS 3
240/** The parameter type is a kind of floating point. */
241#define REMPARMDESC_FLAGS_FLOAT 4
242/** The parameter value is a struct. This type takes a size. */
243#define REMPARMDESC_FLAGS_STRUCT 5
244/** The parameter is an elipsis. */
245#define REMPARMDESC_FLAGS_ELLIPSIS 6
246/** The parameter is a va_list. */
247#define REMPARMDESC_FLAGS_VALIST 7
248/** The parameter is a function pointer. pvExtra is a PREMFNDESC. */
249#define REMPARMDESC_FLAGS_PFN 8
250/** The parameter type mask. */
251#define REMPARMDESC_FLAGS_TYPE_MASK 15
252/** The parameter size field is valid. */
253#define REMPARMDESC_FLAGS_SIZE RT_BIT(7)
254/** @} */
255
256/**
257 * Function descriptor.
258 */
259typedef struct REMFNDESC
260{
261 /** The function name. */
262 const char *pszName;
263 /** Exports: Pointer to the function pointer.
264 * Imports: Pointer to the function. */
265 void *pv;
266 /** Array of parameter descriptors. */
267 PCREMPARMDESC paParams;
268 /** The number of parameter descriptors pointed to by paParams. */
269 uint8_t cParams;
270 /** Function flags (REMFNDESC_FLAGS_*). */
271 uint8_t fFlags;
272 /** The size of the return value. */
273 uint8_t cbReturn;
274 /** Pointer to the wrapper code for imports. */
275 void *pvWrapper;
276} REMFNDESC, *PREMFNDESC;
277/** Pointer to a constant function descriptor. */
278typedef const REMFNDESC *PCREMFNDESC;
279
280/** @name Function descriptor flags.
281 * @{ */
282/** The return type is void. */
283#define REMFNDESC_FLAGS_RET_VOID 0
284/** The return type is a kind of integer passed in rax/eax. This includes pointers. */
285#define REMFNDESC_FLAGS_RET_INT 1
286/** The return type is a kind of floating point. */
287#define REMFNDESC_FLAGS_RET_FLOAT 2
288/** The return value is a struct. This type take a size. */
289#define REMFNDESC_FLAGS_RET_STRUCT 3
290/** The return type mask. */
291#define REMFNDESC_FLAGS_RET_TYPE_MASK 7
292/** The argument list contains one or more va_list arguments (i.e. problems). */
293#define REMFNDESC_FLAGS_VALIST RT_BIT(6)
294/** The function has an ellipsis (i.e. a problem). */
295#define REMFNDESC_FLAGS_ELLIPSIS RT_BIT(7)
296/** @} */
297
298/**
299 * Chunk of read-write-executable memory.
300 */
301typedef struct REMEXECMEM
302{
303 /** The number of bytes left. */
304 struct REMEXECMEM *pNext;
305 /** The size of this chunk. */
306 uint32_t cb;
307 /** The offset of the next code block. */
308 uint32_t off;
309#if ARCH_BITS == 32
310 uint32_t padding;
311#endif
312} REMEXECMEM, *PREMEXECMEM;
313
314
315/*******************************************************************************
316* Global Variables *
317*******************************************************************************/
318#ifndef USE_REM_STUBS
319/** Loader handle of the REM object/DLL. */
320static RTLDRMOD g_ModREM2;
321/** Pointer to the memory containing the loaded REM2 object/DLL. */
322static void *g_pvREM2;
323
324/** Linux object export addresses.
325 * These are references from the assembly wrapper code.
326 * @{ */
327static DECLCALLBACKPTR(int, pfnREMR3Init)(PVM);
328static DECLCALLBACKPTR(int, pfnREMR3Term)(PVM);
329static DECLCALLBACKPTR(void, pfnREMR3Reset)(PVM);
330static DECLCALLBACKPTR(int, pfnREMR3Step)(PVM);
331static DECLCALLBACKPTR(int, pfnREMR3BreakpointSet)(PVM, RTGCUINTPTR);
332static DECLCALLBACKPTR(int, pfnREMR3BreakpointClear)(PVM, RTGCUINTPTR);
333static DECLCALLBACKPTR(int, pfnREMR3EmulateInstruction)(PVM);
334static DECLCALLBACKPTR(int, pfnREMR3Run)(PVM);
335static DECLCALLBACKPTR(int, pfnREMR3State)(PVM, bool fFlushTBs);
336static DECLCALLBACKPTR(int, pfnREMR3StateBack)(PVM);
337static DECLCALLBACKPTR(void, pfnREMR3StateUpdate)(PVM);
338static DECLCALLBACKPTR(void, pfnREMR3A20Set)(PVM, bool);
339static DECLCALLBACKPTR(void, pfnREMR3ReplayInvalidatedPages)(PVM);
340static DECLCALLBACKPTR(void, pfnREMR3ReplayHandlerNotifications)(PVM pVM);
341static DECLCALLBACKPTR(void, pfnREMR3NotifyPhysRamRegister)(PVM, RTGCPHYS, RTUINT, unsigned);
342static DECLCALLBACKPTR(void, pfnREMR3NotifyPhysRamChunkRegister)(PVM, RTGCPHYS, RTUINT, RTHCUINTPTR, unsigned);
343static DECLCALLBACKPTR(void, pfnREMR3NotifyPhysReserve)(PVM, RTGCPHYS, RTUINT);
344static DECLCALLBACKPTR(void, pfnREMR3NotifyPhysRomRegister)(PVM, RTGCPHYS, RTUINT, void *, bool);
345static DECLCALLBACKPTR(void, pfnREMR3NotifyHandlerPhysicalModify)(PVM, PGMPHYSHANDLERTYPE, RTGCPHYS, RTGCPHYS, RTGCPHYS, bool, bool);
346static DECLCALLBACKPTR(void, pfnREMR3NotifyHandlerPhysicalRegister)(PVM, PGMPHYSHANDLERTYPE, RTGCPHYS, RTGCPHYS, bool);
347static DECLCALLBACKPTR(void, pfnREMR3NotifyHandlerPhysicalDeregister)(PVM, PGMPHYSHANDLERTYPE, RTGCPHYS, RTGCPHYS, bool, bool);
348static DECLCALLBACKPTR(void, pfnREMR3NotifyInterruptSet)(PVM);
349static DECLCALLBACKPTR(void, pfnREMR3NotifyInterruptClear)(PVM);
350static DECLCALLBACKPTR(void, pfnREMR3NotifyTimerPending)(PVM);
351static DECLCALLBACKPTR(void, pfnREMR3NotifyDmaPending)(PVM);
352static DECLCALLBACKPTR(void, pfnREMR3NotifyQueuePending)(PVM);
353static DECLCALLBACKPTR(void, pfnREMR3NotifyFF)(PVM);
354static DECLCALLBACKPTR(int, pfnREMR3NotifyCodePageChanged)(PVM, RTGCPTR);
355static DECLCALLBACKPTR(void, pfnREMR3NotifyPendingInterrupt)(PVM, uint8_t);
356static DECLCALLBACKPTR(uint32_t, pfnREMR3QueryPendingInterrupt)(PVM);
357static DECLCALLBACKPTR(int, pfnREMR3DisasEnableStepping)(PVM, bool);
358static DECLCALLBACKPTR(bool, pfnREMR3IsPageAccessHandled)(PVM, RTGCPHYS);
359/** @} */
360
361/** Export and import parameter descriptors.
362 * @{
363 */
364/* Common args. */
365static const REMPARMDESC g_aArgsSIZE_T[] =
366{
367 { REMPARMDESC_FLAGS_INT, sizeof(size_t) }
368};
369static const REMPARMDESC g_aArgsPTR[] =
370{
371 { REMPARMDESC_FLAGS_INT, sizeof(void *) }
372};
373static const REMPARMDESC g_aArgsVM[] =
374{
375 { REMPARMDESC_FLAGS_INT, sizeof(PVM) }
376};
377
378/* REM args */
379static const REMPARMDESC g_aArgsBreakpoint[] =
380{
381 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
382 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCUINTPTR), NULL }
383};
384static const REMPARMDESC g_aArgsA20Set[] =
385{
386 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
387 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
388};
389static const REMPARMDESC g_aArgsNotifyPhysRamRegister[] =
390{
391 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
392 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
393 { REMPARMDESC_FLAGS_INT, sizeof(RTUINT), NULL },
394 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
395 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL }
396};
397static const REMPARMDESC g_aArgsNotifyPhysRamChunkRegister[] =
398{
399 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
400 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
401 { REMPARMDESC_FLAGS_INT, sizeof(RTUINT), NULL },
402 { REMPARMDESC_FLAGS_INT, sizeof(RTHCUINTPTR), NULL },
403 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL }
404};
405static const REMPARMDESC g_aArgsNotifyPhysReserve[] =
406{
407 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
408 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
409 { REMPARMDESC_FLAGS_INT, sizeof(RTUINT), NULL }
410};
411static const REMPARMDESC g_aArgsNotifyPhysRomRegister[] =
412{
413 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
414 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
415 { REMPARMDESC_FLAGS_INT, sizeof(RTUINT), NULL },
416 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
417 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
418};
419static const REMPARMDESC g_aArgsNotifyHandlerPhysicalModify[] =
420{
421 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
422 { REMPARMDESC_FLAGS_INT, sizeof(PGMPHYSHANDLERTYPE), NULL },
423 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
424 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
425 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
426 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL },
427 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
428};
429static const REMPARMDESC g_aArgsNotifyHandlerPhysicalRegister[] =
430{
431 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
432 { REMPARMDESC_FLAGS_INT, sizeof(PGMPHYSHANDLERTYPE), NULL },
433 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
434 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
435 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
436};
437static const REMPARMDESC g_aArgsNotifyHandlerPhysicalDeregister[] =
438{
439 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
440 { REMPARMDESC_FLAGS_INT, sizeof(PGMPHYSHANDLERTYPE), NULL },
441 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
442 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
443 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL },
444 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
445};
446static const REMPARMDESC g_aArgsNotifyCodePageChanged[] =
447{
448 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
449 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCUINTPTR), NULL }
450};
451static const REMPARMDESC g_aArgsNotifyPendingInterrupt[] =
452{
453 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
454 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t), NULL }
455};
456static const REMPARMDESC g_aArgsDisasEnableStepping[] =
457{
458 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
459 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
460};
461static const REMPARMDESC g_aArgsIsPageAccessHandled[] =
462{
463 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
464 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL }
465};
466
467
468/* VMM args */
469static const REMPARMDESC g_aArgsCPUMGetGuestCpl[] =
470{
471 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
472 { REMPARMDESC_FLAGS_INT, sizeof(PCPUMCTXCORE), NULL },
473};
474
475static const REMPARMDESC g_aArgsCPUMGetGuestCpuId[] =
476{
477 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
478 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
479 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL },
480 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL },
481 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL },
482 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL }
483};
484
485static const REMPARMDESC g_aArgsCPUMSetChangedFlags[] =
486{
487 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
488 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
489};
490
491static const REMPARMDESC g_aArgsCPUMQueryGuestCtxPtr[] =
492{
493 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
494 { REMPARMDESC_FLAGS_INT, sizeof(PCPUMCTX *), NULL }
495};
496static const REMPARMDESC g_aArgsCSAMR3MonitorPage[] =
497{
498 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
499 { REMPARMDESC_FLAGS_INT, sizeof(RTRCPTR), NULL },
500 { REMPARMDESC_FLAGS_INT, sizeof(CSAMTAG), NULL }
501};
502static const REMPARMDESC g_aArgsCSAMR3UnmonitorPage[] =
503{
504 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
505 { REMPARMDESC_FLAGS_INT, sizeof(RTRCPTR), NULL },
506 { REMPARMDESC_FLAGS_INT, sizeof(CSAMTAG), NULL }
507};
508
509static const REMPARMDESC g_aArgsCSAMR3RecordCallAddress[] =
510{
511 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
512 { REMPARMDESC_FLAGS_INT, sizeof(RTRCPTR), NULL }
513};
514
515#if defined(VBOX_WITH_DEBUGGER) && !(defined(RT_OS_WINDOWS) && defined(RT_ARCH_AMD64)) /* the callbacks are problematic */
516static const REMPARMDESC g_aArgsDBGCRegisterCommands[] =
517{
518 { REMPARMDESC_FLAGS_INT, sizeof(PCDBGCCMD), NULL },
519 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL }
520};
521#endif
522static const REMPARMDESC g_aArgsDBGFR3DisasInstrEx[] =
523{
524 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
525 { REMPARMDESC_FLAGS_INT, sizeof(RTSEL), NULL },
526 { REMPARMDESC_FLAGS_INT, sizeof(RTGCPTR), NULL },
527 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
528 { REMPARMDESC_FLAGS_INT, sizeof(char *), NULL },
529 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
530 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL }
531};
532static const REMPARMDESC g_aArgsDBGFR3Info[] =
533{
534 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
535 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
536 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
537 { REMPARMDESC_FLAGS_INT, sizeof(PCDBGFINFOHLP), NULL }
538};
539static const REMPARMDESC g_aArgsDBGFR3SymbolByAddr[] =
540{
541 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
542 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCUINTPTR), NULL },
543 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCINTPTR), NULL },
544 { REMPARMDESC_FLAGS_INT, sizeof(PDBGFSYMBOL), NULL }
545};
546static const REMPARMDESC g_aArgsDISInstr[] =
547{
548 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
549 { REMPARMDESC_FLAGS_INT, sizeof(RTUINTPTR), NULL },
550 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
551 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL },
552 { REMPARMDESC_FLAGS_INT, sizeof(char *), NULL }
553};
554static const REMPARMDESC g_aArgsEMR3FatalError[] =
555{
556 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
557 { REMPARMDESC_FLAGS_INT, sizeof(int), NULL }
558};
559static const REMPARMDESC g_aArgsHWACCMR3CanExecuteGuest[] =
560{
561 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
562 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
563 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
564 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
565};
566static const REMPARMDESC g_aArgsIOMIOPortRead[] =
567{
568 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
569 { REMPARMDESC_FLAGS_INT, sizeof(RTIOPORT), NULL },
570 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL },
571 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
572};
573static const REMPARMDESC g_aArgsIOMIOPortWrite[] =
574{
575 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
576 { REMPARMDESC_FLAGS_INT, sizeof(RTIOPORT), NULL },
577 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
578 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
579};
580static const REMPARMDESC g_aArgsIOMMMIORead[] =
581{
582 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
583 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
584 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL },
585 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
586};
587static const REMPARMDESC g_aArgsIOMMMIOWrite[] =
588{
589 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
590 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
591 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
592 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
593};
594static const REMPARMDESC g_aArgsMMR3HeapAlloc[] =
595{
596 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
597 { REMPARMDESC_FLAGS_INT, sizeof(MMTAG), NULL },
598 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
599};
600static const REMPARMDESC g_aArgsMMR3HeapAllocZ[] =
601{
602 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
603 { REMPARMDESC_FLAGS_INT, sizeof(MMTAG), NULL },
604 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
605};
606static const REMPARMDESC g_aArgsPATMIsPatchGCAddr[] =
607{
608 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
609 { REMPARMDESC_FLAGS_INT, sizeof(RTRCPTR), NULL }
610};
611static const REMPARMDESC g_aArgsPATMR3QueryOpcode[] =
612{
613 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
614 { REMPARMDESC_FLAGS_INT, sizeof(RTRCPTR), NULL },
615 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t *), NULL }
616};
617static const REMPARMDESC g_aArgsPATMR3QueryPatchMem[] =
618{
619 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
620 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL }
621};
622static const REMPARMDESC g_aArgsPDMApicGetBase[] =
623{
624 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
625 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t *), NULL }
626};
627static const REMPARMDESC g_aArgsPDMApicGetTPR[] =
628{
629 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
630 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t *), NULL },
631 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t *), NULL }
632};
633static const REMPARMDESC g_aArgsPDMApicSetBase[] =
634{
635 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
636 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL }
637};
638static const REMPARMDESC g_aArgsPDMApicSetTPR[] =
639{
640 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
641 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t), NULL }
642};
643static const REMPARMDESC g_aArgsPDMGetInterrupt[] =
644{
645 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
646 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t *), NULL }
647};
648static const REMPARMDESC g_aArgsPDMIsaSetIrq[] =
649{
650 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
651 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t), NULL },
652 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t), NULL }
653};
654static const REMPARMDESC g_aArgsPGMGetGuestMode[] =
655{
656 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
657};
658static const REMPARMDESC g_aArgsPGMGstGetPage[] =
659{
660 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
661 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCPTR), NULL },
662 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t *), NULL },
663 { REMPARMDESC_FLAGS_INT, sizeof(PRTGCPHYS), NULL }
664};
665static const REMPARMDESC g_aArgsPGMInvalidatePage[] =
666{
667 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
668 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCPTR), NULL }
669};
670static const REMPARMDESC g_aArgsPGMPhysGCPhys2HCPtr[] =
671{
672 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
673 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
674 { REMPARMDESC_FLAGS_INT, sizeof(RTUINT), NULL },
675 { REMPARMDESC_FLAGS_INT, sizeof(PRTHCPTR), NULL }
676};
677static const REMPARMDESC g_aArgsPGMPhysGCPtr2HCPtrByGstCR3[] =
678{
679 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
680 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
681 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL },
682 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
683 { REMPARMDESC_FLAGS_INT, sizeof(PRTHCPTR), NULL }
684};
685static const REMPARMDESC g_aArgsPGM3PhysGrowRange[] =
686{
687 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
688 { REMPARMDESC_FLAGS_INT, sizeof(PCRTGCPHYS), NULL }
689};
690static const REMPARMDESC g_aArgsPGMPhysIsGCPhysValid[] =
691{
692 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
693 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL }
694};
695static const REMPARMDESC g_aArgsPGMPhysRead[] =
696{
697 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
698 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
699 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
700 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
701};
702static const REMPARMDESC g_aArgsPGMPhysReadGCPtr[] =
703{
704 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
705 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
706 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCPTR), NULL },
707 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
708};
709static const REMPARMDESC g_aArgsPGMPhysWrite[] =
710{
711 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
712 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
713 { REMPARMDESC_FLAGS_INT, sizeof(const void *), NULL },
714 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
715};
716static const REMPARMDESC g_aArgsPGMChangeMode[] =
717{
718 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
719 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL },
720 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL },
721 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL }
722};
723static const REMPARMDESC g_aArgsPGMFlushTLB[] =
724{
725 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
726 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL },
727 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
728};
729static const REMPARMDESC g_aArgsPGMR3PhysReadUxx[] =
730{
731 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
732 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL }
733};
734static const REMPARMDESC g_aArgsPGMR3PhysWriteU8[] =
735{
736 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
737 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
738 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t), NULL }
739};
740static const REMPARMDESC g_aArgsPGMR3PhysWriteU16[] =
741{
742 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
743 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
744 { REMPARMDESC_FLAGS_INT, sizeof(uint16_t), NULL }
745};
746static const REMPARMDESC g_aArgsPGMR3PhysWriteU32[] =
747{
748 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
749 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
750 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL }
751};
752static const REMPARMDESC g_aArgsPGMR3PhysWriteU64[] =
753{
754 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
755 { REMPARMDESC_FLAGS_GCPHYS, sizeof(RTGCPHYS), NULL },
756 { REMPARMDESC_FLAGS_INT, sizeof(uint64_t), NULL }
757};
758static const REMPARMDESC g_aArgsSSMR3GetGCPtr[] =
759{
760 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
761 { REMPARMDESC_FLAGS_INT, sizeof(PRTGCPTR), NULL }
762};
763static const REMPARMDESC g_aArgsSSMR3GetMem[] =
764{
765 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
766 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
767 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
768};
769static const REMPARMDESC g_aArgsSSMR3GetU32[] =
770{
771 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
772 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t *), NULL }
773};
774static const REMPARMDESC g_aArgsSSMR3GetUInt[] =
775{
776 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
777 { REMPARMDESC_FLAGS_INT, sizeof(PRTUINT), NULL }
778};
779static const REMPARMDESC g_aArgsSSMR3PutGCPtr[] =
780{
781 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
782 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCPTR), NULL }
783};
784static const REMPARMDESC g_aArgsSSMR3PutMem[] =
785{
786 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
787 { REMPARMDESC_FLAGS_INT, sizeof(const void *), NULL },
788 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
789};
790static const REMPARMDESC g_aArgsSSMR3PutU32[] =
791{
792 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
793 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
794};
795static const REMPARMDESC g_aArgsSSMR3PutUInt[] =
796{
797 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
798 { REMPARMDESC_FLAGS_INT, sizeof(RTUINT), NULL },
799};
800
801static const REMPARMDESC g_aArgsSSMIntCallback[] =
802{
803 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
804 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
805};
806static REMFNDESC g_SSMIntCallback =
807{
808 "SSMIntCallback", NULL, &g_aArgsSSMIntCallback[0], RT_ELEMENTS(g_aArgsSSMIntCallback), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL
809};
810
811static const REMPARMDESC g_aArgsSSMIntLoadExecCallback[] =
812{
813 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
814 { REMPARMDESC_FLAGS_INT, sizeof(PSSMHANDLE), NULL },
815 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
816};
817static REMFNDESC g_SSMIntLoadExecCallback =
818{
819 "SSMIntLoadExecCallback", NULL, &g_aArgsSSMIntLoadExecCallback[0], RT_ELEMENTS(g_aArgsSSMIntLoadExecCallback), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL
820};
821static const REMPARMDESC g_aArgsSSMR3RegisterInternal[] =
822{
823 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
824 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
825 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
826 { REMPARMDESC_FLAGS_INT, sizeof(uint32_t), NULL },
827 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL },
828 { REMPARMDESC_FLAGS_PFN, sizeof(PFNSSMINTSAVEPREP), &g_SSMIntCallback },
829 { REMPARMDESC_FLAGS_PFN, sizeof(PFNSSMINTSAVEEXEC), &g_SSMIntCallback },
830 { REMPARMDESC_FLAGS_PFN, sizeof(PFNSSMINTSAVEDONE), &g_SSMIntCallback },
831 { REMPARMDESC_FLAGS_PFN, sizeof(PFNSSMINTLOADPREP), &g_SSMIntCallback },
832 { REMPARMDESC_FLAGS_PFN, sizeof(PFNSSMINTLOADEXEC), &g_SSMIntLoadExecCallback },
833 { REMPARMDESC_FLAGS_PFN, sizeof(PFNSSMINTLOADDONE), &g_SSMIntCallback },
834};
835
836static const REMPARMDESC g_aArgsSTAMR3Register[] =
837{
838 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
839 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
840 { REMPARMDESC_FLAGS_INT, sizeof(STAMTYPE), NULL },
841 { REMPARMDESC_FLAGS_INT, sizeof(STAMVISIBILITY), NULL },
842 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
843 { REMPARMDESC_FLAGS_INT, sizeof(STAMUNIT), NULL },
844 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL }
845};
846static const REMPARMDESC g_aArgsTRPMAssertTrap[] =
847{
848 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
849 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t), NULL },
850 { REMPARMDESC_FLAGS_INT, sizeof(TRPMEVENT), NULL }
851};
852static const REMPARMDESC g_aArgsTRPMQueryTrap[] =
853{
854 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
855 { REMPARMDESC_FLAGS_INT, sizeof(uint8_t *), NULL },
856 { REMPARMDESC_FLAGS_INT, sizeof(TRPMEVENT *), NULL }
857};
858static const REMPARMDESC g_aArgsTRPMSetErrorCode[] =
859{
860 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
861 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCUINT), NULL }
862};
863static const REMPARMDESC g_aArgsTRPMSetFaultAddress[] =
864{
865 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
866 { REMPARMDESC_FLAGS_GCPTR, sizeof(RTGCUINT), NULL }
867};
868static const REMPARMDESC g_aArgsVMR3ReqCall[] =
869{
870 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
871 { REMPARMDESC_FLAGS_INT, sizeof(PVMREQ *), NULL },
872 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
873 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
874 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
875 { REMPARMDESC_FLAGS_ELLIPSIS, 0 }
876};
877static const REMPARMDESC g_aArgsVMR3ReqFree[] =
878{
879 { REMPARMDESC_FLAGS_INT, sizeof(PVMREQ), NULL }
880};
881
882
883/* IPRT args */
884static const REMPARMDESC g_aArgsAssertMsg1[] =
885{
886 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
887 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
888 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
889 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL }
890};
891static const REMPARMDESC g_aArgsAssertMsg2[] =
892{
893 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
894 { REMPARMDESC_FLAGS_ELLIPSIS, 0 }
895};
896static const REMPARMDESC g_aArgsRTLogFlags[] =
897{
898 { REMPARMDESC_FLAGS_INT, sizeof(PRTLOGGER), NULL },
899 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL }
900};
901static const REMPARMDESC g_aArgsRTLogFlush[] =
902{
903 { REMPARMDESC_FLAGS_INT, sizeof(PRTLOGGER), NULL }
904};
905static const REMPARMDESC g_aArgsRTLogLoggerEx[] =
906{
907 { REMPARMDESC_FLAGS_INT, sizeof(PRTLOGGER), NULL },
908 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
909 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
910 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
911 { REMPARMDESC_FLAGS_ELLIPSIS, 0 }
912};
913static const REMPARMDESC g_aArgsRTLogLoggerExV[] =
914{
915 { REMPARMDESC_FLAGS_INT, sizeof(PRTLOGGER), NULL },
916 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
917 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL },
918 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
919 { REMPARMDESC_FLAGS_VALIST, 0 }
920};
921static const REMPARMDESC g_aArgsRTLogPrintf[] =
922{
923 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
924 { REMPARMDESC_FLAGS_ELLIPSIS, 0 }
925};
926static const REMPARMDESC g_aArgsRTMemProtect[] =
927{
928 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
929 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL },
930 { REMPARMDESC_FLAGS_INT, sizeof(unsigned), NULL }
931};
932static const REMPARMDESC g_aArgsRTStrPrintf[] =
933{
934 { REMPARMDESC_FLAGS_INT, sizeof(char *), NULL },
935 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL },
936 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
937 { REMPARMDESC_FLAGS_ELLIPSIS, 0 }
938};
939static const REMPARMDESC g_aArgsRTStrPrintfV[] =
940{
941 { REMPARMDESC_FLAGS_INT, sizeof(char *), NULL },
942 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL },
943 { REMPARMDESC_FLAGS_INT, sizeof(const char *), NULL },
944 { REMPARMDESC_FLAGS_VALIST, 0 }
945};
946static const REMPARMDESC g_aArgsThread[] =
947{
948 { REMPARMDESC_FLAGS_INT, sizeof(RTTHREAD), NULL }
949};
950
951
952/* CRT args */
953static const REMPARMDESC g_aArgsmemcpy[] =
954{
955 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
956 { REMPARMDESC_FLAGS_INT, sizeof(const void *), NULL },
957 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
958};
959static const REMPARMDESC g_aArgsmemset[] =
960{
961 { REMPARMDESC_FLAGS_INT, sizeof(void *), NULL },
962 { REMPARMDESC_FLAGS_INT, sizeof(int), NULL },
963 { REMPARMDESC_FLAGS_INT, sizeof(size_t), NULL }
964};
965static const REMPARMDESC g_aArgsState[] =
966{
967 { REMPARMDESC_FLAGS_INT, sizeof(PVM), NULL },
968 { REMPARMDESC_FLAGS_INT, sizeof(bool), NULL }
969};
970
971/** @} */
972
973/**
974 * Descriptors for the exported functions.
975 */
976static const REMFNDESC g_aExports[] =
977{ /* pszName, (void *)pv, pParams, cParams, fFlags, cb, pvWrapper. */
978 { "REMR3Init", (void *)&pfnREMR3Init, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
979 { "REMR3Term", (void *)&pfnREMR3Term, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
980 { "REMR3Reset", (void *)&pfnREMR3Reset, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
981 { "REMR3Step", (void *)&pfnREMR3Step, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
982 { "REMR3BreakpointSet", (void *)&pfnREMR3BreakpointSet, &g_aArgsBreakpoint[0], RT_ELEMENTS(g_aArgsBreakpoint), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
983 { "REMR3BreakpointClear", (void *)&pfnREMR3BreakpointClear, &g_aArgsBreakpoint[0], RT_ELEMENTS(g_aArgsBreakpoint), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
984 { "REMR3EmulateInstruction", (void *)&pfnREMR3EmulateInstruction, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
985 { "REMR3Run", (void *)&pfnREMR3Run, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
986 { "REMR3State", (void *)&pfnREMR3State, &g_aArgsState[0], RT_ELEMENTS(g_aArgsState), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
987 { "REMR3StateBack", (void *)&pfnREMR3StateBack, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
988 { "REMR3StateUpdate", (void *)&pfnREMR3StateUpdate, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
989 { "REMR3A20Set", (void *)&pfnREMR3A20Set, &g_aArgsA20Set[0], RT_ELEMENTS(g_aArgsA20Set), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
990 { "REMR3ReplayInvalidatedPages", (void *)&pfnREMR3ReplayInvalidatedPages, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
991 { "REMR3ReplayHandlerNotifications", (void *)&pfnREMR3ReplayHandlerNotifications, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
992 { "REMR3NotifyPhysRamRegister", (void *)&pfnREMR3NotifyPhysRamRegister, &g_aArgsNotifyPhysRamRegister[0], RT_ELEMENTS(g_aArgsNotifyPhysRamRegister), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
993 { "REMR3NotifyPhysRamChunkRegister", (void *)&pfnREMR3NotifyPhysRamChunkRegister, &g_aArgsNotifyPhysRamChunkRegister[0], RT_ELEMENTS(g_aArgsNotifyPhysRamChunkRegister), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
994 { "REMR3NotifyPhysReserve", (void *)&pfnREMR3NotifyPhysReserve, &g_aArgsNotifyPhysReserve[0], RT_ELEMENTS(g_aArgsNotifyPhysReserve), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
995 { "REMR3NotifyPhysRomRegister", (void *)&pfnREMR3NotifyPhysRomRegister, &g_aArgsNotifyPhysRomRegister[0], RT_ELEMENTS(g_aArgsNotifyPhysRomRegister), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
996 { "REMR3NotifyHandlerPhysicalModify", (void *)&pfnREMR3NotifyHandlerPhysicalModify, &g_aArgsNotifyHandlerPhysicalModify[0], RT_ELEMENTS(g_aArgsNotifyHandlerPhysicalModify), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
997 { "REMR3NotifyHandlerPhysicalRegister", (void *)&pfnREMR3NotifyHandlerPhysicalRegister, &g_aArgsNotifyHandlerPhysicalRegister[0], RT_ELEMENTS(g_aArgsNotifyHandlerPhysicalRegister), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
998 { "REMR3NotifyHandlerPhysicalDeregister", (void *)&pfnREMR3NotifyHandlerPhysicalDeregister, &g_aArgsNotifyHandlerPhysicalDeregister[0], RT_ELEMENTS(g_aArgsNotifyHandlerPhysicalDeregister), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
999 { "REMR3NotifyInterruptSet", (void *)&pfnREMR3NotifyInterruptSet, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1000 { "REMR3NotifyInterruptClear", (void *)&pfnREMR3NotifyInterruptClear, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1001 { "REMR3NotifyTimerPending", (void *)&pfnREMR3NotifyTimerPending, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1002 { "REMR3NotifyDmaPending", (void *)&pfnREMR3NotifyDmaPending, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1003 { "REMR3NotifyQueuePending", (void *)&pfnREMR3NotifyQueuePending, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1004 { "REMR3NotifyFF", (void *)&pfnREMR3NotifyFF, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1005 { "REMR3NotifyCodePageChanged", (void *)&pfnREMR3NotifyCodePageChanged, &g_aArgsNotifyCodePageChanged[0], RT_ELEMENTS(g_aArgsNotifyCodePageChanged), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1006 { "REMR3NotifyPendingInterrupt", (void *)&pfnREMR3NotifyPendingInterrupt, &g_aArgsNotifyPendingInterrupt[0], RT_ELEMENTS(g_aArgsNotifyPendingInterrupt), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1007 { "REMR3QueryPendingInterrupt", (void *)&pfnREMR3QueryPendingInterrupt, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1008 { "REMR3DisasEnableStepping", (void *)&pfnREMR3DisasEnableStepping, &g_aArgsDisasEnableStepping[0], RT_ELEMENTS(g_aArgsDisasEnableStepping), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1009 { "REMR3IsPageAccessHandled", (void *)&pfnREMR3IsPageAccessHandled, &g_aArgsIsPageAccessHandled[0], RT_ELEMENTS(g_aArgsIsPageAccessHandled), REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL }
1010};
1011
1012
1013/**
1014 * Descriptors for the functions imported from VBoxVMM.
1015 */
1016static REMFNDESC g_aVMMImports[] =
1017{
1018 { "CPUMAreHiddenSelRegsValid", (void *)(uintptr_t)&CPUMAreHiddenSelRegsValid, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL },
1019 { "CPUMGetAndClearChangedFlagsREM", (void *)(uintptr_t)&CPUMGetAndClearChangedFlagsREM, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(unsigned), NULL },
1020 { "CPUMSetChangedFlags", (void *)(uintptr_t)&CPUMSetChangedFlags, &g_aArgsCPUMSetChangedFlags[0], RT_ELEMENTS(g_aArgsCPUMSetChangedFlags), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1021 { "CPUMGetGuestCPL", (void *)(uintptr_t)&CPUMGetGuestCPL, &g_aArgsCPUMGetGuestCpl[0], RT_ELEMENTS(g_aArgsCPUMGetGuestCpl), REMFNDESC_FLAGS_RET_INT, sizeof(unsigned), NULL },
1022 { "CPUMGetGuestCpuId", (void *)(uintptr_t)&CPUMGetGuestCpuId, &g_aArgsCPUMGetGuestCpuId[0], RT_ELEMENTS(g_aArgsCPUMGetGuestCpuId), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1023 { "CPUMGetGuestEAX", (void *)(uintptr_t)&CPUMGetGuestEAX, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1024 { "CPUMGetGuestEBP", (void *)(uintptr_t)&CPUMGetGuestEBP, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1025 { "CPUMGetGuestEBX", (void *)(uintptr_t)&CPUMGetGuestEBX, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1026 { "CPUMGetGuestECX", (void *)(uintptr_t)&CPUMGetGuestECX, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1027 { "CPUMGetGuestEDI", (void *)(uintptr_t)&CPUMGetGuestEDI, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1028 { "CPUMGetGuestEDX", (void *)(uintptr_t)&CPUMGetGuestEDX, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1029 { "CPUMGetGuestEIP", (void *)(uintptr_t)&CPUMGetGuestEIP, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1030 { "CPUMGetGuestESI", (void *)(uintptr_t)&CPUMGetGuestESI, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1031 { "CPUMGetGuestESP", (void *)(uintptr_t)&CPUMGetGuestESP, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1032 { "CPUMGetGuestCS", (void *)(uintptr_t)&CPUMGetGuestCS, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(RTSEL), NULL },
1033 { "CPUMGetGuestSS", (void *)(uintptr_t)&CPUMGetGuestSS, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(RTSEL), NULL },
1034 { "CPUMQueryGuestCtxPtr", (void *)(uintptr_t)&CPUMQueryGuestCtxPtr, &g_aArgsCPUMQueryGuestCtxPtr[0], RT_ELEMENTS(g_aArgsCPUMQueryGuestCtxPtr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1035 { "CSAMR3MonitorPage", (void *)(uintptr_t)&CSAMR3MonitorPage, &g_aArgsCSAMR3MonitorPage[0], RT_ELEMENTS(g_aArgsCSAMR3MonitorPage), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1036 { "CSAMR3UnmonitorPage", (void *)(uintptr_t)&CSAMR3UnmonitorPage, &g_aArgsCSAMR3UnmonitorPage[0], RT_ELEMENTS(g_aArgsCSAMR3UnmonitorPage), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1037 { "CSAMR3RecordCallAddress", (void *)(uintptr_t)&CSAMR3RecordCallAddress, &g_aArgsCSAMR3RecordCallAddress[0], RT_ELEMENTS(g_aArgsCSAMR3RecordCallAddress), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1038#if defined(VBOX_WITH_DEBUGGER) && !(defined(RT_OS_WINDOWS) && defined(RT_ARCH_AMD64)) /* the callbacks are problematic */
1039 { "DBGCRegisterCommands", (void *)(uintptr_t)&DBGCRegisterCommands, &g_aArgsDBGCRegisterCommands[0], RT_ELEMENTS(g_aArgsDBGCRegisterCommands), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1040#endif
1041 { "DBGFR3DisasInstrEx", (void *)(uintptr_t)&DBGFR3DisasInstrEx, &g_aArgsDBGFR3DisasInstrEx[0], RT_ELEMENTS(g_aArgsDBGFR3DisasInstrEx), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1042 { "DBGFR3Info", (void *)(uintptr_t)&DBGFR3Info, &g_aArgsDBGFR3Info[0], RT_ELEMENTS(g_aArgsDBGFR3Info), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1043 { "DBGFR3InfoLogRelHlp", (void *)(uintptr_t)&DBGFR3InfoLogRelHlp, NULL, 0, REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1044 { "DBGFR3SymbolByAddr", (void *)(uintptr_t)&DBGFR3SymbolByAddr, &g_aArgsDBGFR3SymbolByAddr[0], RT_ELEMENTS(g_aArgsDBGFR3SymbolByAddr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1045 { "DISInstr", (void *)(uintptr_t)&DISInstr, &g_aArgsDISInstr[0], RT_ELEMENTS(g_aArgsDISInstr), REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL },
1046 { "EMR3FatalError", (void *)(uintptr_t)&EMR3FatalError, &g_aArgsEMR3FatalError[0], RT_ELEMENTS(g_aArgsEMR3FatalError), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1047 { "HWACCMR3CanExecuteGuest", (void *)(uintptr_t)&HWACCMR3CanExecuteGuest, &g_aArgsHWACCMR3CanExecuteGuest[0], RT_ELEMENTS(g_aArgsHWACCMR3CanExecuteGuest), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1048 { "IOMIOPortRead", (void *)(uintptr_t)&IOMIOPortRead, &g_aArgsIOMIOPortRead[0], RT_ELEMENTS(g_aArgsIOMIOPortRead), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1049 { "IOMIOPortWrite", (void *)(uintptr_t)&IOMIOPortWrite, &g_aArgsIOMIOPortWrite[0], RT_ELEMENTS(g_aArgsIOMIOPortWrite), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1050 { "IOMMMIORead", (void *)(uintptr_t)&IOMMMIORead, &g_aArgsIOMMMIORead[0], RT_ELEMENTS(g_aArgsIOMMMIORead), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1051 { "IOMMMIOWrite", (void *)(uintptr_t)&IOMMMIOWrite, &g_aArgsIOMMMIOWrite[0], RT_ELEMENTS(g_aArgsIOMMMIOWrite), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1052 { "MMR3HeapAlloc", (void *)(uintptr_t)&MMR3HeapAlloc, &g_aArgsMMR3HeapAlloc[0], RT_ELEMENTS(g_aArgsMMR3HeapAlloc), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1053 { "MMR3HeapAllocZ", (void *)(uintptr_t)&MMR3HeapAllocZ, &g_aArgsMMR3HeapAllocZ[0], RT_ELEMENTS(g_aArgsMMR3HeapAllocZ), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1054 { "MMR3PhysGetRamSize", (void *)(uintptr_t)&MMR3PhysGetRamSize, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint64_t), NULL },
1055 { "PATMIsPatchGCAddr", (void *)(uintptr_t)&PATMIsPatchGCAddr, &g_aArgsPATMIsPatchGCAddr[0], RT_ELEMENTS(g_aArgsPATMIsPatchGCAddr), REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL },
1056 { "PATMR3QueryOpcode", (void *)(uintptr_t)&PATMR3QueryOpcode, &g_aArgsPATMR3QueryOpcode[0], RT_ELEMENTS(g_aArgsPATMR3QueryOpcode), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1057 { "PATMR3QueryPatchMemGC", (void *)(uintptr_t)&PATMR3QueryPatchMemGC, &g_aArgsPATMR3QueryPatchMem[0], RT_ELEMENTS(g_aArgsPATMR3QueryPatchMem), REMFNDESC_FLAGS_RET_INT, sizeof(RTGCPTR), NULL },
1058 { "PATMR3QueryPatchMemHC", (void *)(uintptr_t)&PATMR3QueryPatchMemHC, &g_aArgsPATMR3QueryPatchMem[0], RT_ELEMENTS(g_aArgsPATMR3QueryPatchMem), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1059 { "PDMApicGetBase", (void *)(uintptr_t)&PDMApicGetBase, &g_aArgsPDMApicGetBase[0], RT_ELEMENTS(g_aArgsPDMApicGetBase), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1060 { "PDMApicGetTPR", (void *)(uintptr_t)&PDMApicGetTPR, &g_aArgsPDMApicGetTPR[0], RT_ELEMENTS(g_aArgsPDMApicGetTPR), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1061 { "PDMApicSetBase", (void *)(uintptr_t)&PDMApicSetBase, &g_aArgsPDMApicSetBase[0], RT_ELEMENTS(g_aArgsPDMApicSetBase), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1062 { "PDMApicSetTPR", (void *)(uintptr_t)&PDMApicSetTPR, &g_aArgsPDMApicSetTPR[0], RT_ELEMENTS(g_aArgsPDMApicSetTPR), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1063 { "PDMR3DmaRun", (void *)(uintptr_t)&PDMR3DmaRun, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1064 { "PDMGetInterrupt", (void *)(uintptr_t)&PDMGetInterrupt, &g_aArgsPDMGetInterrupt[0], RT_ELEMENTS(g_aArgsPDMGetInterrupt), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1065 { "PDMIsaSetIrq", (void *)(uintptr_t)&PDMIsaSetIrq, &g_aArgsPDMIsaSetIrq[0], RT_ELEMENTS(g_aArgsPDMIsaSetIrq), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1066 { "PGMGetGuestMode", (void *)(uintptr_t)&PGMGetGuestMode, &g_aArgsPGMGetGuestMode[0], RT_ELEMENTS(g_aArgsPGMGetGuestMode), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1067 { "PGMGstGetPage", (void *)(uintptr_t)&PGMGstGetPage, &g_aArgsPGMGstGetPage[0], RT_ELEMENTS(g_aArgsPGMGstGetPage), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1068 { "PGMInvalidatePage", (void *)(uintptr_t)&PGMInvalidatePage, &g_aArgsPGMInvalidatePage[0], RT_ELEMENTS(g_aArgsPGMInvalidatePage), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1069 { "PGMPhysGCPhys2HCPtr", (void *)(uintptr_t)&PGMPhysGCPhys2HCPtr, &g_aArgsPGMPhysGCPhys2HCPtr[0], RT_ELEMENTS(g_aArgsPGMPhysGCPhys2HCPtr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1070 { "PGMPhysGCPtr2HCPtrByGstCR3", (void *)(uintptr_t)&PGMPhysGCPtr2HCPtrByGstCR3, &g_aArgsPGMPhysGCPtr2HCPtrByGstCR3[0], RT_ELEMENTS(g_aArgsPGMPhysGCPtr2HCPtrByGstCR3), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1071#ifndef VBOX_WITH_NEW_PHYS_CODE
1072 { "PGM3PhysGrowRange", (void *)(uintptr_t)&PGM3PhysGrowRange, &g_aArgsPGM3PhysGrowRange[0], RT_ELEMENTS(g_aArgsPGM3PhysGrowRange), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1073#endif
1074 { "PGMPhysIsGCPhysValid", (void *)(uintptr_t)&PGMPhysIsGCPhysValid, &g_aArgsPGMPhysIsGCPhysValid[0], RT_ELEMENTS(g_aArgsPGMPhysIsGCPhysValid), REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL },
1075 { "PGMPhysIsA20Enabled", (void *)(uintptr_t)&PGMPhysIsA20Enabled, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL },
1076 { "PGMPhysRead", (void *)(uintptr_t)&PGMPhysRead, &g_aArgsPGMPhysRead[0], RT_ELEMENTS(g_aArgsPGMPhysRead), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1077 { "PGMPhysReadGCPtr", (void *)(uintptr_t)&PGMPhysReadGCPtr, &g_aArgsPGMPhysReadGCPtr[0], RT_ELEMENTS(g_aArgsPGMPhysReadGCPtr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1078 { "PGMPhysReadGCPtr", (void *)(uintptr_t)&PGMPhysReadGCPtr, &g_aArgsPGMPhysReadGCPtr[0], RT_ELEMENTS(g_aArgsPGMPhysReadGCPtr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1079 { "PGMPhysWrite", (void *)(uintptr_t)&PGMPhysWrite, &g_aArgsPGMPhysWrite[0], RT_ELEMENTS(g_aArgsPGMPhysWrite), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1080 { "PGMChangeMode", (void *)(uintptr_t)&PGMChangeMode, &g_aArgsPGMChangeMode[0], RT_ELEMENTS(g_aArgsPGMChangeMode), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1081 { "PGMFlushTLB", (void *)(uintptr_t)&PGMFlushTLB, &g_aArgsPGMFlushTLB[0], RT_ELEMENTS(g_aArgsPGMFlushTLB), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1082 { "PGMR3PhysReadU8", (void *)(uintptr_t)&PGMR3PhysReadU8, &g_aArgsPGMR3PhysReadUxx[0], RT_ELEMENTS(g_aArgsPGMR3PhysReadUxx), REMFNDESC_FLAGS_RET_INT, sizeof(uint8_t), NULL },
1083 { "PGMR3PhysReadU16", (void *)(uintptr_t)&PGMR3PhysReadU16, &g_aArgsPGMR3PhysReadUxx[0], RT_ELEMENTS(g_aArgsPGMR3PhysReadUxx), REMFNDESC_FLAGS_RET_INT, sizeof(uint16_t), NULL },
1084 { "PGMR3PhysReadU32", (void *)(uintptr_t)&PGMR3PhysReadU32, &g_aArgsPGMR3PhysReadUxx[0], RT_ELEMENTS(g_aArgsPGMR3PhysReadUxx), REMFNDESC_FLAGS_RET_INT, sizeof(uint32_t), NULL },
1085 { "PGMR3PhysReadU64", (void *)(uintptr_t)&PGMR3PhysReadU64, &g_aArgsPGMR3PhysReadUxx[0], RT_ELEMENTS(g_aArgsPGMR3PhysReadUxx), REMFNDESC_FLAGS_RET_INT, sizeof(uint64_t), NULL },
1086 { "PGMR3PhysWriteU8", (void *)(uintptr_t)&PGMR3PhysWriteU8, &g_aArgsPGMR3PhysWriteU8[0], RT_ELEMENTS(g_aArgsPGMR3PhysWriteU8), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1087 { "PGMR3PhysWriteU16", (void *)(uintptr_t)&PGMR3PhysWriteU16, &g_aArgsPGMR3PhysWriteU16[0], RT_ELEMENTS(g_aArgsPGMR3PhysWriteU16), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1088 { "PGMR3PhysWriteU32", (void *)(uintptr_t)&PGMR3PhysWriteU32, &g_aArgsPGMR3PhysWriteU32[0], RT_ELEMENTS(g_aArgsPGMR3PhysWriteU32), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1089 { "PGMR3PhysWriteU64", (void *)(uintptr_t)&PGMR3PhysWriteU64, &g_aArgsPGMR3PhysWriteU64[0], RT_ELEMENTS(g_aArgsPGMR3PhysWriteU32), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1090 { "SSMR3GetGCPtr", (void *)(uintptr_t)&SSMR3GetGCPtr, &g_aArgsSSMR3GetGCPtr[0], RT_ELEMENTS(g_aArgsSSMR3GetGCPtr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1091 { "SSMR3GetMem", (void *)(uintptr_t)&SSMR3GetMem, &g_aArgsSSMR3GetMem[0], RT_ELEMENTS(g_aArgsSSMR3GetMem), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1092 { "SSMR3GetU32", (void *)(uintptr_t)&SSMR3GetU32, &g_aArgsSSMR3GetU32[0], RT_ELEMENTS(g_aArgsSSMR3GetU32), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1093 { "SSMR3GetUInt", (void *)(uintptr_t)&SSMR3GetUInt, &g_aArgsSSMR3GetUInt[0], RT_ELEMENTS(g_aArgsSSMR3GetUInt), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1094 { "SSMR3PutGCPtr", (void *)(uintptr_t)&SSMR3PutGCPtr, &g_aArgsSSMR3PutGCPtr[0], RT_ELEMENTS(g_aArgsSSMR3PutGCPtr), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1095 { "SSMR3PutMem", (void *)(uintptr_t)&SSMR3PutMem, &g_aArgsSSMR3PutMem[0], RT_ELEMENTS(g_aArgsSSMR3PutMem), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1096 { "SSMR3PutU32", (void *)(uintptr_t)&SSMR3PutU32, &g_aArgsSSMR3PutU32[0], RT_ELEMENTS(g_aArgsSSMR3PutU32), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1097 { "SSMR3PutUInt", (void *)(uintptr_t)&SSMR3PutUInt, &g_aArgsSSMR3PutUInt[0], RT_ELEMENTS(g_aArgsSSMR3PutUInt), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1098 { "SSMR3RegisterInternal", (void *)(uintptr_t)&SSMR3RegisterInternal, &g_aArgsSSMR3RegisterInternal[0], RT_ELEMENTS(g_aArgsSSMR3RegisterInternal), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1099 { "STAMR3Register", (void *)(uintptr_t)&STAMR3Register, &g_aArgsSTAMR3Register[0], RT_ELEMENTS(g_aArgsSTAMR3Register), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1100 { "TMCpuTickGet", (void *)(uintptr_t)&TMCpuTickGet, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint64_t), NULL },
1101 { "TMCpuTickPause", (void *)(uintptr_t)&TMCpuTickPause, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1102 { "TMCpuTickResume", (void *)(uintptr_t)&TMCpuTickResume, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1103 { "TMNotifyEndOfExecution", (void *)(uintptr_t)&TMNotifyEndOfExecution, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1104 { "TMNotifyStartOfExecution", (void *)(uintptr_t)&TMNotifyStartOfExecution, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1105 { "TMTimerPoll", (void *)(uintptr_t)&TMTimerPoll, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(uint64_t), NULL },
1106 { "TMR3TimerQueuesDo", (void *)(uintptr_t)&TMR3TimerQueuesDo, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1107 { "TMVirtualPause", (void *)(uintptr_t)&TMVirtualPause, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1108 { "TMVirtualResume", (void *)(uintptr_t)&TMVirtualResume, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1109 { "TRPMAssertTrap", (void *)(uintptr_t)&TRPMAssertTrap, &g_aArgsTRPMAssertTrap[0], RT_ELEMENTS(g_aArgsTRPMAssertTrap), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1110 { "TRPMGetErrorCode", (void *)(uintptr_t)&TRPMGetErrorCode, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(RTGCUINT), NULL },
1111 { "TRPMGetFaultAddress", (void *)(uintptr_t)&TRPMGetFaultAddress, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(RTGCUINTPTR),NULL },
1112 { "TRPMQueryTrap", (void *)(uintptr_t)&TRPMQueryTrap, &g_aArgsTRPMQueryTrap[0], RT_ELEMENTS(g_aArgsTRPMQueryTrap), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1113 { "TRPMResetTrap", (void *)(uintptr_t)&TRPMResetTrap, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1114 { "TRPMSetErrorCode", (void *)(uintptr_t)&TRPMSetErrorCode, &g_aArgsTRPMSetErrorCode[0], RT_ELEMENTS(g_aArgsTRPMSetErrorCode), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1115 { "TRPMSetFaultAddress", (void *)(uintptr_t)&TRPMSetFaultAddress, &g_aArgsTRPMSetFaultAddress[0], RT_ELEMENTS(g_aArgsTRPMSetFaultAddress), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1116 { "VMMR3Lock", (void *)(uintptr_t)&VMMR3Lock, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1117 { "VMMR3Unlock", (void *)(uintptr_t)&VMMR3Unlock, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1118 { "VMR3ReqCall", (void *)(uintptr_t)&VMR3ReqCall, &g_aArgsVMR3ReqCall[0], RT_ELEMENTS(g_aArgsVMR3ReqCall), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1119 { "VMR3ReqFree", (void *)(uintptr_t)&VMR3ReqFree, &g_aArgsVMR3ReqFree[0], RT_ELEMENTS(g_aArgsVMR3ReqFree), REMFNDESC_FLAGS_RET_INT | REMFNDESC_FLAGS_ELLIPSIS, sizeof(int), NULL },
1120// { "", (void *)(uintptr_t)&, &g_aArgsVM[0], RT_ELEMENTS(g_aArgsVM), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1121};
1122
1123
1124/**
1125 * Descriptors for the functions imported from VBoxRT.
1126 */
1127static REMFNDESC g_aRTImports[] =
1128{
1129 { "AssertMsg1", (void *)(uintptr_t)&AssertMsg1, &g_aArgsAssertMsg1[0], RT_ELEMENTS(g_aArgsAssertMsg1), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1130 { "AssertMsg2", (void *)(uintptr_t)&AssertMsg2, &g_aArgsAssertMsg2[0], RT_ELEMENTS(g_aArgsAssertMsg2), REMFNDESC_FLAGS_RET_VOID | REMFNDESC_FLAGS_ELLIPSIS, 0, NULL },
1131 { "RTAssertDoBreakpoint", (void *)(uintptr_t)&RTAssertDoBreakpoint, NULL, 0, REMFNDESC_FLAGS_RET_INT, sizeof(bool), NULL },
1132 { "RTLogDefaultInstance", (void *)(uintptr_t)&RTLogDefaultInstance, NULL, 0, REMFNDESC_FLAGS_RET_INT, sizeof(PRTLOGGER), NULL },
1133 { "RTLogRelDefaultInstance", (void *)(uintptr_t)&RTLogRelDefaultInstance, NULL, 0, REMFNDESC_FLAGS_RET_INT, sizeof(PRTLOGGER), NULL },
1134 { "RTLogFlags", (void *)(uintptr_t)&RTLogFlags, &g_aArgsRTLogFlags[0], RT_ELEMENTS(g_aArgsRTLogFlags), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1135 { "RTLogFlush", (void *)(uintptr_t)&RTLogFlush, &g_aArgsRTLogFlush[0], RT_ELEMENTS(g_aArgsRTLogFlush), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1136 { "RTLogLoggerEx", (void *)(uintptr_t)&RTLogLoggerEx, &g_aArgsRTLogLoggerEx[0], RT_ELEMENTS(g_aArgsRTLogLoggerEx), REMFNDESC_FLAGS_RET_VOID | REMFNDESC_FLAGS_ELLIPSIS, 0, NULL },
1137 { "RTLogLoggerExV", (void *)(uintptr_t)&RTLogLoggerExV, &g_aArgsRTLogLoggerExV[0], RT_ELEMENTS(g_aArgsRTLogLoggerExV), REMFNDESC_FLAGS_RET_VOID | REMFNDESC_FLAGS_VALIST, 0, NULL },
1138 { "RTLogPrintf", (void *)(uintptr_t)&RTLogPrintf, &g_aArgsRTLogPrintf[0], RT_ELEMENTS(g_aArgsRTLogPrintf), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1139 { "RTMemAlloc", (void *)(uintptr_t)&RTMemAlloc, &g_aArgsSIZE_T[0], RT_ELEMENTS(g_aArgsSIZE_T), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1140 { "RTMemExecAlloc", (void *)(uintptr_t)&RTMemExecAlloc, &g_aArgsSIZE_T[0], RT_ELEMENTS(g_aArgsSIZE_T), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1141 { "RTMemExecFree", (void *)(uintptr_t)&RTMemExecFree, &g_aArgsPTR[0], RT_ELEMENTS(g_aArgsPTR), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1142 { "RTMemFree", (void *)(uintptr_t)&RTMemFree, &g_aArgsPTR[0], RT_ELEMENTS(g_aArgsPTR), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1143 { "RTMemPageAlloc", (void *)(uintptr_t)&RTMemPageAlloc, &g_aArgsSIZE_T[0], RT_ELEMENTS(g_aArgsSIZE_T), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1144 { "RTMemPageFree", (void *)(uintptr_t)&RTMemPageFree, &g_aArgsPTR[0], RT_ELEMENTS(g_aArgsPTR), REMFNDESC_FLAGS_RET_VOID, 0, NULL },
1145 { "RTMemProtect", (void *)(uintptr_t)&RTMemProtect, &g_aArgsRTMemProtect[0], RT_ELEMENTS(g_aArgsRTMemProtect), REMFNDESC_FLAGS_RET_INT, sizeof(int), NULL },
1146 { "RTStrPrintf", (void *)(uintptr_t)&RTStrPrintf, &g_aArgsRTStrPrintf[0], RT_ELEMENTS(g_aArgsRTStrPrintf), REMFNDESC_FLAGS_RET_INT | REMFNDESC_FLAGS_ELLIPSIS, sizeof(size_t), NULL },
1147 { "RTStrPrintfV", (void *)(uintptr_t)&RTStrPrintfV, &g_aArgsRTStrPrintfV[0], RT_ELEMENTS(g_aArgsRTStrPrintfV), REMFNDESC_FLAGS_RET_INT | REMFNDESC_FLAGS_VALIST, sizeof(size_t), NULL },
1148 { "RTThreadSelf", (void *)(uintptr_t)&RTThreadSelf, NULL, 0, REMFNDESC_FLAGS_RET_INT, sizeof(RTTHREAD), NULL },
1149 { "RTThreadNativeSelf", (void *)(uintptr_t)&RTThreadNativeSelf, NULL, 0, REMFNDESC_FLAGS_RET_INT, sizeof(RTNATIVETHREAD), NULL },
1150 { "RTThreadGetWriteLockCount", (void *)(uintptr_t)&RTThreadGetWriteLockCount, &g_aArgsThread[0], 0, REMFNDESC_FLAGS_RET_INT, sizeof(int32_t), NULL },
1151};
1152
1153
1154/**
1155 * Descriptors for the functions imported from VBoxRT.
1156 */
1157static REMFNDESC g_aCRTImports[] =
1158{
1159 { "memcpy", (void *)(uintptr_t)&memcpy, &g_aArgsmemcpy[0], RT_ELEMENTS(g_aArgsmemcpy), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL },
1160 { "memset", (void *)(uintptr_t)&memset, &g_aArgsmemset[0], RT_ELEMENTS(g_aArgsmemset), REMFNDESC_FLAGS_RET_INT, sizeof(void *), NULL }
1161/*
1162floor floor
1163memcpy memcpy
1164sqrt sqrt
1165sqrtf sqrtf
1166*/
1167};
1168
1169
1170# if defined(USE_REM_CALLING_CONVENTION_GLUE) || defined(USE_REM_IMPORT_JUMP_GLUE)
1171/** LIFO of read-write-executable memory chunks used for wrappers. */
1172static PREMEXECMEM g_pExecMemHead;
1173# endif
1174
1175
1176/*******************************************************************************
1177* Internal Functions *
1178*******************************************************************************/
1179static int remGenerateExportGlue(PRTUINTPTR pValue, PCREMFNDESC pDesc);
1180
1181# ifdef USE_REM_CALLING_CONVENTION_GLUE
1182DECLASM(int) WrapGCC2MSC0Int(void); DECLASM(int) WrapGCC2MSC0Int_EndProc(void);
1183DECLASM(int) WrapGCC2MSC1Int(void); DECLASM(int) WrapGCC2MSC1Int_EndProc(void);
1184DECLASM(int) WrapGCC2MSC2Int(void); DECLASM(int) WrapGCC2MSC2Int_EndProc(void);
1185DECLASM(int) WrapGCC2MSC3Int(void); DECLASM(int) WrapGCC2MSC3Int_EndProc(void);
1186DECLASM(int) WrapGCC2MSC4Int(void); DECLASM(int) WrapGCC2MSC4Int_EndProc(void);
1187DECLASM(int) WrapGCC2MSC5Int(void); DECLASM(int) WrapGCC2MSC5Int_EndProc(void);
1188DECLASM(int) WrapGCC2MSC6Int(void); DECLASM(int) WrapGCC2MSC6Int_EndProc(void);
1189DECLASM(int) WrapGCC2MSC7Int(void); DECLASM(int) WrapGCC2MSC7Int_EndProc(void);
1190DECLASM(int) WrapGCC2MSC8Int(void); DECLASM(int) WrapGCC2MSC8Int_EndProc(void);
1191DECLASM(int) WrapGCC2MSC9Int(void); DECLASM(int) WrapGCC2MSC9Int_EndProc(void);
1192DECLASM(int) WrapGCC2MSC10Int(void); DECLASM(int) WrapGCC2MSC10Int_EndProc(void);
1193DECLASM(int) WrapGCC2MSC11Int(void); DECLASM(int) WrapGCC2MSC11Int_EndProc(void);
1194DECLASM(int) WrapGCC2MSC12Int(void); DECLASM(int) WrapGCC2MSC12Int_EndProc(void);
1195DECLASM(int) WrapGCC2MSCVariadictInt(void); DECLASM(int) WrapGCC2MSCVariadictInt_EndProc(void);
1196DECLASM(int) WrapGCC2MSC_SSMR3RegisterInternal(void); DECLASM(int) WrapGCC2MSC_SSMR3RegisterInternal_EndProc(void);
1197
1198DECLASM(int) WrapMSC2GCC0Int(void); DECLASM(int) WrapMSC2GCC0Int_EndProc(void);
1199DECLASM(int) WrapMSC2GCC1Int(void); DECLASM(int) WrapMSC2GCC1Int_EndProc(void);
1200DECLASM(int) WrapMSC2GCC2Int(void); DECLASM(int) WrapMSC2GCC2Int_EndProc(void);
1201DECLASM(int) WrapMSC2GCC3Int(void); DECLASM(int) WrapMSC2GCC3Int_EndProc(void);
1202DECLASM(int) WrapMSC2GCC4Int(void); DECLASM(int) WrapMSC2GCC4Int_EndProc(void);
1203DECLASM(int) WrapMSC2GCC5Int(void); DECLASM(int) WrapMSC2GCC5Int_EndProc(void);
1204DECLASM(int) WrapMSC2GCC6Int(void); DECLASM(int) WrapMSC2GCC6Int_EndProc(void);
1205DECLASM(int) WrapMSC2GCC7Int(void); DECLASM(int) WrapMSC2GCC7Int_EndProc(void);
1206DECLASM(int) WrapMSC2GCC8Int(void); DECLASM(int) WrapMSC2GCC8Int_EndProc(void);
1207DECLASM(int) WrapMSC2GCC9Int(void); DECLASM(int) WrapMSC2GCC9Int_EndProc(void);
1208# endif
1209
1210
1211# if defined(USE_REM_CALLING_CONVENTION_GLUE) || defined(USE_REM_IMPORT_JUMP_GLUE)
1212/**
1213 * Allocates a block of memory for glue code.
1214 *
1215 * The returned memory is padded with INT3s.
1216 *
1217 * @returns Pointer to the allocated memory.
1218 * @param The amount of memory to allocate.
1219 */
1220static void *remAllocGlue(size_t cb)
1221{
1222 PREMEXECMEM pCur = g_pExecMemHead;
1223 uint32_t cbAligned = (uint32_t)RT_ALIGN_32(cb, 32);
1224 while (pCur)
1225 {
1226 if (pCur->cb - pCur->off >= cbAligned)
1227 {
1228 void *pv = (uint8_t *)pCur + pCur->off;
1229 pCur->off += cbAligned;
1230 return memset(pv, 0xcc, cbAligned);
1231 }
1232 pCur = pCur->pNext;
1233 }
1234
1235 /* add a new chunk */
1236 AssertReturn(_64K - RT_ALIGN_Z(sizeof(*pCur), 32) > cbAligned, NULL);
1237 pCur = (PREMEXECMEM)RTMemExecAlloc(_64K);
1238 AssertReturn(pCur, NULL);
1239 pCur->cb = _64K;
1240 pCur->off = RT_ALIGN_32(sizeof(*pCur), 32) + cbAligned;
1241 pCur->pNext = g_pExecMemHead;
1242 g_pExecMemHead = pCur;
1243 return memset((uint8_t *)pCur + RT_ALIGN_Z(sizeof(*pCur), 32), 0xcc, cbAligned);
1244}
1245# endif /* USE_REM_CALLING_CONVENTION_GLUE || USE_REM_IMPORT_JUMP_GLUE */
1246
1247
1248# ifdef USE_REM_CALLING_CONVENTION_GLUE
1249/**
1250 * Checks if a function is all straight forward integers.
1251 *
1252 * @returns True if it's simple, false if it's bothersome.
1253 * @param pDesc The function descriptor.
1254 */
1255static bool remIsFunctionAllInts(PCREMFNDESC pDesc)
1256{
1257 if ( ( (pDesc->fFlags & REMFNDESC_FLAGS_RET_TYPE_MASK) != REMFNDESC_FLAGS_RET_INT
1258 || pDesc->cbReturn > sizeof(uint64_t))
1259 && (pDesc->fFlags & REMFNDESC_FLAGS_RET_TYPE_MASK) != REMFNDESC_FLAGS_RET_VOID)
1260 return false;
1261 unsigned i = pDesc->cParams;
1262 while (i-- > 0)
1263 switch (pDesc->paParams[i].fFlags & REMPARMDESC_FLAGS_TYPE_MASK)
1264 {
1265 case REMPARMDESC_FLAGS_INT:
1266 case REMPARMDESC_FLAGS_GCPTR:
1267 case REMPARMDESC_FLAGS_GCPHYS:
1268 case REMPARMDESC_FLAGS_HCPHYS:
1269 break;
1270
1271 default:
1272 AssertReleaseMsgFailed(("Invalid param flags %#x for #%d of %s!\n", pDesc->paParams[i].fFlags, i, pDesc->pszName));
1273 case REMPARMDESC_FLAGS_VALIST:
1274 case REMPARMDESC_FLAGS_ELLIPSIS:
1275 case REMPARMDESC_FLAGS_FLOAT:
1276 case REMPARMDESC_FLAGS_STRUCT:
1277 case REMPARMDESC_FLAGS_PFN:
1278 return false;
1279 }
1280 return true;
1281}
1282
1283
1284/**
1285 * Checks if the function has an ellipsis (...) argument.
1286 *
1287 * @returns true if it has an ellipsis, otherwise false.
1288 * @param pDesc The function descriptor.
1289 */
1290static bool remHasFunctionEllipsis(PCREMFNDESC pDesc)
1291{
1292 unsigned i = pDesc->cParams;
1293 while (i-- > 0)
1294 if ((pDesc->paParams[i].fFlags & REMPARMDESC_FLAGS_TYPE_MASK) == REMPARMDESC_FLAGS_ELLIPSIS)
1295 return true;
1296 return false;
1297}
1298
1299
1300/**
1301 * Checks if the function uses floating point (FP) arguments or return value.
1302 *
1303 * @returns true if it uses floating point, otherwise false.
1304 * @param pDesc The function descriptor.
1305 */
1306static bool remIsFunctionUsingFP(PCREMFNDESC pDesc)
1307{
1308 if ((pDesc->fFlags & REMFNDESC_FLAGS_RET_TYPE_MASK) == REMFNDESC_FLAGS_RET_FLOAT)
1309 return true;
1310 unsigned i = pDesc->cParams;
1311 while (i-- > 0)
1312 if ((pDesc->paParams[i].fFlags & REMPARMDESC_FLAGS_TYPE_MASK) == REMPARMDESC_FLAGS_FLOAT)
1313 return true;
1314 return false;
1315}
1316
1317
1318/** @name The export and import fixups.
1319 * @{ */
1320#define REM_FIXUP_32_REAL_STUFF UINT32_C(0xdeadbeef)
1321#define REM_FIXUP_64_REAL_STUFF UINT64_C(0xdeadf00df00ddead)
1322#define REM_FIXUP_64_DESC UINT64_C(0xdead00010001dead)
1323#define REM_FIXUP_64_LOG_ENTRY UINT64_C(0xdead00020002dead)
1324#define REM_FIXUP_64_LOG_EXIT UINT64_C(0xdead00030003dead)
1325#define REM_FIXUP_64_WRAP_GCC_CB UINT64_C(0xdead00040004dead)
1326/** @} */
1327
1328
1329/**
1330 * Entry logger function.
1331 *
1332 * @param pDesc The description.
1333 */
1334DECLASM(void) remLogEntry(PCREMFNDESC pDesc)
1335{
1336 RTPrintf("calling %s\n", pDesc->pszName);
1337}
1338
1339
1340/**
1341 * Exit logger function.
1342 *
1343 * @param pDesc The description.
1344 * @param pvRet The return code.
1345 */
1346DECLASM(void) remLogExit(PCREMFNDESC pDesc, void *pvRet)
1347{
1348 RTPrintf("returning %p from %s\n", pvRet, pDesc->pszName);
1349}
1350
1351
1352/**
1353 * Creates a wrapper for the specified callback function at run time.
1354 *
1355 * @param pDesc The function descriptor.
1356 * @param pValue Upon entry *pValue contains the address of the function to be wrapped.
1357 * Upon return *pValue contains the address of the wrapper glue function.
1358 * @param iParam The parameter index in the function descriptor (0 based).
1359 * If UINT32_MAX pDesc is the descriptor for *pValue.
1360 */
1361DECLASM(void) remWrapGCCCallback(PCREMFNDESC pDesc, PRTUINTPTR pValue, uint32_t iParam)
1362{
1363 AssertPtr(pDesc);
1364 AssertPtr(pValue);
1365
1366 /*
1367 * Simple?
1368 */
1369 if (!*pValue)
1370 return;
1371
1372 /*
1373 * Locate the right function descriptor.
1374 */
1375 if (iParam != UINT32_MAX)
1376 {
1377 AssertRelease(iParam < pDesc->cParams);
1378 pDesc = (PCREMFNDESC)pDesc->paParams[iParam].pvExtra;
1379 AssertPtr(pDesc);
1380 }
1381
1382 /*
1383 * When we get serious, here is where to insert the hash table lookup.
1384 */
1385
1386 /*
1387 * Create a new glue patch.
1388 */
1389#ifdef RT_OS_WINDOWS
1390 int rc = remGenerateExportGlue(pValue, pDesc);
1391#else
1392#error "port me"
1393#endif
1394 AssertReleaseRC(rc);
1395
1396 /*
1397 * Add it to the hash (later)
1398 */
1399}
1400
1401
1402/**
1403 * Fixes export glue.
1404 *
1405 * @param pvGlue The glue code.
1406 * @param cb The size of the glue code.
1407 * @param pvExport The address of the export we're wrapping.
1408 * @param pDesc The export descriptor.
1409 */
1410static void remGenerateExportGlueFixup(void *pvGlue, size_t cb, uintptr_t uExport, PCREMFNDESC pDesc)
1411{
1412 union
1413 {
1414 uint8_t *pu8;
1415 int32_t *pi32;
1416 uint32_t *pu32;
1417 uint64_t *pu64;
1418 void *pv;
1419 } u;
1420 u.pv = pvGlue;
1421
1422 while (cb >= 4)
1423 {
1424 /** @todo add defines for the fixup constants... */
1425 if (*u.pu32 == REM_FIXUP_32_REAL_STUFF)
1426 {
1427 /* 32-bit rel jmp/call to real export. */
1428 *u.pi32 = uExport - (uintptr_t)(u.pi32 + 1);
1429 Assert((uintptr_t)(u.pi32 + 1) + *u.pi32 == uExport);
1430 u.pi32++;
1431 cb -= 4;
1432 continue;
1433 }
1434 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_REAL_STUFF)
1435 {
1436 /* 64-bit address to the real export. */
1437 *u.pu64++ = uExport;
1438 cb -= 8;
1439 continue;
1440 }
1441 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_DESC)
1442 {
1443 /* 64-bit address to the descriptor. */
1444 *u.pu64++ = (uintptr_t)pDesc;
1445 cb -= 8;
1446 continue;
1447 }
1448 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_WRAP_GCC_CB)
1449 {
1450 /* 64-bit address to the entry logger function. */
1451 *u.pu64++ = (uintptr_t)remWrapGCCCallback;
1452 cb -= 8;
1453 continue;
1454 }
1455 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_LOG_ENTRY)
1456 {
1457 /* 64-bit address to the entry logger function. */
1458 *u.pu64++ = (uintptr_t)remLogEntry;
1459 cb -= 8;
1460 continue;
1461 }
1462 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_LOG_EXIT)
1463 {
1464 /* 64-bit address to the entry logger function. */
1465 *u.pu64++ = (uintptr_t)remLogExit;
1466 cb -= 8;
1467 continue;
1468 }
1469
1470 /* move on. */
1471 u.pu8++;
1472 cb--;
1473 }
1474}
1475
1476
1477/**
1478 * Fixes import glue.
1479 *
1480 * @param pvGlue The glue code.
1481 * @param cb The size of the glue code.
1482 * @param pDesc The import descriptor.
1483 */
1484static void remGenerateImportGlueFixup(void *pvGlue, size_t cb, PCREMFNDESC pDesc)
1485{
1486 union
1487 {
1488 uint8_t *pu8;
1489 int32_t *pi32;
1490 uint32_t *pu32;
1491 uint64_t *pu64;
1492 void *pv;
1493 } u;
1494 u.pv = pvGlue;
1495
1496 while (cb >= 4)
1497 {
1498 if (*u.pu32 == REM_FIXUP_32_REAL_STUFF)
1499 {
1500 /* 32-bit rel jmp/call to real function. */
1501 *u.pi32 = (uintptr_t)pDesc->pv - (uintptr_t)(u.pi32 + 1);
1502 Assert((uintptr_t)(u.pi32 + 1) + *u.pi32 == (uintptr_t)pDesc->pv);
1503 u.pi32++;
1504 cb -= 4;
1505 continue;
1506 }
1507 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_REAL_STUFF)
1508 {
1509 /* 64-bit address to the real function. */
1510 *u.pu64++ = (uintptr_t)pDesc->pv;
1511 cb -= 8;
1512 continue;
1513 }
1514 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_DESC)
1515 {
1516 /* 64-bit address to the descriptor. */
1517 *u.pu64++ = (uintptr_t)pDesc;
1518 cb -= 8;
1519 continue;
1520 }
1521 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_WRAP_GCC_CB)
1522 {
1523 /* 64-bit address to the entry logger function. */
1524 *u.pu64++ = (uintptr_t)remWrapGCCCallback;
1525 cb -= 8;
1526 continue;
1527 }
1528 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_LOG_ENTRY)
1529 {
1530 /* 64-bit address to the entry logger function. */
1531 *u.pu64++ = (uintptr_t)remLogEntry;
1532 cb -= 8;
1533 continue;
1534 }
1535 if (cb >= 8 && *u.pu64 == REM_FIXUP_64_LOG_EXIT)
1536 {
1537 /* 64-bit address to the entry logger function. */
1538 *u.pu64++ = (uintptr_t)remLogExit;
1539 cb -= 8;
1540 continue;
1541 }
1542
1543 /* move on. */
1544 u.pu8++;
1545 cb--;
1546 }
1547}
1548
1549# endif /* USE_REM_CALLING_CONVENTION_GLUE */
1550
1551
1552/**
1553 * Generate wrapper glue code for an export.
1554 *
1555 * This is only used on win64 when loading a 64-bit linux module. So, on other
1556 * platforms it will not do anything.
1557 *
1558 * @returns VBox status code.
1559 * @param pValue IN: Where to get the address of the function to wrap.
1560 * OUT: Where to store the glue address.
1561 * @param pDesc The export descriptor.
1562 */
1563static int remGenerateExportGlue(PRTUINTPTR pValue, PCREMFNDESC pDesc)
1564{
1565# ifdef USE_REM_CALLING_CONVENTION_GLUE
1566 uintptr_t *ppfn = (uintptr_t *)pDesc->pv;
1567
1568 uintptr_t pfn = 0; /* a little hack for the callback glue */
1569 if (!ppfn)
1570 ppfn = &pfn;
1571
1572 if (!*ppfn)
1573 {
1574 if (remIsFunctionAllInts(pDesc))
1575 {
1576 static const struct { void *pvStart, *pvEnd; } s_aTemplates[] =
1577 {
1578 { (void *)&WrapMSC2GCC0Int, (void *)&WrapMSC2GCC0Int_EndProc },
1579 { (void *)&WrapMSC2GCC1Int, (void *)&WrapMSC2GCC1Int_EndProc },
1580 { (void *)&WrapMSC2GCC2Int, (void *)&WrapMSC2GCC2Int_EndProc },
1581 { (void *)&WrapMSC2GCC3Int, (void *)&WrapMSC2GCC3Int_EndProc },
1582 { (void *)&WrapMSC2GCC4Int, (void *)&WrapMSC2GCC4Int_EndProc },
1583 { (void *)&WrapMSC2GCC5Int, (void *)&WrapMSC2GCC5Int_EndProc },
1584 { (void *)&WrapMSC2GCC6Int, (void *)&WrapMSC2GCC6Int_EndProc },
1585 { (void *)&WrapMSC2GCC7Int, (void *)&WrapMSC2GCC7Int_EndProc },
1586 { (void *)&WrapMSC2GCC8Int, (void *)&WrapMSC2GCC8Int_EndProc },
1587 { (void *)&WrapMSC2GCC9Int, (void *)&WrapMSC2GCC9Int_EndProc },
1588 };
1589 const unsigned i = pDesc->cParams;
1590 AssertReleaseMsg(i < RT_ELEMENTS(s_aTemplates), ("%d (%s)\n", i, pDesc->pszName));
1591
1592 /* duplicate the patch. */
1593 const size_t cb = (uintptr_t)s_aTemplates[i].pvEnd - (uintptr_t)s_aTemplates[i].pvStart;
1594 uint8_t *pb = (uint8_t *)remAllocGlue(cb);
1595 AssertReturn(pb, VERR_NO_MEMORY);
1596 memcpy(pb, s_aTemplates[i].pvStart, cb);
1597
1598 /* fix it up. */
1599 remGenerateExportGlueFixup(pb, cb, *pValue, pDesc);
1600 *ppfn = (uintptr_t)pb;
1601 }
1602 else
1603 {
1604 /* custom hacks - it's simpler to make assembly templates than writing a more generic code generator... */
1605 static const struct { const char *pszName; PFNRT pvStart, pvEnd; } s_aTemplates[] =
1606 {
1607 { "somefunction", (PFNRT)&WrapMSC2GCC9Int, (PFNRT)&WrapMSC2GCC9Int_EndProc },
1608 };
1609 unsigned i;
1610 for (i = 0; i < RT_ELEMENTS(s_aTemplates); i++)
1611 if (!strcmp(pDesc->pszName, s_aTemplates[i].pszName))
1612 break;
1613 AssertReleaseMsgReturn(i < RT_ELEMENTS(s_aTemplates), ("Not implemented! %s\n", pDesc->pszName), VERR_NOT_IMPLEMENTED);
1614
1615 /* duplicate the patch. */
1616 const size_t cb = (uintptr_t)s_aTemplates[i].pvEnd - (uintptr_t)s_aTemplates[i].pvStart;
1617 uint8_t *pb = (uint8_t *)remAllocGlue(cb);
1618 AssertReturn(pb, VERR_NO_MEMORY);
1619 memcpy(pb, s_aTemplates[i].pvStart, cb);
1620
1621 /* fix it up. */
1622 remGenerateExportGlueFixup(pb, cb, *pValue, pDesc);
1623 *ppfn = (uintptr_t)pb;
1624 }
1625 }
1626 *pValue = *ppfn;
1627 return VINF_SUCCESS;
1628# else /* !USE_REM_CALLING_CONVENTION_GLUE */
1629 return VINF_SUCCESS;
1630# endif /* !USE_REM_CALLING_CONVENTION_GLUE */
1631}
1632
1633
1634/**
1635 * Generate wrapper glue code for an import.
1636 *
1637 * This is only used on win64 when loading a 64-bit linux module. So, on other
1638 * platforms it will simply return the address of the imported function
1639 * without generating any glue code.
1640 *
1641 * @returns VBox status code.
1642 * @param pValue Where to store the glue address.
1643 * @param pDesc The export descriptor.
1644 */
1645static int remGenerateImportGlue(PRTUINTPTR pValue, PREMFNDESC pDesc)
1646{
1647# if defined(USE_REM_CALLING_CONVENTION_GLUE) || defined(USE_REM_IMPORT_JUMP_GLUE)
1648 if (!pDesc->pvWrapper)
1649 {
1650# ifdef USE_REM_CALLING_CONVENTION_GLUE
1651 if (remIsFunctionAllInts(pDesc))
1652 {
1653 static const struct { void *pvStart, *pvEnd; } s_aTemplates[] =
1654 {
1655 { (void *)&WrapGCC2MSC0Int, (void *)&WrapGCC2MSC0Int_EndProc },
1656 { (void *)&WrapGCC2MSC1Int, (void *)&WrapGCC2MSC1Int_EndProc },
1657 { (void *)&WrapGCC2MSC2Int, (void *)&WrapGCC2MSC2Int_EndProc },
1658 { (void *)&WrapGCC2MSC3Int, (void *)&WrapGCC2MSC3Int_EndProc },
1659 { (void *)&WrapGCC2MSC4Int, (void *)&WrapGCC2MSC4Int_EndProc },
1660 { (void *)&WrapGCC2MSC5Int, (void *)&WrapGCC2MSC5Int_EndProc },
1661 { (void *)&WrapGCC2MSC6Int, (void *)&WrapGCC2MSC6Int_EndProc },
1662 { (void *)&WrapGCC2MSC7Int, (void *)&WrapGCC2MSC7Int_EndProc },
1663 { (void *)&WrapGCC2MSC8Int, (void *)&WrapGCC2MSC8Int_EndProc },
1664 { (void *)&WrapGCC2MSC9Int, (void *)&WrapGCC2MSC9Int_EndProc },
1665 { (void *)&WrapGCC2MSC10Int, (void *)&WrapGCC2MSC10Int_EndProc },
1666 { (void *)&WrapGCC2MSC11Int, (void *)&WrapGCC2MSC11Int_EndProc },
1667 { (void *)&WrapGCC2MSC12Int, (void *)&WrapGCC2MSC12Int_EndProc }
1668 };
1669 const unsigned i = pDesc->cParams;
1670 AssertReleaseMsg(i < RT_ELEMENTS(s_aTemplates), ("%d (%s)\n", i, pDesc->pszName));
1671
1672 /* duplicate the patch. */
1673 const size_t cb = (uintptr_t)s_aTemplates[i].pvEnd - (uintptr_t)s_aTemplates[i].pvStart;
1674 pDesc->pvWrapper = remAllocGlue(cb);
1675 AssertReturn(pDesc->pvWrapper, VERR_NO_MEMORY);
1676 memcpy(pDesc->pvWrapper, s_aTemplates[i].pvStart, cb);
1677
1678 /* fix it up. */
1679 remGenerateImportGlueFixup((uint8_t *)pDesc->pvWrapper, cb, pDesc);
1680 }
1681 else if ( remHasFunctionEllipsis(pDesc)
1682 && !remIsFunctionUsingFP(pDesc))
1683 {
1684 /* duplicate the patch. */
1685 const size_t cb = (uintptr_t)&WrapGCC2MSCVariadictInt_EndProc - (uintptr_t)&WrapGCC2MSCVariadictInt;
1686 pDesc->pvWrapper = remAllocGlue(cb);
1687 AssertReturn(pDesc->pvWrapper, VERR_NO_MEMORY);
1688 memcpy(pDesc->pvWrapper, (void *)&WrapGCC2MSCVariadictInt, cb);
1689
1690 /* fix it up. */
1691 remGenerateImportGlueFixup((uint8_t *)pDesc->pvWrapper, cb, pDesc);
1692 }
1693 else
1694 {
1695 /* custom hacks - it's simpler to make assembly templates than writing a more generic code generator... */
1696 static const struct { const char *pszName; PFNRT pvStart, pvEnd; } s_aTemplates[] =
1697 {
1698 { "SSMR3RegisterInternal", (PFNRT)&WrapGCC2MSC_SSMR3RegisterInternal, (PFNRT)&WrapGCC2MSC_SSMR3RegisterInternal_EndProc },
1699 };
1700 unsigned i;
1701 for (i = 0; i < RT_ELEMENTS(s_aTemplates); i++)
1702 if (!strcmp(pDesc->pszName, s_aTemplates[i].pszName))
1703 break;
1704 AssertReleaseMsgReturn(i < RT_ELEMENTS(s_aTemplates), ("Not implemented! %s\n", pDesc->pszName), VERR_NOT_IMPLEMENTED);
1705
1706 /* duplicate the patch. */
1707 const size_t cb = (uintptr_t)s_aTemplates[i].pvEnd - (uintptr_t)s_aTemplates[i].pvStart;
1708 pDesc->pvWrapper = remAllocGlue(cb);
1709 AssertReturn(pDesc->pvWrapper, VERR_NO_MEMORY);
1710 memcpy(pDesc->pvWrapper, s_aTemplates[i].pvStart, cb);
1711
1712 /* fix it up. */
1713 remGenerateImportGlueFixup((uint8_t *)pDesc->pvWrapper, cb, pDesc);
1714 }
1715# else /* !USE_REM_CALLING_CONVENTION_GLUE */
1716
1717 /*
1718 * Generate a jump patch.
1719 */
1720 uint8_t *pb;
1721# ifdef RT_ARCH_AMD64
1722 pDesc->pvWrapper = pb = (uint8_t *)remAllocGlue(32);
1723 AssertReturn(pDesc->pvWrapper, VERR_NO_MEMORY);
1724 /**pb++ = 0xcc;*/
1725 *pb++ = 0xff;
1726 *pb++ = 0x24;
1727 *pb++ = 0x25;
1728 *(uint32_t *)pb = (uintptr_t)pb + 5;
1729 pb += 5;
1730 *(uint64_t *)pb = (uint64_t)pDesc->pv;
1731# else
1732 pDesc->pvWrapper = pb = (uint8_t *)remAllocGlue(8);
1733 AssertReturn(pDesc->pvWrapper, VERR_NO_MEMORY);
1734 *pb++ = 0xea;
1735 *(uint32_t *)pb = (uint32_t)pDesc->pv;
1736# endif
1737# endif /* !USE_REM_CALLING_CONVENTION_GLUE */
1738 }
1739 *pValue = (uintptr_t)pDesc->pvWrapper;
1740# else /* !USE_REM_CALLING_CONVENTION_GLUE */
1741 *pValue = (uintptr_t)pDesc->pv;
1742# endif /* !USE_REM_CALLING_CONVENTION_GLUE */
1743 return VINF_SUCCESS;
1744}
1745
1746
1747/**
1748 * Resolve an external symbol during RTLdrGetBits().
1749 *
1750 * @returns iprt status code.
1751 * @param hLdrMod The loader module handle.
1752 * @param pszModule Module name.
1753 * @param pszSymbol Symbol name, NULL if uSymbol should be used.
1754 * @param uSymbol Symbol ordinal, ~0 if pszSymbol should be used.
1755 * @param pValue Where to store the symbol value (address).
1756 * @param pvUser User argument.
1757 */
1758static DECLCALLBACK(int) remGetImport(RTLDRMOD hLdrMod, const char *pszModule, const char *pszSymbol, unsigned uSymbol, RTUINTPTR *pValue, void *pvUser)
1759{
1760 unsigned i;
1761 for (i = 0; i < RT_ELEMENTS(g_aVMMImports); i++)
1762 if (!strcmp(g_aVMMImports[i].pszName, pszSymbol))
1763 return remGenerateImportGlue(pValue, &g_aVMMImports[i]);
1764 for (i = 0; i < RT_ELEMENTS(g_aRTImports); i++)
1765 if (!strcmp(g_aRTImports[i].pszName, pszSymbol))
1766 return remGenerateImportGlue(pValue, &g_aRTImports[i]);
1767 for (i = 0; i < RT_ELEMENTS(g_aCRTImports); i++)
1768 if (!strcmp(g_aCRTImports[i].pszName, pszSymbol))
1769 return remGenerateImportGlue(pValue, &g_aCRTImports[i]);
1770 LogRel(("Missing REM Import: %s\n", pszSymbol));
1771#if 1
1772 *pValue = 0;
1773 AssertMsgFailed(("%s.%s\n", pszModule, pszSymbol));
1774 return VERR_SYMBOL_NOT_FOUND;
1775#else
1776 return remGenerateImportGlue(pValue, &g_aCRTImports[0]);
1777#endif
1778}
1779
1780/**
1781 * Loads the linux object, resolves all imports and exports.
1782 *
1783 * @returns VBox status code.
1784 */
1785static int remLoadLinuxObj(void)
1786{
1787 size_t offFilename;
1788 char szPath[RTPATH_MAX];
1789 int rc = RTPathAppPrivateArch(szPath, sizeof(szPath) - 32);
1790 AssertRCReturn(rc, rc);
1791 offFilename = strlen(szPath);
1792
1793 /*
1794 * Load the VBoxREM2.rel object/DLL.
1795 */
1796 strcpy(&szPath[offFilename], "/VBoxREM2.rel");
1797 rc = RTLdrOpen(szPath, &g_ModREM2);
1798 if (VBOX_SUCCESS(rc))
1799 {
1800 g_pvREM2 = RTMemExecAlloc(RTLdrSize(g_ModREM2));
1801 if (g_pvREM2)
1802 {
1803#ifdef DEBUG /* How to load the VBoxREM2.rel symbols into the GNU debugger. */
1804 RTPrintf("VBoxREMWrapper: (gdb) add-symbol-file %s 0x%p\n", szPath, g_pvREM2);
1805#endif
1806 LogRel(("REM: Loading %s at 0x%p (%d bytes)\n"
1807 "REM: (gdb) add-symbol-file %s 0x%p\n",
1808 szPath, g_pvREM2, RTLdrSize(g_ModREM2), szPath, g_pvREM2));
1809 rc = RTLdrGetBits(g_ModREM2, g_pvREM2, (RTUINTPTR)g_pvREM2, remGetImport, NULL);
1810 if (VBOX_SUCCESS(rc))
1811 {
1812 /*
1813 * Resolve exports.
1814 */
1815 unsigned i;
1816 for (i = 0; i < RT_ELEMENTS(g_aExports); i++)
1817 {
1818 RTUINTPTR Value;
1819 rc = RTLdrGetSymbolEx(g_ModREM2, g_pvREM2, (RTUINTPTR)g_pvREM2, g_aExports[i].pszName, &Value);
1820 AssertMsgRC(rc, ("%s rc=%Vrc\n", g_aExports[i].pszName, rc));
1821 if (VBOX_FAILURE(rc))
1822 break;
1823 rc = remGenerateExportGlue(&Value, &g_aExports[i]);
1824 if (VBOX_FAILURE(rc))
1825 break;
1826 *(void **)g_aExports[i].pv = (void *)(uintptr_t)Value;
1827 }
1828 return rc;
1829 }
1830 RTMemExecFree(g_pvREM2);
1831 }
1832 RTLdrClose(g_ModREM2);
1833 g_ModREM2 = NIL_RTLDRMOD;
1834 }
1835 LogRel(("REM: failed loading '%s', rc=%Vrc\n", szPath, rc));
1836 return rc;
1837}
1838
1839
1840/**
1841 * Unloads the linux object, freeing up all resources (dlls and
1842 * import glue) we allocated during remLoadLinuxObj().
1843 */
1844static void remUnloadLinuxObj(void)
1845{
1846 unsigned i;
1847
1848 /* close modules. */
1849 RTLdrClose(g_ModREM2);
1850 g_ModREM2 = NIL_RTLDRMOD;
1851 RTMemExecFree(g_pvREM2);
1852 g_pvREM2 = NULL;
1853
1854 /* clear the pointers. */
1855 for (i = 0; i < RT_ELEMENTS(g_aExports); i++)
1856 *(void **)g_aExports[i].pv = NULL;
1857# if defined(USE_REM_CALLING_CONVENTION_GLUE) || defined(USE_REM_IMPORT_JUMP_GLUE)
1858 for (i = 0; i < RT_ELEMENTS(g_aVMMImports); i++)
1859 g_aVMMImports[i].pvWrapper = NULL;
1860 for (i = 0; i < RT_ELEMENTS(g_aRTImports); i++)
1861 g_aRTImports[i].pvWrapper = NULL;
1862 for (i = 0; i < RT_ELEMENTS(g_aCRTImports); i++)
1863 g_aCRTImports[i].pvWrapper = NULL;
1864
1865 /* free wrapper memory. */
1866 while (g_pExecMemHead)
1867 {
1868 PREMEXECMEM pCur = g_pExecMemHead;
1869 g_pExecMemHead = pCur->pNext;
1870 memset(pCur, 0xcc, pCur->cb);
1871 RTMemExecFree(pCur);
1872 }
1873# endif
1874}
1875#endif
1876
1877
1878REMR3DECL(int) REMR3Init(PVM pVM)
1879{
1880#ifdef USE_REM_STUBS
1881 return VINF_SUCCESS;
1882#else
1883 if (!pfnREMR3Init)
1884 {
1885 int rc = remLoadLinuxObj();
1886 if (VBOX_FAILURE(rc))
1887 return rc;
1888 }
1889 return pfnREMR3Init(pVM);
1890#endif
1891}
1892
1893REMR3DECL(int) REMR3Term(PVM pVM)
1894{
1895#ifdef USE_REM_STUBS
1896 return VINF_SUCCESS;
1897#else
1898 int rc;
1899 Assert(VALID_PTR(pfnREMR3Term));
1900 rc = pfnREMR3Term(pVM);
1901 remUnloadLinuxObj();
1902 return rc;
1903#endif
1904}
1905
1906REMR3DECL(void) REMR3Reset(PVM pVM)
1907{
1908#ifndef USE_REM_STUBS
1909 Assert(VALID_PTR(pfnREMR3Reset));
1910 pfnREMR3Reset(pVM);
1911#endif
1912}
1913
1914REMR3DECL(int) REMR3Step(PVM pVM)
1915{
1916#ifdef USE_REM_STUBS
1917 return VERR_NOT_IMPLEMENTED;
1918#else
1919 Assert(VALID_PTR(pfnREMR3Step));
1920 return pfnREMR3Step(pVM);
1921#endif
1922}
1923
1924REMR3DECL(int) REMR3BreakpointSet(PVM pVM, RTGCUINTPTR Address)
1925{
1926#ifdef USE_REM_STUBS
1927 return VERR_REM_NO_MORE_BP_SLOTS;
1928#else
1929 Assert(VALID_PTR(pfnREMR3BreakpointSet));
1930 return pfnREMR3BreakpointSet(pVM, Address);
1931#endif
1932}
1933
1934REMR3DECL(int) REMR3BreakpointClear(PVM pVM, RTGCUINTPTR Address)
1935{
1936#ifdef USE_REM_STUBS
1937 return VERR_NOT_IMPLEMENTED;
1938#else
1939 Assert(VALID_PTR(pfnREMR3BreakpointClear));
1940 return pfnREMR3BreakpointClear(pVM, Address);
1941#endif
1942}
1943
1944REMR3DECL(int) REMR3EmulateInstruction(PVM pVM)
1945{
1946#ifdef USE_REM_STUBS
1947 return VERR_NOT_IMPLEMENTED;
1948#else
1949 Assert(VALID_PTR(pfnREMR3EmulateInstruction));
1950 return pfnREMR3EmulateInstruction(pVM);
1951#endif
1952}
1953
1954REMR3DECL(int) REMR3Run(PVM pVM)
1955{
1956#ifdef USE_REM_STUBS
1957 return VERR_NOT_IMPLEMENTED;
1958#else
1959 Assert(VALID_PTR(pfnREMR3Run));
1960 return pfnREMR3Run(pVM);
1961#endif
1962}
1963
1964REMR3DECL(int) REMR3State(PVM pVM, bool fFlushTBs)
1965{
1966#ifdef USE_REM_STUBS
1967 return VERR_NOT_IMPLEMENTED;
1968#else
1969 Assert(VALID_PTR(pfnREMR3State));
1970 return pfnREMR3State(pVM, fFlushTBs);
1971#endif
1972}
1973
1974REMR3DECL(int) REMR3StateBack(PVM pVM)
1975{
1976#ifdef USE_REM_STUBS
1977 return VERR_NOT_IMPLEMENTED;
1978#else
1979 Assert(VALID_PTR(pfnREMR3StateBack));
1980 return pfnREMR3StateBack(pVM);
1981#endif
1982}
1983
1984REMR3DECL(void) REMR3StateUpdate(PVM pVM)
1985{
1986#ifndef USE_REM_STUBS
1987 Assert(VALID_PTR(pfnREMR3StateUpdate));
1988 pfnREMR3StateUpdate(pVM);
1989#endif
1990}
1991
1992REMR3DECL(void) REMR3A20Set(PVM pVM, bool fEnable)
1993{
1994#ifndef USE_REM_STUBS
1995 Assert(VALID_PTR(pfnREMR3A20Set));
1996 pfnREMR3A20Set(pVM, fEnable);
1997#endif
1998}
1999
2000REMR3DECL(void) REMR3ReplayInvalidatedPages(PVM pVM)
2001{
2002#ifndef USE_REM_STUBS
2003 Assert(VALID_PTR(pfnREMR3ReplayInvalidatedPages));
2004 pfnREMR3ReplayInvalidatedPages(pVM);
2005#endif
2006}
2007
2008REMR3DECL(void) REMR3ReplayHandlerNotifications(PVM pVM)
2009{
2010#ifndef USE_REM_STUBS
2011 Assert(VALID_PTR(pfnREMR3ReplayHandlerNotifications));
2012 pfnREMR3ReplayHandlerNotifications(pVM);
2013#endif
2014}
2015
2016REMR3DECL(int) REMR3NotifyCodePageChanged(PVM pVM, RTGCPTR pvCodePage)
2017{
2018#ifdef USE_REM_STUBS
2019 return VINF_SUCCESS;
2020#else
2021 Assert(VALID_PTR(pfnREMR3NotifyCodePageChanged));
2022 return pfnREMR3NotifyCodePageChanged(pVM, pvCodePage);
2023#endif
2024}
2025
2026REMR3DECL(void) REMR3NotifyPhysRamRegister(PVM pVM, RTGCPHYS GCPhys, RTUINT cb, unsigned fFlags)
2027{
2028#ifndef USE_REM_STUBS
2029 Assert(VALID_PTR(pfnREMR3NotifyPhysRamRegister));
2030 pfnREMR3NotifyPhysRamRegister(pVM, GCPhys, cb, fFlags);
2031#endif
2032}
2033
2034#ifndef VBOX_WITH_NEW_PHYS_CODE
2035REMR3DECL(void) REMR3NotifyPhysRamChunkRegister(PVM pVM, RTGCPHYS GCPhys, RTUINT cb, RTHCUINTPTR pvRam, unsigned fFlags)
2036{
2037#ifndef USE_REM_STUBS
2038 Assert(VALID_PTR(pfnREMR3NotifyPhysRamChunkRegister));
2039 pfnREMR3NotifyPhysRamChunkRegister(pVM, GCPhys, cb, pvRam, fFlags);
2040#endif
2041}
2042#endif
2043
2044REMR3DECL(void) REMR3NotifyPhysRomRegister(PVM pVM, RTGCPHYS GCPhys, RTUINT cb, void *pvCopy, bool fShadow)
2045{
2046#ifndef USE_REM_STUBS
2047 Assert(VALID_PTR(pfnREMR3NotifyPhysRomRegister));
2048 pfnREMR3NotifyPhysRomRegister(pVM, GCPhys, cb, pvCopy, fShadow);
2049#endif
2050}
2051
2052REMR3DECL(void) REMR3NotifyPhysReserve(PVM pVM, RTGCPHYS GCPhys, RTUINT cb)
2053{
2054#ifndef USE_REM_STUBS
2055 Assert(VALID_PTR(pfnREMR3NotifyPhysReserve));
2056 pfnREMR3NotifyPhysReserve(pVM, GCPhys, cb);
2057#endif
2058}
2059
2060REMR3DECL(void) REMR3NotifyHandlerPhysicalRegister(PVM pVM, PGMPHYSHANDLERTYPE enmType, RTGCPHYS GCPhys, RTGCPHYS cb, bool fHasHCHandler)
2061{
2062#ifndef USE_REM_STUBS
2063 Assert(VALID_PTR(pfnREMR3NotifyHandlerPhysicalRegister));
2064 pfnREMR3NotifyHandlerPhysicalRegister(pVM, enmType, GCPhys, cb, fHasHCHandler);
2065#endif
2066}
2067
2068REMR3DECL(void) REMR3NotifyHandlerPhysicalDeregister(PVM pVM, PGMPHYSHANDLERTYPE enmType, RTGCPHYS GCPhys, RTGCPHYS cb, bool fHasHCHandler, bool fRestoreAsRAM)
2069{
2070#ifndef USE_REM_STUBS
2071 Assert(VALID_PTR(pfnREMR3NotifyHandlerPhysicalDeregister));
2072 pfnREMR3NotifyHandlerPhysicalDeregister(pVM, enmType, GCPhys, cb, fHasHCHandler, fRestoreAsRAM);
2073#endif
2074}
2075
2076REMR3DECL(void) REMR3NotifyHandlerPhysicalModify(PVM pVM, PGMPHYSHANDLERTYPE enmType, RTGCPHYS GCPhysOld, RTGCPHYS GCPhysNew, RTGCPHYS cb, bool fHasHCHandler, bool fRestoreAsRAM)
2077{
2078#ifndef USE_REM_STUBS
2079 Assert(VALID_PTR(pfnREMR3NotifyHandlerPhysicalModify));
2080 pfnREMR3NotifyHandlerPhysicalModify(pVM, enmType, GCPhysOld, GCPhysNew, cb, fHasHCHandler, fRestoreAsRAM);
2081#endif
2082}
2083
2084REMR3DECL(bool) REMR3IsPageAccessHandled(PVM pVM, RTGCPHYS GCPhys)
2085{
2086#ifdef USE_REM_STUBS
2087 return false;
2088#else
2089 Assert(VALID_PTR(pfnREMR3IsPageAccessHandled));
2090 return pfnREMR3IsPageAccessHandled(pVM, GCPhys);
2091#endif
2092}
2093
2094REMR3DECL(int) REMR3DisasEnableStepping(PVM pVM, bool fEnable)
2095{
2096#ifdef USE_REM_STUBS
2097 return VERR_NOT_IMPLEMENTED;
2098#else
2099 Assert(VALID_PTR(pfnREMR3DisasEnableStepping));
2100 return pfnREMR3DisasEnableStepping(pVM, fEnable);
2101#endif
2102}
2103
2104REMR3DECL(void) REMR3NotifyPendingInterrupt(PVM pVM, uint8_t u8Interrupt)
2105{
2106#ifndef USE_REM_STUBS
2107 Assert(VALID_PTR(pfnREMR3NotifyPendingInterrupt));
2108 pfnREMR3NotifyPendingInterrupt(pVM, u8Interrupt);
2109#endif
2110}
2111
2112REMR3DECL(uint32_t) REMR3QueryPendingInterrupt(PVM pVM)
2113{
2114#ifdef USE_REM_STUBS
2115 return REM_NO_PENDING_IRQ;
2116#else
2117 Assert(VALID_PTR(pfnREMR3QueryPendingInterrupt));
2118 return pfnREMR3QueryPendingInterrupt(pVM);
2119#endif
2120}
2121
2122REMR3DECL(void) REMR3NotifyInterruptSet(PVM pVM)
2123{
2124#ifndef USE_REM_STUBS
2125 Assert(VALID_PTR(pfnREMR3NotifyInterruptSet));
2126 pfnREMR3NotifyInterruptSet(pVM);
2127#endif
2128}
2129
2130REMR3DECL(void) REMR3NotifyInterruptClear(PVM pVM)
2131{
2132#ifndef USE_REM_STUBS
2133 Assert(VALID_PTR(pfnREMR3NotifyInterruptClear));
2134 pfnREMR3NotifyInterruptClear(pVM);
2135#endif
2136}
2137
2138REMR3DECL(void) REMR3NotifyTimerPending(PVM pVM)
2139{
2140#ifndef USE_REM_STUBS
2141 Assert(VALID_PTR(pfnREMR3NotifyTimerPending));
2142 pfnREMR3NotifyTimerPending(pVM);
2143#endif
2144}
2145
2146REMR3DECL(void) REMR3NotifyDmaPending(PVM pVM)
2147{
2148#ifndef USE_REM_STUBS
2149 Assert(VALID_PTR(pfnREMR3NotifyDmaPending));
2150 pfnREMR3NotifyDmaPending(pVM);
2151#endif
2152}
2153
2154REMR3DECL(void) REMR3NotifyQueuePending(PVM pVM)
2155{
2156#ifndef USE_REM_STUBS
2157 Assert(VALID_PTR(pfnREMR3NotifyQueuePending));
2158 pfnREMR3NotifyQueuePending(pVM);
2159#endif
2160}
2161
2162REMR3DECL(void) REMR3NotifyFF(PVM pVM)
2163{
2164#ifndef USE_REM_STUBS
2165 /* the timer can call this early on, so don't be picky. */
2166 if (pfnREMR3NotifyFF)
2167 pfnREMR3NotifyFF(pVM);
2168#endif
2169}
2170
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