VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/test.cpp@ 100312

Last change on this file since 100312 was 100312, checked in by vboxsync, 19 months ago

Runtime/r3/test.cpp: Replace occurences of PAGE_SIZE/PAGE_OFFSET_MASK with RTSystemGetPageSize()/RTSystemGetPageOffsetMask() in code which is used by the linux.arm64 guest additions / validationkit, bugref:10476

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 51.4 KB
Line 
1/* $Id: test.cpp 100312 2023-06-28 10:28:59Z vboxsync $ */
2/** @file
3 * IPRT - Testcase Framework.
4 */
5
6/*
7 * Copyright (C) 2009-2023 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#include <iprt/test.h>
42
43#include <iprt/asm.h>
44#include <iprt/critsect.h>
45#include <iprt/env.h>
46#include <iprt/err.h>
47#include <iprt/file.h>
48#include <iprt/initterm.h>
49#include <iprt/mem.h>
50#include <iprt/once.h>
51#include <iprt/param.h>
52#include <iprt/pipe.h>
53#include <iprt/string.h>
54#include <iprt/stream.h>
55#include <iprt/system.h>
56
57#include "internal/magics.h"
58
59
60/*********************************************************************************************************************************
61* Structures and Typedefs *
62*********************************************************************************************************************************/
63/**
64 * Guarded memory allocation record.
65 */
66typedef struct RTTESTGUARDEDMEM
67{
68 /** Pointer to the next record. */
69 struct RTTESTGUARDEDMEM *pNext;
70 /** The address we return to the user. */
71 void *pvUser;
72 /** The base address of the allocation. */
73 void *pvAlloc;
74 /** The size of the allocation. */
75 size_t cbAlloc;
76 /** Guards. */
77 struct
78 {
79 /** The guard address. */
80 void *pv;
81 /** The guard size. */
82 size_t cb;
83 } aGuards[2];
84} RTTESTGUARDEDMEM;
85/** Pointer to an guarded memory allocation. */
86typedef RTTESTGUARDEDMEM *PRTTESTGUARDEDMEM;
87
88/**
89 * Test instance structure.
90 */
91typedef struct RTTESTINT
92{
93 /** Magic. */
94 uint32_t u32Magic;
95 /** The number of errors. */
96 volatile uint32_t cErrors;
97 /** The test name. */
98 const char *pszTest;
99 /** The length of the test name. */
100 size_t cchTest;
101 /** The size of a guard. Multiple of system page size. */
102 uint32_t cbGuard;
103 /** The verbosity level. */
104 RTTESTLVL enmMaxLevel;
105 /** The creation flags. */
106 uint32_t fFlags;
107
108
109 /** Critical section serializing output. */
110 RTCRITSECT OutputLock;
111 /** The output stream. */
112 PRTSTREAM pOutStrm;
113 /** Whether we're currently at a newline. */
114 bool fNewLine;
115
116
117 /** Critical section serializing access to the members following it. */
118 RTCRITSECT Lock;
119
120 /** The list of guarded memory allocations. */
121 PRTTESTGUARDEDMEM pGuardedMem;
122
123 /** The current sub-test. */
124 const char *pszSubTest;
125 /** The length of the sub-test name. */
126 size_t cchSubTest;
127 /** Whether the current subtest should figure as 'SKIPPED'. */
128 bool fSubTestSkipped;
129 /** Whether we've reported the sub-test result or not. */
130 bool fSubTestReported;
131 /** The start error count of the current subtest. */
132 uint32_t cSubTestAtErrors;
133
134 /** The number of sub tests. */
135 uint32_t cSubTests;
136 /** The number of sub tests that failed. */
137 uint32_t cSubTestsFailed;
138
139 /** Error context message. */
140 char *pszErrCtx;
141
142 /** Set if XML output is enabled. */
143 bool fXmlEnabled;
144 /** Set if we omit the top level test in the XML report. */
145 bool fXmlOmitTopTest;
146 /** Set if we've reported the top test (for RTTEST_C_XML_DELAY_TOP_TEST). */
147 bool fXmlTopTestDone;
148 enum {
149 kXmlPos_ValueStart,
150 kXmlPos_Value,
151 kXmlPos_ElementEnd
152 } eXmlState;
153 /** Test pipe for the XML output stream going to the server. */
154 RTPIPE hXmlPipe;
155 /** File where the XML output stream might be directed. */
156 RTFILE hXmlFile;
157 /** The number of XML elements on the stack. */
158 size_t cXmlElements;
159 /** XML element stack. */
160 const char *apszXmlElements[10];
161
162 /** Number of times assertions has been disabled and quieted. */
163 uint32_t volatile cAssertionsDisabledAndQuieted;
164 /** Saved RTAssertSetQuiet return code. */
165 bool fAssertSavedQuiet;
166 /** Saved RTAssertSetMayPanic return code. */
167 bool fAssertSavedMayPanic;
168} RTTESTINT;
169/** Pointer to a test instance. */
170typedef RTTESTINT *PRTTESTINT;
171
172
173/*********************************************************************************************************************************
174* Defined Constants And Macros *
175*********************************************************************************************************************************/
176/** Validate a test instance. */
177#define RTTEST_VALID_RETURN(pTest) \
178 do { \
179 AssertPtrReturn(pTest, VERR_INVALID_HANDLE); \
180 AssertReturn(pTest->u32Magic == RTTESTINT_MAGIC, VERR_INVALID_HANDLE); \
181 } while (0)
182
183/** Gets and validates a test instance.
184 * If the handle is nil, we will try retrieve it from the test TLS entry.
185 */
186#define RTTEST_GET_VALID_RETURN(pTest) \
187 do { \
188 if (pTest == NIL_RTTEST) \
189 pTest = (PRTTESTINT)RTTlsGet(g_iTestTls); \
190 AssertPtrReturn(pTest, VERR_INVALID_HANDLE); \
191 AssertReturn(pTest->u32Magic == RTTESTINT_MAGIC, VERR_INVALID_MAGIC); \
192 } while (0)
193
194
195/** Gets and validates a test instance.
196 * If the handle is nil, we will try retrieve it from the test TLS entry.
197 */
198#define RTTEST_GET_VALID_RETURN_RC(pTest, rc) \
199 do { \
200 if (pTest == NIL_RTTEST) \
201 pTest = (PRTTESTINT)RTTlsGet(g_iTestTls); \
202 AssertPtrReturn(pTest, (rc)); \
203 AssertReturn(pTest->u32Magic == RTTESTINT_MAGIC, (rc)); \
204 } while (0)
205
206
207/*********************************************************************************************************************************
208* Internal Functions *
209*********************************************************************************************************************************/
210static void rtTestGuardedFreeOne(PRTTESTGUARDEDMEM pMem);
211static int rtTestPrintf(PRTTESTINT pTest, const char *pszFormat, ...);
212static void rtTestXmlStart(PRTTESTINT pTest, const char *pszTest);
213static void rtTestXmlElemV(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, va_list va);
214static void rtTestXmlElem(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, ...);
215static void rtTestXmlElemStartV(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, va_list va);
216static void rtTestXmlElemStart(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, ...);
217static void rtTestXmlElemEnd(PRTTESTINT pTest, const char *pszTag);
218static void rtTestXmlEnd(PRTTESTINT pTest);
219
220
221/*********************************************************************************************************************************
222* Global Variables *
223*********************************************************************************************************************************/
224/** For serializing TLS init. */
225static RTONCE g_TestInitOnce = RTONCE_INITIALIZER;
226/** Our TLS entry. */
227static RTTLS g_iTestTls = NIL_RTTLS;
228
229
230
231/**
232 * Init TLS index once.
233 *
234 * @returns IPRT status code.
235 * @param pvUser Ignored.
236 */
237static DECLCALLBACK(int32_t) rtTestInitOnce(void *pvUser)
238{
239 NOREF(pvUser);
240 return RTTlsAllocEx(&g_iTestTls, NULL);
241}
242
243
244RTR3DECL(int) RTTestCreateEx(const char *pszTest, uint32_t fFlags, RTTESTLVL enmMaxLevel,
245 RTHCINTPTR iNativeTestPipe, const char *pszXmlFile, PRTTEST phTest)
246{
247 AssertReturn(!(fFlags & ~RTTEST_C_VALID_MASK), VERR_INVALID_PARAMETER);
248 AssertPtrNull(phTest);
249 AssertPtrNull(pszXmlFile);
250 /* RTTESTLVL_INVALID is valid! */
251 AssertReturn(enmMaxLevel >= RTTESTLVL_INVALID && enmMaxLevel < RTTESTLVL_END, VERR_INVALID_PARAMETER);
252
253 /*
254 * Global init.
255 */
256 int rc = RTOnce(&g_TestInitOnce, rtTestInitOnce, NULL);
257 if (RT_FAILURE(rc))
258 return rc;
259
260 /*
261 * Create the instance.
262 */
263 PRTTESTINT pTest = (PRTTESTINT)RTMemAllocZ(sizeof(*pTest));
264 if (!pTest)
265 return VERR_NO_MEMORY;
266 pTest->u32Magic = RTTESTINT_MAGIC;
267 pTest->pszTest = RTStrDup(pszTest);
268 pTest->cchTest = strlen(pszTest);
269 pTest->cbGuard = RTSystemGetPageSize() * 7;
270 pTest->enmMaxLevel = enmMaxLevel == RTTESTLVL_INVALID ? RTTESTLVL_INFO : enmMaxLevel;
271 pTest->fFlags = fFlags;
272
273 pTest->pOutStrm = g_pStdOut;
274 pTest->fNewLine = true;
275
276 pTest->pGuardedMem = NULL;
277
278 pTest->pszSubTest = NULL;
279 pTest->cchSubTest = 0;
280 pTest->fSubTestSkipped = false;
281 pTest->fSubTestReported = true;
282 pTest->cSubTestAtErrors = 0;
283 pTest->cSubTests = 0;
284 pTest->cSubTestsFailed = 0;
285
286 pTest->fXmlEnabled = false;
287 pTest->fXmlTopTestDone = false;
288 pTest->eXmlState = RTTESTINT::kXmlPos_ElementEnd;
289 pTest->hXmlPipe = NIL_RTPIPE;
290 pTest->hXmlFile = NIL_RTFILE;
291 pTest->cXmlElements = 0;
292 pTest->cAssertionsDisabledAndQuieted = 0;
293 pTest->fAssertSavedMayPanic = true;
294 pTest->fAssertSavedQuiet = false;
295
296 rc = RTCritSectInit(&pTest->Lock);
297 if (RT_SUCCESS(rc))
298 {
299 rc = RTCritSectInit(&pTest->OutputLock);
300 if (RT_SUCCESS(rc))
301 {
302 /*
303 * Associate it with our TLS entry unless there is already
304 * an instance there.
305 */
306 if ( !(fFlags & RTTEST_C_NO_TLS)
307 && !RTTlsGet(g_iTestTls))
308 rc = RTTlsSet(g_iTestTls, pTest);
309 if (RT_SUCCESS(rc))
310 {
311 /*
312 * Output level override?
313 */
314 char szEnvVal[RTPATH_MAX];
315 if ((fFlags & RTTEST_C_USE_ENV) && enmMaxLevel == RTTESTLVL_INVALID)
316 {
317 rc = RTEnvGetEx(RTENV_DEFAULT, "IPRT_TEST_MAX_LEVEL", szEnvVal, sizeof(szEnvVal), NULL);
318 if (RT_SUCCESS(rc))
319 {
320 char *pszMaxLevel = RTStrStrip(szEnvVal);
321 if (!strcmp(pszMaxLevel, "all"))
322 pTest->enmMaxLevel = RTTESTLVL_DEBUG;
323 if (!strcmp(pszMaxLevel, "quiet"))
324 pTest->enmMaxLevel = RTTESTLVL_FAILURE;
325 else if (!strcmp(pszMaxLevel, "debug"))
326 pTest->enmMaxLevel = RTTESTLVL_DEBUG;
327 else if (!strcmp(pszMaxLevel, "info"))
328 pTest->enmMaxLevel = RTTESTLVL_INFO;
329 else if (!strcmp(pszMaxLevel, "sub_test"))
330 pTest->enmMaxLevel = RTTESTLVL_SUB_TEST;
331 else if (!strcmp(pszMaxLevel, "failure"))
332 pTest->enmMaxLevel = RTTESTLVL_FAILURE;
333 }
334 else if (rc != VERR_ENV_VAR_NOT_FOUND)
335 RTStrmPrintf(g_pStdErr, "%s: test pipe error: RTEnvGetEx(IPRT_TEST_MAX_LEVEL) -> %Rrc\n", pszTest, rc);
336 }
337
338 /*
339 * Any test driver we are connected or should connect to?
340 */
341 if (!(fFlags & RTTEST_C_NO_XML_REPORTING_PIPE))
342 {
343 if ( (fFlags & RTTEST_C_USE_ENV)
344 && iNativeTestPipe == -1)
345 {
346 rc = RTEnvGetEx(RTENV_DEFAULT, "IPRT_TEST_PIPE", szEnvVal, sizeof(szEnvVal), NULL);
347 if (RT_SUCCESS(rc))
348 {
349#if ARCH_BITS == 64
350 rc = RTStrToInt64Full(szEnvVal, 0, &iNativeTestPipe);
351#else
352 rc = RTStrToInt32Full(szEnvVal, 0, &iNativeTestPipe);
353#endif
354 if (RT_FAILURE(rc))
355 {
356 RTStrmPrintf(g_pStdErr, "%s: test pipe error: RTStrToInt32Full(\"%s\") -> %Rrc\n",
357 pszTest, szEnvVal, rc);
358 iNativeTestPipe = -1;
359 }
360 }
361 else if (rc != VERR_ENV_VAR_NOT_FOUND)
362 RTStrmPrintf(g_pStdErr, "%s: test pipe error: RTEnvGetEx(IPRT_TEST_PIPE) -> %Rrc\n", pszTest, rc);
363 }
364 if (iNativeTestPipe != -1)
365 {
366 rc = RTPipeFromNative(&pTest->hXmlPipe, iNativeTestPipe, RTPIPE_N_WRITE);
367 if (RT_SUCCESS(rc))
368 pTest->fXmlEnabled = true;
369 else
370 {
371 RTStrmPrintf(g_pStdErr, "%s: test pipe error: RTPipeFromNative(,%p,WRITE) -> %Rrc\n",
372 pszTest, iNativeTestPipe, rc);
373 pTest->hXmlPipe = NIL_RTPIPE;
374 }
375 }
376 }
377
378 /*
379 * Any test file we should write the test report to?
380 */
381 if (!(fFlags & RTTEST_C_NO_XML_REPORTING_FILE))
382 {
383 if ((fFlags & RTTEST_C_USE_ENV) && pszXmlFile == NULL)
384 {
385 rc = RTEnvGetEx(RTENV_DEFAULT, "IPRT_TEST_FILE", szEnvVal, sizeof(szEnvVal), NULL);
386 if (RT_SUCCESS(rc))
387 pszXmlFile = szEnvVal;
388 else if (rc != VERR_ENV_VAR_NOT_FOUND)
389 RTStrmPrintf(g_pStdErr, "%s: test file error: RTEnvGetEx(IPRT_TEST_MAX_LEVEL) -> %Rrc\n", pszTest, rc);
390 }
391 if (pszXmlFile && *pszXmlFile)
392 {
393 rc = RTFileOpen(&pTest->hXmlFile, pszXmlFile,
394 RTFILE_O_WRITE | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN_CREATE | RTFILE_O_TRUNCATE);
395 if (RT_SUCCESS(rc))
396 pTest->fXmlEnabled = true;
397 else
398 {
399 RTStrmPrintf(g_pStdErr, "%s: test file error: RTFileOpen(,\"%s\",) -> %Rrc\n",
400 pszTest, pszXmlFile, rc);
401 pTest->hXmlFile = NIL_RTFILE;
402 }
403 }
404 }
405
406 /*
407 * What do we report in the XML stream/file.?
408 */
409 pTest->fXmlOmitTopTest = (fFlags & RTTEST_C_XML_OMIT_TOP_TEST)
410 || ( (fFlags & RTTEST_C_USE_ENV)
411 && RTEnvExistEx(RTENV_DEFAULT, "IPRT_TEST_OMIT_TOP_TEST"));
412
413 /*
414 * Tell the test driver that we're up to.
415 */
416 rtTestXmlStart(pTest, pszTest);
417
418 *phTest = pTest;
419 return VINF_SUCCESS;
420 }
421
422 /* bail out. */
423 RTCritSectDelete(&pTest->OutputLock);
424 }
425 RTCritSectDelete(&pTest->Lock);
426 }
427 pTest->u32Magic = 0;
428 RTStrFree((char *)pTest->pszTest);
429 RTMemFree(pTest);
430 return rc;
431}
432
433
434RTR3DECL(int) RTTestCreate(const char *pszTest, PRTTEST phTest)
435{
436 return RTTestCreateEx(pszTest, RTTEST_C_USE_ENV, RTTESTLVL_INVALID, -1 /*iNativeTestPipe*/, NULL /*pszXmlFile*/, phTest);
437}
438
439
440RTR3DECL(int) RTTestCreateChild(const char *pszTest, PRTTEST phTest)
441{
442 return RTTestCreateEx(pszTest, RTTEST_C_USE_ENV | RTTEST_C_NO_XML_REPORTING,
443 RTTESTLVL_INVALID, -1 /*iNativeTestPipe*/, NULL /*pszXmlFile*/, phTest);
444}
445
446
447RTR3DECL(RTEXITCODE) RTTestInitAndCreate(const char *pszTest, PRTTEST phTest)
448{
449 int rc = RTR3InitExeNoArguments(0);
450 if (RT_FAILURE(rc))
451 {
452 RTStrmPrintf(g_pStdErr, "%s: fatal error: RTR3InitExeNoArguments failed with rc=%Rrc\n", pszTest, rc);
453 return RTEXITCODE_INIT;
454 }
455
456 rc = RTTestCreate(pszTest, phTest);
457 if (RT_FAILURE(rc))
458 {
459 RTStrmPrintf(g_pStdErr, "%s: fatal error: RTTestCreate failed with rc=%Rrc\n", pszTest, rc);
460 return RTEXITCODE_INIT;
461 }
462 return RTEXITCODE_SUCCESS;
463}
464
465
466RTR3DECL(RTEXITCODE) RTTestInitExAndCreate(int cArgs, char ***ppapszArgs, uint32_t fRtInit, const char *pszTest, PRTTEST phTest)
467{
468 int rc;
469 if (cArgs <= 0 && ppapszArgs == NULL)
470 rc = RTR3InitExeNoArguments(fRtInit);
471 else
472 rc = RTR3InitExe(cArgs, ppapszArgs, fRtInit);
473 if (RT_FAILURE(rc))
474 {
475 RTStrmPrintf(g_pStdErr, "%s: fatal error: RTR3InitExe(,,%#x) failed with rc=%Rrc\n", pszTest, fRtInit, rc);
476 return RTEXITCODE_INIT;
477 }
478
479 rc = RTTestCreate(pszTest, phTest);
480 if (RT_FAILURE(rc))
481 {
482 RTStrmPrintf(g_pStdErr, "%s: fatal error: RTTestCreate failed with rc=%Rrc\n", pszTest, rc);
483 return RTEXITCODE_INIT;
484 }
485 return RTEXITCODE_SUCCESS;
486}
487
488
489RTR3DECL(int) RTTestDestroy(RTTEST hTest)
490{
491 /*
492 * Validate
493 */
494 if (hTest == NIL_RTTEST)
495 return VINF_SUCCESS;
496 RTTESTINT *pTest = hTest;
497 RTTEST_VALID_RETURN(pTest);
498
499 /*
500 * Make sure we end with a new line and have finished up the XML.
501 */
502 if (!pTest->fNewLine)
503 rtTestPrintf(pTest, "\n");
504 rtTestXmlEnd(pTest);
505
506 /*
507 * Clean up.
508 */
509 if ((RTTESTINT *)RTTlsGet(g_iTestTls) == pTest)
510 RTTlsSet(g_iTestTls, NULL);
511
512 ASMAtomicWriteU32(&pTest->u32Magic, ~RTTESTINT_MAGIC);
513 RTCritSectDelete(&pTest->Lock);
514 RTCritSectDelete(&pTest->OutputLock);
515
516 /* free guarded memory. */
517 PRTTESTGUARDEDMEM pMem = pTest->pGuardedMem;
518 pTest->pGuardedMem = NULL;
519 while (pMem)
520 {
521 PRTTESTGUARDEDMEM pFree = pMem;
522 pMem = pMem->pNext;
523 rtTestGuardedFreeOne(pFree);
524 }
525
526 RTStrFree((char *)pTest->pszSubTest);
527 pTest->pszSubTest = NULL;
528 RTStrFree((char *)pTest->pszTest);
529 pTest->pszTest = NULL;
530 RTStrFree(pTest->pszErrCtx);
531 pTest->pszErrCtx = NULL;
532 RTMemFree(pTest);
533 return VINF_SUCCESS;
534}
535
536
537RTR3DECL(int) RTTestSetDefault(RTTEST hNewDefaultTest, PRTTEST phOldTest)
538{
539 if (phOldTest)
540 *phOldTest = (RTTEST)RTTlsGet(g_iTestTls);
541 return RTTlsSet(g_iTestTls, hNewDefaultTest);
542}
543
544
545RTR3DECL(int) RTTestChangeName(RTTEST hTest, const char *pszName)
546{
547 PRTTESTINT pTest = hTest;
548 RTTEST_GET_VALID_RETURN(pTest);
549 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
550 AssertReturn(*pszName, VERR_INVALID_PARAMETER);
551
552 size_t cchName = strlen(pszName);
553 AssertReturn(cchName < 128, VERR_INVALID_PARAMETER);
554 char *pszDupName = RTStrDup(pszName);
555 if (!pszDupName)
556 return VERR_NO_STR_MEMORY;
557
558 RTCritSectEnter(&pTest->Lock);
559 RTCritSectEnter(&pTest->OutputLock);
560
561 char *pszOldName = (char *)pTest->pszTest;
562 pTest->pszTest = pszDupName;
563 pTest->cchTest = cchName;
564
565 RTCritSectLeave(&pTest->OutputLock);
566 RTCritSectLeave(&pTest->Lock);
567
568 RTStrFree(pszOldName);
569 return VINF_SUCCESS;
570}
571
572
573RTR3DECL(int) RTTestGuardedAlloc(RTTEST hTest, size_t cb, uint32_t cbAlign, bool fHead, void **ppvUser)
574{
575 PRTTESTINT pTest = hTest;
576 RTTEST_GET_VALID_RETURN(pTest);
577 if (cbAlign == 0)
578 cbAlign = 1;
579 uint32_t const cbPage = RTSystemGetPageSize();
580 AssertReturn(cbAlign <= cbPage, VERR_INVALID_PARAMETER);
581 AssertReturn(cbAlign == (UINT32_C(1) << (ASMBitFirstSetU32(cbAlign) - 1)), VERR_INVALID_PARAMETER);
582
583 /*
584 * Allocate the record and block and initialize them.
585 */
586 int rc = VERR_NO_MEMORY;
587 PRTTESTGUARDEDMEM pMem = (PRTTESTGUARDEDMEM)RTMemAlloc(sizeof(*pMem));
588 if (RT_LIKELY(pMem))
589 {
590 size_t const cbAligned = RT_ALIGN_Z(cb, cbPage);
591 pMem->aGuards[0].cb = pMem->aGuards[1].cb = pTest->cbGuard;
592 pMem->cbAlloc = pMem->aGuards[0].cb + pMem->aGuards[1].cb + cbAligned;
593 pMem->pvAlloc = RTMemPageAlloc(pMem->cbAlloc);
594 if (pMem->pvAlloc)
595 {
596 pMem->aGuards[0].pv = pMem->pvAlloc;
597 pMem->pvUser = (uint8_t *)pMem->pvAlloc + pMem->aGuards[0].cb;
598 pMem->aGuards[1].pv = (uint8_t *)pMem->pvUser + cbAligned;
599 if (!fHead)
600 {
601 size_t off = cb & RTSystemGetPageOffsetMask();
602 if (off)
603 {
604 off = cbPage - RT_ALIGN_Z(off, cbAlign);
605 pMem->pvUser = (uint8_t *)pMem->pvUser + off;
606 }
607 }
608
609 /*
610 * Set up the guards and link the record.
611 */
612 ASMMemFill32(pMem->aGuards[0].pv, pMem->aGuards[0].cb, 0xdeadbeef);
613 ASMMemFill32(pMem->aGuards[1].pv, pMem->aGuards[1].cb, 0xdeadbeef);
614 rc = RTMemProtect(pMem->aGuards[0].pv, pMem->aGuards[0].cb, RTMEM_PROT_NONE);
615 if (RT_SUCCESS(rc))
616 {
617 rc = RTMemProtect(pMem->aGuards[1].pv, pMem->aGuards[1].cb, RTMEM_PROT_NONE);
618 if (RT_SUCCESS(rc))
619 {
620 *ppvUser = pMem->pvUser;
621
622 RTCritSectEnter(&pTest->Lock);
623 pMem->pNext = pTest->pGuardedMem;
624 pTest->pGuardedMem = pMem;
625 RTCritSectLeave(&pTest->Lock);
626
627 return VINF_SUCCESS;
628 }
629
630 RTMemProtect(pMem->aGuards[0].pv, pMem->aGuards[0].cb, RTMEM_PROT_WRITE | RTMEM_PROT_READ);
631 }
632
633 RTMemPageFree(pMem->pvAlloc, pMem->cbAlloc);
634 }
635 RTMemFree(pMem);
636 }
637 return rc;
638}
639
640
641RTR3DECL(void *) RTTestGuardedAllocTail(RTTEST hTest, size_t cb)
642{
643 void *pvUser;
644 int rc = RTTestGuardedAlloc(hTest, cb, 1 /* cbAlign */, false /* fHead */, &pvUser);
645 if (RT_SUCCESS(rc))
646 return pvUser;
647 return NULL;
648}
649
650
651RTR3DECL(void *) RTTestGuardedAllocHead(RTTEST hTest, size_t cb)
652{
653 void *pvUser;
654 int rc = RTTestGuardedAlloc(hTest, cb, 1 /* cbAlign */, true /* fHead */, &pvUser);
655 if (RT_SUCCESS(rc))
656 return pvUser;
657 return NULL;
658}
659
660
661/**
662 * Frees one block of guarded memory.
663 *
664 * The caller is responsible for unlinking it.
665 *
666 * @param pMem The memory record.
667 */
668static void rtTestGuardedFreeOne(PRTTESTGUARDEDMEM pMem)
669{
670 int rc;
671 rc = RTMemProtect(pMem->aGuards[0].pv, pMem->aGuards[0].cb, RTMEM_PROT_WRITE | RTMEM_PROT_READ); AssertRC(rc);
672 rc = RTMemProtect(pMem->aGuards[1].pv, pMem->aGuards[1].cb, RTMEM_PROT_WRITE | RTMEM_PROT_READ); AssertRC(rc);
673 RTMemPageFree(pMem->pvAlloc, pMem->cbAlloc);
674 RTMemFree(pMem);
675}
676
677
678RTR3DECL(int) RTTestGuardedFree(RTTEST hTest, void *pv)
679{
680 PRTTESTINT pTest = hTest;
681 RTTEST_GET_VALID_RETURN(pTest);
682 if (!pv)
683 return VINF_SUCCESS;
684
685 /*
686 * Find it.
687 */
688 int rc = VERR_INVALID_POINTER;
689 PRTTESTGUARDEDMEM pPrev = NULL;
690
691 RTCritSectEnter(&pTest->Lock);
692 for (PRTTESTGUARDEDMEM pMem = pTest->pGuardedMem; pMem; pMem = pMem->pNext)
693 {
694 if (pMem->pvUser == pv)
695 {
696 if (pPrev)
697 pPrev->pNext = pMem->pNext;
698 else
699 pTest->pGuardedMem = pMem->pNext;
700 rtTestGuardedFreeOne(pMem);
701 rc = VINF_SUCCESS;
702 break;
703 }
704 pPrev = pMem;
705 }
706 RTCritSectLeave(&pTest->Lock);
707
708 return rc;
709}
710
711
712/**
713 * Outputs the formatted XML.
714 *
715 * @param pTest The test instance.
716 * @param pszFormat The format string.
717 * @param va The format arguments.
718 */
719static void rtTestXmlOutputV(PRTTESTINT pTest, const char *pszFormat, va_list va)
720{
721 if (pTest->fXmlEnabled)
722 {
723 char *pszStr;
724 ssize_t cchStr = RTStrAPrintfV(&pszStr, pszFormat, va);
725 if (pszStr)
726 {
727 if (pTest->hXmlPipe != NIL_RTPIPE)
728 RTPipeWriteBlocking(pTest->hXmlPipe, pszStr, cchStr, NULL);
729 if (pTest->hXmlFile != NIL_RTFILE)
730 RTFileWrite(pTest->hXmlFile, pszStr, cchStr, NULL);
731 RTStrFree(pszStr);
732 }
733 }
734}
735
736
737/**
738 * Outputs the formatted XML.
739 *
740 * @param pTest The test instance.
741 * @param pszFormat The format string.
742 * @param ... The format arguments.
743 */
744static void rtTestXmlOutput(PRTTESTINT pTest, const char *pszFormat, ...)
745{
746 va_list va;
747 va_start(va, pszFormat);
748 rtTestXmlOutputV(pTest, pszFormat, va);
749 va_end(va);
750}
751
752
753/**
754 * Starts the XML stream.
755 *
756 * @param pTest The test instance.
757 * @param pszTest The test name.
758 */
759static void rtTestXmlStart(PRTTESTINT pTest, const char *pszTest)
760{
761 pTest->cXmlElements = 0;
762 if (pTest->fXmlEnabled)
763 {
764 rtTestXmlOutput(pTest, "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n");
765 pTest->eXmlState = RTTESTINT::kXmlPos_ElementEnd;
766 pTest->fXmlTopTestDone = !(pTest->fFlags & RTTEST_C_XML_DELAY_TOP_TEST) || pTest->fXmlOmitTopTest;
767 if (pTest->fXmlTopTestDone && !pTest->fXmlOmitTopTest)
768 rtTestXmlElemStart(pTest, "Test", "name=%RMas", pszTest);
769 }
770}
771
772
773/**
774 * Emit an XML element that doesn't have any value and instead ends immediately.
775 *
776 * The caller must own the instance lock.
777 *
778 * @param pTest The test instance.
779 * @param pszTag The element tag.
780 * @param pszAttrFmt The element attributes as a format string. Use
781 * NULL if none.
782 * @param va Format string arguments.
783 */
784static void rtTestXmlElemV(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, va_list va)
785{
786 if (pTest->fXmlEnabled)
787 {
788 RTTIMESPEC TimeSpec;
789 RTTIME Time;
790 char szTS[80];
791 RTTimeToString(RTTimeExplode(&Time, RTTimeNow(&TimeSpec)), szTS, sizeof(szTS));
792
793 if (pTest->eXmlState != RTTESTINT::kXmlPos_ElementEnd)
794 rtTestXmlOutput(pTest, "\n");
795
796 if (!pszAttrFmt || !*pszAttrFmt)
797 rtTestXmlOutput(pTest, "%*s<%s timestamp=%RMas/>\n",
798 pTest->cXmlElements * 2, "", pszTag, szTS);
799 else
800 {
801 va_list va2;
802 va_copy(va2, va);
803 rtTestXmlOutput(pTest, "%*s<%s timestamp=%RMas %N/>\n",
804 pTest->cXmlElements * 2, "", pszTag, szTS, pszAttrFmt, &va2);
805 va_end(va2);
806 }
807 pTest->eXmlState = RTTESTINT::kXmlPos_ElementEnd;
808 }
809}
810
811
812/**
813 * Wrapper around rtTestXmlElemV.
814 */
815static void rtTestXmlElem(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, ...)
816{
817 va_list va;
818 va_start(va, pszAttrFmt);
819 rtTestXmlElemV(pTest, pszTag, pszAttrFmt, va);
820 va_end(va);
821}
822
823
824/**
825 * Starts a new XML element.
826 *
827 * The caller must own the instance lock.
828 *
829 * @param pTest The test instance.
830 * @param pszTag The element tag.
831 * @param pszAttrFmt The element attributes as a format string. Use
832 * NULL if none.
833 * @param va Format string arguments.
834 */
835static void rtTestXmlElemStartV(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, va_list va)
836{
837 /* Push it onto the stack. */
838 size_t i = pTest->cXmlElements;
839 AssertReturnVoid(i < RT_ELEMENTS(pTest->apszXmlElements));
840 pTest->apszXmlElements[i] = pszTag;
841 pTest->cXmlElements = i + 1;
842
843 if (pTest->fXmlEnabled)
844 {
845 RTTIMESPEC TimeSpec;
846 RTTIME Time;
847 char szTS[80];
848 RTTimeToString(RTTimeExplode(&Time, RTTimeNow(&TimeSpec)), szTS, sizeof(szTS));
849
850 if (pTest->eXmlState != RTTESTINT::kXmlPos_ElementEnd)
851 rtTestXmlOutput(pTest, "\n");
852
853 if (!pszAttrFmt || !*pszAttrFmt)
854 rtTestXmlOutput(pTest, "%*s<%s timestamp=%RMas>",
855 i * 2, "", pszTag, szTS);
856 else
857 {
858 va_list va2;
859 va_copy(va2, va);
860 rtTestXmlOutput(pTest, "%*s<%s timestamp=%RMas %N>",
861 i * 2, "", pszTag, szTS, pszAttrFmt, &va2);
862 va_end(va2);
863 }
864 pTest->eXmlState = RTTESTINT::kXmlPos_ValueStart;
865 }
866}
867
868
869/**
870 * Wrapper around rtTestXmlElemStartV.
871 */
872static void rtTestXmlElemStart(PRTTESTINT pTest, const char *pszTag, const char *pszAttrFmt, ...)
873{
874 va_list va;
875 va_start(va, pszAttrFmt);
876 rtTestXmlElemStartV(pTest, pszTag, pszAttrFmt, va);
877 va_end(va);
878}
879
880
881/**
882 * Ends the current element.
883 *
884 * The caller must own the instance lock.
885 *
886 * @param pTest The test instance.
887 * @param pszTag The tag we're ending (chiefly for sanity
888 * checking).
889 */
890static void rtTestXmlElemEnd(PRTTESTINT pTest, const char *pszTag)
891{
892 /* pop the element */
893 size_t i = pTest->cXmlElements;
894 AssertReturnVoid(i > 0);
895 i--;
896 AssertReturnVoid(!strcmp(pszTag, pTest->apszXmlElements[i]));
897 pTest->cXmlElements = i;
898
899 /* Do the closing. */
900 if (pTest->fXmlEnabled)
901 {
902 if (pTest->eXmlState == RTTESTINT::kXmlPos_ValueStart)
903 rtTestXmlOutput(pTest, "\n%*s</%s>\n", i * 2, "", pszTag);
904 else if (pTest->eXmlState == RTTESTINT::kXmlPos_ElementEnd)
905 rtTestXmlOutput(pTest, "%*s</%s>\n", i * 2, "", pszTag);
906 else
907 rtTestXmlOutput(pTest, "</%s>\n", pszTag);
908 pTest->eXmlState = RTTESTINT::kXmlPos_ElementEnd;
909 }
910}
911
912
913/**
914 * Ends the XML stream, closing all open elements.
915 *
916 * The caller must own the instance lock.
917 *
918 * @param pTest The test instance.
919 */
920static void rtTestXmlEnd(PRTTESTINT pTest)
921{
922 if (pTest->fXmlEnabled)
923 {
924 /*
925 * Close all the elements and add the final TestEnd one to get a
926 * final timestamp and some certainty that the XML is valid.
927 */
928 size_t i = pTest->cXmlElements;
929 AssertReturnVoid(i > 0 || pTest->fXmlOmitTopTest || !pTest->fXmlTopTestDone);
930 while (i-- > 1)
931 {
932 const char *pszTag = pTest->apszXmlElements[pTest->cXmlElements];
933 if (pTest->eXmlState == RTTESTINT::kXmlPos_ValueStart)
934 rtTestXmlOutput(pTest, "\n%*s</%s>\n", i * 2, "", pszTag);
935 else if (pTest->eXmlState == RTTESTINT::kXmlPos_ElementEnd)
936 rtTestXmlOutput(pTest, "%*s</%s>\n", i * 2, "", pszTag);
937 else
938 rtTestXmlOutput(pTest, "</%s>\n", pszTag);
939 pTest->eXmlState = RTTESTINT::kXmlPos_ElementEnd;
940 }
941
942 if (!pTest->fXmlOmitTopTest && pTest->fXmlTopTestDone)
943 {
944 rtTestXmlElem(pTest, "End", "SubTests=\"%u\" SubTestsFailed=\"%u\" errors=\"%u\"",
945 pTest->cSubTests, pTest->cSubTestsFailed, pTest->cErrors);
946 rtTestXmlOutput(pTest, "</Test>\n");
947 }
948
949 /*
950 * Close the XML outputs.
951 */
952 if (pTest->hXmlPipe != NIL_RTPIPE)
953 {
954 RTPipeClose(pTest->hXmlPipe);
955 pTest->hXmlPipe = NIL_RTPIPE;
956 }
957 if (pTest->hXmlFile != NIL_RTFILE)
958 {
959 RTFileClose(pTest->hXmlFile);
960 pTest->hXmlFile = NIL_RTFILE;
961 }
962 pTest->fXmlEnabled = false;
963 pTest->eXmlState = RTTESTINT::kXmlPos_ElementEnd;
964 }
965 pTest->cXmlElements = 0;
966}
967
968/**
969 * Output callback.
970 *
971 * @returns number of bytes written.
972 * @param pvArg User argument.
973 * @param pachChars Pointer to an array of utf-8 characters.
974 * @param cbChars Number of bytes in the character array pointed to by pachChars.
975 */
976static DECLCALLBACK(size_t) rtTestPrintfOutput(void *pvArg, const char *pachChars, size_t cbChars)
977{
978 size_t cch = 0;
979 PRTTESTINT pTest = (PRTTESTINT)pvArg;
980 if (cbChars)
981 {
982 do
983 {
984 /* insert prefix if at a newline. */
985 if (pTest->fNewLine)
986 {
987 RTStrmWrite(pTest->pOutStrm, pTest->pszTest, pTest->cchTest);
988 RTStrmWrite(pTest->pOutStrm, ": ", 2);
989 cch += 2 + pTest->cchTest;
990 }
991
992 /* look for newline and write the stuff. */
993 const char *pchEnd = (const char *)memchr(pachChars, '\n', cbChars);
994 if (!pchEnd)
995 {
996 pTest->fNewLine = false;
997 RTStrmWrite(pTest->pOutStrm, pachChars, cbChars);
998 cch += cbChars;
999 break;
1000 }
1001
1002 pTest->fNewLine = true;
1003 size_t const cchPart = pchEnd - pachChars + 1;
1004 RTStrmWrite(pTest->pOutStrm, pachChars, cchPart);
1005 cch += cchPart;
1006 pachChars += cchPart;
1007 cbChars -= cchPart;
1008 } while (cbChars);
1009 }
1010 else
1011 RTStrmFlush(pTest->pOutStrm);
1012 return cch;
1013}
1014
1015
1016/**
1017 * Internal output worker.
1018 *
1019 * Caller takes the lock.
1020 *
1021 * @returns Number of chars printed.
1022 * @param pTest The test instance.
1023 * @param pszFormat The message.
1024 * @param va The arguments.
1025 */
1026static int rtTestPrintfV(PRTTESTINT pTest, const char *pszFormat, va_list va)
1027{
1028 return (int)RTStrFormatV(rtTestPrintfOutput, pTest, NULL, NULL, pszFormat, va);
1029}
1030
1031
1032/**
1033 * Internal output worker.
1034 *
1035 * Caller takes the lock.
1036 *
1037 * @returns Number of chars printed.
1038 * @param pTest The test instance.
1039 * @param pszFormat The message.
1040 * @param ... The arguments.
1041 */
1042static int rtTestPrintf(PRTTESTINT pTest, const char *pszFormat, ...)
1043{
1044 va_list va;
1045
1046 va_start(va, pszFormat);
1047 int cch = rtTestPrintfV(pTest, pszFormat, va);
1048 va_end(va);
1049
1050 return cch;
1051}
1052
1053
1054RTR3DECL(int) RTTestPrintfNlV(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, va_list va)
1055{
1056 PRTTESTINT pTest = hTest;
1057 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
1058
1059 RTCritSectEnter(&pTest->OutputLock);
1060
1061 int cch = 0;
1062 if (enmLevel <= pTest->enmMaxLevel)
1063 {
1064 if (!pTest->fNewLine)
1065 cch += rtTestPrintf(pTest, "\n");
1066 cch += rtTestPrintfV(pTest, pszFormat, va);
1067 }
1068
1069 RTCritSectLeave(&pTest->OutputLock);
1070
1071 return cch;
1072}
1073
1074
1075RTR3DECL(int) RTTestPrintfNl(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, ...)
1076{
1077 va_list va;
1078
1079 va_start(va, pszFormat);
1080 int cch = RTTestPrintfNlV(hTest, enmLevel, pszFormat, va);
1081 va_end(va);
1082
1083 return cch;
1084}
1085
1086
1087RTR3DECL(int) RTTestPrintfV(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, va_list va)
1088{
1089 PRTTESTINT pTest = hTest;
1090 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
1091
1092 RTCritSectEnter(&pTest->OutputLock);
1093 int cch = 0;
1094 if (enmLevel <= pTest->enmMaxLevel)
1095 cch += rtTestPrintfV(pTest, pszFormat, va);
1096 RTCritSectLeave(&pTest->OutputLock);
1097
1098 return cch;
1099}
1100
1101
1102RTR3DECL(int) RTTestPrintf(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, ...)
1103{
1104 va_list va;
1105
1106 va_start(va, pszFormat);
1107 int cch = RTTestPrintfV(hTest, enmLevel, pszFormat, va);
1108 va_end(va);
1109
1110 return cch;
1111}
1112
1113
1114RTR3DECL(int) RTTestBanner(RTTEST hTest)
1115{
1116 return RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "TESTING...\n");
1117}
1118
1119
1120/**
1121 * Prints the result of a sub-test if necessary.
1122 *
1123 * @returns Number of chars printed.
1124 * @param pTest The test instance.
1125 * @remarks Caller own the test Lock.
1126 */
1127static int rtTestSubTestReport(PRTTESTINT pTest)
1128{
1129 int cch = 0;
1130 if ( !pTest->fSubTestReported
1131 && pTest->pszSubTest)
1132 {
1133 pTest->fSubTestReported = true;
1134 uint32_t cErrors = ASMAtomicUoReadU32(&pTest->cErrors) - pTest->cSubTestAtErrors;
1135 if (!cErrors)
1136 {
1137 if (!pTest->fSubTestSkipped)
1138 {
1139 rtTestXmlElem(pTest, "Passed", NULL);
1140 rtTestXmlElemEnd(pTest, "Test");
1141 cch += RTTestPrintfNl(pTest, RTTESTLVL_SUB_TEST, "%-60s: PASSED\n", pTest->pszSubTest);
1142 }
1143 else
1144 {
1145 rtTestXmlElem(pTest, "Skipped", NULL);
1146 rtTestXmlElemEnd(pTest, "Test");
1147 cch += RTTestPrintfNl(pTest, RTTESTLVL_SUB_TEST, "%-60s: SKIPPED\n", pTest->pszSubTest);
1148 }
1149 }
1150 else
1151 {
1152 pTest->cSubTestsFailed++;
1153 rtTestXmlElem(pTest, "Failed", "errors=\"%u\"", cErrors);
1154 rtTestXmlElemEnd(pTest, "Test");
1155 cch += RTTestPrintfNl(pTest, RTTESTLVL_SUB_TEST, "%-60s: FAILED (%u errors)\n",
1156 pTest->pszSubTest, cErrors);
1157 }
1158 }
1159 return cch;
1160}
1161
1162
1163/**
1164 * RTTestSub and RTTestSubDone worker that cleans up the current (if any)
1165 * sub test.
1166 *
1167 * @returns Number of chars printed.
1168 * @param pTest The test instance.
1169 * @remarks Caller own the test Lock.
1170 */
1171static int rtTestSubCleanup(PRTTESTINT pTest)
1172{
1173 int cch = 0;
1174 if (pTest->pszSubTest)
1175 {
1176 cch += rtTestSubTestReport(pTest);
1177
1178 RTStrFree((char *)pTest->pszSubTest);
1179 pTest->pszSubTest = NULL;
1180 pTest->fSubTestReported = true;
1181 }
1182 RTStrFree(pTest->pszErrCtx);
1183 pTest->pszErrCtx = NULL;
1184 return cch;
1185}
1186
1187
1188RTR3DECL(RTEXITCODE) RTTestSummaryAndDestroy(RTTEST hTest)
1189{
1190 PRTTESTINT pTest = hTest;
1191 RTTEST_GET_VALID_RETURN_RC(pTest, RTEXITCODE_FAILURE);
1192
1193 RTCritSectEnter(&pTest->Lock);
1194 rtTestSubTestReport(pTest);
1195 RTCritSectLeave(&pTest->Lock);
1196
1197 RTEXITCODE enmExitCode;
1198 if (!pTest->cErrors)
1199 {
1200 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "SUCCESS\n");
1201 enmExitCode = RTEXITCODE_SUCCESS;
1202 }
1203 else
1204 {
1205 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "FAILURE - %u errors\n", pTest->cErrors);
1206 enmExitCode = RTEXITCODE_FAILURE;
1207 }
1208
1209 RTTestDestroy(pTest);
1210 return enmExitCode;
1211}
1212
1213
1214RTR3DECL(RTEXITCODE) RTTestSkipAndDestroyV(RTTEST hTest, const char *pszReasonFmt, va_list va)
1215{
1216 PRTTESTINT pTest = hTest;
1217 RTTEST_GET_VALID_RETURN_RC(pTest, RTEXITCODE_SKIPPED);
1218
1219 RTCritSectEnter(&pTest->Lock);
1220 rtTestSubTestReport(pTest);
1221 RTCritSectLeave(&pTest->Lock);
1222
1223 RTEXITCODE enmExitCode;
1224 if (!pTest->cErrors)
1225 {
1226 if (pszReasonFmt)
1227 RTTestPrintfNlV(hTest, RTTESTLVL_FAILURE, pszReasonFmt, va);
1228 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "SKIPPED\n");
1229 enmExitCode = RTEXITCODE_SKIPPED;
1230 }
1231 else
1232 {
1233 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "FAILURE - %u errors\n", pTest->cErrors);
1234 enmExitCode = RTEXITCODE_FAILURE;
1235 }
1236
1237 RTTestDestroy(pTest);
1238 return enmExitCode;
1239}
1240
1241
1242RTR3DECL(RTEXITCODE) RTTestSkipAndDestroy(RTTEST hTest, const char *pszReasonFmt, ...)
1243{
1244 va_list va;
1245 va_start(va, pszReasonFmt);
1246 RTEXITCODE enmExitCode = RTTestSkipAndDestroyV(hTest, pszReasonFmt, va);
1247 va_end(va);
1248 return enmExitCode;
1249}
1250
1251
1252RTR3DECL(int) RTTestSub(RTTEST hTest, const char *pszSubTest)
1253{
1254 PRTTESTINT pTest = hTest;
1255 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
1256
1257 RTCritSectEnter(&pTest->Lock);
1258
1259 /* Cleanup, reporting if necessary previous sub test. */
1260 rtTestSubCleanup(pTest);
1261
1262 /* Start new sub test. */
1263 pTest->cSubTests++;
1264 pTest->cSubTestAtErrors = ASMAtomicUoReadU32(&pTest->cErrors);
1265 pTest->pszSubTest = RTStrDup(pszSubTest);
1266 pTest->cchSubTest = strlen(pszSubTest);
1267 AssertMsg(pTest->cchSubTest < 64 /* See g_kcchMaxTestResultName in testmanager/config.py. */,
1268 ("cchSubTest=%u: '%s'\n", pTest->cchSubTest, pTest->pszSubTest));
1269 pTest->fSubTestSkipped = false;
1270 pTest->fSubTestReported = false;
1271
1272 int cch = 0;
1273 if (pTest->enmMaxLevel >= RTTESTLVL_DEBUG)
1274 cch = RTTestPrintfNl(hTest, RTTESTLVL_DEBUG, "debug: Starting sub-test '%s'\n", pszSubTest);
1275
1276 if (!pTest->fXmlTopTestDone)
1277 {
1278 pTest->fXmlTopTestDone = true;
1279 rtTestXmlElemStart(pTest, "Test", "name=%RMas", pTest->pszTest);
1280 }
1281
1282 rtTestXmlElemStart(pTest, "Test", "name=%RMas", pszSubTest);
1283
1284 RTCritSectLeave(&pTest->Lock);
1285
1286 return cch;
1287}
1288
1289
1290RTR3DECL(int) RTTestSubF(RTTEST hTest, const char *pszSubTestFmt, ...)
1291{
1292 va_list va;
1293 va_start(va, pszSubTestFmt);
1294 int cch = RTTestSubV(hTest, pszSubTestFmt, va);
1295 va_end(va);
1296 return cch;
1297}
1298
1299
1300RTR3DECL(int) RTTestSubV(RTTEST hTest, const char *pszSubTestFmt, va_list va)
1301{
1302 char *pszSubTest;
1303 RTStrAPrintfV(&pszSubTest, pszSubTestFmt, va);
1304 if (pszSubTest)
1305 {
1306 int cch = RTTestSub(hTest, pszSubTest);
1307 RTStrFree(pszSubTest);
1308 return cch;
1309 }
1310 return 0;
1311}
1312
1313
1314RTR3DECL(int) RTTestSubDone(RTTEST hTest)
1315{
1316 PRTTESTINT pTest = hTest;
1317 RTTEST_GET_VALID_RETURN_RC(pTest, VERR_INVALID_HANDLE);
1318
1319 RTCritSectEnter(&pTest->Lock);
1320 int cch = rtTestSubCleanup(pTest);
1321 RTCritSectLeave(&pTest->Lock);
1322
1323 return cch;
1324}
1325
1326
1327RTR3DECL(int) RTTestPassedV(RTTEST hTest, const char *pszFormat, va_list va)
1328{
1329 PRTTESTINT pTest = hTest;
1330 AssertPtr(pszFormat);
1331 RTTEST_GET_VALID_RETURN_RC(pTest, VERR_INVALID_HANDLE);
1332
1333 int cch = 0;
1334 if (pTest->enmMaxLevel >= RTTESTLVL_INFO)
1335 {
1336 va_list va2;
1337 va_copy(va2, va);
1338
1339 RTCritSectEnter(&pTest->OutputLock);
1340 cch += rtTestPrintf(pTest, "%N\n", pszFormat, &va2);
1341 RTCritSectLeave(&pTest->OutputLock);
1342
1343 va_end(va2);
1344 }
1345
1346 return cch;
1347}
1348
1349
1350RTR3DECL(int) RTTestPassed(RTTEST hTest, const char *pszFormat, ...)
1351{
1352 va_list va;
1353
1354 va_start(va, pszFormat);
1355 int cch = RTTestPassedV(hTest, pszFormat, va);
1356 va_end(va);
1357
1358 return cch;
1359}
1360
1361
1362RTR3DECL(int) RTTestSkippedV(RTTEST hTest, const char *pszFormat, va_list va)
1363{
1364 PRTTESTINT pTest = hTest;
1365 AssertPtrNull(pszFormat);
1366 RTTEST_GET_VALID_RETURN_RC(pTest, VERR_INVALID_HANDLE);
1367
1368 pTest->fSubTestSkipped = true;
1369
1370 int cch = 0;
1371 if (pszFormat && *pszFormat && pTest->enmMaxLevel >= RTTESTLVL_INFO)
1372 {
1373 va_list va2;
1374 va_copy(va2, va);
1375
1376 RTCritSectEnter(&pTest->OutputLock);
1377 cch += rtTestPrintf(pTest, "%N\n", pszFormat, &va2);
1378 RTCritSectLeave(&pTest->OutputLock);
1379
1380 va_end(va2);
1381 }
1382
1383 return cch;
1384}
1385
1386
1387RTR3DECL(int) RTTestSkipped(RTTEST hTest, const char *pszFormat, ...)
1388{
1389 va_list va;
1390
1391 va_start(va, pszFormat);
1392 int cch = RTTestSkippedV(hTest, pszFormat, va);
1393 va_end(va);
1394
1395 return cch;
1396}
1397
1398
1399
1400/**
1401 * Gets the unit name.
1402 *
1403 * @returns Unit name.
1404 * @param enmUnit The unit.
1405 */
1406static const char *rtTestUnitName(RTTESTUNIT enmUnit)
1407{
1408 switch (enmUnit)
1409 {
1410 case RTTESTUNIT_PCT: return "%";
1411 case RTTESTUNIT_BYTES: return "bytes";
1412 case RTTESTUNIT_BYTES_PER_SEC: return "bytes/s";
1413 case RTTESTUNIT_KILOBYTES: return "KB";
1414 case RTTESTUNIT_KILOBYTES_PER_SEC: return "KB/s";
1415 case RTTESTUNIT_MEGABYTES: return "MB";
1416 case RTTESTUNIT_MEGABYTES_PER_SEC: return "MB/s";
1417 case RTTESTUNIT_PACKETS: return "packets";
1418 case RTTESTUNIT_PACKETS_PER_SEC: return "packets/s";
1419 case RTTESTUNIT_FRAMES: return "frames";
1420 case RTTESTUNIT_FRAMES_PER_SEC: return "frames/s";
1421 case RTTESTUNIT_OCCURRENCES: return "occurrences";
1422 case RTTESTUNIT_OCCURRENCES_PER_SEC: return "occurrences/s";
1423 case RTTESTUNIT_ROUND_TRIP: return "roundtrips";
1424 case RTTESTUNIT_CALLS: return "calls";
1425 case RTTESTUNIT_CALLS_PER_SEC: return "calls/s";
1426 case RTTESTUNIT_SECS: return "s";
1427 case RTTESTUNIT_MS: return "ms";
1428 case RTTESTUNIT_NS: return "ns";
1429 case RTTESTUNIT_NS_PER_CALL: return "ns/call";
1430 case RTTESTUNIT_NS_PER_FRAME: return "ns/frame";
1431 case RTTESTUNIT_NS_PER_OCCURRENCE: return "ns/occurrence";
1432 case RTTESTUNIT_NS_PER_PACKET: return "ns/packet";
1433 case RTTESTUNIT_NS_PER_ROUND_TRIP: return "ns/roundtrip";
1434 case RTTESTUNIT_INSTRS: return "ins";
1435 case RTTESTUNIT_INSTRS_PER_SEC: return "ins/sec";
1436 case RTTESTUNIT_NONE: return "";
1437 case RTTESTUNIT_PP1K: return "pp1k";
1438 case RTTESTUNIT_PP10K: return "pp10k";
1439 case RTTESTUNIT_PPM: return "ppm";
1440 case RTTESTUNIT_PPB: return "ppb";
1441 case RTTESTUNIT_TICKS: return "ticks";
1442 case RTTESTUNIT_TICKS_PER_CALL: return "ticks/call";
1443 case RTTESTUNIT_TICKS_PER_OCCURENCE: return "ticks/occ";
1444 case RTTESTUNIT_PAGES: return "pages";
1445 case RTTESTUNIT_PAGES_PER_SEC: return "pages/s";
1446 case RTTESTUNIT_TICKS_PER_PAGE: return "ticks/page";
1447 case RTTESTUNIT_NS_PER_PAGE: return "ns/page";
1448 case RTTESTUNIT_PS: return "ps";
1449 case RTTESTUNIT_PS_PER_CALL: return "ps/call";
1450 case RTTESTUNIT_PS_PER_FRAME: return "ps/frame";
1451 case RTTESTUNIT_PS_PER_OCCURRENCE: return "ps/occurrence";
1452 case RTTESTUNIT_PS_PER_PACKET: return "ps/packet";
1453 case RTTESTUNIT_PS_PER_ROUND_TRIP: return "ps/roundtrip";
1454 case RTTESTUNIT_PS_PER_PAGE: return "ps/page";
1455
1456 /* No default so gcc helps us keep this up to date. */
1457 case RTTESTUNIT_INVALID:
1458 case RTTESTUNIT_END:
1459 break;
1460 }
1461 AssertMsgFailed(("%d\n", enmUnit));
1462 return "unknown";
1463}
1464
1465
1466RTR3DECL(int) RTTestValue(RTTEST hTest, const char *pszName, uint64_t u64Value, RTTESTUNIT enmUnit)
1467{
1468 PRTTESTINT pTest = hTest;
1469 RTTEST_GET_VALID_RETURN(pTest);
1470
1471 Assert(strlen(pszName) < 56 /* See g_kcchMaxTestValueName in testmanager/config.py. */);
1472
1473 const char *pszUnit = rtTestUnitName(enmUnit);
1474
1475 RTCritSectEnter(&pTest->Lock);
1476 rtTestXmlElem(pTest, "Value", "name=%RMas unit=%RMas value=\"%llu\"", pszName, pszUnit, u64Value);
1477 RTCritSectLeave(&pTest->Lock);
1478
1479 RTCritSectEnter(&pTest->OutputLock);
1480 rtTestPrintf(pTest, " %-58s: %'16llu %s\n", pszName, u64Value, pszUnit);
1481 RTCritSectLeave(&pTest->OutputLock);
1482
1483 return VINF_SUCCESS;
1484}
1485
1486
1487RTR3DECL(int) RTTestValueF(RTTEST hTest, uint64_t u64Value, RTTESTUNIT enmUnit, const char *pszNameFmt, ...)
1488{
1489 va_list va;
1490 va_start(va, pszNameFmt);
1491 int rc = RTTestValueV(hTest, u64Value, enmUnit, pszNameFmt, va);
1492 va_end(va);
1493 return rc;
1494}
1495
1496
1497RTR3DECL(int) RTTestValueV(RTTEST hTest, uint64_t u64Value, RTTESTUNIT enmUnit, const char *pszNameFmt, va_list va)
1498{
1499 char *pszName;
1500 RTStrAPrintfV(&pszName, pszNameFmt, va);
1501 if (!pszName)
1502 return VERR_NO_MEMORY;
1503 int rc = RTTestValue(hTest, pszName, u64Value, enmUnit);
1504 RTStrFree(pszName);
1505 return rc;
1506}
1507
1508
1509RTR3DECL(int) RTTestErrorInc(RTTEST hTest)
1510{
1511 PRTTESTINT pTest = hTest;
1512 RTTEST_GET_VALID_RETURN(pTest);
1513
1514 ASMAtomicIncU32(&pTest->cErrors);
1515
1516 return VINF_SUCCESS;
1517}
1518
1519
1520RTR3DECL(uint32_t) RTTestErrorCount(RTTEST hTest)
1521{
1522 PRTTESTINT pTest = hTest;
1523 RTTEST_GET_VALID_RETURN_RC(pTest, UINT32_MAX);
1524
1525 return ASMAtomicReadU32(&pTest->cErrors);
1526}
1527
1528
1529RTR3DECL(uint32_t) RTTestSubErrorCount(RTTEST hTest)
1530{
1531 PRTTESTINT pTest = hTest;
1532 RTTEST_GET_VALID_RETURN_RC(pTest, UINT32_MAX);
1533
1534 return ASMAtomicReadU32(&pTest->cErrors) - pTest->cSubTestAtErrors;
1535}
1536
1537
1538RTR3DECL(int) RTTestFailedV(RTTEST hTest, const char *pszFormat, va_list va)
1539{
1540 PRTTESTINT pTest = hTest;
1541 RTTEST_GET_VALID_RETURN_RC(pTest, VERR_INVALID_HANDLE);
1542
1543 RTTestErrorInc(pTest);
1544
1545 int cch = 0;
1546 if (pTest->enmMaxLevel >= RTTESTLVL_FAILURE)
1547 {
1548 va_list va2;
1549 va_copy(va2, va);
1550
1551 const char *pszEnd = strchr(pszFormat, '\0');
1552 bool fHasNewLine = pszFormat != pszEnd
1553 && pszEnd[-1] == '\n';
1554
1555 RTCritSectEnter(&pTest->OutputLock);
1556 cch += rtTestPrintf(pTest, fHasNewLine ? "%N" : "%N\n", pszFormat, &va2);
1557 if (pTest->pszErrCtx)
1558 {
1559 cch += rtTestPrintf(pTest, "context: %s\n", pTest->pszErrCtx);
1560 RTStrFree(pTest->pszErrCtx);
1561 pTest->pszErrCtx = NULL;
1562 }
1563 RTCritSectLeave(&pTest->OutputLock);
1564
1565 va_end(va2);
1566 }
1567
1568 return cch;
1569}
1570
1571
1572RTR3DECL(int) RTTestFailed(RTTEST hTest, const char *pszFormat, ...)
1573{
1574 va_list va;
1575
1576 va_start(va, pszFormat);
1577 int cch = RTTestFailedV(hTest, pszFormat, va);
1578 va_end(va);
1579
1580 return cch;
1581}
1582
1583
1584RTR3DECL(int) RTTestFailureDetailsV(RTTEST hTest, const char *pszFormat, va_list va)
1585{
1586 return RTTestPrintfV(hTest, RTTESTLVL_FAILURE, pszFormat, va);
1587}
1588
1589
1590RTR3DECL(int) RTTestFailureDetails(RTTEST hTest, const char *pszFormat, ...)
1591{
1592 va_list va;
1593 va_start(va, pszFormat);
1594 int cch = RTTestFailureDetailsV(hTest, pszFormat, va);
1595 va_end(va);
1596 return cch;
1597}
1598
1599
1600RTR3DECL(int) RTTestErrContextV(RTTEST hTest, const char *pszFormat, va_list va)
1601{
1602 PRTTESTINT pTest = hTest;
1603 RTTEST_GET_VALID_RETURN(pTest);
1604
1605 RTStrFree(pTest->pszErrCtx);
1606 pTest->pszErrCtx = NULL;
1607
1608 if (pszFormat && *pszFormat)
1609 {
1610 pTest->pszErrCtx = RTStrAPrintf2V(pszFormat, va);
1611 AssertReturn(pTest->pszErrCtx, VERR_NO_STR_MEMORY);
1612 RTStrStripR(pTest->pszErrCtx);
1613 }
1614
1615 return VINF_SUCCESS;
1616}
1617
1618
1619RTR3DECL(int) RTTestErrContext(RTTEST hTest, const char *pszFormat, ...)
1620{
1621 va_list va;
1622 va_start(va, pszFormat);
1623 int rc = RTTestErrContextV(hTest, pszFormat, va);
1624 va_end(va);
1625 return rc;
1626}
1627
1628
1629RTR3DECL(int) RTTestDisableAssertions(RTTEST hTest)
1630{
1631 PRTTESTINT pTest = hTest;
1632 RTTEST_GET_VALID_RETURN(pTest);
1633
1634 uint32_t cTimes = ASMAtomicIncU32(&pTest->cAssertionsDisabledAndQuieted);
1635 if (cTimes >= 2 && cTimes <= 8)
1636 return VINF_SUCCESS;
1637 if (cTimes > 8)
1638 {
1639 RTAssertSetMayPanic(pTest->fAssertSavedMayPanic);
1640 RTAssertSetQuiet(pTest->fAssertSavedQuiet);
1641 Assert(cTimes <= 8);
1642 }
1643 pTest->fAssertSavedMayPanic = RTAssertSetMayPanic(false);
1644 pTest->fAssertSavedQuiet = RTAssertSetQuiet(true);
1645 return VINF_SUCCESS;
1646}
1647
1648
1649RTR3DECL(int) RTTestRestoreAssertions(RTTEST hTest)
1650{
1651 PRTTESTINT pTest = hTest;
1652 RTTEST_GET_VALID_RETURN(pTest);
1653
1654 uint32_t cTimes = ASMAtomicDecU32(&pTest->cAssertionsDisabledAndQuieted);
1655 if (cTimes == 0)
1656 {
1657 RTAssertSetMayPanic(pTest->fAssertSavedMayPanic);
1658 RTAssertSetQuiet(pTest->fAssertSavedQuiet);
1659 }
1660 else
1661 AssertStmt(cTimes < UINT32_MAX / 2, ASMAtomicIncU32(&pTest->cAssertionsDisabledAndQuieted));
1662 return VINF_SUCCESS;
1663}
1664
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