/* $Id: tstGuestCtrlParseBuffer.cpp 42214 2012-07-18 18:02:58Z vboxsync $ */ /** @file * * Output stream parsing test cases. */ /* * 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 LOG_ENABLED #define LOG_GROUP LOG_GROUP_MAIN #define LOG_INSTANCE NULL #include #include "../include/GuestCtrlImplPrivate.h" using namespace com; #include #include #include #ifndef BYTE # define BYTE uint8_t #endif typedef struct VBOXGUESTCTRL_BUFFER_VALUE { char *pszValue; } VBOXGUESTCTRL_BUFFER_VALUE, *PVBOXGUESTCTRL_BUFFER_VALUE; typedef std::map< RTCString, VBOXGUESTCTRL_BUFFER_VALUE > GuestBufferMap; typedef std::map< RTCString, VBOXGUESTCTRL_BUFFER_VALUE >::iterator GuestBufferMapIter; typedef std::map< RTCString, VBOXGUESTCTRL_BUFFER_VALUE >::const_iterator GuestBufferMapIterConst; char szUnterm1[] = { 'a', 's', 'd', 'f' }; char szUnterm2[] = { 'f', 'o', 'o', '3', '=', 'b', 'a', 'r', '3' }; static struct { const char *pbData; size_t cbData; uint32_t uOffsetStart; uint32_t uOffsetAfter; uint32_t uMapElements; int iResult; } aTestBlock[] = { /* * Single object parsing. * An object is represented by one or multiple key=value pairs which are * separated by a single "\0". If this termination is missing it will be assumed * that we need to collect more data to do a successful parsing. */ /* Invalid stuff. */ { NULL, 0, 0, 0, 0, VERR_INVALID_POINTER }, { NULL, 512, 0, 0, 0, VERR_INVALID_POINTER }, { "", 0, 0, 0, 0, VERR_INVALID_PARAMETER }, { "", 0, 0, 0, 0, VERR_INVALID_PARAMETER }, { "foo=bar1", 0, 0, 0, 0, VERR_INVALID_PARAMETER }, { "foo=bar2", 0, 50, 50, 0, VERR_INVALID_PARAMETER }, /* Empty buffers. */ { "", 1, 0, 1, 0, VINF_SUCCESS }, { "\0", 1, 0, 1, 0, VINF_SUCCESS }, /* Unterminated values (missing "\0"). */ { "test1", sizeof("test1"), 0, 0, 0, VERR_MORE_DATA }, { "test2=", sizeof("test2="), 0, 0, 0, VERR_MORE_DATA }, { "test3=test3", sizeof("test3=test3"), 0, 0, 0, VERR_MORE_DATA }, { "test4=test4\0t41", sizeof("test4=test4\0t41"), 0, sizeof("test4=test4\0") - 1, 1, VERR_MORE_DATA }, { "test5=test5\0t51=t51", sizeof("test5=test5\0t51=t51"), 0, sizeof("test5=test5\0") - 1, 1, VERR_MORE_DATA }, /* Next block unterminated. */ { "t51=t51\0t52=t52\0\0t53=t53", sizeof("t51=t51\0t52=t52\0\0t53=t53"), 0, sizeof("t51=t51\0t52=t52\0") - 1, 2, VINF_SUCCESS }, { "test6=test6\0\0t61=t61", sizeof("test6=test6\0\0t61=t61"), 0, sizeof("test6=test6\0") - 1, 1, VINF_SUCCESS }, /* Good stuff. */ { "test61=\0test611=test611\0", sizeof("test61=\0test611=test611\0"), 0, sizeof("test61=\0test611=test611\0") - 1, 2, VINF_SUCCESS }, { "test7=test7\0\0", sizeof("test7=test7\0\0"), 0, sizeof("test7=test7\0") - 1, 1, VINF_SUCCESS }, { "test8=test8\0t81=t81\0\0", sizeof("test8=test8\0t81=t81\0\0"), 0, sizeof("test8=test8\0t81=t81\0") - 1, 2, VINF_SUCCESS }, /* Good stuff, but with a second block -- should be *not* taken into account since * we're only interested in parsing/handling the first object. */ { "t9=t9\0t91=t91\0\0t92=t92\0\0", sizeof("t9=t9\0t91=t91\0\0t92=t92\0\0"), 0, sizeof("t9=t9\0t91=t91\0") - 1, 2, VINF_SUCCESS }, /* Nasty stuff. */ { "äöü=fäö\0\0", sizeof("äöü=fäö\0\0"), 0, sizeof("äöü=fäö\0") - 1, 1, VINF_SUCCESS }, { "äöü=fäö\0ööö=äää", sizeof("äöü=fäö\0ööö=äää"), 0, sizeof("äöü=fäö\0") - 1, 1, VERR_MORE_DATA }, /* Some "real world" examples. */ { "hdr_id=vbt_stat\0hdr_ver=1\0name=foo.txt\0\0", sizeof("hdr_id=vbt_stat\0hdr_ver=1\0name=foo.txt\0\0"), 0, sizeof("hdr_id=vbt_stat\0hdr_ver=1\0name=foo.txt\0") - 1, 3, VINF_SUCCESS } }; static struct { const char *pbData; size_t cbData; /** Number of data blocks retrieved. These are separated by "\0\0". */ uint32_t uNumBlocks; /** Overall result when done parsing. */ int iResult; } aTestStream[] = { /* No blocks. */ { "\0\0\0\0", sizeof("\0\0\0\0"), 0, VERR_NO_DATA }, /* Good stuff. */ { "\0b1=b1\0\0", sizeof("\0b1=b1\0\0"), 1, VERR_NO_DATA }, { "b1=b1\0\0", sizeof("b1=b1\0\0"), 1, VERR_NO_DATA }, { "b1=b1\0b2=b2\0\0", sizeof("b1=b1\0b2=b2\0\0"), 1, VERR_NO_DATA }, { "b1=b1\0b2=b2\0\0\0", sizeof("b1=b1\0b2=b2\0\0\0"), 1, VERR_NO_DATA } }; int manualTest() { int rc; static struct { const char *pbData; size_t cbData; uint32_t uOffsetStart; uint32_t uOffsetAfter; uint32_t uMapElements; int iResult; } aTest[] = { { "test5=test5\0t51=t51", sizeof("test5=test5\0t51=t51"), 0, sizeof("test5=test5\0") - 1, 1, VERR_MORE_DATA }, { "\0\0test5=test5\0t51=t51", sizeof("\0\0test5=test5\0t51=t51"), 0, sizeof("\0\0test5=test5\0") - 1, 1, VERR_MORE_DATA }, }; unsigned iTest = 0; for (iTest; iTest < RT_ELEMENTS(aTest); iTest++) { RTTestIPrintf(RTTESTLVL_DEBUG, "Manual test #%d\n", iTest); GuestProcessStream stream; rc = stream.AddData((BYTE*)aTest[iTest].pbData, aTest[iTest].cbData); for (;;) { GuestProcessStreamBlock block; rc = stream.ParseBlock(block); RTTestIPrintf(RTTESTLVL_DEBUG, "\tReturned with rc=%Rrc, numItems=%ld\n", rc, block.GetCount()); if (block.GetCount()) break; } } return rc; } int main() { RTTEST hTest; int rc = RTTestInitAndCreate("tstParseBuffer", &hTest); if (rc) return rc; RTTestBanner(hTest); RTTestIPrintf(RTTESTLVL_DEBUG, "Initializing COM...\n"); HRESULT hrc = com::Initialize(); if (FAILED(hrc)) { RTTestFailed(hTest, "Failed to initialize COM (%Rhrc)!\n", hrc); return RTEXITCODE_FAILURE; } #ifdef DEBUG_andy rc = manualTest(); #endif RTTestIPrintf(RTTESTLVL_INFO, "Doing basic tests ...\n"); if (sizeof("sizecheck") != 10) RTTestFailed(hTest, "Basic size test #1 failed (%u <-> 10)", sizeof("sizecheck")); if (sizeof("off=rab") != 8) RTTestFailed(hTest, "Basic size test #2 failed (%u <-> 7)", sizeof("off=rab")); if (sizeof("off=rab\0\0") != 10) RTTestFailed(hTest, "Basic size test #3 failed (%u <-> 10)", sizeof("off=rab\0\0")); RTTestIPrintf(RTTESTLVL_INFO, "Doing line tests ...\n"); /* Don't let the assertions trigger here * -- we rely on the return values in the test(s) below. */ RTAssertSetQuiet(true); unsigned iTest = 0; for (iTest; iTest < RT_ELEMENTS(aTestBlock); iTest++) { RTTestIPrintf(RTTESTLVL_DEBUG, "=> Test #%u\n", iTest); GuestProcessStream stream; int iResult = stream.AddData((BYTE*)aTestBlock[iTest].pbData, aTestBlock[iTest].cbData); if (RT_SUCCESS(iResult)) { GuestProcessStreamBlock curBlock; iResult = stream.ParseBlock(curBlock); if (iResult != aTestBlock[iTest].iResult) { RTTestFailed(hTest, "\tReturned %Rrc, expected %Rrc\n", iResult, aTestBlock[iTest].iResult); } else if (stream.GetOffset() != aTestBlock[iTest].uOffsetAfter) { RTTestFailed(hTest, "\tOffset %u wrong, expected %u\n", stream.GetOffset(), aTestBlock[iTest].uOffsetAfter); } else if (iResult == VERR_MORE_DATA) { RTTestIPrintf(RTTESTLVL_DEBUG, "\tMore data (Offset: %u)\n", stream.GetOffset()); } if ( ( RT_SUCCESS(iResult) || iResult == VERR_MORE_DATA)) { if (curBlock.GetCount() != aTestBlock[iTest].uMapElements) { RTTestFailed(hTest, "\tMap has %u elements, expected %u\n", curBlock.GetCount(), aTestBlock[iTest].uMapElements); } } /* There is remaining data left in the buffer (which needs to be merged * with a following buffer) -- print it. */ uint32_t uOffset = stream.GetOffset(); size_t uToWrite = aTestBlock[iTest].cbData - uOffset; if (uToWrite) { const char *pszRemaining = aTestBlock[iTest].pbData; RTTestIPrintf(RTTESTLVL_DEBUG, "\tRemaining (%u):\n", uToWrite); /* How to properly get the current RTTESTLVL (aka IPRT_TEST_MAX_LEVEL) here? * Hack alert: Using RTEnvGet for now. */ if (!RTStrICmp(RTEnvGet("IPRT_TEST_MAX_LEVEL"), "debug")) RTStrmWriteEx(g_pStdOut, &aTestBlock[iTest].pbData[uOffset], uToWrite - 1, NULL); } } } RTTestIPrintf(RTTESTLVL_INFO, "Doing block tests ...\n"); iTest = 0; for (iTest; iTest < RT_ELEMENTS(aTestStream); iTest++) { RTTestIPrintf(RTTESTLVL_DEBUG, "=> Block test #%u\n", iTest); GuestProcessStream stream; int iResult = stream.AddData((BYTE*)aTestStream[iTest].pbData, aTestStream[iTest].cbData); if (RT_SUCCESS(iResult)) { uint32_t uNumBlocks = 0; uint8_t uSafeCouunter = 0; do { GuestProcessStreamBlock curBlock; iResult = stream.ParseBlock(curBlock); RTTestIPrintf(RTTESTLVL_DEBUG, "\tReturned with %Rrc\n", iResult); if (RT_SUCCESS(iResult)) { /* Only count block which have at least one pair. */ if (curBlock.GetCount()) uNumBlocks++; } if (uSafeCouunter++ > 32) break; } while (RT_SUCCESS(iResult)); if (iResult != aTestStream[iTest].iResult) { RTTestFailed(hTest, "\tReturned %Rrc, expected %Rrc\n", iResult, aTestStream[iTest].iResult); } else if (uNumBlocks != aTestStream[iTest].uNumBlocks) { RTTestFailed(hTest, "\tReturned %u blocks, expected %u\n", uNumBlocks, aTestStream[iTest].uNumBlocks); } } else RTTestFailed(hTest, "\tAdding data failed with %Rrc", iResult); } RTTestIPrintf(RTTESTLVL_DEBUG, "Shutting down COM...\n"); com::Shutdown(); /* * Summary. */ return RTTestSummaryAndDestroy(hTest); }