VirtualBox

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

Last change on this file since 20371 was 20109, checked in by vboxsync, 16 years ago

RTTestSkipAndDestroy.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.4 KB
Line 
1/* $Id: test.cpp 20109 2009-05-27 22:49:29Z vboxsync $ */
2/** @file
3 * IPRT - Testcase Framework.
4 */
5
6/*
7 * Copyright (C) 2009 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 *
26 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#include <iprt/test.h>
36
37#include <iprt/asm.h>
38#include <iprt/critsect.h>
39#include <iprt/env.h>
40#include <iprt/err.h>
41#include <iprt/mem.h>
42#include <iprt/once.h>
43#include <iprt/param.h>
44#include <iprt/string.h>
45#include <iprt/stream.h>
46
47#include "internal/magics.h"
48
49
50/*******************************************************************************
51* Structures and Typedefs *
52*******************************************************************************/
53/**
54 * Guarded memory allocation record.
55 */
56typedef struct RTTESTGUARDEDMEM
57{
58 /** Pointer to the next record. */
59 struct RTTESTGUARDEDMEM *pNext;
60 /** The address we return to the user. */
61 void *pvUser;
62 /** The base address of the allocation. */
63 void *pvAlloc;
64 /** The size of the allocation. */
65 size_t cbAlloc;
66 /** Guards. */
67 struct
68 {
69 /** The guard address. */
70 void *pv;
71 /** The guard size. */
72 size_t cb;
73 } aGuards[2];
74} RTTESTGUARDEDMEM;
75/** Pointer to an guarded memory allocation. */
76typedef RTTESTGUARDEDMEM *PRTTESTGUARDEDMEM;
77
78/**
79 * Test instance structure.
80 */
81typedef struct RTTESTINT
82{
83 /** Magic. */
84 uint32_t u32Magic;
85 /** The number of errors. */
86 volatile uint32_t cErrors;
87 /** The test name. */
88 const char *pszTest;
89 /** The length of the test name. */
90 size_t cchTest;
91 /** The size of a guard. Multiple of PAGE_SIZE. */
92 uint32_t cbGuard;
93 /** The verbosity level. */
94 RTTESTLVL enmMaxLevel;
95
96
97 /** Critical section seralizing output. */
98 RTCRITSECT OutputLock;
99 /** The output stream. */
100 PRTSTREAM pOutStrm;
101 /** Whether we're currently at a newline. */
102 bool fNewLine;
103
104
105 /** Critical section seralizing access to the members following it. */
106 RTCRITSECT Lock;
107
108 /** The list of guarded memory allocations. */
109 PRTTESTGUARDEDMEM pGuardedMem;
110
111 /** The current sub-test. */
112 const char *pszSubTest;
113 /** The lenght of the sub-test name. */
114 size_t cchSubTest;
115 /** Whether we've reported the sub-test result or not. */
116 bool fSubTestReported;
117 /** The start error count of the current subtest. */
118 uint32_t cSubTestAtErrors;
119
120 /** The number of sub tests. */
121 uint32_t cSubTests;
122 /** The number of sub tests that failed. */
123 uint32_t cSubTestsFailed;
124
125} RTTESTINT;
126/** Pointer to a test instance. */
127typedef RTTESTINT *PRTTESTINT;
128
129
130/*******************************************************************************
131* Defined Constants And Macros *
132*******************************************************************************/
133/** Validate a test instance. */
134#define RTTEST_VALID_RETURN(pTest) \
135 do { \
136 AssertPtrReturn(pTest, VERR_INVALID_HANDLE); \
137 AssertReturn(pTest->u32Magic == RTTESTINT_MAGIC, VERR_INVALID_HANDLE); \
138 } while (0)
139
140/** Gets and validates a test instance.
141 * If the handle is nil, we will try retrive it from the test TLS entry.
142 */
143#define RTTEST_GET_VALID_RETURN(pTest) \
144 do { \
145 if (pTest == NIL_RTTEST) \
146 pTest = (PRTTESTINT)RTTlsGet(g_iTestTls); \
147 AssertPtrReturn(pTest, VERR_INVALID_HANDLE); \
148 AssertReturn(pTest->u32Magic == RTTESTINT_MAGIC, VERR_INVALID_MAGIC); \
149 } while (0)
150
151
152/** Gets and validates a test instance.
153 * If the handle is nil, we will try retrive it from the test TLS entry.
154 */
155#define RTTEST_GET_VALID_RETURN_RC(pTest, rc) \
156 do { \
157 if (pTest == NIL_RTTEST) \
158 pTest = (PRTTESTINT)RTTlsGet(g_iTestTls); \
159 AssertPtrReturn(pTest, (rc)); \
160 AssertReturn(pTest->u32Magic == RTTESTINT_MAGIC, (rc)); \
161 } while (0)
162
163
164/*******************************************************************************
165* Internal Functions *
166*******************************************************************************/
167static void rtTestGuardedFreeOne(PRTTESTGUARDEDMEM pMem);
168static int rtTestPrintf(PRTTESTINT pTest, const char *pszFormat, ...);
169
170
171/*******************************************************************************
172* Global Variables *
173*******************************************************************************/
174/** For serializing TLS init. */
175static RTONCE g_TestInitOnce = RTONCE_INITIALIZER;
176/** Our TLS entry. */
177static RTTLS g_iTestTls = NIL_RTTLS;
178
179
180
181/**
182 * Init TLS index once.
183 *
184 * @returns IPRT status code.
185 * @param pvUser1 Ignored.
186 * @param pvUser2 Ignored.
187 */
188static DECLCALLBACK(int32_t) rtTestInitOnce(void *pvUser1, void *pvUser2)
189{
190 NOREF(pvUser1);
191 NOREF(pvUser2);
192 return RTTlsAllocEx(&g_iTestTls, NULL);
193}
194
195
196
197/**
198 * Creates a test instance.
199 *
200 * @returns IPRT status code.
201 * @param pszTest The test name.
202 * @param phTest Where to store the test instance handle.
203 */
204RTR3DECL(int) RTTestCreate(const char *pszTest, PRTTEST phTest)
205{
206 /*
207 * Global init.
208 */
209 int rc = RTOnce(&g_TestInitOnce, rtTestInitOnce, NULL, NULL);
210 if (RT_FAILURE(rc))
211 return rc;
212
213 /*
214 * Create the instance.
215 */
216 PRTTESTINT pTest = (PRTTESTINT)RTMemAllocZ(sizeof(*pTest));
217 if (!pTest)
218 return VERR_NO_MEMORY;
219 pTest->u32Magic = RTTESTINT_MAGIC;
220 pTest->pszTest = RTStrDup(pszTest);
221 pTest->cchTest = strlen(pszTest);
222 pTest->cbGuard = PAGE_SIZE * 7;
223 pTest->enmMaxLevel = RTTESTLVL_SUB_TEST;
224
225 pTest->pOutStrm = g_pStdOut;
226 pTest->fNewLine = true;
227
228 pTest->pGuardedMem = NULL;
229
230 pTest->pszSubTest = NULL;
231 pTest->cchSubTest = 0;
232 pTest->fSubTestReported = true;
233 pTest->cSubTestAtErrors = 0;
234 pTest->cSubTests = 0;
235 pTest->cSubTestsFailed = 0;
236
237 rc = RTCritSectInit(&pTest->Lock);
238 if (RT_SUCCESS(rc))
239 {
240 rc = RTCritSectInit(&pTest->OutputLock);
241 if (RT_SUCCESS(rc))
242 {
243
244 /*
245 * Associate it with our TLS entry unless there is already
246 * an instance there.
247 */
248 if (!RTTlsGet(g_iTestTls))
249 rc = RTTlsSet(g_iTestTls, pTest);
250 if (RT_SUCCESS(rc))
251 {
252 /*
253 * Finally, pick up overrides from the environment.
254 */
255 char szMaxLevel[80];
256 rc = RTEnvGetEx(RTENV_DEFAULT, "IPRT_TEST_MAX_LEVEL", szMaxLevel, sizeof(szMaxLevel), NULL);
257 if (RT_SUCCESS(rc))
258 {
259 char *pszMaxLevel = RTStrStrip(szMaxLevel);
260 if (!strcmp(pszMaxLevel, "all"))
261 pTest->enmMaxLevel = RTTESTLVL_DEBUG;
262 if (!strcmp(pszMaxLevel, "quiet"))
263 pTest->enmMaxLevel = RTTESTLVL_FAILURE;
264 else if (!strcmp(pszMaxLevel, "debug"))
265 pTest->enmMaxLevel = RTTESTLVL_DEBUG;
266 else if (!strcmp(pszMaxLevel, "info"))
267 pTest->enmMaxLevel = RTTESTLVL_INFO;
268 else if (!strcmp(pszMaxLevel, "sub_test"))
269 pTest->enmMaxLevel = RTTESTLVL_SUB_TEST;
270 else if (!strcmp(pszMaxLevel, "failure"))
271 pTest->enmMaxLevel = RTTESTLVL_FAILURE;
272 }
273
274 *phTest = pTest;
275 return VINF_SUCCESS;
276 }
277
278 /* bail out. */
279 RTCritSectDelete(&pTest->OutputLock);
280 }
281 RTCritSectDelete(&pTest->Lock);
282 }
283 pTest->u32Magic = 0;
284 RTStrFree((char *)pTest->pszTest);
285 RTMemFree(pTest);
286 return rc;
287}
288
289
290/**
291 * Destroys a test instance previously created by RTTestCreate.
292 *
293 * @returns IPRT status code.
294 * @param hTest The test handle. NIL_RTTEST is ignored.
295 */
296RTR3DECL(int) RTTestDestroy(RTTEST hTest)
297{
298 /*
299 * Validate
300 */
301 if (hTest == NIL_RTTEST)
302 return VINF_SUCCESS;
303 RTTESTINT *pTest = hTest;
304 RTTEST_VALID_RETURN(pTest);
305
306 /*
307 * Make sure we end with a new line.
308 */
309 if (!pTest->fNewLine)
310 rtTestPrintf(pTest, "\n");
311
312 /*
313 * Clean up.
314 */
315 if ((RTTESTINT *)RTTlsGet(g_iTestTls) == pTest)
316 RTTlsSet(g_iTestTls, NULL);
317
318 ASMAtomicWriteU32(&pTest->u32Magic, ~RTTESTINT_MAGIC);
319 RTCritSectDelete(&pTest->Lock);
320 RTCritSectDelete(&pTest->OutputLock);
321
322 /* free guarded memory. */
323 PRTTESTGUARDEDMEM pMem = pTest->pGuardedMem;
324 pTest->pGuardedMem = NULL;
325 while (pMem)
326 {
327 PRTTESTGUARDEDMEM pFree = pMem;
328 pMem = pMem->pNext;
329 rtTestGuardedFreeOne(pFree);
330 }
331
332 RTStrFree((char *)pTest->pszSubTest);
333 pTest->pszSubTest = NULL;
334 RTStrFree((char *)pTest->pszTest);
335 pTest->pszTest = NULL;
336 RTMemFree(pTest);
337 return VINF_SUCCESS;
338}
339
340
341
342/**
343 * Allocate a block of guarded memory.
344 *
345 * @returns IPRT status code.
346 * @param hTest The test handle. If NIL_RTTEST we'll use the one
347 * associated with the calling thread.
348 * @param cb The amount of memory to allocate.
349 * @param cbAlign The alignment of the returned block.
350 * @param fHead Head or tail optimized guard.
351 * @param ppvUser Where to return the pointer to the block.
352 */
353RTR3DECL(int) RTTestGuardedAlloc(RTTEST hTest, size_t cb, uint32_t cbAlign, bool fHead, void **ppvUser)
354{
355 PRTTESTINT pTest = hTest;
356 RTTEST_GET_VALID_RETURN(pTest);
357 if (cbAlign == 0)
358 cbAlign = 1;
359 AssertReturn(cbAlign <= PAGE_SIZE, VERR_INVALID_PARAMETER);
360 AssertReturn(cbAlign == (UINT32_C(1) << (ASMBitFirstSetU32(cbAlign) - 1)), VERR_INVALID_PARAMETER);
361
362 /*
363 * Allocate the record and block and initialize them.
364 */
365 int rc = VERR_NO_MEMORY;
366 PRTTESTGUARDEDMEM pMem = (PRTTESTGUARDEDMEM)RTMemAlloc(sizeof(*pMem));
367 if (RT_LIKELY(pMem))
368 {
369 size_t const cbAligned = RT_ALIGN_Z(cb, PAGE_SIZE);
370 pMem->aGuards[0].cb = pMem->aGuards[1].cb = pTest->cbGuard;
371 pMem->cbAlloc = pMem->aGuards[0].cb + pMem->aGuards[1].cb + cbAligned;
372 pMem->pvAlloc = RTMemPageAlloc(pMem->cbAlloc);
373 if (pMem->pvAlloc)
374 {
375 pMem->aGuards[0].pv = pMem->pvAlloc;
376 pMem->pvUser = (uint8_t *)pMem->pvAlloc + pMem->aGuards[0].cb;
377 pMem->aGuards[1].pv = (uint8_t *)pMem->pvUser + cbAligned;
378 if (!fHead)
379 {
380 size_t off = cb & PAGE_OFFSET_MASK;
381 if (off)
382 {
383 off = PAGE_SIZE - RT_ALIGN_Z(off, cbAlign);
384 pMem->pvUser = (uint8_t *)pMem->pvUser + off;
385 }
386 }
387
388 /*
389 * Set up the guards and link the record.
390 */
391 ASMMemFill32(pMem->aGuards[0].pv, pMem->aGuards[0].cb, 0xdeadbeef);
392 ASMMemFill32(pMem->aGuards[1].pv, pMem->aGuards[1].cb, 0xdeadbeef);
393 rc = RTMemProtect(pMem->aGuards[0].pv, pMem->aGuards[0].cb, RTMEM_PROT_NONE);
394 if (RT_SUCCESS(rc))
395 {
396 rc = RTMemProtect(pMem->aGuards[1].pv, pMem->aGuards[1].cb, RTMEM_PROT_NONE);
397 if (RT_SUCCESS(rc))
398 {
399 *ppvUser = pMem->pvUser;
400
401 RTCritSectEnter(&pTest->Lock);
402 pMem->pNext = pTest->pGuardedMem;
403 pTest->pGuardedMem = pMem;
404 RTCritSectLeave(&pTest->Lock);
405
406 return VINF_SUCCESS;
407 }
408
409 RTMemProtect(pMem->aGuards[0].pv, pMem->aGuards[0].cb, RTMEM_PROT_WRITE | RTMEM_PROT_READ);
410 }
411
412 RTMemPageFree(pMem->pvAlloc);
413 }
414 RTMemFree(pMem);
415 }
416 return rc;
417}
418
419
420/**
421 * Allocates a block of guarded memory where the guarded is immediately after
422 * the user memory.
423 *
424 * @returns Pointer to the allocated memory. NULL on failure.
425 * @param hTest The test handle. If NIL_RTTEST we'll use the one
426 * associated with the calling thread.
427 * @param cb The amount of memory to allocate.
428 */
429RTR3DECL(void *) RTTestGuardedAllocTail(RTTEST hTest, size_t cb)
430{
431 void *pvUser;
432 int rc = RTTestGuardedAlloc(hTest, cb, 1, false /*fHead*/, &pvUser);
433 if (RT_SUCCESS(rc))
434 return pvUser;
435 return NULL;
436}
437
438
439/**
440 * Allocates a block of guarded memory where the guarded is right in front of
441 * the user memory.
442 *
443 * @returns Pointer to the allocated memory. NULL on failure.
444 * @param hTest The test handle. If NIL_RTTEST we'll use the one
445 * associated with the calling thread.
446 * @param cb The amount of memory to allocate.
447 */
448RTR3DECL(void *) RTTestGuardedAllocHead(RTTEST hTest, size_t cb)
449{
450 void *pvUser;
451 int rc = RTTestGuardedAlloc(hTest, cb, 1, true /*fHead*/, &pvUser);
452 if (RT_SUCCESS(rc))
453 return pvUser;
454 return NULL;
455}
456
457
458/**
459 * Frees one block of guarded memory.
460 *
461 * The caller is responsible for unlinking it.
462 *
463 * @param pMem The memory record.
464 */
465static void rtTestGuardedFreeOne(PRTTESTGUARDEDMEM pMem)
466{
467 int rc;
468 rc = RTMemProtect(pMem->aGuards[0].pv, pMem->aGuards[0].cb, RTMEM_PROT_WRITE | RTMEM_PROT_READ); AssertRC(rc);
469 rc = RTMemProtect(pMem->aGuards[1].pv, pMem->aGuards[1].cb, RTMEM_PROT_WRITE | RTMEM_PROT_READ); AssertRC(rc);
470 RTMemPageFree(pMem->pvAlloc);
471 RTMemFree(pMem);
472}
473
474
475/**
476 * Frees a block of guarded memory.
477 *
478 * @returns IPRT status code.
479 * @param hTest The test handle. If NIL_RTTEST we'll use the one
480 * associated with the calling thread.
481 * @param pv The memory. NULL is ignored.
482 */
483RTR3DECL(int) RTTestGuardedFree(RTTEST hTest, void *pv)
484{
485 PRTTESTINT pTest = hTest;
486 RTTEST_GET_VALID_RETURN(pTest);
487 if (!pv)
488 return VINF_SUCCESS;
489
490 /*
491 * Find it.
492 */
493 int rc = VERR_INVALID_POINTER;
494 PRTTESTGUARDEDMEM pPrev = NULL;
495
496 RTCritSectEnter(&pTest->Lock);
497 for (PRTTESTGUARDEDMEM pMem = pTest->pGuardedMem; pMem; pMem = pMem->pNext)
498 {
499 if (pMem->pvUser == pv)
500 {
501 if (pPrev)
502 pPrev->pNext = pMem->pNext;
503 else
504 pTest->pGuardedMem = pMem->pNext;
505 rtTestGuardedFreeOne(pMem);
506 rc = VINF_SUCCESS;
507 break;
508 }
509 pPrev = pMem;
510 }
511 RTCritSectLeave(&pTest->Lock);
512
513 return VINF_SUCCESS;
514}
515
516
517/**
518 * Output callback.
519 *
520 * @returns number of bytes written.
521 * @param pvArg User argument.
522 * @param pachChars Pointer to an array of utf-8 characters.
523 * @param cbChars Number of bytes in the character array pointed to by pachChars.
524 */
525static DECLCALLBACK(size_t) rtTestPrintfOutput(void *pvArg, const char *pachChars, size_t cbChars)
526{
527 size_t cch = 0;
528 PRTTESTINT pTest = (PRTTESTINT)pvArg;
529 if (cbChars)
530 {
531 do
532 {
533 /* insert prefix if at a newline. */
534 if (pTest->fNewLine)
535 {
536 RTStrmWrite(pTest->pOutStrm, pTest->pszTest, pTest->cchTest);
537 RTStrmWrite(pTest->pOutStrm, ": ", 2);
538 cch += 2 + pTest->cchTest;
539 }
540
541 /* look for newline and write the stuff. */
542 const char *pchEnd = (const char *)memchr(pachChars, '\n', cbChars);
543 if (!pchEnd)
544 {
545 pTest->fNewLine = false;
546 RTStrmWrite(pTest->pOutStrm, pachChars, cbChars);
547 cch += cbChars;
548 break;
549 }
550
551 pTest->fNewLine = true;
552 size_t const cchPart = pchEnd - pachChars + 1;
553 RTStrmWrite(pTest->pOutStrm, pachChars, cchPart);
554 cch += cchPart;
555 pachChars += cchPart;
556 cbChars -= cchPart;
557 } while (cbChars);
558 }
559 else
560 RTStrmFlush(pTest->pOutStrm);
561 return cch;
562}
563
564
565/**
566 * Internal output worker.
567 *
568 * Caller takes the lock.
569 *
570 * @returns Number of chars printed.
571 * @param pTest The test instance.
572 * @param pszFormat The message.
573 * @param va The arguments.
574 */
575static int rtTestPrintfV(PRTTESTINT pTest, const char *pszFormat, va_list va)
576{
577 return (int)RTStrFormatV(rtTestPrintfOutput, pTest, NULL, NULL, pszFormat, va);
578}
579
580
581/**
582 * Internal output worker.
583 *
584 * Caller takes the lock.
585 *
586 * @returns Number of chars printed.
587 * @param pTest The test instance.
588 * @param pszFormat The message.
589 * @param ... The arguments.
590 */
591static int rtTestPrintf(PRTTESTINT pTest, const char *pszFormat, ...)
592{
593 va_list va;
594
595 va_start(va, pszFormat);
596 int cch = rtTestPrintfV(pTest, pszFormat, va);
597 va_end(va);
598
599 return cch;
600}
601
602
603/**
604 * Test vprintf making sure the output starts on a new line.
605 *
606 * @returns Number of chars printed.
607 * @param hTest The test handle. If NIL_RTTEST we'll use the one
608 * associated with the calling thread.
609 * @param enmLevel Message importance level.
610 * @param pszFormat The message.
611 * @param va Arguments.
612 */
613RTR3DECL(int) RTTestPrintfNlV(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, va_list va)
614{
615 PRTTESTINT pTest = hTest;
616 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
617
618 RTCritSectEnter(&pTest->OutputLock);
619
620 int cch = 0;
621 if (enmLevel <= pTest->enmMaxLevel)
622 {
623 if (!pTest->fNewLine)
624 cch += rtTestPrintf(pTest, "\n");
625 cch += rtTestPrintfV(pTest, pszFormat, va);
626 }
627
628 RTCritSectLeave(&pTest->OutputLock);
629
630 return cch;
631}
632
633
634/**
635 * Test printf making sure the output starts on a new line.
636 *
637 * @returns Number of chars printed.
638 * @param hTest The test handle. If NIL_RTTEST we'll use the one
639 * associated with the calling thread.
640 * @param enmLevel Message importance level.
641 * @param pszFormat The message.
642 * @param ... Arguments.
643 */
644RTR3DECL(int) RTTestPrintfNl(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, ...)
645{
646 va_list va;
647
648 va_start(va, pszFormat);
649 int cch = RTTestPrintfNlV(hTest, enmLevel, pszFormat, va);
650 va_end(va);
651
652 return cch;
653}
654
655
656/**
657 * Test vprintf, makes sure lines are prefixed and so forth.
658 *
659 * @returns Number of chars printed.
660 * @param hTest The test handle. If NIL_RTTEST we'll use the one
661 * associated with the calling thread.
662 * @param enmLevel Message importance level.
663 * @param pszFormat The message.
664 * @param va Arguments.
665 */
666RTR3DECL(int) RTTestPrintfV(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, va_list va)
667{
668 PRTTESTINT pTest = hTest;
669 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
670
671 RTCritSectEnter(&pTest->OutputLock);
672 int cch = 0;
673 if (enmLevel <= pTest->enmMaxLevel)
674 cch += rtTestPrintfV(pTest, pszFormat, va);
675 RTCritSectLeave(&pTest->OutputLock);
676
677 return cch;
678}
679
680
681/**
682 * Test printf, makes sure lines are prefixed and so forth.
683 *
684 * @returns Number of chars printed.
685 * @param hTest The test handle. If NIL_RTTEST we'll use the one
686 * associated with the calling thread.
687 * @param enmLevel Message importance level.
688 * @param pszFormat The message.
689 * @param ... Arguments.
690 */
691RTR3DECL(int) RTTestPrintf(RTTEST hTest, RTTESTLVL enmLevel, const char *pszFormat, ...)
692{
693 va_list va;
694
695 va_start(va, pszFormat);
696 int cch = RTTestPrintfV(hTest, enmLevel, pszFormat, va);
697 va_end(va);
698
699 return cch;
700}
701
702
703/**
704 * Prints the test banner.
705 *
706 * @returns Number of chars printed.
707 * @param hTest The test handle. If NIL_RTTEST we'll use the one
708 * associated with the calling thread.
709 */
710RTR3DECL(int) RTTestBanner(RTTEST hTest)
711{
712 return RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "TESTING...\n");
713}
714
715
716/**
717 * Prints the result of a sub-test if necessary.
718 *
719 * @returns Number of chars printed.
720 * @param pTest The test instance.
721 * @remarks Caller own the test Lock.
722 */
723static int rtTestSubTestReport(PRTTESTINT pTest)
724{
725 int cch = 0;
726 if ( !pTest->fSubTestReported
727 && pTest->pszSubTest)
728 {
729 pTest->fSubTestReported = true;
730 uint32_t cErrors = ASMAtomicUoReadU32(&pTest->cErrors) - pTest->cSubTestAtErrors;
731 if (!cErrors)
732 cch += RTTestPrintfNl(pTest, RTTESTLVL_SUB_TEST, "%-50s: PASSED\n", pTest->pszSubTest);
733 else
734 {
735 pTest->cSubTestsFailed++;
736 cch += RTTestPrintfNl(pTest, RTTESTLVL_SUB_TEST, "%-50s: FAILED (%u errors)\n",
737 pTest->pszSubTest, cErrors);
738 }
739 }
740 return cch;
741}
742
743
744/**
745 * RTTestSub and RTTestSubDone worker that cleans up the current (if any)
746 * sub test.
747 *
748 * @returns Number of chars printed.
749 * @param pTest The test instance.
750 * @remarks Caller own the test Lock.
751 */
752static int rtTestSubCleanup(PRTTESTINT pTest)
753{
754 int cch = 0;
755 if (pTest->pszSubTest)
756 {
757 cch += rtTestSubTestReport(pTest);
758
759 RTStrFree((char *)pTest->pszSubTest);
760 pTest->pszSubTest = NULL;
761 pTest->fSubTestReported = true;
762 }
763 return cch;
764}
765
766
767/**
768 * Summaries the test, destroys the test instance and return an exit code.
769 *
770 * @returns Test program exit code.
771 * @param hTest The test handle. If NIL_RTTEST we'll use the one
772 * associated with the calling thread.
773 */
774RTR3DECL(int) RTTestSummaryAndDestroy(RTTEST hTest)
775{
776 PRTTESTINT pTest = hTest;
777 RTTEST_GET_VALID_RETURN_RC(pTest, 2);
778
779 RTCritSectEnter(&pTest->Lock);
780 rtTestSubTestReport(pTest);
781 RTCritSectLeave(&pTest->Lock);
782
783 int rc;
784 if (!pTest->cErrors)
785 {
786 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "SUCCESS\n", pTest->cErrors);
787 rc = 0;
788 }
789 else
790 {
791 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "FAILURE - %u errors\n", pTest->cErrors);
792 rc = 1;
793 }
794
795 RTTestDestroy(pTest);
796 return rc;
797}
798
799
800RTR3DECL(int) RTTestSkipAndDestroyV(RTTEST hTest, const char *pszReason, va_list va)
801{
802 PRTTESTINT pTest = hTest;
803 RTTEST_GET_VALID_RETURN_RC(pTest, 2);
804
805 RTCritSectEnter(&pTest->Lock);
806 rtTestSubTestReport(pTest);
807 RTCritSectLeave(&pTest->Lock);
808
809 int rc;
810 if (!pTest->cErrors)
811 {
812 if (pszReason)
813 RTTestPrintfNlV(hTest, RTTESTLVL_FAILURE, pszReason, va);
814 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "SKIPPED\n", pTest->cErrors);
815 rc = 2;
816 }
817 else
818 {
819 RTTestPrintfNl(hTest, RTTESTLVL_ALWAYS, "FAILURE - %u errors\n", pTest->cErrors);
820 rc = 1;
821 }
822
823 RTTestDestroy(pTest);
824 return rc;
825}
826
827
828RTR3DECL(int) RTTestSkipAndDestroy(RTTEST hTest, const char *pszReason, ...)
829{
830 va_list va;
831 va_start(va, pszReason);
832 int rc = RTTestSkipAndDestroyV(hTest, pszReason, va);
833 va_end(va);
834 return rc;
835}
836
837
838/**
839 * Starts a sub-test.
840 *
841 * This will perform an implicit RTTestSubDone() call if that has not been done
842 * since the last RTTestSub call.
843 *
844 * @returns Number of chars printed.
845 * @param hTest The test handle. If NIL_RTTEST we'll use the one
846 * associated with the calling thread.
847 * @param pszSubTest The sub-test name
848 */
849RTR3DECL(int) RTTestSub(RTTEST hTest, const char *pszSubTest)
850{
851 PRTTESTINT pTest = hTest;
852 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
853
854 RTCritSectEnter(&pTest->Lock);
855
856 /* Cleanup, reporting if necessary previous sub test. */
857 rtTestSubCleanup(pTest);
858
859 /* Start new sub test. */
860 pTest->cSubTests++;
861 pTest->cSubTestAtErrors = ASMAtomicUoReadU32(&pTest->cErrors);
862 pTest->pszSubTest = RTStrDup(pszSubTest);
863 pTest->cchSubTest = strlen(pszSubTest);
864 pTest->fSubTestReported = false;
865
866 int cch = 0;
867 if (pTest->enmMaxLevel >= RTTESTLVL_DEBUG)
868 cch = RTTestPrintfNl(hTest, RTTESTLVL_DEBUG, "debug: Starting sub-test '%s'\n", pszSubTest);
869
870 RTCritSectLeave(&pTest->Lock);
871
872 return cch;
873}
874
875
876/**
877 * Format string version of RTTestSub.
878 *
879 * See RTTestSub for details.
880 *
881 * @returns Number of chars printed.
882 * @param hTest The test handle. If NIL_RTTEST we'll use the one
883 * associated with the calling thread.
884 * @param pszSubTestFmt The sub-test name format string.
885 * @param ... Arguments.
886 */
887RTR3DECL(int) RTTestSubF(RTTEST hTest, const char *pszSubTestFmt, ...)
888{
889 va_list va;
890 va_start(va, pszSubTestFmt);
891 int cch = RTTestSubV(hTest, pszSubTestFmt, va);
892 va_end(va);
893 return cch;
894}
895
896
897/**
898 * Format string version of RTTestSub.
899 *
900 * See RTTestSub for details.
901 *
902 * @returns Number of chars printed.
903 * @param hTest The test handle. If NIL_RTTEST we'll use the one
904 * associated with the calling thread.
905 * @param pszSubTestFmt The sub-test name format string.
906 * @param ... Arguments.
907 */
908RTR3DECL(int) RTTestSubV(RTTEST hTest, const char *pszSubTestFmt, va_list va)
909{
910 char *pszSubTest;
911 RTStrAPrintfV(&pszSubTest, pszSubTestFmt, va);
912 if (pszSubTest)
913 {
914 int cch = RTTestSub(hTest, pszSubTest);
915 RTStrFree(pszSubTest);
916 return cch;
917 }
918 return 0;
919}
920
921
922/**
923 * Completes a sub-test.
924 *
925 * @returns Number of chars printed.
926 * @param hTest The test handle. If NIL_RTTEST we'll use the one
927 * associated with the calling thread.
928 */
929RTR3DECL(int) RTTestSubDone(RTTEST hTest)
930{
931 PRTTESTINT pTest = hTest;
932 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
933
934 RTCritSectEnter(&pTest->Lock);
935 int cch = rtTestSubCleanup(pTest);
936 RTCritSectLeave(&pTest->Lock);
937
938 return cch;
939}
940
941/**
942 * Prints an extended PASSED message, optional.
943 *
944 * This does not conclude the sub-test, it could be used to report the passing
945 * of a sub-sub-to-the-power-of-N-test.
946 *
947 * @returns IPRT status code.
948 * @param hTest The test handle. If NIL_RTTEST we'll use the one
949 * associated with the calling thread.
950 * @param pszFormat The message. No trailing newline.
951 * @param va The arguments.
952 */
953RTR3DECL(int) RTTestPassedV(RTTEST hTest, const char *pszFormat, va_list va)
954{
955 PRTTESTINT pTest = hTest;
956 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
957
958 int cch = 0;
959 if (pTest->enmMaxLevel >= RTTESTLVL_INFO)
960 {
961 va_list va2;
962 va_copy(va2, va);
963
964 RTCritSectEnter(&pTest->OutputLock);
965 cch += rtTestPrintf(pTest, "%N\n", pszFormat, &va2);
966 RTCritSectLeave(&pTest->OutputLock);
967
968 va_end(va2);
969 }
970
971 return cch;
972}
973
974
975/**
976 * Prints an extended PASSED message, optional.
977 *
978 * This does not conclude the sub-test, it could be used to report the passing
979 * of a sub-sub-to-the-power-of-N-test.
980 *
981 * @returns IPRT status code.
982 * @param hTest The test handle. If NIL_RTTEST we'll use the one
983 * associated with the calling thread.
984 * @param pszFormat The message. No trailing newline.
985 * @param ... The arguments.
986 */
987RTR3DECL(int) RTTestPassed(RTTEST hTest, const char *pszFormat, ...)
988{
989 va_list va;
990
991 va_start(va, pszFormat);
992 int cch = RTTestPassedV(hTest, pszFormat, va);
993 va_end(va);
994
995 return cch;
996}
997
998
999/**
1000 * Increments the error counter.
1001 *
1002 * @returns IPRT status code.
1003 * @param hTest The test handle. If NIL_RTTEST we'll use the one
1004 * associated with the calling thread.
1005 */
1006RTR3DECL(int) RTTestErrorInc(RTTEST hTest)
1007{
1008 PRTTESTINT pTest = hTest;
1009 RTTEST_GET_VALID_RETURN(pTest);
1010
1011 ASMAtomicIncU32(&pTest->cErrors);
1012
1013 return VINF_SUCCESS;
1014}
1015
1016
1017/**
1018 * Increments the error counter and prints a failure message.
1019 *
1020 * @returns IPRT status code.
1021 * @param hTest The test handle. If NIL_RTTEST we'll use the one
1022 * associated with the calling thread.
1023 * @param pszFormat The message. No trailing newline.
1024 * @param va The arguments.
1025 */
1026RTR3DECL(int) RTTestFailedV(RTTEST hTest, const char *pszFormat, va_list va)
1027{
1028 PRTTESTINT pTest = hTest;
1029 RTTEST_GET_VALID_RETURN_RC(pTest, -1);
1030
1031 RTTestErrorInc(pTest);
1032
1033 int cch = 0;
1034 if (pTest->enmMaxLevel >= RTTESTLVL_FAILURE)
1035 {
1036 va_list va2;
1037 va_copy(va2, va);
1038
1039 const char *pszEnd = strchr(pszFormat, '\0');
1040 bool fHasNewLine = pszFormat != pszEnd
1041 && pszEnd[-1] == '\n';
1042
1043 RTCritSectEnter(&pTest->OutputLock);
1044 cch += rtTestPrintf(pTest, fHasNewLine ? "%N" : "%N\n", pszFormat, &va2);
1045 RTCritSectLeave(&pTest->OutputLock);
1046
1047 va_end(va2);
1048 }
1049
1050 return cch;
1051}
1052
1053
1054/**
1055 * Increments the error counter and prints a failure message.
1056 *
1057 * @returns IPRT status code.
1058 * @param hTest The test handle. If NIL_RTTEST we'll use the one
1059 * associated with the calling thread.
1060 * @param pszFormat The message. No trailing newline.
1061 * @param ... The arguments.
1062 */
1063RTR3DECL(int) RTTestFailed(RTTEST hTest, const char *pszFormat, ...)
1064{
1065 va_list va;
1066
1067 va_start(va, pszFormat);
1068 int cch = RTTestFailedV(hTest, pszFormat, va);
1069 va_end(va);
1070
1071 return cch;
1072}
1073
1074
1075/**
1076 * Same as RTTestPrintfV with RTTESTLVL_FAILURE.
1077 *
1078 * @returns Number of chars printed.
1079 * @param hTest The test handle. If NIL_RTTEST we'll use the one
1080 * associated with the calling thread.
1081 * @param enmLevel Message importance level.
1082 * @param pszFormat The message.
1083 * @param va Arguments.
1084 */
1085RTR3DECL(int) RTTestFailureDetailsV(RTTEST hTest, const char *pszFormat, va_list va)
1086{
1087 return RTTestPrintfV(hTest, RTTESTLVL_FAILURE, pszFormat, va);
1088}
1089
1090
1091/**
1092 * Same as RTTestPrintf with RTTESTLVL_FAILURE.
1093 *
1094 * @returns Number of chars printed.
1095 * @param hTest The test handle. If NIL_RTTEST we'll use the one
1096 * associated with the calling thread.
1097 * @param enmLevel Message importance level.
1098 * @param pszFormat The message.
1099 * @param ... Arguments.
1100 */
1101RTR3DECL(int) RTTestFailureDetails(RTTEST hTest, const char *pszFormat, ...)
1102{
1103 va_list va;
1104 va_start(va, pszFormat);
1105 int cch = RTTestFailureDetailsV(hTest, pszFormat, va);
1106 va_end(va);
1107 return cch;
1108}
1109
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