Blame tests/clar.c

Packit Service 20376f
/*
Packit Service 20376f
 * Copyright (c) Vicent Marti. All rights reserved.
Packit Service 20376f
 *
Packit Service 20376f
 * This file is part of clar, distributed under the ISC license.
Packit Service 20376f
 * For full terms see the included COPYING file.
Packit Service 20376f
 */
Packit Service 20376f
#include <assert.h>
Packit Service 20376f
#include <setjmp.h>
Packit Service 20376f
#include <stdlib.h>
Packit Service 20376f
#include <stdio.h>
Packit Service 20376f
#include <string.h>
Packit Service 20376f
#include <math.h>
Packit Service 20376f
#include <stdarg.h>
Packit Service 20376f
#include <wchar.h>
Packit Service 20376f
Packit Service 20376f
/* required for sandboxing */
Packit Service 20376f
#include <sys/types.h>
Packit Service 20376f
#include <sys/stat.h>
Packit Service 20376f
Packit Service 20376f
#ifdef _WIN32
Packit Service 20376f
#	include <windows.h>
Packit Service 20376f
#	include <io.h>
Packit Service 20376f
#	include <shellapi.h>
Packit Service 20376f
#	include <direct.h>
Packit Service 20376f
Packit Service 20376f
#	define _MAIN_CC __cdecl
Packit Service 20376f
Packit Service 20376f
#	ifndef stat
Packit Service 20376f
#		define stat(path, st) _stat(path, st)
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef mkdir
Packit Service 20376f
#		define mkdir(path, mode) _mkdir(path)
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef chdir
Packit Service 20376f
#		define chdir(path) _chdir(path)
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef access
Packit Service 20376f
#		define access(path, mode) _access(path, mode)
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef strdup
Packit Service 20376f
#		define strdup(str) _strdup(str)
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef strcasecmp
Packit Service 20376f
#		define strcasecmp(a,b) _stricmp(a,b)
Packit Service 20376f
#	endif
Packit Service 20376f
Packit Service 20376f
#	ifndef __MINGW32__
Packit Service 20376f
#		pragma comment(lib, "shell32")
Packit Service 20376f
#		ifndef strncpy
Packit Service 20376f
#			define strncpy(to, from, to_size) strncpy_s(to, to_size, from, _TRUNCATE)
Packit Service 20376f
#		endif
Packit Service 20376f
#		ifndef W_OK
Packit Service 20376f
#			define W_OK 02
Packit Service 20376f
#		endif
Packit Service 20376f
#		ifndef S_ISDIR
Packit Service 20376f
#			define S_ISDIR(x) ((x & _S_IFDIR) != 0)
Packit Service 20376f
#		endif
Packit Service 20376f
#		define p_snprintf(buf,sz,fmt,...) _snprintf_s(buf,sz,_TRUNCATE,fmt,__VA_ARGS__)
Packit Service 20376f
#	else
Packit Service 20376f
#		define p_snprintf snprintf
Packit Service 20376f
#	endif
Packit Service 20376f
Packit Service 20376f
#	ifndef PRIuZ
Packit Service 20376f
#		define PRIuZ "Iu"
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef PRIxZ
Packit Service 20376f
#		define PRIxZ "Ix"
Packit Service 20376f
#	endif
Packit Service 20376f
Packit Service 20376f
#	ifdef _MSC_VER
Packit Service 20376f
	typedef struct stat STAT_T;
Packit Service 20376f
#	else
Packit Service 20376f
	typedef struct _stat STAT_T;
Packit Service 20376f
#	endif
Packit Service 20376f
#else
Packit Service 20376f
#	include <sys/wait.h> /* waitpid(2) */
Packit Service 20376f
#	include <unistd.h>
Packit Service 20376f
#	define _MAIN_CC
Packit Service 20376f
#	define p_snprintf snprintf
Packit Service 20376f
#	ifndef PRIuZ
Packit Service 20376f
#		define PRIuZ "zu"
Packit Service 20376f
#	endif
Packit Service 20376f
#	ifndef PRIxZ
Packit Service 20376f
#		define PRIxZ "zx"
Packit Service 20376f
#	endif
Packit Service 20376f
	typedef struct stat STAT_T;
Packit Service 20376f
#endif
Packit Service 20376f
Packit Service 20376f
#include "clar.h"
Packit Service 20376f
Packit Service 20376f
static void fs_rm(const char *_source);
Packit Service 20376f
static void fs_copy(const char *_source, const char *dest);
Packit Service 20376f
Packit Service 20376f
static const char *
Packit Service 20376f
fixture_path(const char *base, const char *fixture_name);
Packit Service 20376f
Packit Service 20376f
struct clar_error {
Packit Service 20376f
	const char *file;
Packit Service 20376f
	int line_number;
Packit Service 20376f
	const char *error_msg;
Packit Service 20376f
	char *description;
Packit Service 20376f
Packit Service 20376f
	struct clar_error *next;
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
struct clar_explicit {
Packit Service 20376f
	size_t suite_idx;
Packit Service 20376f
	const char *filter;
Packit Service 20376f
Packit Service 20376f
	struct clar_explicit *next;
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
struct clar_report {
Packit Service 20376f
	const char *test;
Packit Service 20376f
	int test_number;
Packit Service 20376f
	const char *suite;
Packit Service 20376f
Packit Service 20376f
	enum cl_test_status status;
Packit Service 20376f
Packit Service 20376f
	struct clar_error *errors;
Packit Service 20376f
	struct clar_error *last_error;
Packit Service 20376f
Packit Service 20376f
	struct clar_report *next;
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
struct clar_summary {
Packit Service 20376f
	const char *filename;
Packit Service 20376f
	FILE *fp;
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
static struct {
Packit Service 20376f
	enum cl_test_status test_status;
Packit Service 20376f
Packit Service 20376f
	const char *active_test;
Packit Service 20376f
	const char *active_suite;
Packit Service 20376f
Packit Service 20376f
	int total_skipped;
Packit Service 20376f
	int total_errors;
Packit Service 20376f
Packit Service 20376f
	int tests_ran;
Packit Service 20376f
	int suites_ran;
Packit Service 20376f
Packit Service 20376f
	int report_errors_only;
Packit Service 20376f
	int exit_on_error;
Packit Service 20376f
	int report_suite_names;
Packit Service 20376f
Packit Service 20376f
	int write_summary;
Packit Service 20376f
	const char *summary_filename;
Packit Service 20376f
	struct clar_summary *summary;
Packit Service 20376f
Packit Service 20376f
	struct clar_explicit *explicit;
Packit Service 20376f
	struct clar_explicit *last_explicit;
Packit Service 20376f
Packit Service 20376f
	struct clar_report *reports;
Packit Service 20376f
	struct clar_report *last_report;
Packit Service 20376f
Packit Service 20376f
	void (*local_cleanup)(void *);
Packit Service 20376f
	void *local_cleanup_payload;
Packit Service 20376f
Packit Service 20376f
	jmp_buf trampoline;
Packit Service 20376f
	int trampoline_enabled;
Packit Service 20376f
Packit Service 20376f
	cl_trace_cb *pfn_trace_cb;
Packit Service 20376f
	void *trace_payload;
Packit Service 20376f
Packit Service 20376f
} _clar;
Packit Service 20376f
Packit Service 20376f
struct clar_func {
Packit Service 20376f
	const char *name;
Packit Service 20376f
	void (*ptr)(void);
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
struct clar_suite {
Packit Service 20376f
	const char *name;
Packit Service 20376f
	struct clar_func initialize;
Packit Service 20376f
	struct clar_func cleanup;
Packit Service 20376f
	const struct clar_func *tests;
Packit Service 20376f
	size_t test_count;
Packit Service 20376f
	int enabled;
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
/* From clar_print_*.c */
Packit Service 20376f
static void clar_print_init(int test_count, int suite_count, const char *suite_names);
Packit Service 20376f
static void clar_print_shutdown(int test_count, int suite_count, int error_count);
Packit Service 20376f
static void clar_print_error(int num, const struct clar_report *report, const struct clar_error *error);
Packit Service 20376f
static void clar_print_ontest(const char *test_name, int test_number, enum cl_test_status failed);
Packit Service 20376f
static void clar_print_onsuite(const char *suite_name, int suite_index);
Packit Service 20376f
static void clar_print_onabort(const char *msg, ...);
Packit Service 20376f
Packit Service 20376f
/* From clar_sandbox.c */
Packit Service 20376f
static void clar_unsandbox(void);
Packit Service 20376f
static int clar_sandbox(void);
Packit Service 20376f
Packit Service 20376f
/* From summary.h */
Packit Service 20376f
static struct clar_summary *clar_summary_init(const char *filename);
Packit Service 20376f
static int clar_summary_shutdown(struct clar_summary *fp);
Packit Service 20376f
Packit Service 20376f
/* Load the declarations for the test suite */
Packit Service 20376f
#include "clar.suite"
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
#define CL_TRACE(ev)													\
Packit Service 20376f
	do {																\
Packit Service 20376f
		if (_clar.pfn_trace_cb)											\
Packit Service 20376f
			_clar.pfn_trace_cb(ev,										\
Packit Service 20376f
							   _clar.active_suite,						\
Packit Service 20376f
							   _clar.active_test,						\
Packit Service 20376f
							   _clar.trace_payload);					\
Packit Service 20376f
	} while (0)
Packit Service 20376f
Packit Service 20376f
void cl_trace_register(cl_trace_cb *cb, void *payload)
Packit Service 20376f
{
Packit Service 20376f
	_clar.pfn_trace_cb = cb;
Packit Service 20376f
	_clar.trace_payload = payload;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
/* Core test functions */
Packit Service 20376f
static void
Packit Service 20376f
clar_report_errors(struct clar_report *report)
Packit Service 20376f
{
Packit Service 20376f
	struct clar_error *error;
Packit Service 20376f
	int i = 1;
Packit Service 20376f
Packit Service 20376f
	for (error = report->errors; error; error = error->next)
Packit Service 20376f
		clar_print_error(i++, _clar.last_report, error);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
clar_report_all(void)
Packit Service 20376f
{
Packit Service 20376f
	struct clar_report *report;
Packit Service 20376f
	struct clar_error *error;
Packit Service 20376f
	int i = 1;
Packit Service 20376f
Packit Service 20376f
	for (report = _clar.reports; report; report = report->next) {
Packit Service 20376f
		if (report->status != CL_TEST_FAILURE)
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		for (error = report->errors; error; error = error->next)
Packit Service 20376f
			clar_print_error(i++, report, error);
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
clar_run_test(
Packit Service 20376f
	const struct clar_func *test,
Packit Service 20376f
	const struct clar_func *initialize,
Packit Service 20376f
	const struct clar_func *cleanup)
Packit Service 20376f
{
Packit Service 20376f
	_clar.trampoline_enabled = 1;
Packit Service 20376f
Packit Service 20376f
	CL_TRACE(CL_TRACE__TEST__BEGIN);
Packit Service 20376f
Packit Service 20376f
	if (setjmp(_clar.trampoline) == 0) {
Packit Service 20376f
		if (initialize->ptr != NULL)
Packit Service 20376f
			initialize->ptr();
Packit Service 20376f
Packit Service 20376f
		CL_TRACE(CL_TRACE__TEST__RUN_BEGIN);
Packit Service 20376f
		test->ptr();
Packit Service 20376f
		CL_TRACE(CL_TRACE__TEST__RUN_END);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	_clar.trampoline_enabled = 0;
Packit Service 20376f
Packit Service 20376f
	if (_clar.last_report->status == CL_TEST_NOTRUN)
Packit Service 20376f
		_clar.last_report->status = CL_TEST_OK;
Packit Service 20376f
Packit Service 20376f
	if (_clar.local_cleanup != NULL)
Packit Service 20376f
		_clar.local_cleanup(_clar.local_cleanup_payload);
Packit Service 20376f
Packit Service 20376f
	if (cleanup->ptr != NULL)
Packit Service 20376f
		cleanup->ptr();
Packit Service 20376f
Packit Service 20376f
	CL_TRACE(CL_TRACE__TEST__END);
Packit Service 20376f
Packit Service 20376f
	_clar.tests_ran++;
Packit Service 20376f
Packit Service 20376f
	/* remove any local-set cleanup methods */
Packit Service 20376f
	_clar.local_cleanup = NULL;
Packit Service 20376f
	_clar.local_cleanup_payload = NULL;
Packit Service 20376f
Packit Service 20376f
	if (_clar.report_errors_only) {
Packit Service 20376f
		clar_report_errors(_clar.last_report);
Packit Service 20376f
	} else {
Packit Service 20376f
		clar_print_ontest(test->name, _clar.tests_ran, _clar.last_report->status);
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
clar_run_suite(const struct clar_suite *suite, const char *filter)
Packit Service 20376f
{
Packit Service 20376f
	const struct clar_func *test = suite->tests;
Packit Service 20376f
	size_t i, matchlen;
Packit Service 20376f
	struct clar_report *report;
Packit Service 20376f
Packit Service 20376f
	if (!suite->enabled)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	if (_clar.exit_on_error && _clar.total_errors)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	if (!_clar.report_errors_only)
Packit Service 20376f
		clar_print_onsuite(suite->name, ++_clar.suites_ran);
Packit Service 20376f
Packit Service 20376f
	_clar.active_suite = suite->name;
Packit Service 20376f
	_clar.active_test = NULL;
Packit Service 20376f
	CL_TRACE(CL_TRACE__SUITE_BEGIN);
Packit Service 20376f
Packit Service 20376f
	if (filter) {
Packit Service 20376f
		size_t suitelen = strlen(suite->name);
Packit Service 20376f
		matchlen = strlen(filter);
Packit Service 20376f
		if (matchlen <= suitelen) {
Packit Service 20376f
			filter = NULL;
Packit Service 20376f
		} else {
Packit Service 20376f
			filter += suitelen;
Packit Service 20376f
			while (*filter == ':')
Packit Service 20376f
				++filter;
Packit Service 20376f
			matchlen = strlen(filter);
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	for (i = 0; i < suite->test_count; ++i) {
Packit Service 20376f
		if (filter && strncmp(test[i].name, filter, matchlen))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		_clar.active_test = test[i].name;
Packit Service 20376f
Packit Service 20376f
		report = calloc(1, sizeof(struct clar_report));
Packit Service 20376f
		report->suite = _clar.active_suite;
Packit Service 20376f
		report->test = _clar.active_test;
Packit Service 20376f
		report->test_number = _clar.tests_ran;
Packit Service 20376f
		report->status = CL_TEST_NOTRUN;
Packit Service 20376f
Packit Service 20376f
		if (_clar.reports == NULL)
Packit Service 20376f
			_clar.reports = report;
Packit Service 20376f
Packit Service 20376f
		if (_clar.last_report != NULL)
Packit Service 20376f
			_clar.last_report->next = report;
Packit Service 20376f
Packit Service 20376f
		_clar.last_report = report;
Packit Service 20376f
Packit Service 20376f
		clar_run_test(&test[i], &suite->initialize, &suite->cleanup);
Packit Service 20376f
Packit Service 20376f
		if (_clar.exit_on_error && _clar.total_errors)
Packit Service 20376f
			return;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	_clar.active_test = NULL;
Packit Service 20376f
	CL_TRACE(CL_TRACE__SUITE_END);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
clar_usage(const char *arg)
Packit Service 20376f
{
Packit Service 20376f
	printf("Usage: %s [options]\n\n", arg);
Packit Service 20376f
	printf("Options:\n");
Packit Service 20376f
	printf("  -sname        Run only the suite with `name` (can go to individual test name)\n");
Packit Service 20376f
	printf("  -iname        Include the suite with `name`\n");
Packit Service 20376f
	printf("  -xname        Exclude the suite with `name`\n");
Packit Service 20376f
	printf("  -v            Increase verbosity (show suite names)\n");
Packit Service 20376f
	printf("  -q            Only report tests that had an error\n");
Packit Service 20376f
	printf("  -Q            Quit as soon as a test fails\n");
Packit Service 20376f
	printf("  -l            Print suite names\n");
Packit Service 20376f
	printf("  -r[filename]  Write summary file (to the optional filename)\n");
Packit Service 20376f
	exit(-1);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void
Packit Service 20376f
clar_parse_args(int argc, char **argv)
Packit Service 20376f
{
Packit Service 20376f
	int i;
Packit Service 20376f
Packit Service 20376f
	/* Verify options before execute */
Packit Service 20376f
	for (i = 1; i < argc; ++i) {
Packit Service 20376f
		char *argument = argv[i];
Packit Service 20376f
Packit Service 20376f
		if (argument[0] != '-' || argument[1] == '\0'
Packit Service 20376f
		    || strchr("sixvqQlr", argument[1]) == NULL) {
Packit Service 20376f
			clar_usage(argv[0]);
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	for (i = 1; i < argc; ++i) {
Packit Service 20376f
		char *argument = argv[i];
Packit Service 20376f
Packit Service 20376f
		switch (argument[1]) {
Packit Service 20376f
		case 's':
Packit Service 20376f
		case 'i':
Packit Service 20376f
		case 'x': { /* given suite name */
Packit Service 20376f
			int offset = (argument[2] == '=') ? 3 : 2, found = 0;
Packit Service 20376f
			char action = argument[1];
Packit Service 20376f
			size_t j, arglen, suitelen, cmplen;
Packit Service 20376f
Packit Service 20376f
			argument += offset;
Packit Service 20376f
			arglen = strlen(argument);
Packit Service 20376f
Packit Service 20376f
			if (arglen == 0)
Packit Service 20376f
				clar_usage(argv[0]);
Packit Service 20376f
Packit Service 20376f
			for (j = 0; j < _clar_suite_count; ++j) {
Packit Service 20376f
				suitelen = strlen(_clar_suites[j].name);
Packit Service 20376f
				cmplen = (arglen < suitelen) ? arglen : suitelen;
Packit Service 20376f
Packit Service 20376f
				if (strncmp(argument, _clar_suites[j].name, cmplen) == 0) {
Packit Service 20376f
					int exact = (arglen >= suitelen);
Packit Service 20376f
Packit Service 20376f
					/* Do we have a real suite prefix separated by a
Packit Service 20376f
					 * trailing '::' or just a matching substring? */
Packit Service 20376f
					if (arglen > suitelen && (argument[suitelen] != ':'
Packit Service 20376f
						    || argument[suitelen + 1] != ':'))
Packit Service 20376f
					    continue;
Packit Service 20376f
Packit Service 20376f
					++found;
Packit Service 20376f
Packit Service 20376f
					if (!exact)
Packit Service 20376f
						_clar.report_suite_names = 1;
Packit Service 20376f
Packit Service 20376f
					switch (action) {
Packit Service 20376f
					case 's': {
Packit Service 20376f
						struct clar_explicit *explicit =
Packit Service 20376f
							calloc(1, sizeof(struct clar_explicit));
Packit Service 20376f
						assert(explicit);
Packit Service 20376f
Packit Service 20376f
						explicit->suite_idx = j;
Packit Service 20376f
						explicit->filter = argument;
Packit Service 20376f
Packit Service 20376f
						if (_clar.explicit == NULL)
Packit Service 20376f
							_clar.explicit = explicit;
Packit Service 20376f
Packit Service 20376f
						if (_clar.last_explicit != NULL)
Packit Service 20376f
							_clar.last_explicit->next = explicit;
Packit Service 20376f
Packit Service 20376f
						_clar_suites[j].enabled = 1;
Packit Service 20376f
						_clar.last_explicit = explicit;
Packit Service 20376f
						break;
Packit Service 20376f
					}
Packit Service 20376f
					case 'i': _clar_suites[j].enabled = 1; break;
Packit Service 20376f
					case 'x': _clar_suites[j].enabled = 0; break;
Packit Service 20376f
					}
Packit Service 20376f
Packit Service 20376f
					if (exact)
Packit Service 20376f
						break;
Packit Service 20376f
				}
Packit Service 20376f
			}
Packit Service 20376f
Packit Service 20376f
			if (!found) {
Packit Service 20376f
				clar_print_onabort("No suite matching '%s' found.\n", argument);
Packit Service 20376f
				exit(-1);
Packit Service 20376f
			}
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		case 'q':
Packit Service 20376f
			_clar.report_errors_only = 1;
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		case 'Q':
Packit Service 20376f
			_clar.exit_on_error = 1;
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		case 'l': {
Packit Service 20376f
			size_t j;
Packit Service 20376f
			printf("Test suites (use -s<name> to run just one):\n");
Packit Service 20376f
			for (j = 0; j < _clar_suite_count; ++j)
Packit Service 20376f
				printf(" %3d: %s\n", (int)j, _clar_suites[j].name);
Packit Service 20376f
Packit Service 20376f
			exit(0);
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		case 'v':
Packit Service 20376f
			_clar.report_suite_names = 1;
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		case 'r':
Packit Service 20376f
			_clar.write_summary = 1;
Packit Service 20376f
			_clar.summary_filename = *(argument + 2) ? (argument + 2) :
Packit Service 20376f
			    "summary.xml";
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		default:
Packit Service 20376f
			assert(!"Unexpected commandline argument!");
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void
Packit Service 20376f
clar_test_init(int argc, char **argv)
Packit Service 20376f
{
Packit Service 20376f
	clar_print_init(
Packit Service 20376f
		(int)_clar_callback_count,
Packit Service 20376f
		(int)_clar_suite_count,
Packit Service 20376f
		""
Packit Service 20376f
	);
Packit Service 20376f
Packit Service 20376f
	if (argc > 1)
Packit Service 20376f
		clar_parse_args(argc, argv);
Packit Service 20376f
Packit Service 20376f
	if (_clar.write_summary &&
Packit Service 20376f
	    !(_clar.summary = clar_summary_init(_clar.summary_filename))) {
Packit Service 20376f
		clar_print_onabort("Failed to open the summary file\n");
Packit Service 20376f
		exit(-1);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (clar_sandbox() < 0) {
Packit Service 20376f
		clar_print_onabort("Failed to sandbox the test runner.\n");
Packit Service 20376f
		exit(-1);
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int
Packit Service 20376f
clar_test_run(void)
Packit Service 20376f
{
Packit Service 20376f
	size_t i;
Packit Service 20376f
	struct clar_explicit *explicit;
Packit Service 20376f
Packit Service 20376f
	if (_clar.explicit) {
Packit Service 20376f
		for (explicit = _clar.explicit; explicit; explicit = explicit->next)
Packit Service 20376f
			clar_run_suite(&_clar_suites[explicit->suite_idx], explicit->filter);
Packit Service 20376f
	} else {
Packit Service 20376f
		for (i = 0; i < _clar_suite_count; ++i)
Packit Service 20376f
			clar_run_suite(&_clar_suites[i], NULL);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return _clar.total_errors;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void
Packit Service 20376f
clar_test_shutdown(void)
Packit Service 20376f
{
Packit Service 20376f
	struct clar_explicit *explicit, *explicit_next;
Packit Service 20376f
	struct clar_report *report, *report_next;
Packit Service 20376f
Packit Service 20376f
	clar_print_shutdown(
Packit Service 20376f
		_clar.tests_ran,
Packit Service 20376f
		(int)_clar_suite_count,
Packit Service 20376f
		_clar.total_errors
Packit Service 20376f
	);
Packit Service 20376f
Packit Service 20376f
	clar_unsandbox();
Packit Service 20376f
Packit Service 20376f
	if (_clar.write_summary && clar_summary_shutdown(_clar.summary) < 0) {
Packit Service 20376f
		clar_print_onabort("Failed to write the summary file\n");
Packit Service 20376f
		exit(-1);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	for (explicit = _clar.explicit; explicit; explicit = explicit_next) {
Packit Service 20376f
		explicit_next = explicit->next;
Packit Service 20376f
		free(explicit);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	for (report = _clar.reports; report; report = report_next) {
Packit Service 20376f
		report_next = report->next;
Packit Service 20376f
		free(report);
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int
Packit Service 20376f
clar_test(int argc, char **argv)
Packit Service 20376f
{
Packit Service 20376f
	int errors;
Packit Service 20376f
Packit Service 20376f
	clar_test_init(argc, argv);
Packit Service 20376f
	errors = clar_test_run();
Packit Service 20376f
	clar_test_shutdown();
Packit Service 20376f
Packit Service 20376f
	return errors;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void abort_test(void)
Packit Service 20376f
{
Packit Service 20376f
	if (!_clar.trampoline_enabled) {
Packit Service 20376f
		clar_print_onabort(
Packit Service 20376f
				"Fatal error: a cleanup method raised an exception.");
Packit Service 20376f
		clar_report_errors(_clar.last_report);
Packit Service 20376f
		exit(-1);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	CL_TRACE(CL_TRACE__TEST__LONGJMP);
Packit Service 20376f
	longjmp(_clar.trampoline, -1);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void clar__skip(void)
Packit Service 20376f
{
Packit Service 20376f
	_clar.last_report->status = CL_TEST_SKIP;
Packit Service 20376f
	_clar.total_skipped++;
Packit Service 20376f
	abort_test();
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void clar__fail(
Packit Service 20376f
	const char *file,
Packit Service 20376f
	int line,
Packit Service 20376f
	const char *error_msg,
Packit Service 20376f
	const char *description,
Packit Service 20376f
	int should_abort)
Packit Service 20376f
{
Packit Service 20376f
	struct clar_error *error = calloc(1, sizeof(struct clar_error));
Packit Service 20376f
Packit Service 20376f
	if (_clar.last_report->errors == NULL)
Packit Service 20376f
		_clar.last_report->errors = error;
Packit Service 20376f
Packit Service 20376f
	if (_clar.last_report->last_error != NULL)
Packit Service 20376f
		_clar.last_report->last_error->next = error;
Packit Service 20376f
Packit Service 20376f
	_clar.last_report->last_error = error;
Packit Service 20376f
Packit Service 20376f
	error->file = file;
Packit Service 20376f
	error->line_number = line;
Packit Service 20376f
	error->error_msg = error_msg;
Packit Service 20376f
Packit Service 20376f
	if (description != NULL)
Packit Service 20376f
		error->description = strdup(description);
Packit Service 20376f
Packit Service 20376f
	_clar.total_errors++;
Packit Service 20376f
	_clar.last_report->status = CL_TEST_FAILURE;
Packit Service 20376f
Packit Service 20376f
	if (should_abort)
Packit Service 20376f
		abort_test();
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void clar__assert(
Packit Service 20376f
	int condition,
Packit Service 20376f
	const char *file,
Packit Service 20376f
	int line,
Packit Service 20376f
	const char *error_msg,
Packit Service 20376f
	const char *description,
Packit Service 20376f
	int should_abort)
Packit Service 20376f
{
Packit Service 20376f
	if (condition)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	clar__fail(file, line, error_msg, description, should_abort);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void clar__assert_equal(
Packit Service 20376f
	const char *file,
Packit Service 20376f
	int line,
Packit Service 20376f
	const char *err,
Packit Service 20376f
	int should_abort,
Packit Service 20376f
	const char *fmt,
Packit Service 20376f
	...)
Packit Service 20376f
{
Packit Service 20376f
	va_list args;
Packit Service 20376f
	char buf[4096];
Packit Service 20376f
	int is_equal = 1;
Packit Service 20376f
Packit Service 20376f
	va_start(args, fmt);
Packit Service 20376f
Packit Service 20376f
	if (!strcmp("%s", fmt)) {
Packit Service 20376f
		const char *s1 = va_arg(args, const char *);
Packit Service 20376f
		const char *s2 = va_arg(args, const char *);
Packit Service 20376f
		is_equal = (!s1 || !s2) ? (s1 == s2) : !strcmp(s1, s2);
Packit Service 20376f
Packit Service 20376f
		if (!is_equal) {
Packit Service 20376f
			if (s1 && s2) {
Packit Service 20376f
				int pos;
Packit Service 20376f
				for (pos = 0; s1[pos] == s2[pos] && s1[pos] && s2[pos]; ++pos)
Packit Service 20376f
					/* find differing byte offset */;
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%s' != '%s' (at byte %d)",
Packit Service 20376f
					s1, s2, pos);
Packit Service 20376f
			} else {
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%s' != '%s'", s1, s2);
Packit Service 20376f
			}
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
	else if(!strcmp("%.*s", fmt)) {
Packit Service 20376f
		const char *s1 = va_arg(args, const char *);
Packit Service 20376f
		const char *s2 = va_arg(args, const char *);
Packit Service 20376f
		int len = va_arg(args, int);
Packit Service 20376f
		is_equal = (!s1 || !s2) ? (s1 == s2) : !strncmp(s1, s2, len);
Packit Service 20376f
Packit Service 20376f
		if (!is_equal) {
Packit Service 20376f
			if (s1 && s2) {
Packit Service 20376f
				int pos;
Packit Service 20376f
				for (pos = 0; s1[pos] == s2[pos] && pos < len; ++pos)
Packit Service 20376f
					/* find differing byte offset */;
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%.*s' != '%.*s' (at byte %d)",
Packit Service 20376f
					len, s1, len, s2, pos);
Packit Service 20376f
			} else {
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%.*s' != '%.*s'", len, s1, len, s2);
Packit Service 20376f
			}
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
	else if (!strcmp("%ls", fmt)) {
Packit Service 20376f
		const wchar_t *wcs1 = va_arg(args, const wchar_t *);
Packit Service 20376f
		const wchar_t *wcs2 = va_arg(args, const wchar_t *);
Packit Service 20376f
		is_equal = (!wcs1 || !wcs2) ? (wcs1 == wcs2) : !wcscmp(wcs1, wcs2);
Packit Service 20376f
Packit Service 20376f
		if (!is_equal) {
Packit Service 20376f
			if (wcs1 && wcs2) {
Packit Service 20376f
				int pos;
Packit Service 20376f
				for (pos = 0; wcs1[pos] == wcs2[pos] && wcs1[pos] && wcs2[pos]; ++pos)
Packit Service 20376f
					/* find differing byte offset */;
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%ls' != '%ls' (at byte %d)",
Packit Service 20376f
					wcs1, wcs2, pos);
Packit Service 20376f
			} else {
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%ls' != '%ls'", wcs1, wcs2);
Packit Service 20376f
			}
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
	else if(!strcmp("%.*ls", fmt)) {
Packit Service 20376f
		const wchar_t *wcs1 = va_arg(args, const wchar_t *);
Packit Service 20376f
		const wchar_t *wcs2 = va_arg(args, const wchar_t *);
Packit Service 20376f
		int len = va_arg(args, int);
Packit Service 20376f
		is_equal = (!wcs1 || !wcs2) ? (wcs1 == wcs2) : !wcsncmp(wcs1, wcs2, len);
Packit Service 20376f
Packit Service 20376f
		if (!is_equal) {
Packit Service 20376f
			if (wcs1 && wcs2) {
Packit Service 20376f
				int pos;
Packit Service 20376f
				for (pos = 0; wcs1[pos] == wcs2[pos] && pos < len; ++pos)
Packit Service 20376f
					/* find differing byte offset */;
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%.*ls' != '%.*ls' (at byte %d)",
Packit Service 20376f
					len, wcs1, len, wcs2, pos);
Packit Service 20376f
			} else {
Packit Service 20376f
				p_snprintf(buf, sizeof(buf), "'%.*ls' != '%.*ls'", len, wcs1, len, wcs2);
Packit Service 20376f
			}
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
	else if (!strcmp("%"PRIuZ, fmt) || !strcmp("%"PRIxZ, fmt)) {
Packit Service 20376f
		size_t sz1 = va_arg(args, size_t), sz2 = va_arg(args, size_t);
Packit Service 20376f
		is_equal = (sz1 == sz2);
Packit Service 20376f
		if (!is_equal) {
Packit Service 20376f
			int offset = p_snprintf(buf, sizeof(buf), fmt, sz1);
Packit Service 20376f
			strncat(buf, " != ", sizeof(buf) - offset);
Packit Service 20376f
			p_snprintf(buf + offset + 4, sizeof(buf) - offset - 4, fmt, sz2);
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
	else if (!strcmp("%p", fmt)) {
Packit Service 20376f
		void *p1 = va_arg(args, void *), *p2 = va_arg(args, void *);
Packit Service 20376f
		is_equal = (p1 == p2);
Packit Service 20376f
		if (!is_equal)
Packit Service 20376f
			p_snprintf(buf, sizeof(buf), "%p != %p", p1, p2);
Packit Service 20376f
	}
Packit Service 20376f
	else {
Packit Service 20376f
		int i1 = va_arg(args, int), i2 = va_arg(args, int);
Packit Service 20376f
		is_equal = (i1 == i2);
Packit Service 20376f
		if (!is_equal) {
Packit Service 20376f
			int offset = p_snprintf(buf, sizeof(buf), fmt, i1);
Packit Service 20376f
			strncat(buf, " != ", sizeof(buf) - offset);
Packit Service 20376f
			p_snprintf(buf + offset + 4, sizeof(buf) - offset - 4, fmt, i2);
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	va_end(args);
Packit Service 20376f
Packit Service 20376f
	if (!is_equal)
Packit Service 20376f
		clar__fail(file, line, err, buf, should_abort);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void cl_set_cleanup(void (*cleanup)(void *), void *opaque)
Packit Service 20376f
{
Packit Service 20376f
	_clar.local_cleanup = cleanup;
Packit Service 20376f
	_clar.local_cleanup_payload = opaque;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
#include "clar/sandbox.h"
Packit Service 20376f
#include "clar/fixtures.h"
Packit Service 20376f
#include "clar/fs.h"
Packit Service 20376f
#include "clar/print.h"
Packit Service 20376f
#include "clar/summary.h"