Add TLS offset probing to mach-amd64 in the same way it's done on x86.
[mono-project.git] / eglib / test / test.c
blob6dd7bf47469a891c8ca708bf157b99f04936fd02
1 /*
2 * EGLib Unit Group/Test Runners
4 * Author:
5 * Aaron Bockover (abockover@novell.com)
7 * (C) 2006 Novell, Inc.
9 * Permission is hereby granted, free of charge, to any person obtaining
10 * a copy of this software and associated documentation files (the
11 * "Software"), to deal in the Software without restriction, including
12 * without limitation the rights to use, copy, modify, merge, publish,
13 * distribute, sublicense, and/or sell copies of the Software, and to
14 * permit persons to whom the Software is furnished to do so, subject to
15 * the following conditions:
17 * The above copyright notice and this permission notice shall be
18 * included in all copies or substantial portions of the Software.
20 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 #ifndef _GNU_SOURCE
30 #define _GNU_SOURCE
31 #endif
33 #include <config.h>
34 #include <stdlib.h>
35 #include <stdio.h>
36 #include <string.h>
37 #include <stdarg.h>
38 #include <glib.h>
39 #ifdef HAVE_SYS_TIME_H
40 #include <sys/time.h>
41 #endif
42 #ifdef G_OS_WIN32
43 #include <winsock2.h>
44 #endif
46 #include "test.h"
48 extern gint global_passed, global_tests;
50 #ifndef HAVE_VASPRINTF
51 /* systen does not provide a vasprintf function, use the one
52 provided within eglib itself */
53 extern int vasprintf(char **ret, const char *format, va_list ap);
54 #endif
56 static gchar *last_result = NULL;
58 gboolean
59 run_test(Test *test, gchar **result_out)
61 gchar *result;
63 if((result = test->handler()) == NULL) {
64 *result_out = NULL;
65 return TRUE;
66 } else {
67 *result_out = result;
68 return FALSE;
72 gboolean
73 run_group(Group *group, gint iterations, gboolean quiet,
74 gboolean time, gchar *tests_to_run_s)
76 Test *tests = group->handler();
77 gint i, j, passed = 0, total = 0;
78 gdouble start_time_group, start_time_test;
79 gchar **tests_to_run = NULL;
81 if(!quiet) {
82 if(iterations > 1) {
83 printf("[%s] (%dx)\n", group->name, iterations);
84 } else {
85 printf("[%s]\n", group->name);
89 if(tests_to_run_s != NULL) {
90 tests_to_run = eg_strsplit(tests_to_run_s, ",", -1);
93 start_time_group = get_timestamp();
95 for(i = 0; tests[i].name != NULL; i++) {
96 gchar *result = "";
97 gboolean iter_pass, run;
99 iter_pass = FALSE;
100 if(tests_to_run != NULL) {
101 gint j;
102 run = FALSE;
103 for(j = 0; tests_to_run[j] != NULL; j++) {
104 if(strcmp(tests_to_run[j], tests[i].name) == 0) {
105 run = TRUE;
106 break;
109 } else {
110 run = TRUE;
113 if(!run) {
114 continue;
117 total++;
119 if(!quiet) {
120 printf(" %s: ", tests[i].name);
123 start_time_test = get_timestamp();
125 for(j = 0; j < iterations; j++) {
126 iter_pass = run_test(&(tests[i]), &result);
127 if(!iter_pass) {
128 break;
132 if(iter_pass) {
133 passed++;
134 if(!quiet) {
135 if(time) {
136 printf("OK (%g)\n", get_timestamp() - start_time_test);
137 } else {
138 printf("OK\n");
141 } else {
142 if(!quiet) {
143 printf("FAILED (%s)\n", result);
146 if(last_result == result) {
147 last_result = NULL;
148 g_free(result);
153 global_passed += passed;
154 global_tests += total;
156 if(!quiet) {
157 gdouble pass_percentage = ((gdouble)passed / (gdouble)total) * 100.0;
158 if(time) {
159 printf(" %d / %d (%g%%, %g)\n", passed, total,
160 pass_percentage, get_timestamp() - start_time_group);
161 } else {
162 printf(" %d / %d (%g%%)\n", passed, total, pass_percentage);
166 if(tests_to_run != NULL) {
167 eg_strfreev(tests_to_run);
170 return passed == total;
173 RESULT
174 FAILED(const gchar *format, ...)
176 gchar *ret;
177 va_list args;
178 gint n;
180 #if !defined(HAVE_VASPRINTF) && !defined(_EGLIB_MAJOR)
181 /* We are linked against the real glib, no vasprintf */
182 g_assert_not_reached ();
183 return NULL;
184 #else
185 va_start(args, format);
186 n = vasprintf(&ret, format, args);
187 va_end(args);
189 if(n == -1) {
190 last_result = NULL;
191 return NULL;
194 last_result = ret;
195 return ret;
196 #endif
199 gdouble
200 get_timestamp()
202 /* FIXME: We should use g_get_current_time here */
203 GTimeVal res;
204 g_get_current_time (&res);
205 return res.tv_sec + (1.e-6) * res.tv_usec;
209 * Duplicating code here from EGlib to avoid g_strsplit skew between
210 * EGLib and GLib
213 gchar **
214 eg_strsplit (const gchar *string, const gchar *delimiter, gint max_tokens)
216 gchar *string_c;
217 gchar *strtok_save, **vector;
218 gchar *token, *token_c;
219 gint size = 1;
220 size_t token_length;
222 g_return_val_if_fail(string != NULL, NULL);
223 g_return_val_if_fail(delimiter != NULL, NULL);
224 g_return_val_if_fail(delimiter[0] != 0, NULL);
226 token_length = strlen(string);
227 string_c = (gchar *)g_malloc(token_length + 1);
228 memcpy(string_c, string, token_length);
229 string_c[token_length] = 0;
231 vector = NULL;
232 token = (gchar *)strtok_r(string_c, delimiter, &strtok_save);
234 while(token != NULL) {
235 token_length = strlen(token);
236 token_c = (gchar *)g_malloc(token_length + 1);
237 memcpy(token_c, token, token_length);
238 token_c[token_length] = 0;
240 vector = vector == NULL ?
241 (gchar **)g_malloc(2 * sizeof(vector)) :
242 (gchar **)g_realloc(vector, (size + 1) * sizeof(vector));
244 vector[size - 1] = token_c;
245 size++;
247 if(max_tokens > 0 && size >= max_tokens) {
248 if(size > max_tokens) {
249 break;
252 token = strtok_save;
253 } else {
254 token = (gchar *)strtok_r(NULL, delimiter, &strtok_save);
258 if(vector != NULL && size > 0) {
259 vector[size - 1] = NULL;
262 g_free(string_c);
263 string_c = NULL;
265 return vector;
268 void
269 eg_strfreev (gchar **str_array)
271 gchar **orig = str_array;
272 if (str_array == NULL)
273 return;
274 while (*str_array != NULL){
275 g_free (*str_array);
276 str_array++;
278 g_free (orig);