This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
runner.c
221 lines (179 loc) · 5.45 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/*
Copyright (C) 2011 Markus Kauppila <markus.kauppila@gmail.com>
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
*/
21
#include "SDL/SDL.h"
22
23
24
25
26
27
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
28
#include "SDL_test.h"
29
30
31
32
//!< Function pointer to a test case function
typedef int (*TestCase)(void *arg);
33
34
35
/*!
* Loads test suite which is implemented as dynamic library.
*
36
* \return Pointer to loaded test suite, or NULL if library could not be loaded
37
*/
38
39
40
void *
LoadTestSuite()
{
41
42
#if defined(linux) || defined( __linux)
char *libName = "tests/libtest.so";
43
44
#else
char *libName = "tests/libtest.dylib";
45
46
#endif
47
48
void *library = SDL_LoadObject(libName);
if(library == NULL) {
49
50
fprintf(stderr, "Loading %s failed\n", libName);
fprintf(stderr, "%s\n", SDL_GetError());
51
52
}
53
54
55
return library;
}
56
57
58
59
/*!
* Loads the test case references from the given test suite.
* \param library Previously loaded dynamic library AKA test suite
60
* \return Pointer to array of TestCaseReferences or NULL if function failed
61
*/
62
63
64
TestCaseReference **
QueryTestCases(void *library)
{
65
TestCaseReference **(*suite)(void);
66
67
suite = (TestCaseReference **(*)(void)) SDL_LoadFunction(library, "QueryTestSuite");
68
if(suite == NULL) {
69
70
fprintf(stderr, "Loading QueryTestCaseReferences() failed.\n");
fprintf(stderr, "%s\n", SDL_GetError());
71
72
}
73
TestCaseReference **tests = suite();
74
if(tests == NULL) {
75
76
fprintf(stderr, "Failed to load test references.\n");
fprintf(stderr, "%s\n", SDL_GetError());
77
78
79
80
81
}
return tests;
}
82
83
84
/*!
* Loads test case from a test suite
*
85
86
* \param suite a test suite
* \param testName Name of the test that is going to be loaded
87
*
88
* \return Function Pointer (TestCase) to loaded test case, NULL if function failed
89
*/
90
91
92
TestCase
LoadTestCase(void *suite, char *testName)
{
93
94
TestCase test = (int (*)(void *)) SDL_LoadFunction(suite, testName);
if(test == NULL) {
95
96
fprintf(stderr, "Loading test failed, tests == NULL\n");
fprintf(stderr, "%s\n", SDL_GetError());
97
98
99
100
101
102
}
return test;
}
103
/*!
104
105
106
107
* If using out-of-proc execution of tests. This function
* will handle the return value of the child process
* and interprets it to the runner. Also prints warnings
* if child was aborted by a signela.
108
*
109
* \param stat_lock information about the exited child process
110
*
111
* \return 0 if test case succeeded, 1 otherwise
112
*/
113
114
115
116
int
HandleTestReturnValue(int stat_lock)
{
//! \todo rename to: HandleChildProcessReturnValue?
117
int returnValue = -1;
118
119
120
if(WIFEXITED(stat_lock)) {
returnValue = WEXITSTATUS(stat_lock);
121
122
} else if(WIFSIGNALED(stat_lock)) {
int signal = WTERMSIG(stat_lock);
123
fprintf(stderr, "FAILURE: test was aborted due to signal no %d\n", signal);
124
returnValue = 1;
125
126
}
127
return returnValue;
128
129
}
130
//!< Flag for executing tests in-process
131
static int execute_inproc = 0;
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
/*!
* Parse command line arguments
*/
void
ParseOptions(int argc, char *argv[])
{
int i;
for (i = 1; i < argc; ++i) {
const char *arg = argv[i];
if (SDL_strcmp(arg, "--in-proc") == 0) {
execute_inproc = 1;
}
}
}
148
149
150
151
152
153
154
/*!
* Entry point for test runner
*
* \param argc Count of command line arguments
* \param argv Array of commond lines arguments
*/
155
156
157
158
int
main(int argc, char *argv[])
{
ParseOptions(argc, argv);
159
160
// print: Testing against SDL version fuu (rev: bar) if verbose == true
161
162
163
164
165
166
167
int failureCount = 0, passCount = 0;
char *libName = "libtest";
const Uint32 startTicks = SDL_GetTicks();
168
169
void *suite = LoadTestSuite();
TestCaseReference **tests = QueryTestCases(suite);
170
171
172
173
174
TestCaseReference *reference = NULL;
int counter = 0;
for(reference = tests[counter]; reference; reference = tests[++counter]) {
175
176
if(reference->enabled == TEST_DISABLED) {
printf("Test %s (in %s) disabled. Omitting...\n", reference->name, libName);
177
} else {
178
179
180
181
char *testname = reference->name;
printf("Running %s (in %s):\n", testname, libName);
182
183
184
185
int retVal = 1;
if(execute_inproc) {
TestCase test = (TestCase) LoadTestCase(suite, testname);
retVal = test(0x0);
186
} else {
187
188
189
190
int childpid = fork();
if(childpid == 0) {
TestCase test = (TestCase) LoadTestCase(suite, testname);
return test(0x0);
191
} else {
192
193
194
195
int stat_lock = -1;
int child = wait(&stat_lock);
retVal = HandleTestReturnValue(stat_lock);
196
}
197
}
198
199
200
201
202
203
204
205
if(retVal) {
failureCount++;
printf("%s (in %s): FAILED\n", testname, libName);
} else {
passCount++;
printf("%s (in %s): ok\n", testname, libName);
}
206
}
207
208
printf("\n");
209
210
}
211
SDL_UnloadObject(suite);
212
213
214
const Uint32 endTicks = SDL_GetTicks();
215
printf("Ran %d tests in %0.5f seconds.\n", (passCount + failureCount), (endTicks-startTicks)/1000.0f);
216
217
218
219
printf("%d tests passed\n", passCount);
printf("%d tests failed\n", failureCount);
220
221
return 0;
}