/* $Id: tstVDIo.cpp 37308 2011-06-02 12:53:10Z vboxsync $ */ /** @file * * VBox HDD container test utility - I/O replay. */ /* * Copyright (C) 2011 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ #define LOGGROUP LOGGROUP_DEFAULT #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "VDMemDisk.h" #include "VDIoBackendMem.h" #include "VDIoRnd.h" /** * A virtual file backed by memory. */ typedef struct VDFILE { /** Pointer to the next file. */ RTLISTNODE Node; /** Name of the file. */ char *pszName; /** Memory file baking the file. */ PVDMEMDISK pMemDisk; /** Flag whether the file is read locked. */ bool fReadLock; /** Flag whether the file is write locked. */ bool fWriteLock; } VDFILE, *PVDFILE; /** * VD storage object. */ typedef struct VDSTORAGE { /** Pointer to the file. */ PVDFILE pFile; /** Completion callback of the VD layer. */ PFNVDCOMPLETED pfnComplete; } VDSTORAGE, *PVDSTORAGE; /** * A virtual disk. */ typedef struct VDDISK { /** List node. */ RTLISTNODE ListNode; /** Name of the disk handle for identification. */ char *pszName; /** HDD handle to operate on. */ PVBOXHDD pVD; /** Memory disk used for data verification. */ PVDMEMDISK pMemDiskVerify; /** Critical section to serialize access to the memory disk. */ RTCRITSECT CritSectVerify; /** Physical CHS Geometry. */ VDGEOMETRY PhysGeom; /** Logical CHS geometry. */ VDGEOMETRY LogicalGeom; } VDDISK, *PVDDISK; /** * A data buffer with a pattern. */ typedef struct VDPATTERN { /** List node. */ RTLISTNODE ListNode; /** Name of the pattern. */ char *pszName; /** Size of the pattern. */ size_t cbPattern; /** Pointer to the buffer containing the pattern. */ void *pvPattern; } VDPATTERN, *PVDPATTERN; /** * Global VD test state. */ typedef struct VDTESTGLOB { /** List of active virtual disks. */ RTLISTNODE ListDisks; /** Head of the active file list. */ RTLISTNODE ListFiles; /** Head of the pattern list. */ RTLISTNODE ListPatterns; /** Memory I/O backend. */ PVDIOBACKENDMEM pIoBackend; /** Error interface. */ VDINTERFACE VDIError; /** Error interface callbacks. */ VDINTERFACEERROR VDIErrorCallbacks; /** Pointer to the per disk interface list. */ PVDINTERFACE pInterfacesDisk; /** I/O interface. */ VDINTERFACE VDIIo; /** I/O interface callbacks. */ VDINTERFACEIO VDIIoCallbacks; /** Pointer to the per image interface list. */ PVDINTERFACE pInterfacesImages; /** I/O RNG handle. */ PVDIORND pIoRnd; } VDTESTGLOB, *PVDTESTGLOB; /** * Transfer direction. */ typedef enum VDIOREQTXDIR { VDIOREQTXDIR_READ = 0, VDIOREQTXDIR_WRITE, VDIOREQTXDIR_FLUSH } VDIOREQTXDIR; /** * I/O request. */ typedef struct VDIOREQ { /** Transfer type. */ VDIOREQTXDIR enmTxDir; /** slot index. */ unsigned idx; /** Start offset. */ uint64_t off; /** Size to transfer. */ size_t cbReq; /** S/G Buffer */ RTSGBUF SgBuf; /** Data segment */ RTSGSEG DataSeg; /** Flag whether the request is outstanding or not. */ volatile bool fOutstanding; /** Buffer to use for reads. */ void *pvBufRead; /** Opaque user data. */ void *pvUser; } VDIOREQ, *PVDIOREQ; /** * I/O test data. */ typedef struct VDIOTEST { /** Start offset. */ uint64_t offStart; /** End offset. */ uint64_t offEnd; /** Flag whether random or sequential access is wanted */ bool fRandomAccess; /** Block size. */ size_t cbBlkIo; /** Number of bytes to transfer. */ uint64_t cbIo; /** Chance in percent to get a write. */ unsigned uWriteChance; /** Pointer to the I/O data generator. */ PVDIORND pIoRnd; /** Pointer to the data pattern to use. */ PVDPATTERN pPattern; /** Data dependent on the I/O mode (sequential or random). */ union { /** Next offset for sequential access. */ uint64_t offNext; /** Data for random acess. */ struct { /** Number of valid entries in the bitmap. */ uint32_t cBlocks; /** Pointer to the bitmap marking accessed blocks. */ uint8_t *pbMapAccessed; /** Number of unaccessed blocks. */ uint32_t cBlocksLeft; } Rnd; } u; } VDIOTEST, *PVDIOTEST; /** * Argument types. */ typedef enum VDSCRIPTARGTYPE { /** Argument is a string. */ VDSCRIPTARGTYPE_STRING = 0, /** Argument is a 64bit unsigned number. */ VDSCRIPTARGTYPE_UNSIGNED_NUMBER, /** Argument is a 64bit signed number. */ VDSCRIPTARGTYPE_SIGNED_NUMBER, /** Arugment is a unsigned 64bit range */ VDSCRIPTARGTYPE_UNSIGNED_RANGE, /** Arugment is a boolean. */ VDSCRIPTARGTYPE_BOOL } VDSCRIPTARGTYPE; /** * Script argument. */ typedef struct VDSCRIPTARG { /** Argument identifier. */ char chId; /** Type of the argument. */ VDSCRIPTARGTYPE enmType; /** Type depndent data. */ union { /** String. */ const char *pcszString; /** Bool. */ bool fFlag; /** unsigned number. */ uint64_t u64; /** Signed number. */ int64_t i64; /** Unsigned range. */ struct { uint64_t Start; uint64_t End; } Range; } u; } VDSCRIPTARG, *PVDSCRIPTARG; /** Script action handler. */ typedef DECLCALLBACK(int) FNVDSCRIPTACTION(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); /** Pointer to a script action handler. */ typedef FNVDSCRIPTACTION *PFNVDSCRIPTACTION; /** * Script argument descriptor. */ typedef struct VDSCRIPTARGDESC { /** Name of the arugment. */ const char *pcszName; /** Identifier for the argument. */ char chId; /** Type of the argument. */ VDSCRIPTARGTYPE enmType; /** Flags */ uint32_t fFlags; } VDSCRIPTARGDESC, *PVDSCRIPTARGDESC; /** Pointer to a const script argument descriptor. */ typedef const VDSCRIPTARGDESC *PCVDSCRIPTARGDESC; /** Flag whether the argument is mandatory. */ #define VDSCRIPTARGDESC_FLAG_MANDATORY RT_BIT(0) /** Flag whether the number can have a size suffix (K|M|G) */ #define VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX RT_BIT(1) /** * Script action. */ typedef struct VDSCRIPTACTION { /** Action name. */ const char *pcszAction; /** Pointer to the arguments. */ const PCVDSCRIPTARGDESC paArgDesc; /** Number of arugments in the array. */ unsigned cArgDescs; /** Pointer to the action handler. */ PFNVDSCRIPTACTION pfnHandler; } VDSCRIPTACTION, *PVDSCRIPTACTION; typedef const VDSCRIPTACTION *PCVDSCRIPTACTION; static DECLCALLBACK(int) vdScriptHandlerCreate(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerOpen(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerIo(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerFlush(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerMerge(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerCompact(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerClose(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerIoRngCreate(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerIoRngDestroy(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerIoPatternCreateFromNumber(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerIoPatternCreateFromFile(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerIoPatternDestroy(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerSleep(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerDumpFile(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerCreateDisk(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerDestroyDisk(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerCompareDisks(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerDumpDiskInfo(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); static DECLCALLBACK(int) vdScriptHandlerPrintMsg(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs); /* create action */ const VDSCRIPTARGDESC g_aArgCreate[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"mode", 'm', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"type", 't', VDSCRIPTARGTYPE_STRING, 0}, {"backend", 'b', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"size", 's', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY | VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX} }; /* open action */ const VDSCRIPTARGDESC g_aArgOpen[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"backend", 'b', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"shareable", 's', VDSCRIPTARGTYPE_BOOL, 0}, {"readonly", 'r', VDSCRIPTARGTYPE_BOOL, 0} }; /* I/O action */ const VDSCRIPTARGDESC g_aArgIo[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"async", 'a', VDSCRIPTARGTYPE_BOOL, 0}, {"max-reqs", 'l', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, 0}, {"mode", 'm', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"size", 's', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX}, {"blocksize", 'b', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY | VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX}, {"off", 'o', VDSCRIPTARGTYPE_UNSIGNED_RANGE, VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX}, {"writes", 'w', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"pattern", 'p', VDSCRIPTARGTYPE_STRING, 0}, }; /* flush action */ const VDSCRIPTARGDESC g_aArgFlush[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"async", 'a', VDSCRIPTARGTYPE_BOOL, 0} }; /* merge action */ const VDSCRIPTARGDESC g_aArgMerge[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"from", 'f', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"to", 't', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY}, }; /* Compact a disk */ const VDSCRIPTARGDESC g_aArgCompact[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"image", 'i', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY}, }; /* close action */ const VDSCRIPTARGDESC g_aArgClose[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"mode", 'm', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"delete", 'r', VDSCRIPTARGTYPE_BOOL, VDSCRIPTARGDESC_FLAG_MANDATORY} }; /* I/O RNG create action */ const VDSCRIPTARGDESC g_aArgIoRngCreate[] = { /* pcszName chId enmType fFlags */ {"size", 'd', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY | VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX}, {"mode", 'm', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"seed", 's', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, 0} }; /* I/O pattern create action */ const VDSCRIPTARGDESC g_aArgIoPatternCreateFromNumber[] = { /* pcszName chId enmType fFlags */ {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"size", 's', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY | VDSCRIPTARGDESC_FLAG_SIZE_SUFFIX}, {"pattern", 'p', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY}, }; /* I/O pattern create action */ const VDSCRIPTARGDESC g_aArgIoPatternCreateFromFile[] = { /* pcszName chId enmType fFlags */ {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"file", 'f', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, }; /* I/O pattern destroy action */ const VDSCRIPTARGDESC g_aArgIoPatternDestroy[] = { /* pcszName chId enmType fFlags */ {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY} }; /* Sleep */ const VDSCRIPTARGDESC g_aArgSleep[] = { /* pcszName chId enmType fFlags */ {"time", 't', VDSCRIPTARGTYPE_UNSIGNED_NUMBER, VDSCRIPTARGDESC_FLAG_MANDATORY} }; /* Dump memory file */ const VDSCRIPTARGDESC g_aArgDumpFile[] = { /* pcszName chId enmType fFlags */ {"file", 'f', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"path", 'p', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY} }; /* Create virtual disk handle */ const VDSCRIPTARGDESC g_aArgCreateDisk[] = { /* pcszName chId enmType fFlags */ {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"verify", 'v', VDSCRIPTARGTYPE_BOOL, 0} }; /* Create virtual disk handle */ const VDSCRIPTARGDESC g_aArgDestroyDisk[] = { /* pcszName chId enmType fFlags */ {"name", 'n', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY} }; /* Compare virtual disks */ const VDSCRIPTARGDESC g_aArgCompareDisks[] = { /* pcszName chId enmType fFlags */ {"disk1", '1', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, {"disk2", '2', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY} }; /* Dump disk info */ const VDSCRIPTARGDESC g_aArgDumpDiskInfo[] = { /* pcszName chId enmType fFlags */ {"disk", 'd', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, }; /* Print message */ const VDSCRIPTARGDESC g_aArgPrintMsg[] = { /* pcszName chId enmType fFlags */ {"msg", 'm', VDSCRIPTARGTYPE_STRING, VDSCRIPTARGDESC_FLAG_MANDATORY}, }; const VDSCRIPTACTION g_aScriptActions[] = { /* pcszAction paArgDesc cArgDescs pfnHandler */ {"create", g_aArgCreate, RT_ELEMENTS(g_aArgCreate), vdScriptHandlerCreate}, {"open", g_aArgOpen, RT_ELEMENTS(g_aArgOpen), vdScriptHandlerOpen}, {"io", g_aArgIo, RT_ELEMENTS(g_aArgIo), vdScriptHandlerIo}, {"flush", g_aArgFlush, RT_ELEMENTS(g_aArgFlush), vdScriptHandlerFlush}, {"close", g_aArgClose, RT_ELEMENTS(g_aArgClose), vdScriptHandlerClose}, {"merge", g_aArgMerge, RT_ELEMENTS(g_aArgMerge), vdScriptHandlerMerge}, {"compact", g_aArgCompact, RT_ELEMENTS(g_aArgCompact), vdScriptHandlerCompact}, {"iorngcreate", g_aArgIoRngCreate, RT_ELEMENTS(g_aArgIoRngCreate), vdScriptHandlerIoRngCreate}, {"iorngdestroy", NULL, 0, vdScriptHandlerIoRngDestroy}, {"iopatterncreatefromnumber", g_aArgIoPatternCreateFromNumber, RT_ELEMENTS(g_aArgIoPatternCreateFromNumber), vdScriptHandlerIoPatternCreateFromNumber}, {"iopatterncreatefromfile", g_aArgIoPatternCreateFromFile, RT_ELEMENTS(g_aArgIoPatternCreateFromFile), vdScriptHandlerIoPatternCreateFromFile}, {"iopatterndestroy", g_aArgIoPatternDestroy, RT_ELEMENTS(g_aArgIoPatternDestroy), vdScriptHandlerIoPatternDestroy}, {"sleep", g_aArgSleep, RT_ELEMENTS(g_aArgSleep), vdScriptHandlerSleep}, {"dumpfile", g_aArgDumpFile, RT_ELEMENTS(g_aArgDumpFile), vdScriptHandlerDumpFile}, {"createdisk", g_aArgCreateDisk, RT_ELEMENTS(g_aArgCreateDisk), vdScriptHandlerCreateDisk}, {"destroydisk", g_aArgDestroyDisk, RT_ELEMENTS(g_aArgDestroyDisk), vdScriptHandlerDestroyDisk}, {"comparedisks", g_aArgCompareDisks, RT_ELEMENTS(g_aArgCompareDisks), vdScriptHandlerCompareDisks}, {"dumpdiskinfo", g_aArgDumpDiskInfo, RT_ELEMENTS(g_aArgDumpDiskInfo), vdScriptHandlerDumpDiskInfo}, {"print", g_aArgPrintMsg, RT_ELEMENTS(g_aArgPrintMsg), vdScriptHandlerPrintMsg} }; const unsigned g_cScriptActions = RT_ELEMENTS(g_aScriptActions); static void tstVDError(void *pvUser, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va) { RTPrintf("tstVD: Error %Rrc at %s:%u (%s): ", rc, RT_SRC_POS_ARGS); RTPrintfV(pszFormat, va); RTPrintf("\n"); } static int tstVDMessage(void *pvUser, const char *pszFormat, va_list va) { RTPrintf("tstVD: "); RTPrintfV(pszFormat, va); return VINF_SUCCESS; } static int tstVDIoTestInit(PVDIOTEST pIoTest, PVDTESTGLOB pGlob, bool fRandomAcc, uint64_t cbIo, size_t cbBlkSize, uint64_t offStart, uint64_t offEnd, unsigned uWriteChance, PVDPATTERN pPattern); static bool tstVDIoTestRunning(PVDIOTEST pIoTest); static void tstVDIoTestDestroy(PVDIOTEST pIoTest); static bool tstVDIoTestReqOutstanding(PVDIOREQ pIoReq); static int tstVDIoTestReqInit(PVDIOTEST pIoTest, PVDIOREQ pIoReq, void *pvUser); static void tstVDIoTestReqComplete(void *pvUser1, void *pvUser2, int rcReq); static PVDDISK tstVDIoGetDiskByName(PVDTESTGLOB pGlob, const char *pcszDisk); static PVDPATTERN tstVDIoGetPatternByName(PVDTESTGLOB pGlob, const char *pcszName); static PVDPATTERN tstVDIoPatternCreate(const char *pcszName, size_t cbPattern); static int tstVDIoPatternGetBuffer(PVDPATTERN pPattern, void **ppv, size_t cb); static DECLCALLBACK(int) vdScriptHandlerCreate(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; uint64_t cbSize = 0; const char *pcszBackend = NULL; const char *pcszImage = NULL; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; bool fBase = false; bool fDynamic = true; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'm': { if (!RTStrICmp(paScriptArgs[i].u.pcszString, "base")) fBase = true; else if (!RTStrICmp(paScriptArgs[i].u.pcszString, "diff")) fBase = false; else { RTPrintf("Invalid image mode '%s' given\n", paScriptArgs[i].u.pcszString); rc = VERR_INVALID_PARAMETER; } break; } case 'n': { pcszImage = paScriptArgs[i].u.pcszString; break; } case 'b': { pcszBackend = paScriptArgs[i].u.pcszString; break; } case 's': { cbSize = paScriptArgs[i].u.u64; break; } case 't': { if (!RTStrICmp(paScriptArgs[i].u.pcszString, "fixed")) fDynamic = false; else if (!RTStrICmp(paScriptArgs[i].u.pcszString, "dynamic")) fDynamic = true; else { RTPrintf("Invalid image type '%s' given\n", paScriptArgs[i].u.pcszString); rc = VERR_INVALID_PARAMETER; } break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (pDisk) { unsigned fImageFlags = VD_IMAGE_FLAGS_NONE; if (!fDynamic) fImageFlags |= VD_IMAGE_FLAGS_FIXED; if (fBase) rc = VDCreateBase(pDisk->pVD, pcszBackend, pcszImage, cbSize, fImageFlags, NULL, &pDisk->PhysGeom, &pDisk->LogicalGeom, NULL, VD_OPEN_FLAGS_ASYNC_IO, pGlob->pInterfacesImages, NULL); else rc = VDCreateDiff(pDisk->pVD, pcszBackend, pcszImage, fImageFlags, NULL, NULL, NULL, VD_OPEN_FLAGS_ASYNC_IO, pGlob->pInterfacesImages, NULL); } else rc = VERR_NOT_FOUND; } return rc; } static DECLCALLBACK(int) vdScriptHandlerOpen(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszBackend = NULL; const char *pcszImage = NULL; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; bool fShareable = false; bool fReadonly = false; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'n': { pcszImage = paScriptArgs[i].u.pcszString; break; } case 'b': { pcszBackend = paScriptArgs[i].u.pcszString; break; } case 's': { fShareable = paScriptArgs[i].u.fFlag; break; } case 'r': { fReadonly = paScriptArgs[i].u.fFlag; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (pDisk) { unsigned fOpenFlags = VD_OPEN_FLAGS_ASYNC_IO; if (fShareable) fOpenFlags |= VD_OPEN_FLAGS_SHAREABLE; if (fReadonly) fOpenFlags |= VD_OPEN_FLAGS_READONLY; rc = VDOpen(pDisk->pVD, pcszBackend, pcszImage, fOpenFlags, pGlob->pInterfacesImages); } else rc = VERR_NOT_FOUND; } return rc; } static DECLCALLBACK(int) vdScriptHandlerIo(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; bool fAsync = false; bool fRandomAcc = false; uint64_t cbIo = 0; uint64_t cbBlkSize = 0; bool fDataProviderRnd = false; bool fPrintStats = false; uint64_t offStart = 0; uint64_t offEnd = 0; unsigned cMaxReqs = 0; uint8_t uWriteChance = 0; const char *pcszDisk = NULL; const char *pcszPattern = NULL; PVDDISK pDisk = NULL; PVDPATTERN pPattern = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'a': { fAsync = paScriptArgs[i].u.fFlag; break; } case 'l': { cMaxReqs = paScriptArgs[i].u.u64; break; } case 'm': { if (!RTStrICmp(paScriptArgs[i].u.pcszString, "seq")) fRandomAcc = false; else if (!RTStrICmp(paScriptArgs[i].u.pcszString, "rnd")) fRandomAcc = true; else { RTPrintf("Invalid access mode '%s'\n", paScriptArgs[i].u.pcszString); rc = VERR_INVALID_PARAMETER; } break; } case 's': { cbIo = paScriptArgs[i].u.u64; break; } case 'b': { cbBlkSize = paScriptArgs[i].u.u64; break; } case 'o': { offStart = paScriptArgs[i].u.Range.Start; offEnd = paScriptArgs[i].u.Range.End; break; } case 'w': { uWriteChance = (uint8_t)paScriptArgs[i].u.u64; break; } case 'p': { pcszPattern = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (!pDisk) rc = VERR_NOT_FOUND; } if (RT_SUCCESS(rc)) { /* Set defaults if not set by the user. */ if (offStart == 0 && offEnd == 0) { offEnd = VDGetSize(pDisk->pVD, VD_LAST_IMAGE); if (offEnd == 0) return VERR_INVALID_STATE; } if (!cbIo) cbIo = offEnd; } if ( RT_SUCCESS(rc) && pcszPattern) { pPattern = tstVDIoGetPatternByName(pGlob, pcszPattern); if (!pPattern) rc = VERR_NOT_FOUND; } if (RT_SUCCESS(rc)) { VDIOTEST IoTest; rc = tstVDIoTestInit(&IoTest, pGlob, fRandomAcc, cbIo, cbBlkSize, offStart, offEnd, uWriteChance, pPattern); if (RT_SUCCESS(rc)) { PVDIOREQ paIoReq = NULL; unsigned cMaxTasksOutstanding = fAsync ? cMaxReqs : 1; RTSEMEVENT EventSem; rc = RTSemEventCreate(&EventSem); paIoReq = (PVDIOREQ)RTMemAllocZ(cMaxTasksOutstanding * sizeof(VDIOREQ)); if (paIoReq && RT_SUCCESS(rc)) { uint64_t NanoTS = RTTimeNanoTS(); /* Init requests. */ for (unsigned i = 0; i < cMaxTasksOutstanding; i++) { paIoReq[i].idx = i; paIoReq[i].pvBufRead = RTMemAlloc(cbBlkSize); if (!paIoReq[i].pvBufRead) { rc = VERR_NO_MEMORY; break; } } while ( tstVDIoTestRunning(&IoTest) && RT_SUCCESS(rc)) { bool fTasksOutstanding = false; unsigned idx = 0; /* Submit all idling requests. */ while ( idx < cMaxTasksOutstanding && tstVDIoTestRunning(&IoTest)) { if (!tstVDIoTestReqOutstanding(&paIoReq[idx])) { rc = tstVDIoTestReqInit(&IoTest, &paIoReq[idx], pDisk); AssertRC(rc); if (RT_SUCCESS(rc)) { if (!fAsync) { switch (paIoReq[idx].enmTxDir) { case VDIOREQTXDIR_READ: { rc = VDRead(pDisk->pVD, paIoReq[idx].off, paIoReq[idx].DataSeg.pvSeg, paIoReq[idx].cbReq); if (RT_SUCCESS(rc) && pDisk->pMemDiskVerify) { RTSGBUF SgBuf; RTSgBufInit(&SgBuf, &paIoReq[idx].DataSeg, 1); if (VDMemDiskCmp(pDisk->pMemDiskVerify, paIoReq[idx].off, paIoReq[idx].cbReq, &SgBuf)) { RTPrintf("Corrupted disk at offset %llu!\n", paIoReq[idx].off); rc = VERR_INVALID_STATE; } } break; } case VDIOREQTXDIR_WRITE: { rc = VDWrite(pDisk->pVD, paIoReq[idx].off, paIoReq[idx].DataSeg.pvSeg, paIoReq[idx].cbReq); if (RT_SUCCESS(rc) && pDisk->pMemDiskVerify) { RTSGBUF SgBuf; RTSgBufInit(&SgBuf, &paIoReq[idx].DataSeg, 1); rc = VDMemDiskWrite(pDisk->pMemDiskVerify, paIoReq[idx].off, paIoReq[idx].cbReq, &SgBuf); } break; } case VDIOREQTXDIR_FLUSH: { rc = VDFlush(pDisk->pVD); break; } } ASMAtomicXchgBool(&paIoReq[idx].fOutstanding, false); if (RT_SUCCESS(rc)) idx++; } else { LogFlow(("Queuing request %d\n", idx)); switch (paIoReq[idx].enmTxDir) { case VDIOREQTXDIR_READ: { rc = VDAsyncRead(pDisk->pVD, paIoReq[idx].off, paIoReq[idx].cbReq, &paIoReq[idx].SgBuf, tstVDIoTestReqComplete, &paIoReq[idx], EventSem); break; } case VDIOREQTXDIR_WRITE: { rc = VDAsyncWrite(pDisk->pVD, paIoReq[idx].off, paIoReq[idx].cbReq, &paIoReq[idx].SgBuf, tstVDIoTestReqComplete, &paIoReq[idx], EventSem); break; } case VDIOREQTXDIR_FLUSH: { rc = VDAsyncFlush(pDisk->pVD, tstVDIoTestReqComplete, &paIoReq[idx], EventSem); break; } } if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS) { idx++; fTasksOutstanding = true; rc = VINF_SUCCESS; } else if (rc == VINF_VD_ASYNC_IO_FINISHED) { LogFlow(("Request %d completed\n", idx)); switch (paIoReq[idx].enmTxDir) { case VDIOREQTXDIR_READ: { if (pDisk->pMemDiskVerify) { RTCritSectEnter(&pDisk->CritSectVerify); RTSgBufReset(&paIoReq[idx].SgBuf); if (VDMemDiskCmp(pDisk->pMemDiskVerify, paIoReq[idx].off, paIoReq[idx].cbReq, &paIoReq[idx].SgBuf)) { RTPrintf("Corrupted disk at offset %llu!\n", paIoReq[idx].off); rc = VERR_INVALID_STATE; } RTCritSectLeave(&pDisk->CritSectVerify); } break; } case VDIOREQTXDIR_WRITE: { if (pDisk->pMemDiskVerify) { RTCritSectEnter(&pDisk->CritSectVerify); RTSgBufReset(&paIoReq[idx].SgBuf); rc = VDMemDiskWrite(pDisk->pMemDiskVerify, paIoReq[idx].off, paIoReq[idx].cbReq, &paIoReq[idx].SgBuf); RTCritSectLeave(&pDisk->CritSectVerify); } break; } case VDIOREQTXDIR_FLUSH: break; } ASMAtomicXchgBool(&paIoReq[idx].fOutstanding, false); if (rc != VERR_INVALID_STATE) rc = VINF_SUCCESS; } } if (RT_FAILURE(rc)) RTPrintf("Error submitting task %u rc=%Rrc\n", paIoReq[idx].idx, rc); } } } /* Wait for a request to complete. */ if ( fAsync && fTasksOutstanding) { rc = RTSemEventWait(EventSem, RT_INDEFINITE_WAIT); AssertRC(rc); } } /* Cleanup, wait for all tasks to complete. */ while (fAsync) { unsigned idx = 0; bool fAllIdle = true; while (idx < cMaxTasksOutstanding) { if (tstVDIoTestReqOutstanding(&paIoReq[idx])) { fAllIdle = false; break; } idx++; } if (!fAllIdle) { rc = RTSemEventWait(EventSem, 100); Assert(RT_SUCCESS(rc) || rc == VERR_TIMEOUT); } else break; } NanoTS = RTTimeNanoTS() - NanoTS; uint64_t SpeedKBs = (uint64_t)(cbIo / (NanoTS / 1000000000.0) / 1024); RTPrintf("I/O Test: Throughput %lld kb/s\n", SpeedKBs); RTSemEventDestroy(EventSem); RTMemFree(paIoReq); } else rc = VERR_NO_MEMORY; tstVDIoTestDestroy(&IoTest); } } return rc; } static DECLCALLBACK(int) vdScriptHandlerFlush(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; bool fAsync = false; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'a': { fAsync = paScriptArgs[i].u.fFlag; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (!pDisk) rc = VERR_NOT_FOUND; else if (fAsync) { VDIOREQ IoReq; RTSEMEVENT EventSem; rc = RTSemEventCreate(&EventSem); if (RT_SUCCESS(rc)) { memset(&IoReq, 0, sizeof(VDIOREQ)); IoReq.enmTxDir = VDIOREQTXDIR_FLUSH; IoReq.pvUser = pDisk; IoReq.idx = 0; rc = VDAsyncFlush(pDisk->pVD, tstVDIoTestReqComplete, &IoReq, EventSem); if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS) { rc = RTSemEventWait(EventSem, RT_INDEFINITE_WAIT); AssertRC(rc); } else if (rc == VINF_VD_ASYNC_IO_FINISHED) rc = VINF_SUCCESS; RTSemEventDestroy(EventSem); } } else rc = VDFlush(pDisk->pVD); } return rc; } static DECLCALLBACK(int) vdScriptHandlerMerge(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; unsigned nImageFrom = 0; unsigned nImageTo = 0; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'f': { nImageFrom = (unsigned)paScriptArgs[i].u.u64; break; } case 't': { nImageTo = (unsigned)paScriptArgs[i].u.u64; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (!pDisk) rc = VERR_NOT_FOUND; else { /** @todo: Provide progress interface to test that cancelation * doesn't corrupt the data. */ rc = VDMerge(pDisk->pVD, nImageFrom, nImageTo, NULL); } } return rc; } static DECLCALLBACK(int) vdScriptHandlerCompact(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; unsigned nImage = 0; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'i': { nImage = (unsigned)paScriptArgs[i].u.u64; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (!pDisk) rc = VERR_NOT_FOUND; else { /** @todo: Provide progress interface to test that cancelation * doesn't corrupt the data. */ rc = VDCompact(pDisk->pVD, nImage, NULL); } } return rc; } static DECLCALLBACK(int) vdScriptHandlerClose(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; bool fAll = false; bool fDelete = false; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'm': { if (!RTStrICmp(paScriptArgs[i].u.pcszString, "all")) fAll = true; else if (!RTStrICmp(paScriptArgs[i].u.pcszString, "single")) fAll = false; else { RTPrintf("Invalid mode '%s' given\n", paScriptArgs[i].u.pcszString); rc = VERR_INVALID_PARAMETER; } break; } case 'r': { fDelete = paScriptArgs[i].u.fFlag; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } if (RT_FAILURE(rc)) break; } if ( RT_SUCCESS(rc) && fAll && fDelete) { RTPrintf("mode=all doesn't work with delete=yes\n"); rc = VERR_INVALID_PARAMETER; } if (RT_SUCCESS(rc)) { pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (pDisk) { if (fAll) rc = VDCloseAll(pDisk->pVD); else rc = VDClose(pDisk->pVD, fDelete); } else rc = VERR_NOT_FOUND; } return rc; } static DECLCALLBACK(int) vdScriptHandlerIoRngCreate(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; size_t cbPattern = 0; uint64_t uSeed = 0; const char *pcszSeeder = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { cbPattern = paScriptArgs[i].u.u64; break; } case 's': { uSeed = paScriptArgs[i].u.u64; break; } case 'm': { pcszSeeder = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } if (pGlob->pIoRnd) { RTPrintf("I/O RNG already exists\n"); rc = VERR_INVALID_STATE; } else { uint64_t uSeedToUse = 0; if (!RTStrICmp(pcszSeeder, "manual")) uSeedToUse = uSeed; else if (!RTStrICmp(pcszSeeder, "time")) uSeedToUse = RTTimeSystemMilliTS(); else if (!RTStrICmp(pcszSeeder, "system")) { RTRAND hRand; rc = RTRandAdvCreateSystemTruer(&hRand); if (RT_SUCCESS(rc)) { RTRandAdvBytes(hRand, &uSeedToUse, sizeof(uSeedToUse)); RTRandAdvDestroy(hRand); } } if (RT_SUCCESS(rc)) rc = VDIoRndCreate(&pGlob->pIoRnd, cbPattern, uSeed); } return rc; } static DECLCALLBACK(int) vdScriptHandlerIoRngDestroy(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { if (pGlob->pIoRnd) { VDIoRndDestroy(pGlob->pIoRnd); pGlob->pIoRnd = NULL; } else RTPrintf("WARNING: No I/O RNG active, faulty script. Continuing\n"); return VINF_SUCCESS; } static DECLCALLBACK(int) vdScriptHandlerIoPatternCreateFromNumber(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; size_t cbPattern = 0; const char *pcszName = NULL; uint64_t u64Pattern = 0; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'n': { pcszName = paScriptArgs[i].u.pcszString; break; } case 's': { cbPattern = paScriptArgs[i].u.u64; break; } case 'p': { u64Pattern = paScriptArgs[i].u.u64; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } PVDPATTERN pPattern = tstVDIoGetPatternByName(pGlob, pcszName); if (!pPattern) { pPattern = tstVDIoPatternCreate(pcszName, RT_ALIGN_Z(cbPattern, sizeof(uint64_t))); if (pPattern) { /* Fill the buffer. */ void *pv = pPattern->pvPattern; while (pPattern->cbPattern > 0) { *((uint64_t*)pv) = u64Pattern; pPattern->cbPattern -= sizeof(uint64_t); pv = (uint64_t *)pv + 1; } pPattern->cbPattern = cbPattern; /* Set to the desired size. (could be unaligned) */ RTListAppend(&pGlob->ListPatterns, &pPattern->ListNode); } else rc = VERR_NO_MEMORY; } else rc = VERR_ALREADY_EXISTS; return rc; } static DECLCALLBACK(int) vdScriptHandlerIoPatternCreateFromFile(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszName = NULL; const char *pcszFile = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'n': { pcszName = paScriptArgs[i].u.pcszString; break; } case 'f': { pcszFile = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } PVDPATTERN pPattern = tstVDIoGetPatternByName(pGlob, pcszName); if (!pPattern) { RTFILE hFile; uint64_t cbPattern = 0; rc = RTFileOpen(&hFile, pcszFile, RTFILE_O_DENY_NONE | RTFILE_O_OPEN | RTFILE_O_READ); if (RT_SUCCESS(rc)) { rc = RTFileGetSize(hFile, &cbPattern); if (RT_SUCCESS(rc)) { pPattern = tstVDIoPatternCreate(pcszName, (size_t)cbPattern); if (pPattern) { rc = RTFileRead(hFile, pPattern->pvPattern, (size_t)cbPattern, NULL); if (RT_SUCCESS(rc)) RTListAppend(&pGlob->ListPatterns, &pPattern->ListNode); else { RTMemFree(pPattern->pvPattern); RTStrFree(pPattern->pszName); RTMemFree(pPattern); } } else rc = VERR_NO_MEMORY; } RTFileClose(hFile); } } else rc = VERR_ALREADY_EXISTS; return rc; } static DECLCALLBACK(int) vdScriptHandlerIoPatternDestroy(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszName = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'n': { pcszName = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } PVDPATTERN pPattern = tstVDIoGetPatternByName(pGlob, pcszName); if (pPattern) { RTListNodeRemove(&pPattern->ListNode); RTMemFree(pPattern->pvPattern); RTStrFree(pPattern->pszName); RTMemFree(pPattern); } else rc = VERR_NOT_FOUND; return rc; } static DECLCALLBACK(int) vdScriptHandlerSleep(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; uint64_t cMillies = 0; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 't': { cMillies = paScriptArgs[i].u.u64; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } rc = RTThreadSleep(cMillies); return rc; } static DECLCALLBACK(int) vdScriptHandlerDumpFile(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszFile = NULL; const char *pcszPathToDump = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'f': { pcszFile = paScriptArgs[i].u.pcszString; break; } case 'p': { pcszPathToDump = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } /* Check for the file. */ PVDFILE pIt = NULL; bool fFound = false; RTListForEach(&pGlob->ListFiles, pIt, VDFILE, Node) { if (!RTStrCmp(pIt->pszName, pcszFile)) { fFound = true; break; } } if (fFound) { RTPrintf("Dumping memory file %s to %s, this might take some time\n", pcszFile, pcszPathToDump); rc = VDMemDiskWriteToFile(pIt->pMemDisk, pcszPathToDump); } else rc = VERR_FILE_NOT_FOUND; return rc; } static DECLCALLBACK(int) vdScriptHandlerCreateDisk(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; bool fVerify = false; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'n': { pcszDisk = paScriptArgs[i].u.pcszString; break; } case 'v': { fVerify = paScriptArgs[i].u.fFlag; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (pDisk) rc = VERR_ALREADY_EXISTS; else { pDisk = (PVDDISK)RTMemAllocZ(sizeof(VDDISK)); if (pDisk) { pDisk->pszName = RTStrDup(pcszDisk); if (pDisk->pszName) { rc = VINF_SUCCESS; if (fVerify) { rc = VDMemDiskCreate(&pDisk->pMemDiskVerify, 0 /* Growing */); if (RT_SUCCESS(rc)) { rc = RTCritSectInit(&pDisk->CritSectVerify); if (RT_FAILURE(rc)) VDMemDiskDestroy(pDisk->pMemDiskVerify); } } if (RT_SUCCESS(rc)) { rc = VDCreate(pGlob->pInterfacesDisk, VDTYPE_HDD, &pDisk->pVD); if (RT_SUCCESS(rc)) RTListAppend(&pGlob->ListDisks, &pDisk->ListNode); else { if (fVerify) { RTCritSectDelete(&pDisk->CritSectVerify); VDMemDiskDestroy(pDisk->pMemDiskVerify); } RTStrFree(pDisk->pszName); } } } else rc = VERR_NO_MEMORY; if (RT_FAILURE(rc)) RTMemFree(pDisk); } else rc = VERR_NO_MEMORY; } return rc; } static DECLCALLBACK(int) vdScriptHandlerDestroyDisk(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'n': { pcszDisk = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (pDisk) { RTListNodeRemove(&pDisk->ListNode); VDDestroy(pDisk->pVD); if (pDisk->pMemDiskVerify) { VDMemDiskDestroy(pDisk->pMemDiskVerify); RTCritSectDelete(&pDisk->CritSectVerify); } RTStrFree(pDisk->pszName); RTMemFree(pDisk); } else rc = VERR_NOT_FOUND; return rc; } static DECLCALLBACK(int) vdScriptHandlerCompareDisks(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszDisk1 = NULL; PVDDISK pDisk1 = NULL; const char *pcszDisk2 = NULL; PVDDISK pDisk2 = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case '1': { pcszDisk1 = paScriptArgs[i].u.pcszString; break; } case '2': { pcszDisk2 = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } pDisk1 = tstVDIoGetDiskByName(pGlob, pcszDisk1); pDisk2 = tstVDIoGetDiskByName(pGlob, pcszDisk2); if (pDisk1 && pDisk2) { uint8_t *pbBuf1 = (uint8_t *)RTMemAllocZ(16 * _1M); uint8_t *pbBuf2 = (uint8_t *)RTMemAllocZ(16 * _1M); if (pbBuf1 && pbBuf2) { uint64_t cbDisk1, cbDisk2; uint64_t uOffCur = 0; cbDisk1 = VDGetSize(pDisk1->pVD, VD_LAST_IMAGE); cbDisk2 = VDGetSize(pDisk2->pVD, VD_LAST_IMAGE); if (cbDisk1 != cbDisk2) RTPrintf("Disks differ in size %llu vs %llu\n", cbDisk1, cbDisk2); else { while (uOffCur < cbDisk1) { size_t cbRead = RT_MIN(cbDisk1, 16 * _1M); rc = VDRead(pDisk1->pVD, uOffCur, pbBuf1, cbRead); if (RT_SUCCESS(rc)) rc = VDRead(pDisk2->pVD, uOffCur, pbBuf2, cbRead); if (RT_SUCCESS(rc)) { if (memcmp(pbBuf1, pbBuf2, cbRead)) { RTPrintf("Disks differ at offset %llu\n", uOffCur); rc = VERR_DEV_IO_ERROR; break; } } else { RTPrintf("Reading one disk at offset %llu failed\n", uOffCur); break; } uOffCur += cbRead; cbDisk1 -= cbRead; } } RTMemFree(pbBuf1); RTMemFree(pbBuf2); } else { if (pbBuf1) RTMemFree(pbBuf1); if (pbBuf2) RTMemFree(pbBuf2); rc = VERR_NO_MEMORY; } } else rc = VERR_NOT_FOUND; return rc; } static DECLCALLBACK(int) vdScriptHandlerDumpDiskInfo(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { int rc = VINF_SUCCESS; const char *pcszDisk = NULL; PVDDISK pDisk = NULL; for (unsigned i = 0; i < cScriptArgs; i++) { switch (paScriptArgs[i].chId) { case 'd': { pcszDisk = paScriptArgs[i].u.pcszString; break; } default: AssertMsgFailed(("Invalid argument given!\n")); } } pDisk = tstVDIoGetDiskByName(pGlob, pcszDisk); if (pDisk) VDDumpImages(pDisk->pVD); else rc = VERR_NOT_FOUND; return rc; } static DECLCALLBACK(int) vdScriptHandlerPrintMsg(PVDTESTGLOB pGlob, PVDSCRIPTARG paScriptArgs, unsigned cScriptArgs) { RTPrintf("%s\n", paScriptArgs[0].u.pcszString); return VINF_SUCCESS; } static DECLCALLBACK(int) tstVDIoFileOpen(void *pvUser, const char *pszLocation, uint32_t fOpen, PFNVDCOMPLETED pfnCompleted, void **ppStorage) { int rc = VINF_SUCCESS; PVDTESTGLOB pGlob = (PVDTESTGLOB)pvUser; bool fFound = false; /* * Some backends use ./ for paths, strip it. * @todo: Implement proper directory support for the * memory filesystem. */ if ( strlen(pszLocation) >= 2 && *pszLocation == '.' && pszLocation[1] == '/') pszLocation += 2; /* Check if the file exists. */ PVDFILE pIt = NULL; RTListForEach(&pGlob->ListFiles, pIt, VDFILE, Node) { if (!RTStrCmp(pIt->pszName, pszLocation)) { fFound = true; break; } } if ((fOpen & RTFILE_O_ACTION_MASK) == RTFILE_O_CREATE) { /* If the file exists delete the memory disk. */ if (fFound) rc = VDMemDiskSetSize(pIt->pMemDisk, 0); else { /* Create completey new. */ pIt = (PVDFILE)RTMemAllocZ(sizeof(VDFILE)); if (pIt) { pIt->pszName = RTStrDup(pszLocation); if (pIt->pszName) { rc = VDMemDiskCreate(&pIt->pMemDisk, 0); } else rc = VERR_NO_MEMORY; if (RT_FAILURE(rc)) { if (pIt->pszName) RTStrFree(pIt->pszName); RTMemFree(pIt); } } else rc = VERR_NO_MEMORY; RTListAppend(&pGlob->ListFiles, &pIt->Node); } } else if ((fOpen & RTFILE_O_ACTION_MASK) == RTFILE_O_OPEN) { if (!fFound) rc = VERR_FILE_NOT_FOUND; } else rc = VERR_INVALID_PARAMETER; if (RT_SUCCESS(rc)) { AssertPtr(pIt); PVDSTORAGE pStorage = (PVDSTORAGE)RTMemAllocZ(sizeof(VDSTORAGE)); if (!pStorage) rc = VERR_NO_MEMORY; pStorage->pFile = pIt; pStorage->pfnComplete = pfnCompleted; *ppStorage = pStorage; } return rc; } static DECLCALLBACK(int) tstVDIoFileClose(void *pvUser, void *pStorage) { PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; RTMemFree(pIoStorage); return VINF_SUCCESS; } static DECLCALLBACK(int) tstVDIoFileDelete(void *pvUser, const char *pcszFilename) { int rc = VINF_SUCCESS; PVDTESTGLOB pGlob = (PVDTESTGLOB)pvUser; bool fFound = false; /* * Some backends use ./ for paths, strip it. * @todo: Implement proper directory support for the * memory filesystem. */ if ( strlen(pcszFilename) >= 2 && *pcszFilename == '.' && pcszFilename[1] == '/') pcszFilename += 2; /* Check if the file exists. */ PVDFILE pIt = NULL; RTListForEach(&pGlob->ListFiles, pIt, VDFILE, Node) { if (!RTStrCmp(pIt->pszName, pcszFilename)) { fFound = true; break; } } if (fFound) { RTListNodeRemove(&pIt->Node); VDMemDiskDestroy(pIt->pMemDisk); RTStrFree(pIt->pszName); RTMemFree(pIt); } else rc = VERR_FILE_NOT_FOUND; return rc; } static DECLCALLBACK(int) tstVDIoFileMove(void *pvUser, const char *pcszSrc, const char *pcszDst, unsigned fMove) { int rc = VINF_SUCCESS; PVDTESTGLOB pGlob = (PVDTESTGLOB)pvUser; bool fFound = false; /* Check if the file exists. */ PVDFILE pIt = NULL; RTListForEach(&pGlob->ListFiles, pIt, VDFILE, Node) { if (!RTStrCmp(pIt->pszName, pcszSrc)) { fFound = true; break; } } if (fFound) { char *pszNew = RTStrDup(pcszDst); if (pszNew) { RTStrFree(pIt->pszName); pIt->pszName = pszNew; } else rc = VERR_NO_MEMORY; } else rc = VERR_FILE_NOT_FOUND; return rc; } static DECLCALLBACK(int) tstVDIoFileGetFreeSpace(void *pvUser, const char *pcszFilename, int64_t *pcbFreeSpace) { AssertPtrReturn(pcbFreeSpace, VERR_INVALID_POINTER); *pcbFreeSpace = ~0ULL; /** @todo: Implement */ return VINF_SUCCESS; } static DECLCALLBACK(int) tstVDIoFileGetModificationTime(void *pvUser, const char *pcszFilename, PRTTIMESPEC pModificationTime) { AssertPtrReturn(pModificationTime, VERR_INVALID_POINTER); /** @todo: Implement */ return VINF_SUCCESS; } static DECLCALLBACK(int) tstVDIoFileGetSize(void *pvUser, void *pStorage, uint64_t *pcbSize) { PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; return VDMemDiskGetSize(pIoStorage->pFile->pMemDisk, pcbSize); } static DECLCALLBACK(int) tstVDIoFileSetSize(void *pvUser, void *pStorage, uint64_t cbSize) { PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; return VDMemDiskSetSize(pIoStorage->pFile->pMemDisk, cbSize); } static DECLCALLBACK(int) tstVDIoFileWriteSync(void *pvUser, void *pStorage, uint64_t uOffset, const void *pvBuffer, size_t cbBuffer, size_t *pcbWritten) { int rc = VINF_SUCCESS; PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; RTSGBUF SgBuf; RTSGSEG Seg; Seg.pvSeg = (void *)pvBuffer; Seg.cbSeg = cbBuffer; RTSgBufInit(&SgBuf, &Seg, 1); rc = VDMemDiskWrite(pIoStorage->pFile->pMemDisk, uOffset, cbBuffer, &SgBuf); if (RT_SUCCESS(rc) && pcbWritten) *pcbWritten = cbBuffer; return rc; } static DECLCALLBACK(int) tstVDIoFileReadSync(void *pvUser, void *pStorage, uint64_t uOffset, void *pvBuffer, size_t cbBuffer, size_t *pcbRead) { int rc = VINF_SUCCESS; PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; RTSGBUF SgBuf; RTSGSEG Seg; Seg.pvSeg = pvBuffer; Seg.cbSeg = cbBuffer; RTSgBufInit(&SgBuf, &Seg, 1); rc = VDMemDiskRead(pIoStorage->pFile->pMemDisk, uOffset, cbBuffer, &SgBuf); if (RT_SUCCESS(rc) && pcbRead) *pcbRead = cbBuffer; return rc; } static DECLCALLBACK(int) tstVDIoFileFlushSync(void *pvUser, void *pStorage) { /* nothing to do. */ return VINF_SUCCESS; } static DECLCALLBACK(int) tstVDIoFileReadAsync(void *pvUser, void *pStorage, uint64_t uOffset, PCRTSGSEG paSegments, size_t cSegments, size_t cbRead, void *pvCompletion, void **ppTask) { int rc = VINF_SUCCESS; PVDTESTGLOB pGlob = (PVDTESTGLOB)pvUser; PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; rc = VDIoBackendMemTransfer(pGlob->pIoBackend, pIoStorage->pFile->pMemDisk, VDIOTXDIR_READ, uOffset, cbRead, paSegments, cSegments, pIoStorage->pfnComplete, pvCompletion); if (RT_SUCCESS(rc)) rc = VERR_VD_ASYNC_IO_IN_PROGRESS; return rc; } static DECLCALLBACK(int) tstVDIoFileWriteAsync(void *pvUser, void *pStorage, uint64_t uOffset, PCRTSGSEG paSegments, size_t cSegments, size_t cbWrite, void *pvCompletion, void **ppTask) { int rc = VINF_SUCCESS; PVDTESTGLOB pGlob = (PVDTESTGLOB)pvUser; PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; rc = VDIoBackendMemTransfer(pGlob->pIoBackend, pIoStorage->pFile->pMemDisk, VDIOTXDIR_WRITE, uOffset, cbWrite, paSegments, cSegments, pIoStorage->pfnComplete, pvCompletion); if (RT_SUCCESS(rc)) rc = VERR_VD_ASYNC_IO_IN_PROGRESS; return rc; } static DECLCALLBACK(int) tstVDIoFileFlushAsync(void *pvUser, void *pStorage, void *pvCompletion, void **ppTask) { int rc = VINF_SUCCESS; PVDTESTGLOB pGlob = (PVDTESTGLOB)pvUser; PVDSTORAGE pIoStorage = (PVDSTORAGE)pStorage; rc = VDIoBackendMemTransfer(pGlob->pIoBackend, pIoStorage->pFile->pMemDisk, VDIOTXDIR_FLUSH, 0, 0, NULL, 0, pIoStorage->pfnComplete, pvCompletion); if (RT_SUCCESS(rc)) rc = VERR_VD_ASYNC_IO_IN_PROGRESS; return rc; } static int tstVDIoTestInit(PVDIOTEST pIoTest, PVDTESTGLOB pGlob, bool fRandomAcc, uint64_t cbIo, size_t cbBlkSize, uint64_t offStart, uint64_t offEnd, unsigned uWriteChance, PVDPATTERN pPattern) { int rc = VINF_SUCCESS; pIoTest->fRandomAccess = fRandomAcc; pIoTest->cbIo = cbIo; pIoTest->cbBlkIo = cbBlkSize; pIoTest->offStart = offStart; pIoTest->offEnd = offEnd; pIoTest->uWriteChance = uWriteChance; pIoTest->pIoRnd = pGlob->pIoRnd; pIoTest->pPattern = pPattern; if (fRandomAcc) { uint64_t cbRange = pIoTest->offEnd < pIoTest->offStart ? pIoTest->offStart - pIoTest->offEnd : pIoTest->offEnd - pIoTest->offStart; pIoTest->u.Rnd.cBlocks = cbRange / cbBlkSize + ((cbRange % cbBlkSize) ? 1 : 0); pIoTest->u.Rnd.cBlocksLeft = pIoTest->u.Rnd.cBlocks; pIoTest->u.Rnd.pbMapAccessed = (uint8_t *)RTMemAllocZ(pIoTest->u.Rnd.cBlocks / 8 + ((pIoTest->u.Rnd.cBlocks % 8) ? 1 : 0)); if (!pIoTest->u.Rnd.pbMapAccessed) rc = VERR_NO_MEMORY; } else pIoTest->u.offNext = pIoTest->offEnd < pIoTest->offStart ? pIoTest->offStart - cbBlkSize : 0; return rc; } static void tstVDIoTestDestroy(PVDIOTEST pIoTest) { if (pIoTest->fRandomAccess) RTMemFree(pIoTest->u.Rnd.pbMapAccessed); } static bool tstVDIoTestRunning(PVDIOTEST pIoTest) { return pIoTest->cbIo > 0; } static bool tstVDIoTestReqOutstanding(PVDIOREQ pIoReq) { return pIoReq->fOutstanding; } /** * Returns true with the given chance in percent. * * @returns true or false * @param iPercentage The percentage of the chance to return true. */ static bool tstVDIoTestIsTrue(PVDIOTEST pIoTest, int iPercentage) { int uRnd = VDIoRndGetU32Ex(pIoTest->pIoRnd, 0, 100); return (uRnd < iPercentage); /* This should be enough for our purpose */ } static int tstVDIoTestReqInit(PVDIOTEST pIoTest, PVDIOREQ pIoReq, void *pvUser) { int rc = VINF_SUCCESS; if (pIoTest->cbIo) { /* Read or Write? */ pIoReq->enmTxDir = tstVDIoTestIsTrue(pIoTest, pIoTest->uWriteChance) ? VDIOREQTXDIR_WRITE : VDIOREQTXDIR_READ; pIoReq->cbReq = RT_MIN(pIoTest->cbBlkIo, pIoTest->cbIo); pIoTest->cbIo -= pIoReq->cbReq; pIoReq->DataSeg.cbSeg = pIoReq->cbReq; if (pIoReq->enmTxDir == VDIOREQTXDIR_WRITE) { if (pIoTest->pPattern) rc = tstVDIoPatternGetBuffer(pIoTest->pPattern, &pIoReq->DataSeg.pvSeg, pIoReq->cbReq); else rc = VDIoRndGetBuffer(pIoTest->pIoRnd, &pIoReq->DataSeg.pvSeg, pIoReq->cbReq); AssertRC(rc); } else { /* Read */ pIoReq->DataSeg.pvSeg = pIoReq->pvBufRead; } if (RT_SUCCESS(rc)) { RTSgBufInit(&pIoReq->SgBuf, &pIoReq->DataSeg, 1); if (pIoTest->fRandomAccess) { int idx = -1; idx = ASMBitFirstClear(pIoTest->u.Rnd.pbMapAccessed, pIoTest->u.Rnd.cBlocks); /* In case this is the last request we don't need to search further. */ if (pIoTest->u.Rnd.cBlocksLeft > 1) { int idxIo; idxIo = VDIoRndGetU32Ex(pIoTest->pIoRnd, idx, pIoTest->u.Rnd.cBlocks - 1); /* * If the bit is marked free use it, otherwise search for the next free bit * and if that doesn't work use the first free bit. */ if (ASMBitTest(pIoTest->u.Rnd.pbMapAccessed, idxIo)) { idxIo = ASMBitNextClear(pIoTest->u.Rnd.pbMapAccessed, pIoTest->u.Rnd.cBlocks, idxIo); if (idxIo != -1) idx = idxIo; } else idx = idxIo; } Assert(idx != -1); pIoReq->off = idx * pIoTest->cbBlkIo; pIoTest->u.Rnd.cBlocksLeft--; if (!pIoTest->u.Rnd.cBlocksLeft) { /* New round, clear everything. */ ASMBitClearRange(pIoTest->u.Rnd.pbMapAccessed, 0, pIoTest->u.Rnd.cBlocks); pIoTest->u.Rnd.cBlocksLeft = pIoTest->u.Rnd.cBlocks; } else ASMBitSet(pIoTest->u.Rnd.pbMapAccessed, idx); } else { pIoReq->off = pIoTest->u.offNext; if (pIoTest->offEnd < pIoTest->offStart) { pIoTest->u.offNext = pIoTest->u.offNext == 0 ? pIoTest->offEnd - pIoTest->cbBlkIo : RT_MAX(pIoTest->offEnd, pIoTest->u.offNext - pIoTest->cbBlkIo); } else { pIoTest->u.offNext = pIoTest->u.offNext + pIoTest->cbBlkIo >= pIoTest->offEnd ? 0 : RT_MIN(pIoTest->offEnd, pIoTest->u.offNext + pIoTest->cbBlkIo); } } pIoReq->pvUser = pvUser; pIoReq->fOutstanding = true; } } else rc = VERR_ACCESS_DENIED; return rc; } static void tstVDIoTestReqComplete(void *pvUser1, void *pvUser2, int rcReq) { PVDIOREQ pIoReq = (PVDIOREQ)pvUser1; RTSEMEVENT hEventSem = (RTSEMEVENT)pvUser2; PVDDISK pDisk = (PVDDISK)pIoReq->pvUser; LogFlow(("Request %d completed\n", pIoReq->idx)); if (pDisk->pMemDiskVerify) { switch (pIoReq->enmTxDir) { case VDIOREQTXDIR_READ: { RTCritSectEnter(&pDisk->CritSectVerify); RTSgBufReset(&pIoReq->SgBuf); if (VDMemDiskCmp(pDisk->pMemDiskVerify, pIoReq->off, pIoReq->cbReq, &pIoReq->SgBuf)) RTPrintf("Corrupted disk at offset %llu!\n", pIoReq->off); RTCritSectLeave(&pDisk->CritSectVerify); } case VDIOREQTXDIR_WRITE: { RTCritSectEnter(&pDisk->CritSectVerify); RTSgBufReset(&pIoReq->SgBuf); int rc = VDMemDiskWrite(pDisk->pMemDiskVerify, pIoReq->off, pIoReq->cbReq, &pIoReq->SgBuf); AssertRC(rc); RTCritSectLeave(&pDisk->CritSectVerify); break; } case VDIOREQTXDIR_FLUSH: break; } } ASMAtomicXchgBool(&pIoReq->fOutstanding, false); RTSemEventSignal(hEventSem); return; } /** * Returns the disk handle by name or NULL if not found * * @returns Disk handle or NULL if the disk could not be found. * * @param pGlob Global test state. * @param pcszDisk Name of the disk to get. */ static PVDDISK tstVDIoGetDiskByName(PVDTESTGLOB pGlob, const char *pcszDisk) { PVDDISK pIt = NULL; bool fFound = false; LogFlowFunc(("pGlob=%#p pcszDisk=%s\n", pGlob, pcszDisk)); RTListForEach(&pGlob->ListDisks, pIt, VDDISK, ListNode) { if (!RTStrCmp(pIt->pszName, pcszDisk)) { fFound = true; break; } } LogFlowFunc(("return %#p\n", fFound ? pIt : NULL)); return fFound ? pIt : NULL; } /** * Returns the I/O pattern handle by name of NULL if not found. * * @returns I/O pattern handle or NULL if the pattern could not be found. * * @param pGlob Global test state. * @param pcszName Name of the pattern. */ static PVDPATTERN tstVDIoGetPatternByName(PVDTESTGLOB pGlob, const char *pcszName) { PVDPATTERN pIt = NULL; bool fFound = false; LogFlowFunc(("pGlob=%#p pcszName=%s\n", pGlob, pcszName)); RTListForEach(&pGlob->ListPatterns, pIt, VDPATTERN, ListNode) { if (!RTStrCmp(pIt->pszName, pcszName)) { fFound = true; break; } } LogFlowFunc(("return %#p\n", fFound ? pIt : NULL)); return fFound ? pIt : NULL; } /** * Creates a new pattern with the given name and an * allocated pattern buffer. * * @returns Pointer to a new pattern buffer or NULL on failure. * @param pcszName Name of the pattern. * @param cbPattern Size of the pattern buffer. */ static PVDPATTERN tstVDIoPatternCreate(const char *pcszName, size_t cbPattern) { PVDPATTERN pPattern = (PVDPATTERN)RTMemAllocZ(sizeof(VDPATTERN)); char *pszName = RTStrDup(pcszName); void *pvPattern = RTMemAllocZ(cbPattern); if (pPattern && pszName && pvPattern) { pPattern->pszName = pszName; pPattern->pvPattern = pvPattern; pPattern->cbPattern = cbPattern; } else { if (pPattern) RTMemFree(pPattern); if (pszName) RTStrFree(pszName); if (pvPattern) RTMemFree(pvPattern); pPattern = NULL; pszName = NULL; pvPattern = NULL; } return pPattern; } static int tstVDIoPatternGetBuffer(PVDPATTERN pPattern, void **ppv, size_t cb) { AssertPtrReturn(pPattern, VERR_INVALID_POINTER); AssertPtrReturn(ppv, VERR_INVALID_POINTER); AssertReturn(cb > 0, VERR_INVALID_PARAMETER); if (cb > pPattern->cbPattern) return VERR_INVALID_PARAMETER; *ppv = pPattern->pvPattern; return VINF_SUCCESS; } /** * Skips the characters until the given character is reached. * * @returns Start of the string with the given character * or NULL if the string ended before. * * @param psz The string to skip. * @param ch The character. */ static char *tstVDIoScriptSkipUntil(char *psz, char ch) { while ( *psz != '\0' && *psz != ch) psz++; return psz; } /** * Skips the spaces of the current string. * * @returns Start of the string with a non space character * or NULL if the string ended before. * * @param psz The string to skip. */ static char *tstVDIoScriptSkipSpace(char *psz) { while ( *psz != '\0' && RT_C_IS_SPACE(*psz)) psz++; return psz; } /** * Skips all characters until a space is reached of the current * string. * * @returns Start of the string with a space character * or NULL if the string ended before. * * @param psz The string to skip. */ static char *tstVDIoScriptSkipNonSpace(char *psz) { while ( *psz != '\0' && !RT_C_IS_SPACE(*psz)) psz++; return psz; } /** * Returns true if the first character of the given string * contains a character marking a line end (comment or \0 * terminator). * * @returns true if the line contains no more characters of * interest and false otherwise. * * @param psz The string to check for. */ static bool tstVDIoIsLineEnd(const char *psz) { return *psz == '\0' || *psz == '#'; } /** * Parses one argument name, value pair. * * @returns IPRT status code. * * @param pVDScriptAction Script action. * @param pcszName Argument name. * @param pcszValue Argument value. * @param pScriptArg Where to fill in the parsed * argument. * @param pfMandatory Where to store whether the argument * is mandatory. */ static int tstVDIoScriptArgumentParse(PCVDSCRIPTACTION pVDScriptAction, const char *pcszName, const char *pcszValue, PVDSCRIPTARG pScriptArg, bool *pfMandatory) { int rc = VERR_NOT_FOUND; for (unsigned i = 0; i < pVDScriptAction->cArgDescs; i++) { if (!RTStrCmp(pVDScriptAction->paArgDesc[i].pcszName, pcszName)) { rc = VINF_SUCCESS; switch (pVDScriptAction->paArgDesc[i].enmType) { case VDSCRIPTARGTYPE_BOOL: { pScriptArg->enmType = VDSCRIPTARGTYPE_BOOL; if (!RTStrICmp(pcszValue, "yes") || !RTStrICmp(pcszValue, "on")) pScriptArg->u.fFlag = true; else if (!RTStrICmp(pcszValue, "no") || !RTStrICmp(pcszValue, "off")) pScriptArg->u.fFlag = false; else { RTPrintf("Boolean argument malformed '%s'\n", pcszValue); rc = VERR_INVALID_PARAMETER; } break; } case VDSCRIPTARGTYPE_SIGNED_NUMBER: { pScriptArg->enmType = VDSCRIPTARGTYPE_SIGNED_NUMBER; AssertMsgFailed(("todo\n")); break; } case VDSCRIPTARGTYPE_STRING: { pScriptArg->enmType = VDSCRIPTARGTYPE_STRING; pScriptArg->u.pcszString = pcszValue; break; } case VDSCRIPTARGTYPE_UNSIGNED_NUMBER: { char *pszSuffix = NULL; pScriptArg->enmType = VDSCRIPTARGTYPE_UNSIGNED_NUMBER; rc = RTStrToUInt64Ex(pcszValue, &pszSuffix, 10, &pScriptArg->u.u64); if (rc == VWRN_TRAILING_CHARS) { switch (*pszSuffix) { case 'k': case 'K': { pScriptArg->u.u64 *= _1K; break; } case 'm': case 'M': { pScriptArg->u.u64 *= _1M; break; } case 'g': case 'G': { pScriptArg->u.u64 *= _1G; break; } default: { RTPrintf("Invalid size suffix '%s'\n", pszSuffix); rc = VERR_INVALID_PARAMETER; } } if (rc != VERR_INVALID_PARAMETER) rc = VINF_SUCCESS; } break; } case VDSCRIPTARGTYPE_UNSIGNED_RANGE: { char *pszSuffix = NULL; pScriptArg->enmType = VDSCRIPTARGTYPE_UNSIGNED_RANGE; rc = RTStrToUInt64Ex(pcszValue, &pszSuffix, 10, &pScriptArg->u.Range.Start); if (rc == VWRN_TRAILING_CHARS) { if (*pszSuffix != '-') { switch (*pszSuffix) { case 'k': case 'K': { pScriptArg->u.u64 *= _1K; break; } case 'm': case 'M': { pScriptArg->u.u64 *= _1M; break; } case 'g': case 'G': { pScriptArg->u.u64 *= _1G; break; } default: { RTPrintf("Invalid size suffix '%s'\n", pszSuffix); rc = VERR_INVALID_PARAMETER; } } if (RT_SUCCESS(rc)) pszSuffix++; } if (*pszSuffix == '-') { pszSuffix++; rc = RTStrToUInt64Ex(pszSuffix, &pszSuffix, 10, &pScriptArg->u.Range.End); if (rc == VWRN_TRAILING_CHARS) { switch (*pszSuffix) { case 'k': case 'K': { pScriptArg->u.Range.End *= _1K; break; } case 'm': case 'M': { pScriptArg->u.Range.End *= _1M; break; } case 'g': case 'G': { pScriptArg->u.Range.End *= _1G; break; } default: { RTPrintf("Invalid size suffix '%s'\n", pszSuffix); rc = VERR_INVALID_PARAMETER; } } } } else rc = VERR_INVALID_PARAMETER; } else rc = VERR_INVALID_PARAMETER; if (rc == VERR_INVALID_PARAMETER) RTPrintf("Invalid range format\n"); break; } default: AssertMsgFailed(("Invalid script argument type\n")); } if (RT_SUCCESS(rc)) { pScriptArg->chId = pVDScriptAction->paArgDesc[i].chId; *pfMandatory = !!(pVDScriptAction->paArgDesc[i].fFlags & VDSCRIPTARGDESC_FLAG_MANDATORY); } break; } } if (rc == VERR_NOT_FOUND) RTPrintf("Argument '%s' not found\n", pcszName); return rc; } /** * Parses the arguments of a action in the script. * * @returns IPRT status code. * * @param psz Argument string. * @param pVDScriptAction The script action to parses * arguments for. * @param paScriptArgs Where to store the arguments. * @param pcScriptArgs Where to store the actual number of * arguments parsed. */ static int tstVDIoScriptArgumentListParse(char *psz, PCVDSCRIPTACTION pVDScriptAction, PVDSCRIPTARG paScriptArgs, unsigned *pcScriptArgs) { int rc = VINF_SUCCESS; unsigned cMandatoryArgsReq = 0; unsigned cScriptArgs = 0; /* Count the number of mandatory arguments first. */ for (unsigned i = 0; i < pVDScriptAction->cArgDescs; i++) if (pVDScriptAction->paArgDesc[i].fFlags & VDSCRIPTARGDESC_FLAG_MANDATORY) cMandatoryArgsReq++; /* One argument is given in the form name=value. */ *pcScriptArgs = 0; while ( psz && !tstVDIoIsLineEnd(psz)) { const char *pcszName = psz; psz = tstVDIoScriptSkipUntil(psz, '='); if (!tstVDIoIsLineEnd(psz)) { *psz = '\0'; /* Overwrite */ psz++; const char *pcszValue = psz; psz = tstVDIoScriptSkipNonSpace(psz); if (!tstVDIoIsLineEnd(psz)) { *psz = '\0'; /* Overwrite */ psz++; psz = tstVDIoScriptSkipSpace(psz); } pcszValue = tstVDIoScriptSkipSpace((char *)pcszValue); if (*pcszValue == '\0') { RTPrintf("Value missing for argument '%s'\n", pcszName); rc = VERR_INVALID_STATE; break; } /* We have the name and value pair now. */ bool fMandatory = false; /* Shut up gcc */ rc = tstVDIoScriptArgumentParse(pVDScriptAction, pcszName, pcszValue, &paScriptArgs[cScriptArgs], &fMandatory); if (RT_SUCCESS(rc)) { if (fMandatory) cMandatoryArgsReq--; cScriptArgs++; } } else { RTPrintf("Argument in invalid form\n"); rc = VERR_INVALID_STATE; break; } } if ( RT_SUCCESS(rc) && cMandatoryArgsReq) { /* No arguments anymore but there are still mandatory arguments left. */ RTPrintf("There are %u arguments missing for script action '%s\n", pVDScriptAction->pcszAction); rc = VERR_INVALID_STATE; } if (RT_SUCCESS(rc)) *pcScriptArgs = cScriptArgs; return rc; } /** * Executes the script pointed to by the given stream. * * @returns IPRT status code. * * @param pStrm The stream handle of the script. * @param pGlob Global test data. */ static int tstVDIoScriptExecute(PRTSTREAM pStrm, PVDTESTGLOB pGlob) { int rc = VINF_SUCCESS; char abBuffer[0x1000]; /* Current assumption that a line is never longer than 4096 bytes. */ PVDSCRIPTARG paScriptArgs = NULL; unsigned cScriptArgsMax = 0; do { memset(abBuffer, 0, sizeof(abBuffer)); rc = RTStrmGetLine(pStrm, abBuffer, sizeof(abBuffer)); if (RT_SUCCESS(rc)) { const char *pcszAction = NULL; char *psz = abBuffer; /* Skip space */ psz = tstVDIoScriptSkipSpace(psz); if (!tstVDIoIsLineEnd(psz)) { PCVDSCRIPTACTION pVDScriptAction = NULL; /* Get the action name. */ pcszAction = psz; psz = tstVDIoScriptSkipNonSpace(psz); if (!tstVDIoIsLineEnd(psz)) { Assert(RT_C_IS_SPACE(*psz)); *psz++ = '\0'; } /* Find the action. */ for (unsigned i = 0; i < g_cScriptActions; i++) { if (!RTStrCmp(pcszAction, g_aScriptActions[i].pcszAction)) { pVDScriptAction = &g_aScriptActions[i]; break; } } if (pVDScriptAction) { /* Parse arguments. */ if (cScriptArgsMax < pVDScriptAction->cArgDescs) { /* Increase arguments array. */ if (paScriptArgs) RTMemFree(paScriptArgs); cScriptArgsMax = pVDScriptAction->cArgDescs; paScriptArgs = (PVDSCRIPTARG)RTMemAllocZ(cScriptArgsMax * sizeof(VDSCRIPTARG)); } if (paScriptArgs) { unsigned cScriptArgs; rc = tstVDIoScriptArgumentListParse(psz, pVDScriptAction, paScriptArgs, &cScriptArgs); if (RT_SUCCESS(rc)) { /* Execute the handler. */ rc = pVDScriptAction->pfnHandler(pGlob, paScriptArgs, cScriptArgs); } } else { RTPrintf("Out of memory while allocating argument array for script action %s\n", pcszAction); rc = VERR_NO_MEMORY; } } else { RTPrintf("Script action %s is not known\n", pcszAction); rc = VERR_NOT_FOUND; } } /* else empty line, just continue */ } } while(RT_SUCCESS(rc)); if (rc == VERR_EOF) { RTPrintf("Successfully executed I/O script\n"); rc = VINF_SUCCESS; } return rc; } /** * Executes the given I/O script. * * @returns nothing. * * @param pcszFilename The script to execute. */ static void tstVDIoScriptRun(const char *pcszFilename) { int rc = VINF_SUCCESS; PRTSTREAM pScriptStrm; /**< Stream of the script file. */ VDTESTGLOB GlobTest; /**< Global test data. */ memset(&GlobTest, 0, sizeof(VDTESTGLOB)); RTListInit(&GlobTest.ListFiles); RTListInit(&GlobTest.ListDisks); RTListInit(&GlobTest.ListPatterns); rc = RTStrmOpen(pcszFilename, "r", &pScriptStrm); if (RT_SUCCESS(rc)) { /* Init global test data. */ GlobTest.VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR); GlobTest.VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR; GlobTest.VDIErrorCallbacks.pfnError = tstVDError; GlobTest.VDIErrorCallbacks.pfnMessage = tstVDMessage; rc = VDInterfaceAdd(&GlobTest.VDIError, "tstVDIo_VDIError", VDINTERFACETYPE_ERROR, &GlobTest.VDIErrorCallbacks, NULL, &GlobTest.pInterfacesDisk); AssertRC(rc); GlobTest.VDIIoCallbacks.cbSize = sizeof(VDINTERFACEIO); GlobTest.VDIIoCallbacks.enmInterface = VDINTERFACETYPE_IO; GlobTest.VDIIoCallbacks.pfnOpen = tstVDIoFileOpen; GlobTest.VDIIoCallbacks.pfnClose = tstVDIoFileClose; GlobTest.VDIIoCallbacks.pfnDelete = tstVDIoFileDelete; GlobTest.VDIIoCallbacks.pfnMove = tstVDIoFileMove; GlobTest.VDIIoCallbacks.pfnGetFreeSpace = tstVDIoFileGetFreeSpace; GlobTest.VDIIoCallbacks.pfnGetModificationTime = tstVDIoFileGetModificationTime; GlobTest.VDIIoCallbacks.pfnGetSize = tstVDIoFileGetSize; GlobTest.VDIIoCallbacks.pfnSetSize = tstVDIoFileSetSize; GlobTest.VDIIoCallbacks.pfnWriteSync = tstVDIoFileWriteSync; GlobTest.VDIIoCallbacks.pfnReadSync = tstVDIoFileReadSync; GlobTest.VDIIoCallbacks.pfnFlushSync = tstVDIoFileFlushSync; GlobTest.VDIIoCallbacks.pfnReadAsync = tstVDIoFileReadAsync; GlobTest.VDIIoCallbacks.pfnWriteAsync = tstVDIoFileWriteAsync; GlobTest.VDIIoCallbacks.pfnFlushAsync = tstVDIoFileFlushAsync; rc = VDInterfaceAdd(&GlobTest.VDIIo, "tstVDIo_VDIIo", VDINTERFACETYPE_IO, &GlobTest.VDIIoCallbacks, &GlobTest, &GlobTest.pInterfacesImages); AssertRC(rc); /* Init I/O backend. */ rc = VDIoBackendMemCreate(&GlobTest.pIoBackend); if (RT_SUCCESS(rc)) { /* Execute the script. */ rc = tstVDIoScriptExecute(pScriptStrm, &GlobTest); if (RT_FAILURE(rc)) { RTPrintf("Executing the script stream failed rc=%Rrc\n", rc); } VDIoBackendMemDestroy(GlobTest.pIoBackend); } else RTPrintf("Creating the I/O backend failed rc=%Rrc\n"); RTStrmClose(pScriptStrm); } else RTPrintf("Opening script failed rc=%Rrc\n", rc); } /** * Shows help message. */ static void printUsage(void) { RTPrintf("Usage:\n" "--script Script to execute\n" "--replay Log to replay (not implemented yet)\n"); } static const RTGETOPTDEF g_aOptions[] = { { "--script", 's', RTGETOPT_REQ_STRING }, { "--replay", 'r', RTGETOPT_REQ_STRING }, }; int main(int argc, char *argv[]) { RTR3Init(); int rc; RTGETOPTUNION ValueUnion; RTGETOPTSTATE GetState; char c; if (argc != 3) { printUsage(); return RTEXITCODE_FAILURE; } rc = VDInit(); if (RT_FAILURE(rc)) return RTEXITCODE_FAILURE; RTGetOptInit(&GetState, argc, argv, g_aOptions, RT_ELEMENTS(g_aOptions), 1, RTGETOPTINIT_FLAGS_NO_STD_OPTS); while ( RT_SUCCESS(rc) && (c = RTGetOpt(&GetState, &ValueUnion))) { switch (c) { case 's': tstVDIoScriptRun(ValueUnion.psz); break; case 'r': RTPrintf("Replaying I/O logs is not implemented yet\n"); break; default: printUsage(); } } rc = VDShutdown(); if (RT_FAILURE(rc)) RTPrintf("tstVDIo: unloading backends failed! rc=%Rrc\n", rc); return RTEXITCODE_SUCCESS; }