264 lines
6.8 KiB
C
264 lines
6.8 KiB
C
|
/**
|
||
|
* \file
|
||
|
*
|
||
|
* \brief Test suite core functionality
|
||
|
*
|
||
|
* Copyright (c) 2011-2015 Atmel Corporation. All rights reserved.
|
||
|
*
|
||
|
* \asf_license_start
|
||
|
*
|
||
|
* \page License
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
*
|
||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer.
|
||
|
*
|
||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer in the documentation
|
||
|
* and/or other materials provided with the distribution.
|
||
|
*
|
||
|
* 3. The name of Atmel may not be used to endorse or promote products derived
|
||
|
* from this software without specific prior written permission.
|
||
|
*
|
||
|
* 4. This software may only be redistributed and used in connection with an
|
||
|
* Atmel microcontroller product.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR IMPLIED
|
||
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||
|
* EXPRESSLY AND SPECIFICALLY DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR
|
||
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||
|
*
|
||
|
* \asf_license_stop
|
||
|
*
|
||
|
*/
|
||
|
/*
|
||
|
* Support and FAQ: visit <a href="http://www.atmel.com/design-support/">Atmel Support</a>
|
||
|
*/
|
||
|
|
||
|
#include <compiler.h>
|
||
|
#include <setjmp.h>
|
||
|
#include <stdarg.h>
|
||
|
#include <stdio.h>
|
||
|
#include "suite.h"
|
||
|
|
||
|
/**
|
||
|
* \weakgroup test_suite_group
|
||
|
* @{
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Data pointer for test cases
|
||
|
*
|
||
|
* \see test_set_data(), test_get_data()
|
||
|
*/
|
||
|
void *test_priv_data;
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Context saved before executing a test or fixture function.
|
||
|
*
|
||
|
* This is used for doing non-local jumps from the test cases on failure.
|
||
|
*/
|
||
|
static jmp_buf test_failure_jmpbuf;
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Pointer to current test case
|
||
|
*
|
||
|
* \see test_set_case(), test_get_case()
|
||
|
*/
|
||
|
struct test_case *test_case_ptr = NULL;
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Report a failing test stage
|
||
|
*
|
||
|
* \param test Current test case.
|
||
|
* \param stage Name of test stage that failed.
|
||
|
* \param result Result value of test stage.
|
||
|
*/
|
||
|
static void test_report_failure(const struct test_case *test, const char *stage,
|
||
|
int result)
|
||
|
{
|
||
|
dbg_error("Test '%s' failed during '%s': %d\r\n", test->name, stage,
|
||
|
result);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Call a test or fixture function
|
||
|
*
|
||
|
* This function will initialize \ref test_failure_jmpbuf and call \a func
|
||
|
* with \a test as the parameter.
|
||
|
*
|
||
|
* \param func Pointer to function to call.
|
||
|
* \param test_case Pointer to the function's originating test case.
|
||
|
*
|
||
|
* \return #TEST_PASS if \a func was executed successfully, or the
|
||
|
* result value passed to test_fail() on failure.
|
||
|
*/
|
||
|
static int test_call(void (*func)(const struct test_case *),
|
||
|
const struct test_case *test)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
|
||
|
if (!func) {
|
||
|
return TEST_PASS;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* The first call to setjmp() always return 0. However, if the
|
||
|
* call to func() below goes wrong, we'll return here again with
|
||
|
* a nonzero value.
|
||
|
*/
|
||
|
ret = setjmp(test_failure_jmpbuf);
|
||
|
if (ret) {
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
func(test);
|
||
|
|
||
|
return TEST_PASS;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Run a test case
|
||
|
*
|
||
|
* This function will call the \e setup, \e run and \e cleanup functions of the
|
||
|
* specified test case, outputting debug information as it goes, then returning
|
||
|
* the result value of the test case.
|
||
|
*
|
||
|
* If the setup stage does not pass, the rest of the test case if skipped.
|
||
|
* If the test stage itself does not pass, the cleanup is still executed.
|
||
|
*
|
||
|
* The result from the first non-passing function in the test case is returned,
|
||
|
* meaning a failing cleanup will override a successful test.
|
||
|
*
|
||
|
* \param test_case Test case to run.
|
||
|
*
|
||
|
* \return \ref TEST_PASS if all functions execute successfully. Otherwise, the
|
||
|
* first non-zero value which is returned from the test case.
|
||
|
*/
|
||
|
static int test_case_run(const struct test_case *test)
|
||
|
{
|
||
|
int result;
|
||
|
|
||
|
// Store test pointer for access by test_get_data()
|
||
|
#if defined(_ASSERT_ENABLE_) && defined(TEST_SUITE_DEFINE_ASSERT_MACRO)
|
||
|
test_set_case(test);
|
||
|
#endif
|
||
|
|
||
|
dbg_info("Running test: %s\r\n", test->name);
|
||
|
if (test->setup) {
|
||
|
int ret;
|
||
|
dbg("Setting up fixture\r\n");
|
||
|
ret = test_call(test->setup, test);
|
||
|
if (ret) {
|
||
|
test_report_failure(test, "setup", ret);
|
||
|
return ret;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
result = test_call(test->run, test);
|
||
|
if (result) {
|
||
|
test_report_failure(test, "test", result);
|
||
|
}
|
||
|
|
||
|
if (test->cleanup) {
|
||
|
int ret;
|
||
|
dbg("Cleaning up fixture\r\n");
|
||
|
ret = test_call(test->cleanup, test);
|
||
|
if (ret && !result) {
|
||
|
test_report_failure(test, "cleanup", ret);
|
||
|
result = ret;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Test is done, clear the stored test pointer
|
||
|
#if defined(_ASSERT_ENABLE_) && defined(TEST_SUITE_DEFINE_ASSERT_MACRO)
|
||
|
test_set_case(NULL);
|
||
|
#endif
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \internal
|
||
|
* \brief Report a failure and jump out of current test case function
|
||
|
*
|
||
|
* \param test Current test case.
|
||
|
* \param result Result value of failure. \note Should not be \ref TEST_PASS.
|
||
|
* \param file Name of file in which function resides.
|
||
|
* \param line Line number at which failure occurred.
|
||
|
* \param fmt Failure message, as printf-formatted string.
|
||
|
* \param ... Values to insert into failure message.
|
||
|
*/
|
||
|
void test_case_fail(const struct test_case *test, int result,
|
||
|
const char *file, unsigned int line,
|
||
|
const char *fmt, ...)
|
||
|
{
|
||
|
va_list ap;
|
||
|
|
||
|
dbg_error("Test '%s' failed at %s:%u:\r\n\t", test->name, file, line);
|
||
|
|
||
|
va_start(ap, fmt);
|
||
|
dbg_vprintf_pgm(fmt, ap);
|
||
|
va_end(ap);
|
||
|
dbg_putchar('\r');
|
||
|
dbg_putchar('\n');
|
||
|
|
||
|
/*
|
||
|
* This will cause the setjmp() call in test_call() to return
|
||
|
* TEST_FAIL.
|
||
|
*/
|
||
|
longjmp(test_failure_jmpbuf, result);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* \brief Run a test suite
|
||
|
*
|
||
|
* Run all tests in \a suite, in the order in which they are found in
|
||
|
* the array.
|
||
|
*
|
||
|
* \return The number of tests that didn't pass.
|
||
|
*/
|
||
|
int test_suite_run(const struct test_suite *suite)
|
||
|
{
|
||
|
unsigned int nr_failures = 0;
|
||
|
unsigned int nr_errors = 0;
|
||
|
unsigned int i;
|
||
|
int ret;
|
||
|
|
||
|
dbg_info("Running test suite '%s'...\r\n", suite->name);
|
||
|
|
||
|
for (i = 0; i < suite->nr_tests; i++) {
|
||
|
const struct test_case *test;
|
||
|
|
||
|
test = suite->tests[i];
|
||
|
ret = test_case_run(test);
|
||
|
if (ret < TEST_PASS) {
|
||
|
nr_errors++;
|
||
|
} else if (ret > TEST_PASS) {
|
||
|
nr_failures++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dbg_info("Test suite '%s' complete: %u tests, %u failures, %u errors\r\n\r\n",
|
||
|
suite->name, suite->nr_tests, nr_failures, nr_errors);
|
||
|
|
||
|
return nr_errors + nr_failures;
|
||
|
}
|
||
|
|
||
|
//! @}
|