VirtualBox

source: vbox/trunk/src/VBox/Devices/testcase/tstDevice.cpp@ 69288

Last change on this file since 69288 was 69183, checked in by vboxsync, 7 years ago

Devices/testcase: Updates for the PDM unit test framework

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.5 KB
Line 
1/* $Id: tstDevice.cpp 69183 2017-10-23 18:47:18Z vboxsync $ */
2/** @file
3 * tstDevice - Test framework for PDM devices/drivers
4 */
5
6/*
7 * Copyright (C) 2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DEFAULT /** @todo */
23#include <VBox/types.h>
24#include <VBox/sup.h>
25#include <VBox/version.h>
26#include <iprt/assert.h>
27#include <iprt/ctype.h>
28#include <iprt/getopt.h>
29#include <iprt/initterm.h>
30#include <iprt/ldr.h>
31#include <iprt/log.h>
32#include <iprt/list.h>
33#include <iprt/mem.h>
34#include <iprt/once.h>
35#include <iprt/path.h>
36#include <iprt/string.h>
37#include <iprt/stream.h>
38#include <iprt/trace.h>
39
40#include "tstDeviceInternal.h"
41
42/*********************************************************************************************************************************
43* Defined Constants And Macros *
44*********************************************************************************************************************************/
45
46
47/*********************************************************************************************************************************
48* Structures and Typedefs *
49*********************************************************************************************************************************/
50
51
52/**
53 * Testcase plugin descriptor.
54 */
55typedef struct TSTDEVPLUGIN
56{
57 /** Node for the plugin list. */
58 RTLISTNODE NdPlugins;
59 /** Copy of the filename. */
60 char *pszFilename;
61 /** Loader handle. */
62 RTLDRMOD hMod;
63 /** Number of references to this plugin. */
64 volatile uint32_t cRefs;
65} TSTDEVPLUGIN;
66/** Pointer to a device plugin descriptor. */
67typedef TSTDEVPLUGIN *PTSTDEVPLUGIN;
68/** Pointer to a const plugin descriptor. */
69typedef const TSTDEVPLUGIN *PCTSTDEVPLUGIN;
70
71
72/**
73 * Testcase descriptor.
74 */
75typedef struct TSTDEVTESTCASE
76{
77 /** Node for the list of registered testcases. */
78 RTLISTNODE NdTestcases;
79 /** Pointer to the plugin the testcase belongs to. */
80 PCTSTDEVPLUGIN pPlugin;
81 /** Pointer to the testcase descriptor. */
82 PCTSTDEVTESTCASEREG pTestcaseReg;
83} TSTDEVTESTCASE;
84/** Pointer to a testcase descriptor. */
85typedef TSTDEVTESTCASE *PTSTDEVTESTCASE;
86/** Pointer to a constant testcase descriptor. */
87typedef const TSTDEVTESTCASE *PCTSTDEVTESTCASE;
88
89
90/**
91 * PDM R0/RC module trampoline descriptor.
92 */
93#pragma pack(1)
94typedef struct TSTDEVPDMMODTRAMPOLINE
95{
96 /** Jump instruction. */
97 uint8_t abJmp[6];
98 /** Address to jump to. */
99 uintptr_t AddrTarget;
100 /** Padding to get a 16byte aligned structure. */
101 uint8_t abPadding[HC_ARCH_BITS == 64 ? 2 : 5];
102} TSTDEVPDMMODTRAMPOLINE;
103#pragma pack()
104AssertCompileSize(TSTDEVPDMMODTRAMPOLINE, 16);
105/** Pointer to a trampoline descriptor. */
106typedef TSTDEVPDMMODTRAMPOLINE *PTSTDEVPDMMODTRAMPOLINE;
107
108/**
109 * PDM module descriptor.
110 */
111typedef struct TSTDEVPDMMOD
112{
113 /** Node for the module list. */
114 RTLISTNODE NdPdmMods;
115 /** Type of module (R3/R0/RC). */
116 TSTDEVPDMMODTYPE enmType;
117 /** Copy of the filename. */
118 char *pszFilename;
119 /** Loader handle. */
120 RTLDRMOD hLdrMod;
121 /** Number of references to this plugin. */
122 volatile uint32_t cRefs;
123 /** R0/RC Module type dependent data. */
124 struct
125 {
126 /** The exectuable image bits. */
127 void *pvBits;
128 /** Size of the memory buffer. */
129 size_t cbBits;
130 /** Pointer to the executable memory containing the trampoline code. */
131 uint8_t *pbTrampoline;
132 /** Number of trampoline entries supported. */
133 uint32_t cTrampolinesMax;
134 /** Number of trampoline entries used. */
135 uint32_t cTrampolines;
136 /** Pointer to the next unused trampoline entry. */
137 PTSTDEVPDMMODTRAMPOLINE pTrampolineNext;
138 } R0Rc;
139} TSTDEVPDMMOD;
140/** Pointer to a PDM module descriptor. */
141typedef TSTDEVPDMMOD *PTSTDEVPDMMOD;
142/** Pointer to a const PDM module descriptor. */
143typedef const TSTDEVPDMMOD *PCTSTDEVPDMMOD;
144
145/**
146 * PDM device descriptor.
147 */
148typedef struct TSTDEVPDMDEV
149{
150 /** Node for the known device list. */
151 RTLISTNODE NdPdmDevs;
152 /** Pointer to the PDM module containing the device. */
153 PCTSTDEVPDMMOD pPdmMod;
154 /** Device registration structure. */
155 const struct PDMDEVREG *pReg;
156} TSTDEVPDMDEV;
157/** Pointer to a PDM device descriptor .*/
158typedef TSTDEVPDMDEV *PTSTDEVPDMDEV;
159/** Pointer to a constant PDM device descriptor .*/
160typedef const TSTDEVPDMDEV *PCTSTDEVPDMDEV;
161
162
163/**
164 * Internal callback structure pointer.
165 * The main purpose is to define the extra data we associate
166 * with PDMDEVREGCB so we can find the plugin the device is associated with etc.
167 */
168typedef struct TSTDEVPDMDEVREGCBINT
169{
170 /** The callback structure. */
171 PDMDEVREGCB Core;
172 /** A bit of padding. */
173 uint32_t u32[4];
174 /** Pointer to plugin. */
175 PTSTDEVPDMMOD pMod;
176} TSTDEVPDMDEVREGCBINT;
177/** Pointer to a PDMDEVREGCBINT structure. */
178typedef TSTDEVPDMDEVREGCBINT *PTSTDEVPDMDEVREGCBINT;
179/** Pointer to a const PDMDEVREGCBINT structure. */
180typedef const TSTDEVPDMDEVREGCBINT *PCTSTDEVPDMDEVREGCBINT;
181
182
183typedef struct TSTDEVPDMR0IMPORTS
184{
185 /** The symbol name. */
186 const char *pszSymbol;
187 /** The pointer. */
188 PFNRT pfn;
189} TSTDEVPDMR0IMPORTS;
190typedef const TSTDEVPDMR0IMPORTS *PCTSTDEVPDMR0IMPORTS;
191
192/*********************************************************************************************************************************
193* Global Variables *
194*********************************************************************************************************************************/
195/** List of registered testcase plugins. */
196RTLISTANCHOR g_LstPlugins;
197/** List of registered testcases. */
198RTLISTANCHOR g_LstTestcases;
199/** List of registered PDM modules. */
200RTLISTANCHOR g_LstPdmMods;
201/** List of registered PDM devices. */
202RTLISTANCHOR g_LstPdmDevs;
203
204extern const TSTDEVVMMCALLBACKS g_tstDevVmmCallbacks;
205
206/**
207 * PDM R0 imports we implement.
208 */
209static const TSTDEVPDMR0IMPORTS g_aPdmR0Imports[] =
210{
211 {"IOMMMIOMapMMIO2Page", (PFNRT)IOMMMIOMapMMIO2Page},
212 {"IOMMMIOResetRegion", (PFNRT)IOMMMIOResetRegion},
213 {"IntNetR0IfSend", (PFNRT)/*IntNetR0IfSend*/NULL},
214 {"IntNetR0IfSetPromiscuousMode", (PFNRT)/*IntNetR0IfSetPromiscuousMode*/NULL},
215 {"PDMCritSectEnterDebug", (PFNRT)PDMCritSectEnterDebug},
216 {"PDMCritSectIsOwner", (PFNRT)PDMCritSectIsOwner},
217 {"PDMCritSectLeave", (PFNRT)PDMCritSectLeave},
218 {"PDMCritSectTryEnterDebug", (PFNRT)PDMCritSectTryEnterDebug},
219 {"PDMHCCritSectScheduleExitEvent", (PFNRT)PDMHCCritSectScheduleExitEvent},
220 {"PDMNsAllocateBandwidth", (PFNRT)PDMNsAllocateBandwidth},
221 {"PDMQueueAlloc", (PFNRT)PDMQueueAlloc},
222 {"PDMQueueInsert", (PFNRT)PDMQueueInsert},
223 {"PGMHandlerPhysicalPageTempOff", (PFNRT)PGMHandlerPhysicalPageTempOff},
224 {"PGMShwMakePageWritable", (PFNRT)PGMShwMakePageWritable},
225 {"RTAssertMsg1Weak", (PFNRT)RTAssertMsg1Weak},
226 {"RTAssertMsg2Weak", (PFNRT)RTAssertMsg2Weak},
227 {"RTAssertShouldPanic", (PFNRT)RTAssertShouldPanic},
228 {"RTLogDefaultInstanceEx", (PFNRT)RTLogDefaultInstanceEx},
229 {"RTLogLoggerEx", (PFNRT)RTLogLoggerEx},
230 {"RTLogRelGetDefaultInstanceEx", (PFNRT)RTLogRelGetDefaultInstanceEx},
231 {"RTOnceSlow", (PFNRT)RTOnceSlow},
232 {"RTR0AssertPanicSystem", (PFNRT)0x10101010},
233 {"RTThreadSleep", (PFNRT)RTThreadSleep},
234 {"RTTimeMilliTS", (PFNRT)RTTimeMilliTS},
235 {"RTTimeNanoTS", (PFNRT)RTTimeNanoTS},
236 {"RTTraceBufAddMsgF", (PFNRT)RTTraceBufAddMsgF},
237 {"SUPSemEventSignal", (PFNRT)SUPSemEventSignal},
238 {"TMTimerGet", (PFNRT)TMTimerGet},
239 {"TMTimerGetFreq", (PFNRT)TMTimerGetFreq},
240 {"TMTimerIsActive", (PFNRT)TMTimerIsActive},
241 {"TMTimerIsLockOwner", (PFNRT)TMTimerIsLockOwner},
242 {"TMTimerLock", (PFNRT)TMTimerLock},
243 {"TMTimerSet", (PFNRT)TMTimerSet},
244 {"TMTimerSetFrequencyHint", (PFNRT)TMTimerSetFrequencyHint},
245 {"TMTimerSetMicro", (PFNRT)TMTimerSetMicro},
246 {"TMTimerSetMillies", (PFNRT)TMTimerSetMillies},
247 {"TMTimerSetNano", (PFNRT)TMTimerSetNano},
248 {"TMTimerStop", (PFNRT)TMTimerStop},
249 {"TMTimerUnlock", (PFNRT)TMTimerUnlock},
250 {"nocrt_memcmp", (PFNRT)memcmp},
251 {"nocrt_memcpy", (PFNRT)memcpy},
252 {"nocrt_memmove", (PFNRT)memmove},
253 {"nocrt_memset", (PFNRT)memset},
254 {"nocrt_strlen", (PFNRT)strlen},
255};
256
257
258/*********************************************************************************************************************************
259* Internal Functions *
260*********************************************************************************************************************************/
261
262#if 0
263/**
264 * Parses the options given to the testcase.
265 *
266 * @returns Process status code.
267 * @param cArgs Number of arguments given.
268 * @param paszArgs Pointer to the argument vector.
269 */
270static RTEXITCODE tstDevParseOptions(int cArgs, char *paszArgs[])
271{
272 static RTGETOPTDEF const s_aOptions[] =
273 {
274 };
275
276
277 int ch;
278 RTGETOPTUNION Value;
279 RTGETOPTSTATE GetState;
280 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, 0 /* fFlags */);
281 while ((ch = RTGetOpt(&GetState, &Value)))
282 {
283 switch (ch)
284 {
285 default:
286 return RTGetOptPrintError(ch, &Value);
287 }
288 }
289}
290#endif
291
292
293/**
294 * Checks whether the given testcase name is already existing.
295 *
296 * @returns Pointer to existing testcase, NULL if not found.
297 * @param pszFilename The filename to check.
298 */
299static PCTSTDEVTESTCASE tstDevTestcaseFind(const char *pszName)
300{
301 PCTSTDEVTESTCASE pIt;
302 RTListForEach(&g_LstTestcases, pIt, TSTDEVTESTCASE, NdTestcases)
303 {
304 if (!RTStrCmp(pIt->pTestcaseReg->szName, pszName))
305 return pIt;
306 }
307
308 return NULL;
309}
310
311
312/**
313 * @interface_method_impl{TSTDEVPLUGINREGISTER,pfnRegisterTestcase}
314 */
315static DECLCALLBACK(int) tstDevRegisterTestcase(void *pvUser, PCTSTDEVTESTCASEREG pTestcaseReg)
316{
317 int rc = VINF_SUCCESS;
318 PTSTDEVPLUGIN pPlugin = (PTSTDEVPLUGIN)pvUser;
319
320 /* Try to find a testcase with the name first. */
321 if (!tstDevTestcaseFind(pTestcaseReg->szName))
322 {
323 PTSTDEVTESTCASE pTestcase = (PTSTDEVTESTCASE)RTMemAllocZ(sizeof(TSTDEVPLUGIN));
324 if (RT_LIKELY(pTestcase))
325 {
326 pTestcase->pPlugin = pPlugin;
327 pPlugin->cRefs++;
328 pTestcase->pTestcaseReg = pTestcaseReg;
329 RTListAppend(&g_LstTestcases, &pTestcase->NdTestcases);
330 return VINF_SUCCESS;
331 }
332 else
333 rc = VERR_NO_MEMORY;
334 }
335 else
336 rc = VERR_ALREADY_EXISTS;
337
338 return rc;
339}
340
341
342/**
343 * Checks whether the given plugin filename was already loaded.
344 *
345 * @returns Pointer to already loaded plugin, NULL if not found.
346 * @param pszFilename The filename to check.
347 */
348static PCTSTDEVPLUGIN tstDevPluginFind(const char *pszFilename)
349{
350 PCTSTDEVPLUGIN pIt;
351 RTListForEach(&g_LstPlugins, pIt, TSTDEVPLUGIN, NdPlugins)
352 {
353 if (!RTStrCmp(pIt->pszFilename, pszFilename))
354 return pIt;
355 }
356
357 return NULL;
358}
359
360
361/**
362 * Tries to loads the given plugin.
363 *
364 * @returns VBox status code.
365 * @param pszFilename The filename to load.
366 */
367static int tstDevLoadPlugin(const char *pszFilename)
368{
369 int rc = VINF_SUCCESS;
370
371 /* Check whether the plugin is loaded first. */
372 if (!tstDevPluginFind(pszFilename))
373 {
374 PTSTDEVPLUGIN pPlugin = (PTSTDEVPLUGIN)RTMemAllocZ(sizeof(TSTDEVPLUGIN));
375 if (RT_LIKELY(pPlugin))
376 {
377 pPlugin->pszFilename = RTStrDup(pszFilename);
378 pPlugin->cRefs = 1;
379 rc = RTLdrLoad(pszFilename, &pPlugin->hMod);
380 if (RT_SUCCESS(rc))
381 {
382 TSTDEVPLUGINREGISTER TestcaseRegister;
383 PFNTSTDEVPLUGINLOAD pfnPluginLoad = NULL;
384
385 TestcaseRegister.pfnRegisterTestcase = tstDevRegisterTestcase;
386
387 rc = RTLdrGetSymbol(pPlugin->hMod, TSTDEV_PLUGIN_LOAD_NAME, (void**)&pfnPluginLoad);
388 if (RT_FAILURE(rc) || !pfnPluginLoad)
389 {
390 LogFunc(("error resolving the entry point %s in plugin %s, rc=%Rrc, pfnPluginLoad=%#p\n",
391 TSTDEV_PLUGIN_LOAD_NAME, pszFilename, rc, pfnPluginLoad));
392 if (RT_SUCCESS(rc))
393 rc = VERR_SYMBOL_NOT_FOUND;
394 }
395
396 if (RT_SUCCESS(rc))
397 {
398 /* Get the function table. */
399 rc = pfnPluginLoad(pPlugin, &TestcaseRegister);
400 }
401 else
402 LogFunc(("ignored plugin '%s': rc=%Rrc\n", pszFilename, rc));
403
404 /* Create a plugin entry on success. */
405 if (RT_SUCCESS(rc))
406 {
407 RTListAppend(&g_LstPlugins, &pPlugin->NdPlugins);
408 return VINF_SUCCESS;
409 }
410 else
411 RTLdrClose(pPlugin->hMod);
412 }
413
414 RTMemFree(pPlugin);
415 }
416 else
417 rc = VERR_NO_MEMORY;
418 }
419
420 return rc;
421}
422
423
424/**
425 * Checks whether the given testcase name is already existing.
426 *
427 * @returns Pointer to already loaded plugin, NULL if not found.
428 * @param pszFilename The filename to check.
429 */
430static PCTSTDEVPDMDEV tstDevPdmDeviceFind(const char *pszName)
431{
432 PCTSTDEVPDMDEV pIt;
433 RTListForEach(&g_LstPdmDevs, pIt, TSTDEVPDMDEV, NdPdmDevs)
434 {
435 if (!RTStrCmp(pIt->pReg->szName, pszName))
436 return pIt;
437 }
438
439 return NULL;
440}
441
442
443/**
444 * Checks that a PDMDRVREG::szName, PDMDEVREG::szName or PDMUSBREG::szName
445 * field contains only a limited set of ASCII characters.
446 *
447 * @returns true / false.
448 * @param pszName The name to validate.
449 */
450bool tstDevPdmR3IsValidName(const char *pszName)
451{
452 char ch;
453 while ( (ch = *pszName) != '\0'
454 && ( RT_C_IS_ALNUM(ch)
455 || ch == '-'
456 || ch == ' ' /** @todo disallow this! */
457 || ch == '_') )
458 pszName++;
459 return ch == '\0';
460}
461
462
463/**
464 * @interface_method_impl{PDMDEVREGCB,pfnRegister}
465 */
466static DECLCALLBACK(int) tstDevPdmR3DevReg_Register(PPDMDEVREGCB pCallbacks, PCPDMDEVREG pReg)
467{
468 /*
469 * Validate the registration structure (mostly copy and paste from PDMDevice.cpp).
470 */
471 Assert(pReg);
472 AssertMsgReturn(pReg->u32Version == PDM_DEVREG_VERSION,
473 ("Unknown struct version %#x!\n", pReg->u32Version),
474 VERR_PDM_UNKNOWN_DEVREG_VERSION);
475
476 AssertMsgReturn( pReg->szName[0]
477 && strlen(pReg->szName) < sizeof(pReg->szName)
478 && tstDevPdmR3IsValidName(pReg->szName),
479 ("Invalid name '%.*s'\n", sizeof(pReg->szName), pReg->szName),
480 VERR_PDM_INVALID_DEVICE_REGISTRATION);
481 AssertMsgReturn( !(pReg->fFlags & PDM_DEVREG_FLAGS_RC)
482 || ( pReg->szRCMod[0]
483 && strlen(pReg->szRCMod) < sizeof(pReg->szRCMod)),
484 ("Invalid GC module name '%s' - (Device %s)\n", pReg->szRCMod, pReg->szName),
485 VERR_PDM_INVALID_DEVICE_REGISTRATION);
486 AssertMsgReturn( !(pReg->fFlags & PDM_DEVREG_FLAGS_R0)
487 || ( pReg->szR0Mod[0]
488 && strlen(pReg->szR0Mod) < sizeof(pReg->szR0Mod)),
489 ("Invalid R0 module name '%s' - (Device %s)\n", pReg->szR0Mod, pReg->szName),
490 VERR_PDM_INVALID_DEVICE_REGISTRATION);
491 AssertMsgReturn((pReg->fFlags & PDM_DEVREG_FLAGS_HOST_BITS_MASK) == PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT,
492 ("Invalid host bits flags! fFlags=%#x (Device %s)\n", pReg->fFlags, pReg->szName),
493 VERR_PDM_INVALID_DEVICE_HOST_BITS);
494 AssertMsgReturn((pReg->fFlags & PDM_DEVREG_FLAGS_GUEST_BITS_MASK),
495 ("Invalid guest bits flags! fFlags=%#x (Device %s)\n", pReg->fFlags, pReg->szName),
496 VERR_PDM_INVALID_DEVICE_REGISTRATION);
497 AssertMsgReturn(pReg->fClass,
498 ("No class! (Device %s)\n", pReg->szName),
499 VERR_PDM_INVALID_DEVICE_REGISTRATION);
500 AssertMsgReturn(pReg->cMaxInstances > 0,
501 ("Max instances %u! (Device %s)\n", pReg->cMaxInstances, pReg->szName),
502 VERR_PDM_INVALID_DEVICE_REGISTRATION);
503 AssertMsgReturn(pReg->cbInstance <= (uint32_t)(pReg->fFlags & (PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0) ? 96 * _1K : _1M),
504 ("Instance size %d bytes! (Device %s)\n", pReg->cbInstance, pReg->szName),
505 VERR_PDM_INVALID_DEVICE_REGISTRATION);
506 AssertMsgReturn(pReg->pfnConstruct,
507 ("No constructor! (Device %s)\n", pReg->szName),
508 VERR_PDM_INVALID_DEVICE_REGISTRATION);
509 AssertLogRelMsgReturn((pReg->fFlags & PDM_DEVREG_FLAGS_GUEST_BITS_MASK) == PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT,
510 ("PDM: Rejected device '%s' because it didn't match the guest bits.\n", pReg->szName),
511 VERR_PDM_INVALID_DEVICE_GUEST_BITS);
512 AssertLogRelMsg(pReg->u32VersionEnd == PDM_DEVREG_VERSION,
513 ("u32VersionEnd=%#x, expected %#x. (szName=%s)\n",
514 pReg->u32VersionEnd, PDM_DEVREG_VERSION, pReg->szName));
515
516 /*
517 * Check for duplicates.
518 */
519 int rc = VINF_SUCCESS;
520 PCTSTDEVPDMDEVREGCBINT pRegCB = (PCTSTDEVPDMDEVREGCBINT)pCallbacks;
521 if (!tstDevPdmDeviceFind(pReg->szName))
522 {
523 PTSTDEVPDMDEV pPdmDev = (PTSTDEVPDMDEV)RTMemAllocZ(sizeof(TSTDEVPDMDEV));
524 if (RT_LIKELY(pPdmDev))
525 {
526 pPdmDev->pPdmMod = pRegCB->pMod;
527 pRegCB->pMod->cRefs++;
528 pPdmDev->pReg = pReg;
529 RTListAppend(&g_LstPdmDevs, &pPdmDev->NdPdmDevs);
530 return VINF_SUCCESS;
531 }
532 else
533 rc = VERR_NO_MEMORY;
534 }
535 else
536 rc = VERR_PDM_DEVICE_NAME_CLASH;
537
538 return rc;
539}
540
541
542/**
543 * Checks whether the given PDM module filename was already loaded.
544 *
545 * @returns Pointer to already loaded plugin, NULL if not found.
546 * @param pszFilename The filename to check.
547 */
548static PCTSTDEVPDMMOD tstDevPdmModFind(const char *pszFilename)
549{
550 PCTSTDEVPDMMOD pIt;
551 RTListForEach(&g_LstPdmMods, pIt, TSTDEVPDMMOD, NdPdmMods)
552 {
553 if (!RTStrCmp(pIt->pszFilename, pszFilename))
554 return pIt;
555 }
556
557 return NULL;
558}
559
560
561/**
562 * Resolve an external symbol during RTLdrGetBits().
563 *
564 * @returns iprt status code.
565 * @param hLdrMod The loader module handle.
566 * @param pszModule Module name.
567 * @param pszSymbol Symbol name, NULL if uSymbol should be used.
568 * @param uSymbol Symbol ordinal, ~0 if pszSymbol should be used.
569 * @param pValue Where to store the symbol value (address).
570 * @param pvUser User argument.
571 */
572static DECLCALLBACK(int) tstDevPdmLoadR0RcModGetImport(RTLDRMOD hLdrMod, const char *pszModule, const char *pszSymbol,
573 unsigned uSymbol, RTUINTPTR *pValue, void *pvUser)
574{
575 RT_NOREF(hLdrMod, uSymbol, pszModule);
576 PTSTDEVPDMMOD pMod = (PTSTDEVPDMMOD)pvUser;
577
578 /* Resolve the import. */
579 PCTSTDEVPDMR0IMPORTS pImpDesc = NULL;
580 bool fFound = false;
581 for (uint32_t i = 0; i < RT_ELEMENTS(g_aPdmR0Imports); i++)
582 {
583 pImpDesc = &g_aPdmR0Imports[i];
584 if (!strcmp(pszSymbol, pImpDesc->pszSymbol))
585 {
586 fFound = true;
587 break;
588 }
589 }
590
591 int rc = VERR_SYMBOL_NOT_FOUND;
592 if (fFound)
593 {
594 /* Check whether the symbol has a trampoline already. */
595 PTSTDEVPDMMODTRAMPOLINE pTrampoline = (PTSTDEVPDMMODTRAMPOLINE)pMod->R0Rc.pbTrampoline;
596 for (uint32_t i = 0; i < pMod->R0Rc.cTrampolines; i++)
597 {
598 if (pTrampoline->AddrTarget == (uintptr_t)pImpDesc->pfn)
599 break;
600 pTrampoline++;
601 }
602
603 /* Create new trampoline if not found. */
604 if (pTrampoline->AddrTarget != (uintptr_t)pImpDesc->pfn)
605 {
606 if (pMod->R0Rc.cTrampolines < pMod->R0Rc.cTrampolinesMax)
607 {
608 pTrampoline = pMod->R0Rc.pTrampolineNext;
609 pMod->R0Rc.pTrampolineNext++;
610 pMod->R0Rc.cTrampolines++;
611 pTrampoline->abJmp[0] = 0xff; /* jmp */
612 pTrampoline->abJmp[1] = 0x25; /* rip */
613 pTrampoline->abJmp[2] = 0x00; /* offset */
614 pTrampoline->abJmp[3] = 0x00;
615 pTrampoline->abJmp[4] = 0x00;
616 pTrampoline->abJmp[5] = 0x00;
617 pTrampoline->AddrTarget = (uintptr_t)pImpDesc->pfn;
618 rc = VINF_SUCCESS;
619 }
620 else
621 {
622 rc = VERR_SYMBOL_NOT_FOUND;
623 AssertFailed();
624 }
625 }
626 else
627 rc = VINF_SUCCESS;
628
629 if (RT_SUCCESS(rc))
630 *pValue = (RTUINTPTR)pTrampoline;
631 }
632 else
633 AssertFailed();
634
635 return rc;
636}
637
638
639/**
640 * Loads a new R0 modules given by the filename.
641 *
642 * @returns VBox status code.
643 * @param pMod Pointer to module structure.
644 */
645static int tstDevPdmLoadR0RcMod(PTSTDEVPDMMOD pMod)
646{
647 int rc = VINF_SUCCESS;
648 const char *pszFile = RTPathFilename(pMod->pszFilename);
649
650 /* Check whether the plugin is loaded first. */
651 if (!tstDevPdmModFind(pszFile))
652 {
653 /*
654 * R0 modules need special treatment as these are relocatable images
655 * which are supposed to run in ring 0.
656 */
657 rc = RTLdrOpen(pMod->pszFilename, 0, RTLDRARCH_HOST, &pMod->hLdrMod);
658 if (RT_SUCCESS(rc))
659 {
660 size_t cb = RTLdrSize(pMod->hLdrMod) + 1024 * sizeof(TSTDEVPDMMODTRAMPOLINE);
661
662 /* Allocate bits. */
663 uint32_t fFlags = RTMEMALLOCEX_FLAGS_EXEC;
664#ifdef RT_OS_LINUX
665 /*
666 * amd64 ELF binaries support only a 2GB code segment everything must be in
667 * (X86_64_PC32 relocation) so we have to use a trampoline to the final destination
668 * which is kept close to the imported module.
669 */
670 fFlags |= RTMEMALLOCEX_FLAGS_32BIT_REACH;
671#endif
672 rc = RTMemAllocEx(cb, 0, fFlags, (void **)&pMod->R0Rc.pbTrampoline);
673 pMod->R0Rc.cbBits = cb;
674 if (RT_SUCCESS(rc))
675 {
676 pMod->R0Rc.pvBits = pMod->R0Rc.pbTrampoline + 1024 * sizeof(TSTDEVPDMMODTRAMPOLINE);
677 pMod->R0Rc.cTrampolinesMax = 1024;
678 pMod->R0Rc.cTrampolines = 0;
679 pMod->R0Rc.pTrampolineNext = (PTSTDEVPDMMODTRAMPOLINE)pMod->R0Rc.pbTrampoline;
680 /* Get the bits. */
681 rc = RTLdrGetBits(pMod->hLdrMod, pMod->R0Rc.pvBits, (uintptr_t)pMod->R0Rc.pvBits,
682 tstDevPdmLoadR0RcModGetImport, pMod);
683 if (RT_FAILURE(rc))
684 RTMemFreeEx(pMod->R0Rc.pbTrampoline, pMod->R0Rc.cbBits);
685 }
686
687 if (RT_FAILURE(rc))
688 RTLdrClose(pMod->hLdrMod);
689 }
690 }
691
692 return rc;
693}
694
695
696/**
697 * Loads the given
698 */
699static int tstDevPdmLoadR3Mod(PTSTDEVPDMMOD pMod)
700{
701 int rc = RTLdrLoad(pMod->pszFilename, &pMod->hLdrMod);
702 if (RT_SUCCESS(rc))
703 {
704 FNPDMVBOXDEVICESREGISTER *pfnVBoxDevicesRegister;
705 rc = RTLdrGetSymbol(pMod->hLdrMod, "VBoxDevicesRegister", (void**)&pfnVBoxDevicesRegister);
706 if (RT_FAILURE(rc) || !pfnVBoxDevicesRegister)
707 {
708 LogFunc(("error resolving the entry point %s in plugin %s, rc=%Rrc, pfnPluginLoad=%#p\n",
709 "VBoxDevicesRegister", pMod->pszFilename, rc, pfnVBoxDevicesRegister));
710 if (RT_SUCCESS(rc))
711 rc = VERR_SYMBOL_NOT_FOUND;
712 }
713
714 if (RT_SUCCESS(rc))
715 {
716 TSTDEVPDMDEVREGCBINT RegCB;
717 RegCB.Core.u32Version = PDM_DEVREG_CB_VERSION;
718 RegCB.Core.pfnRegister = tstDevPdmR3DevReg_Register;
719 RegCB.pMod = pMod;
720 rc = pfnVBoxDevicesRegister(&RegCB.Core, VBOX_VERSION);
721 }
722 else
723 LogFunc(("ignored plugin '%s': rc=%Rrc\n", pMod->pszFilename, rc));
724
725 if (RT_FAILURE(rc))
726 RTLdrClose(pMod->hLdrMod);
727 }
728
729 return rc;
730}
731
732
733/**
734 * Tries to loads the given PDM module.
735 *
736 * @returns VBox status code.
737 * @param pszFilename The filename to load.
738 * @param enmModType The module type.
739 */
740static int tstDevPdmLoadMod(const char *pszFilename, TSTDEVPDMMODTYPE enmModType)
741{
742 int rc = VINF_SUCCESS;
743
744 /* Check whether the plugin is loaded first. */
745 if (!tstDevPdmModFind(pszFilename))
746 {
747 PTSTDEVPDMMOD pMod = (PTSTDEVPDMMOD)RTMemAllocZ(sizeof(TSTDEVPDMMOD));
748 if (RT_LIKELY(pMod))
749 {
750 pMod->pszFilename = RTStrDup(pszFilename);
751 pMod->cRefs = 1;
752 pMod->enmType = enmModType;
753
754 if (enmModType == TSTDEVPDMMODTYPE_R3)
755 rc = tstDevPdmLoadR3Mod(pMod);
756 else if (enmModType == TSTDEVPDMMODTYPE_RC || enmModType == TSTDEVPDMMODTYPE_R0)
757 rc = tstDevPdmLoadR0RcMod(pMod);
758
759 if (RT_SUCCESS(rc))
760 RTListAppend(&g_LstPdmMods, &pMod->NdPdmMods);
761 else
762 RTMemFree(pMod);
763 }
764 else
765 rc = VERR_NO_MEMORY;
766 }
767
768 return rc;
769}
770
771
772/**
773 * Tries to resolve the given symbol from the module given.
774 *
775 * @returns VBox status code.
776 * @param pThis The device under test instance.
777 * @param pszMod The module name.
778 * @param enmModType The module type if the module needs to be loaded.
779 * @param pszSymbol The symbol to resolve.
780 * @param ppfn Where to store the value on success.
781 */
782DECLHIDDEN(int) tstDevPdmLdrGetSymbol(PTSTDEVDUTINT pThis, const char *pszMod, TSTDEVPDMMODTYPE enmModType,
783 const char *pszSymbol, PFNRT *ppfn)
784{
785 RT_NOREF(pThis);
786
787 int rc = VINF_SUCCESS;
788 PCTSTDEVPDMMOD pMod = tstDevPdmModFind(pszMod);
789 if (!pMod)
790 {
791 /* Try to load the module. */
792 rc = tstDevPdmLoadMod(pszMod, enmModType);
793 if (RT_SUCCESS(rc))
794 {
795 pMod = tstDevPdmModFind(pszMod);
796 AssertPtr(pMod);
797 }
798 }
799
800 if (RT_SUCCESS(rc))
801 {
802 if (pMod->enmType == TSTDEVPDMMODTYPE_R0 || pMod->enmType == TSTDEVPDMMODTYPE_RC)
803 rc = RTLdrGetSymbolEx(pMod->hLdrMod, pMod->R0Rc.pvBits, (uintptr_t)pMod->R0Rc.pvBits,
804 UINT32_MAX, pszSymbol, (PRTLDRADDR)ppfn);
805 else
806 rc = RTLdrGetSymbol(pMod->hLdrMod, pszSymbol, (void **)ppfn);
807 }
808
809 return rc;
810}
811
812
813static TSTDEVCFGITEM s_aTestcaseCfg[] =
814{
815 {"CtrlMemBufSize", TSTDEVCFGITEMTYPE_INTEGER, "0" },
816 {NULL, TSTDEVCFGITEMTYPE_INVALID, NULL }
817};
818
819static TSTDEVTESTCASEREG s_TestcaseDef =
820{
821 "test",
822 "Testcase during implementation",
823 "nvme",
824 0,
825 &s_aTestcaseCfg[0],
826 NULL,
827};
828
829/**
830 * Create a new PDM device with default config.
831 *
832 * @returns VBox status code.
833 * @param pszName Name of the device to create.
834 */
835static int tstDevPdmDevCreate(const char *pszName)
836{
837 int rc = VINF_SUCCESS;
838 PCTSTDEVPDMDEV pPdmDev = tstDevPdmDeviceFind(pszName);
839 if (RT_LIKELY(pPdmDev))
840 {
841 TSTDEVDUTINT Dut;
842 Dut.pTestcaseReg = &s_TestcaseDef;
843 Dut.enmCtx = TSTDEVDUTCTX_R3;
844 Dut.pVm = NULL;
845 Dut.SupSession.pDut = &Dut;
846 RTListInit(&Dut.LstIoPorts);
847 RTListInit(&Dut.LstTimers);
848 RTListInit(&Dut.LstMmHeap);
849 RTListInit(&Dut.LstPdmThreads);
850 RTListInit(&Dut.SupSession.LstSupSem);
851 CFGMNODE Cfg;
852 Cfg.pVmmCallbacks = &g_tstDevVmmCallbacks;
853 Cfg.pDut = &Dut;
854
855 rc = RTCritSectRwInit(&Dut.CritSectLists);
856 AssertRC(rc);
857
858 PPDMDEVINS pDevIns = (PPDMDEVINS)RTMemAllocZ(RT_OFFSETOF(PDMDEVINS, achInstanceData[pPdmDev->pReg->cbInstance]));
859 pDevIns->u32Version = PDM_DEVINS_VERSION;
860 pDevIns->iInstance = 0;
861 pDevIns->pReg = pPdmDev->pReg;
862 pDevIns->pvInstanceDataR3 = &pDevIns->achInstanceData[0];
863 pDevIns->pHlpR3 = &g_tstDevPdmDevHlpR3;
864 pDevIns->pCfg = &Cfg;
865 pDevIns->Internal.s.pVmmCallbacks = &g_tstDevVmmCallbacks;
866 pDevIns->Internal.s.pDut = &Dut;
867 rc = pPdmDev->pReg->pfnConstruct(pDevIns, 0, &Cfg);
868 if (RT_SUCCESS(rc))
869 {
870 PRTDEVDUTIOPORT pIoPort = RTListGetFirst(&Dut.LstIoPorts, RTDEVDUTIOPORT, NdIoPorts);
871 uint32_t uVal = 0;
872 PDMCritSectEnter(pDevIns->pCritSectRoR3, VERR_IGNORED);
873 pIoPort->pfnInR0(pDevIns, pIoPort->pvUserR0, pIoPort->PortStart, &uVal, sizeof(uint8_t));
874 PDMCritSectLeave(pDevIns->pCritSectRoR3);
875 }
876 }
877 else
878 rc = VERR_NOT_FOUND;
879
880 return rc;
881}
882
883int main(int argc, char *argv[])
884{
885 /*
886 * Init the runtime and parse the arguments.
887 */
888 RTEXITCODE rcExit = RTEXITCODE_SUCCESS;
889 int rc = RTR3InitExe(argc, &argv, 0);
890 if (RT_SUCCESS(rc))
891 {
892 RTListInit(&g_LstPlugins);
893 RTListInit(&g_LstTestcases);
894 RTListInit(&g_LstPdmMods);
895 RTListInit(&g_LstPdmDevs);
896
897 rc = tstDevLoadPlugin("TSTDevTestcases");
898 if (RT_SUCCESS(rc) || true)
899 {
900 rc = tstDevPdmLoadMod(argv[1], TSTDEVPDMMODTYPE_R3);
901 if (RT_SUCCESS(rc))
902 rc = tstDevPdmDevCreate("nvme");
903 else
904 rcExit = RTEXITCODE_FAILURE;
905 }
906 else
907 rcExit = RTEXITCODE_FAILURE;
908
909 //rcExit = tstDevParseOptions(argc, argv);
910 }
911 else
912 rcExit = RTEXITCODE_FAILURE;
913
914 return rcExit;
915}
916
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