source: mainline/uspace/lib/pcut/src/run.c@ 0b63dc2

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 0b63dc2 was 4b54bd9, checked in by Vojtech Horky <vojtech.horky@…>, 7 years ago

Update PCUT to latest revision

  • Property mode set to 100644
File size: 7.3 KB
RevLine 
[01579ad]1/*
2 * Copyright (c) 2012-2013 Vojtech Horky
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @file
30 *
31 * Test execution routines.
32 */
33
34#include "internal.h"
[4b54bd9]35
[01579ad]36#ifndef PCUT_NO_LONG_JUMP
[4b54bd9]37#pragma warning(push, 0)
[01579ad]38#include <setjmp.h>
[4b54bd9]39#pragma warning(pop)
[01579ad]40#endif
41
42#ifndef PCUT_NO_LONG_JUMP
43/** Long-jump buffer. */
44static jmp_buf start_test_jump;
45#endif
46
47/** Whether to run a tear-down function on a failure.
48 *
49 * Used to determine whether we are already in a tear-down context.
50 */
51static int execute_teardown_on_failure;
52
53/** Whether to report test result at all.
54 *
55 * Used to determine whether we are the forked or the parent process.
56 */
57static int report_test_result;
58
59/** Whether to print test error.
60 *
61 * Used to determine whether we are the forked or the parent process.
62 */
63static int print_test_error;
64
65/** Whether leaving a test means a process exit. */
66static int leave_means_exit;
67
68/** Pointer to currently running test. */
69static pcut_item_t *current_test = NULL;
70
71/** Pointer to current test suite. */
72static pcut_item_t *current_suite = NULL;
73
74/** A NULL-like suite. */
[9b20126]75static pcut_item_t default_suite;
76static int default_suite_initialized = 0;
77
[4b54bd9]78static void init_default_suite_when_needed() {
[9eb1ff5]79 if (default_suite_initialized) {
[9b20126]80 return;
[9eb1ff5]81 }
[9b20126]82 default_suite.id = -1;
83 default_suite.kind = PCUT_KIND_TESTSUITE;
84 default_suite.previous = NULL;
85 default_suite.next = NULL;
86 default_suite.name = "Default";
87 default_suite.setup_func = NULL;
88 default_suite.teardown_func = NULL;
89}
[01579ad]90
91/** Find the suite given test belongs to.
92 *
93 * @param it The test.
94 * @return Always a valid test suite item.
95 */
[4b54bd9]96static pcut_item_t *pcut_find_parent_suite(pcut_item_t *it) {
[01579ad]97 while (it != NULL) {
[9eb1ff5]98 if (it->kind == PCUT_KIND_TESTSUITE) {
[01579ad]99 return it;
[9eb1ff5]100 }
[01579ad]101 it = it->previous;
102 }
[9b20126]103 init_default_suite_when_needed();
[01579ad]104 return &default_suite;
105}
106
107/** Run a set-up (tear-down) function.
108 *
109 * @param func Function to run (can be NULL).
110 */
[4b54bd9]111static void run_setup_teardown(pcut_setup_func_t func) {
[9eb1ff5]112 if (func != NULL) {
[01579ad]113 func();
[9eb1ff5]114 }
[01579ad]115}
116
117/** Terminate current test with given outcome.
118 *
119 * @warning This function may execute a long jump or terminate
120 * current process.
121 *
122 * @param outcome Outcome of the current test.
123 */
[4b54bd9]124static void leave_test(int outcome) {
[9b20126]125 PCUT_DEBUG("leave_test(outcome=%d), will_exit=%s", outcome,
[4b54bd9]126 leave_means_exit ? "yes" : "no");
[9eb1ff5]127 if (leave_means_exit) {
[01579ad]128 exit(outcome);
[9eb1ff5]129 }
130
[01579ad]131#ifndef PCUT_NO_LONG_JUMP
132 longjmp(start_test_jump, 1);
133#endif
134}
135
136/** Process a failed assertion.
137 *
138 * @warning This function calls leave_test() and typically will not
139 * return.
140 *
141 * @param message Message describing the failure.
142 */
[4b54bd9]143void pcut_failed_assertion(const char *message) {
[01579ad]144 static const char *prev_message = NULL;
145 /*
146 * The assertion failed. We need to abort the current test,
147 * inform the user and perform some clean-up. That could
148 * include running the tear-down routine.
149 */
[9eb1ff5]150 if (print_test_error) {
[01579ad]151 pcut_print_fail_message(message);
[9eb1ff5]152 }
153
[01579ad]154 if (execute_teardown_on_failure) {
155 execute_teardown_on_failure = 0;
156 prev_message = message;
[9b20126]157 run_setup_teardown(current_suite->teardown_func);
[9eb1ff5]158
[01579ad]159 /* Tear-down was okay. */
160 if (report_test_result) {
[9eb1ff5]161 pcut_report_test_done(current_test, PCUT_OUTCOME_FAIL,
[4b54bd9]162 message, NULL, NULL);
[01579ad]163 }
164 } else {
165 if (report_test_result) {
[9eb1ff5]166 pcut_report_test_done(current_test, PCUT_OUTCOME_FAIL,
[4b54bd9]167 prev_message, message, NULL);
[01579ad]168 }
169 }
[9eb1ff5]170
[01579ad]171 prev_message = NULL;
[9eb1ff5]172
173 leave_test(PCUT_OUTCOME_FAIL); /* No return. */
[01579ad]174}
175
176/** Run a test.
177 *
178 * @param test Test to execute.
179 * @return Error status (zero means success).
180 */
[4b54bd9]181static int run_test(pcut_item_t *test) {
[01579ad]182 /*
183 * Set here as the returning point in case of test failure.
184 * If we get here, it means something failed during the
185 * test execution.
186 */
187#ifndef PCUT_NO_LONG_JUMP
188 int test_finished = setjmp(start_test_jump);
[9eb1ff5]189 if (test_finished) {
190 return PCUT_OUTCOME_FAIL;
191 }
[01579ad]192#endif
[9eb1ff5]193
194 if (report_test_result) {
[01579ad]195 pcut_report_test_start(test);
[9eb1ff5]196 }
197
[01579ad]198 current_suite = pcut_find_parent_suite(test);
199 current_test = test;
[9eb1ff5]200
[9b20126]201 pcut_hook_before_test(test);
[9eb1ff5]202
[01579ad]203 /*
204 * If anything goes wrong, execute the tear-down function
205 * as well.
206 */
207 execute_teardown_on_failure = 1;
[9eb1ff5]208
[01579ad]209 /*
210 * Run the set-up function.
211 */
[9b20126]212 run_setup_teardown(current_suite->setup_func);
[9eb1ff5]213
[01579ad]214 /*
215 * The setup function was performed, it is time to run
216 * the actual test.
217 */
[9b20126]218 test->test_func();
[9eb1ff5]219
[01579ad]220 /*
221 * Finally, run the tear-down function. We need to clear
222 * the flag to prevent endless loop.
223 */
224 execute_teardown_on_failure = 0;
[9b20126]225 run_setup_teardown(current_suite->teardown_func);
[9eb1ff5]226
[01579ad]227 /*
228 * If we got here, it means everything went well with
229 * this test.
230 */
[9eb1ff5]231 if (report_test_result) {
232 pcut_report_test_done(current_test, PCUT_OUTCOME_PASS,
[4b54bd9]233 NULL, NULL, NULL);
[9eb1ff5]234 }
235
236 return PCUT_OUTCOME_PASS;
[01579ad]237}
238
239/** Run a test in a forked mode.
240 *
241 * Forked mode means that the caller of the test is already a new
242 * process running this test only.
243 *
244 * @param test Test to execute.
245 * @return Error status (zero means success).
246 */
[4b54bd9]247int pcut_run_test_forked(pcut_item_t *test) {
[9eb1ff5]248 int rc;
249
[01579ad]250 report_test_result = 0;
251 print_test_error = 1;
252 leave_means_exit = 1;
[9eb1ff5]253
254 rc = run_test(test);
255
[01579ad]256 current_test = NULL;
257 current_suite = NULL;
[9eb1ff5]258
[01579ad]259 return rc;
260}
261
262/** Run a test in a single mode.
263 *
264 * Single mode means that the test is called in the context of the
265 * parent process, that is no new process is forked.
266 *
267 * @param test Test to execute.
268 * @return Error status (zero means success).
269 */
[4b54bd9]270int pcut_run_test_single(pcut_item_t *test) {
[9eb1ff5]271 int rc;
272
[01579ad]273 report_test_result = 1;
274 print_test_error = 0;
275 leave_means_exit = 0;
[9eb1ff5]276
277 rc = run_test(test);
278
[01579ad]279 current_test = NULL;
280 current_suite = NULL;
[9eb1ff5]281
[01579ad]282 return rc;
283}
284
[134ac5d]285/** Tells time-out length for a given test.
286 *
287 * @param test Test for which the time-out is questioned.
288 * @return Timeout in seconds.
289 */
[4b54bd9]290int pcut_get_test_timeout(pcut_item_t *test) {
[134ac5d]291 int timeout = PCUT_DEFAULT_TEST_TIMEOUT;
[9b20126]292 pcut_extra_t *extras = test->extras;
[9eb1ff5]293
294
[134ac5d]295 while (extras->type != PCUT_EXTRA_LAST) {
[9eb1ff5]296 if (extras->type == PCUT_EXTRA_TIMEOUT) {
[134ac5d]297 timeout = extras->timeout;
[9eb1ff5]298 }
[134ac5d]299 extras++;
300 }
[9eb1ff5]301
[134ac5d]302 return timeout;
303}
Note: See TracBrowser for help on using the repository browser.