2016-08-25 16:40:23 +08:00
|
|
|
/* Miniature re-implementation of the "check" library.
|
|
|
|
*
|
|
|
|
* This is intended to support just enough of check to run the Expat
|
|
|
|
* tests. This interface is based entirely on the portion of the
|
|
|
|
* check library being used.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <setjmp.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "internal.h" /* for UNUSED_P only */
|
|
|
|
#include "minicheck.h"
|
|
|
|
|
|
|
|
Suite *
|
|
|
|
suite_create(const char *name)
|
|
|
|
{
|
|
|
|
Suite *suite = (Suite *) calloc(1, sizeof(Suite));
|
|
|
|
if (suite != NULL) {
|
|
|
|
suite->name = name;
|
|
|
|
}
|
|
|
|
return suite;
|
|
|
|
}
|
|
|
|
|
|
|
|
TCase *
|
|
|
|
tcase_create(const char *name)
|
|
|
|
{
|
|
|
|
TCase *tc = (TCase *) calloc(1, sizeof(TCase));
|
|
|
|
if (tc != NULL) {
|
|
|
|
tc->name = name;
|
|
|
|
}
|
|
|
|
return tc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
suite_add_tcase(Suite *suite, TCase *tc)
|
|
|
|
{
|
|
|
|
assert(suite != NULL);
|
|
|
|
assert(tc != NULL);
|
|
|
|
assert(tc->next_tcase == NULL);
|
|
|
|
|
|
|
|
tc->next_tcase = suite->tests;
|
|
|
|
suite->tests = tc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tcase_add_checked_fixture(TCase *tc,
|
|
|
|
tcase_setup_function setup,
|
|
|
|
tcase_teardown_function teardown)
|
|
|
|
{
|
|
|
|
assert(tc != NULL);
|
|
|
|
tc->setup = setup;
|
|
|
|
tc->teardown = teardown;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tcase_add_test(TCase *tc, tcase_test_function test)
|
|
|
|
{
|
|
|
|
assert(tc != NULL);
|
|
|
|
if (tc->allocated == tc->ntests) {
|
|
|
|
int nalloc = tc->allocated + 100;
|
|
|
|
size_t new_size = sizeof(tcase_test_function) * nalloc;
|
|
|
|
tcase_test_function *new_tests = realloc(tc->tests, new_size);
|
|
|
|
assert(new_tests != NULL);
|
|
|
|
if (new_tests != tc->tests) {
|
|
|
|
free(tc->tests);
|
|
|
|
tc->tests = new_tests;
|
|
|
|
}
|
|
|
|
tc->allocated = nalloc;
|
|
|
|
}
|
|
|
|
tc->tests[tc->ntests] = test;
|
|
|
|
tc->ntests++;
|
|
|
|
}
|
|
|
|
|
|
|
|
SRunner *
|
|
|
|
srunner_create(Suite *suite)
|
|
|
|
{
|
|
|
|
SRunner *runner = calloc(1, sizeof(SRunner));
|
|
|
|
if (runner != NULL) {
|
|
|
|
runner->suite = suite;
|
|
|
|
}
|
|
|
|
return runner;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jmp_buf env;
|
|
|
|
|
|
|
|
static char const *_check_current_function = NULL;
|
|
|
|
static int _check_current_lineno = -1;
|
|
|
|
static char const *_check_current_filename = NULL;
|
|
|
|
|
|
|
|
void
|
|
|
|
_check_set_test_info(char const *function, char const *filename, int lineno)
|
|
|
|
{
|
|
|
|
_check_current_function = function;
|
|
|
|
_check_current_lineno = lineno;
|
|
|
|
_check_current_filename = filename;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_failure(SRunner *runner, int verbosity)
|
|
|
|
{
|
|
|
|
runner->nfailures++;
|
|
|
|
if (verbosity >= CK_VERBOSE) {
|
|
|
|
printf("%s:%d: %s\n", _check_current_filename,
|
|
|
|
_check_current_lineno, _check_current_function);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-16 15:42:38 +11:00
|
|
|
static void run_test(SRunner *runner, int verbosity, TCase *tc, int i)
|
|
|
|
{
|
|
|
|
if (tc->setup != NULL) {
|
|
|
|
/* setup */
|
|
|
|
if (setjmp(env)) {
|
|
|
|
add_failure(runner, verbosity);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
tc->setup();
|
|
|
|
}
|
|
|
|
/* test */
|
|
|
|
if (setjmp(env)) {
|
|
|
|
add_failure(runner, verbosity);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
(tc->tests[i])();
|
|
|
|
|
|
|
|
/* teardown */
|
|
|
|
if (tc->teardown != NULL) {
|
|
|
|
if (setjmp(env)) {
|
|
|
|
add_failure(runner, verbosity);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
tc->teardown();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-25 16:40:23 +08:00
|
|
|
void
|
|
|
|
srunner_run_all(SRunner *runner, int verbosity)
|
|
|
|
{
|
|
|
|
assert(runner != NULL);
|
2016-11-16 15:42:38 +11:00
|
|
|
assert(runner->suite != NULL);
|
|
|
|
TCase *tc = runner->suite->tests;
|
2016-08-25 16:40:23 +08:00
|
|
|
while (tc != NULL) {
|
2016-11-16 15:42:38 +11:00
|
|
|
for (int i = 0; i < tc->ntests; ++i) {
|
2016-08-25 16:40:23 +08:00
|
|
|
runner->nchecks++;
|
2016-11-16 15:42:38 +11:00
|
|
|
run_test(runner, verbosity, tc, i);
|
|
|
|
tc = tc->next_tcase;
|
2016-08-25 16:40:23 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (verbosity) {
|
|
|
|
int passed = runner->nchecks - runner->nfailures;
|
|
|
|
double percentage = ((double) passed) / runner->nchecks;
|
|
|
|
int display = (int) (percentage * 100);
|
|
|
|
printf("%d%%: Checks: %d, Failed: %d\n",
|
|
|
|
display, runner->nchecks, runner->nfailures);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_fail_unless(int UNUSED_P(condition), const char *UNUSED_P(file), int UNUSED_P(line), const char *msg)
|
|
|
|
{
|
|
|
|
/* Always print the error message so it isn't lost. In this case,
|
|
|
|
we have a failure, so there's no reason to be quiet about what
|
|
|
|
it is.
|
|
|
|
*/
|
|
|
|
if (msg != NULL)
|
|
|
|
printf("%s", msg);
|
|
|
|
longjmp(env, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
srunner_ntests_failed(SRunner *runner)
|
|
|
|
{
|
|
|
|
assert(runner != NULL);
|
|
|
|
return runner->nfailures;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
srunner_free(SRunner *runner)
|
|
|
|
{
|
|
|
|
free(runner->suite);
|
|
|
|
free(runner);
|
|
|
|
}
|