2 * Copyright (c) 2012-2013 Vojtech Horky
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
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.
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.
31 * Test execution routines.
36 #ifndef PCUT_NO_LONG_JUMP
37 #pragma warning(push, 0)
42 #ifndef PCUT_NO_LONG_JUMP
43 /** Long-jump buffer. */
44 static jmp_buf start_test_jump
;
47 /** Whether to run a tear-down function on a failure.
49 * Used to determine whether we are already in a tear-down context.
51 static int execute_teardown_on_failure
;
53 /** Whether to report test result at all.
55 * Used to determine whether we are the forked or the parent process.
57 static int report_test_result
;
59 /** Whether to print test error.
61 * Used to determine whether we are the forked or the parent process.
63 static int print_test_error
;
65 /** Whether leaving a test means a process exit. */
66 static int leave_means_exit
;
68 /** Pointer to currently running test. */
69 static pcut_item_t
*current_test
= NULL
;
71 /** Pointer to current test suite. */
72 static pcut_item_t
*current_suite
= NULL
;
74 /** A NULL-like suite. */
75 static pcut_item_t default_suite
;
76 static int default_suite_initialized
= 0;
78 static void init_default_suite_when_needed() {
79 if (default_suite_initialized
) {
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
;
91 /** Find the suite given test belongs to.
94 * @return Always a valid test suite item.
96 static pcut_item_t
*pcut_find_parent_suite(pcut_item_t
*it
) {
98 if (it
->kind
== PCUT_KIND_TESTSUITE
) {
103 init_default_suite_when_needed();
104 return &default_suite
;
107 /** Run a set-up (tear-down) function.
109 * @param func Function to run (can be NULL).
111 static void run_setup_teardown(pcut_setup_func_t func
) {
117 /** Terminate current test with given outcome.
119 * @warning This function may execute a long jump or terminate
122 * @param outcome Outcome of the current test.
124 static void leave_test(int outcome
) {
125 PCUT_DEBUG("leave_test(outcome=%d), will_exit=%s", outcome
,
126 leave_means_exit
? "yes" : "no");
127 if (leave_means_exit
) {
131 #ifndef PCUT_NO_LONG_JUMP
132 longjmp(start_test_jump
, 1);
136 /** Process a failed assertion.
138 * @warning This function calls leave_test() and typically will not
141 * @param message Message describing the failure.
143 void pcut_failed_assertion(const char *message
) {
144 static const char *prev_message
= NULL
;
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.
150 if (print_test_error
) {
151 pcut_print_fail_message(message
);
154 if (execute_teardown_on_failure
) {
155 execute_teardown_on_failure
= 0;
156 prev_message
= message
;
157 run_setup_teardown(current_suite
->teardown_func
);
159 /* Tear-down was okay. */
160 if (report_test_result
) {
161 pcut_report_test_done(current_test
, PCUT_OUTCOME_FAIL
,
162 message
, NULL
, NULL
);
165 if (report_test_result
) {
166 pcut_report_test_done(current_test
, PCUT_OUTCOME_FAIL
,
167 prev_message
, message
, NULL
);
173 leave_test(PCUT_OUTCOME_FAIL
); /* No return. */
178 * @param test Test to execute.
179 * @return Error status (zero means success).
181 static int run_test(pcut_item_t
*test
) {
183 * Set here as the returning point in case of test failure.
184 * If we get here, it means something failed during the
187 #ifndef PCUT_NO_LONG_JUMP
188 int test_finished
= setjmp(start_test_jump
);
190 return PCUT_OUTCOME_FAIL
;
194 if (report_test_result
) {
195 pcut_report_test_start(test
);
198 current_suite
= pcut_find_parent_suite(test
);
201 pcut_hook_before_test(test
);
204 * If anything goes wrong, execute the tear-down function
207 execute_teardown_on_failure
= 1;
210 * Run the set-up function.
212 run_setup_teardown(current_suite
->setup_func
);
215 * The setup function was performed, it is time to run
221 * Finally, run the tear-down function. We need to clear
222 * the flag to prevent endless loop.
224 execute_teardown_on_failure
= 0;
225 run_setup_teardown(current_suite
->teardown_func
);
228 * If we got here, it means everything went well with
231 if (report_test_result
) {
232 pcut_report_test_done(current_test
, PCUT_OUTCOME_PASS
,
236 return PCUT_OUTCOME_PASS
;
239 /** Run a test in a forked mode.
241 * Forked mode means that the caller of the test is already a new
242 * process running this test only.
244 * @param test Test to execute.
245 * @return Error status (zero means success).
247 int pcut_run_test_forked(pcut_item_t
*test
) {
250 report_test_result
= 0;
251 print_test_error
= 1;
252 leave_means_exit
= 1;
257 current_suite
= NULL
;
262 /** Run a test in a single mode.
264 * Single mode means that the test is called in the context of the
265 * parent process, that is no new process is forked.
267 * @param test Test to execute.
268 * @return Error status (zero means success).
270 int pcut_run_test_single(pcut_item_t
*test
) {
273 report_test_result
= 1;
274 print_test_error
= 0;
275 leave_means_exit
= 0;
280 current_suite
= NULL
;
285 /** Tells time-out length for a given test.
287 * @param test Test for which the time-out is questioned.
288 * @return Timeout in seconds.
290 int pcut_get_test_timeout(pcut_item_t
*test
) {
291 int timeout
= PCUT_DEFAULT_TEST_TIMEOUT
;
292 pcut_extra_t
*extras
= test
->extras
;
295 while (extras
->type
!= PCUT_EXTRA_LAST
) {
296 if (extras
->type
== PCUT_EXTRA_TIMEOUT
) {
297 timeout
= extras
->timeout
;