bc_tester_utils.c 19.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
tester - liblinphone test suite
Copyright (C) 2013  Belledonne Communications SARL

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/


20 21
/* this must be provided at compile time*/
#include BC_CONFIG_FILE
22

Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
23
#include "bc_tester_utils.h"
24 25

#include <stdlib.h>
26
#include <time.h>
27
#include <stdio.h>
28

29
#include "CUnit/Basic.h"
30
#include "CUnit/Automated.h"
31
#include "CUnit/MyMem.h"
32

33 34 35 36 37 38
#ifdef _WIN32
#if defined(__MINGW32__) || !defined(WINAPI_FAMILY_PARTITION) || !defined(WINAPI_PARTITION_DESKTOP)
#define BC_TESTER_WINDOWS_DESKTOP 1
#elif defined(WINAPI_FAMILY_PARTITION)
#if defined(WINAPI_PARTITION_DESKTOP) && WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
#define BC_TESTER_WINDOWS_DESKTOP 1
39
#endif
40 41
#if defined(WINAPI_PARTITION_PHONE_APP) && WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_PHONE_APP)
#define BC_TESTER_WINDOWS_PHONE 1
42
#endif
43 44 45 46 47 48
#if defined(WINAPI_PARTITION_APP) && WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_APP)
#define BC_TESTER_WINDOWS_UNIVERSAL 1
#endif
#endif
#endif

49 50 51 52
#ifdef __linux
/*for monitoring total space allocated via malloc*/
#include <malloc.h>
#endif
53 54

static char *bc_tester_resource_dir_prefix = NULL;
55
// by default writable will always write near the executable
56
static char *bc_tester_writable_dir_prefix = NULL;
57

58 59 60
static char *bc_current_suite_name = NULL;
static char *bc_current_test_name = NULL;

61 62 63
int bc_printf_verbosity_info;
int bc_printf_verbosity_error;

64 65 66
static test_suite_t **test_suite = NULL;
static int nb_test_suites = 0;

67 68
#ifdef HAVE_CU_CURSES
#include "CUnit/CUCurses.h"
69 70 71
static unsigned char curses = 0;
#endif

72
char* xml_file = "CUnitAutomated-Results.xml";
73
int   xml_enabled = 0;
74 75
char * suite_name;
char * test_name;
76 77
static long max_vm_kb = 0;

78
void (*tester_printf_va)(int level, const char *format, va_list args);
79

80
void bc_tester_printf(int level, const char *format, ...) {
81
	va_list args;
82 83
	va_start (args, format);
	tester_printf_va(level, format, args);
84 85 86
	va_end (args);
}

87
int bc_tester_run_suite(test_suite_t *suite) {
88
	int i;
89

90
	CU_pSuite pSuite = CU_add_suite(suite->name, suite->before_all, suite->after_all);
91

92 93 94
	for (i = 0; i < suite->nb_tests; i++) {
		if (NULL == CU_add_test(pSuite, suite->tests[i].name, suite->tests[i].func)) {
			return CU_get_error();
95 96 97
		}
	}

98
	return 0;
99 100
}

101
const char * bc_tester_suite_name(int suite_index) {
102 103
	if (suite_index >= nb_test_suites) return NULL;
	return test_suite[suite_index]->name;
104 105
}

106
int bc_tester_suite_index(const char *suite_name) {
107 108
	int i;

109
	for (i = 0; i < nb_test_suites; i++) {
110
		if (strcmp(suite_name, test_suite[i]->name) == 0) {
111 112 113 114 115 116
			return i;
		}
	}

	return -1;
}
117

118
int bc_tester_nb_suites(void) {
119 120 121 122 123
	return nb_test_suites;
}

const char * bc_tester_test_name(const char *suite_name, int test_index) {
	int suite_index = bc_tester_suite_index(suite_name);
124 125 126
	if ((suite_index < 0) || (suite_index >= nb_test_suites)) return NULL;
	if (test_index >= test_suite[suite_index]->nb_tests) return NULL;
	return test_suite[suite_index]->tests[test_index].name;
127 128
}

129 130
int bc_tester_nb_tests(const char *suite_name) {
	int i = bc_tester_suite_index(suite_name);
131 132 133 134
	if (i < 0) return 0;
	return test_suite[i]->nb_tests;
}

135
void bc_tester_list_suites(void) {
136 137
	int j;
	for(j=0;j<nb_test_suites;j++) {
138
		bc_tester_printf(bc_printf_verbosity_info, "%s", bc_tester_suite_name(j));
139
	}
140 141
}

142
void bc_tester_list_tests(const char *suite_name) {
143
	int j;
144 145
	for( j = 0; j < bc_tester_nb_tests(suite_name); j++) {
		const char *test_name = bc_tester_test_name(suite_name, j);
146
		bc_tester_printf(bc_printf_verbosity_info, "%s", test_name);
147
	}
148 149 150 151 152
}

static void all_complete_message_handler(const CU_pFailureRecord pFailure) {
#ifdef HAVE_CU_GET_SUITE
	char * results = CU_get_run_results_string();
153
	bc_tester_printf(bc_printf_verbosity_info,"\n%s",results);
154
	CU_FREE(results);
155 156 157 158
#endif
}

static void suite_init_failure_message_handler(const CU_pSuite pSuite) {
159
	bc_tester_printf(bc_printf_verbosity_error,"Suite initialization failed for [%s]", pSuite->pName);
160 161 162
}

static void suite_cleanup_failure_message_handler(const CU_pSuite pSuite) {
163
	bc_tester_printf(bc_printf_verbosity_error,"Suite cleanup failed for [%s]", pSuite->pName);
164 165 166
}

#ifdef HAVE_CU_GET_SUITE
167
static time_t suite_start_time = 0;
168
static void suite_start_message_handler(const CU_pSuite pSuite) {
169
	bc_tester_printf(bc_printf_verbosity_info,"Suite [%s] started\n", pSuite->pName);
170
	suite_start_time = time(NULL);
171
	bc_current_suite_name = pSuite->pName;
172 173
}
static void suite_complete_message_handler(const CU_pSuite pSuite, const CU_pFailureRecord pFailure) {
174 175
	bc_tester_printf(bc_printf_verbosity_info, "Suite [%s] ended in %lu sec\n", pSuite->pName,
					 time(NULL) - suite_start_time);
176 177
}

178
static time_t test_start_time = 0;
179
static void test_start_message_handler(const CU_pTest pTest, const CU_pSuite pSuite) {
180 181 182 183
	int suite_index = bc_tester_suite_index(pSuite->pName);
	if (test_suite[suite_index]->before_each) {
		test_suite[suite_index]->before_each();
	}
184 185
	bc_tester_printf(bc_printf_verbosity_info,"Suite [%s] Test [%s] started", pSuite->pName,pTest->pName);
	test_start_time = time(NULL);
186
	bc_current_test_name = pTest->pName;
187 188 189
}

/*derivated from cunit*/
190 191
static void test_complete_message_handler(const CU_pTest pTest, const CU_pSuite pSuite,
										  const CU_pFailureRecord pFailureList) {
192
	int i;
193
	int suite_index = bc_tester_suite_index(pSuite->pName);
194
	CU_pFailureRecord pFailure = pFailureList;
195 196
	char *buffer = NULL;
	char* result = bc_sprintf("Suite [%s] Test [%s] %s in %lu secs", pSuite->pName, pTest->pName,
197
			 pFailure ? "failed" : "passed", (unsigned long)(time(NULL) - test_start_time));
198

199
	if (pFailure) {
200 201 202 203 204 205 206 207 208
		for (i = 1; (NULL != pFailure); pFailure = pFailure->pNext, i++) {
			buffer = bc_sprintf("%s\n    %d. %s:%u  - %s",
									result,
									i,
									(NULL != pFailure->strFileName) ? pFailure->strFileName : "",
									pFailure->uiLineNumber,
									(NULL != pFailure->strCondition) ? pFailure->strCondition : "");
			free(result);
			result = buffer;
209 210
		}
	}
211

212
	bc_tester_printf(bc_printf_verbosity_info,"%s", result);
213
	free(result);
214

215 216 217 218 219 220
	if (test_suite[suite_index]->after_each) {
		test_suite[suite_index]->after_each();
	}
	//insert empty line
	bc_tester_printf(bc_printf_verbosity_info,"");

221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241
#ifdef __linux
	/* use mallinfo() to monitor allocated space. It is linux specific but other methods don't work:
	 * setrlimit() RLIMIT_DATA doesn't count memory allocated via mmap() (which is used internally by malloc)
	 * setrlimit() RLIMIT_AS works but also counts virtual memory allocated by thread stacks, which is very big and
	 * hardly controllable.
	 * setrlimit() RLIMIT_RSS does nothing interesting on linux.
	 * getrusage() of RSS is unreliable: memory blocks can be leaked without being read or written, which would not
	 * appear in RSS.
	 * mallinfo() itself is the less worse solution. Allocated bytes are returned as 'int' so limited to 2GB
	 */
	if (max_vm_kb) {
		struct mallinfo minfo = mallinfo();
		if (minfo.uordblks > max_vm_kb * 1024) {
			bc_tester_printf(
				bc_printf_verbosity_error,
				"The program exceeded the maximum amount of memory allocatable (%i bytes), aborting now.\n",
				minfo.uordblks);
			abort();
		}
	}
#endif
242 243 244
}
#endif

245
int bc_tester_run_tests(const char *suite_name, const char *test_name) {
246
	int i;
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
247

248
	/* initialize the CUnit test registry */
249 250 251
	if (CUE_SUCCESS != CU_initialize_registry())
		return CU_get_error();

252
	for (i = 0; i < nb_test_suites; i++) {
253
		bc_tester_run_suite(test_suite[i]);
254 255 256 257 258 259 260 261 262 263 264
	}
#ifdef HAVE_CU_GET_SUITE
	CU_set_suite_start_handler(suite_start_message_handler);
	CU_set_suite_complete_handler(suite_complete_message_handler);
	CU_set_test_start_handler(test_start_message_handler);
	CU_set_test_complete_handler(test_complete_message_handler);
#endif
	CU_set_all_test_complete_handler(all_complete_message_handler);
	CU_set_suite_init_failure_handler(suite_init_failure_message_handler);
	CU_set_suite_cleanup_failure_handler(suite_cleanup_failure_message_handler);

265
	if( xml_enabled != 0 ){
266 267 268
		CU_automated_run_tests();
	} else {

269
#ifndef HAVE_CU_GET_SUITE
270
		if( suite_name ){
271
			bc_tester_printf(bc_printf_verbosity_info, "Tester compiled without CU_get_suite() function, running all tests instead of suite '%s'", suite_name);
272 273 274 275 276 277
		}
#else
		if (suite_name){
			CU_pSuite suite;
			suite=CU_get_suite(suite_name);
			if (!suite) {
278
				bc_tester_printf(bc_printf_verbosity_error, "Could not find suite '%s'. Available suites are:", suite_name);
279
				bc_tester_list_suites();
280 281 282 283
				return -1;
			} else if (test_name) {
				CU_pTest test=CU_get_test_by_name(test_name, suite);
				if (!test) {
284
					bc_tester_printf(bc_printf_verbosity_error, "Could not find test '%s' in suite '%s'. Available tests are:", test_name, suite_name);
285
					// do not use suite_name here, since this method is case sensitive
286
					bc_tester_list_tests(suite->pName);
287 288 289
					return -2;
				} else {
					CU_ErrorCode err= CU_run_test(suite, test);
290
					if (err != CUE_SUCCESS) bc_tester_printf(bc_printf_verbosity_error, "CU_basic_run_test error %d", err);
291 292 293 294 295 296 297 298
				}
			} else {
				CU_run_suite(suite);
			}
		}
		else
#endif
		{
299
#ifdef HAVE_CU_CURSES
300 301 302 303 304 305 306
			if (curses) {
			/* Run tests using the CUnit curses interface */
				CU_curses_run_tests();
			}
			else
#endif
			{
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
307
				/* Run all tests using the CUnit Basic interface */
308 309 310 311
				CU_run_all_tests();
			}
		}
	}
312 313 314 315 316
#ifdef __linux
	bc_tester_printf(bc_printf_verbosity_info, "Still %i kilobytes allocated when all tests are finished.",
					 mallinfo().uordblks / 1024);
#endif

Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
317
	return CU_get_number_of_tests_failed()!=0;
318 319 320

}

321 322

void bc_tester_helper(const char *name, const char* additionnal_helper) {
323 324
	bc_tester_printf(bc_printf_verbosity_info,
					 "%s --help\n"
325 326 327
#ifdef HAVE_CU_CURSES
					 "\t\t\t--curses\n"
#endif
328 329 330 331
					 "\t\t\t--list-suites\n"
					 "\t\t\t--list-tests <suite>\n"
					 "\t\t\t--suite <suite name>\n"
					 "\t\t\t--test <test name>\n"
332 333
					 "\t\t\t--resource-dir <folder path> (directory where tester resource are located)\n"
					 "\t\t\t--writable-dir <folder path> (directory where temporary files should be created)\n"
334 335 336 337 338
					 "\t\t\t--xml\n"
					 "\t\t\t--xml-file <xml file name>\n"
					 "\t\t\t--max-alloc <size in ko> (maximum ammount of memory obtained via malloc allocator)\n"
					 "And additionally:\n"
					 "%s",
339 340
					 name,
					 additionnal_helper);
341 342
}

343
#if !defined(BC_TESTER_WINDOWS_PHONE) && !defined(BC_TESTER_WINDOWS_UNIVERSAL) && !defined(__QNX__) && !defined(ANDROID) && !defined(IOS)
344
static int file_exists(const char* root_path) {
345 346 347
	char * res_path = bc_sprintf("%s/common/bc_completion", root_path);
	FILE* file = fopen(res_path, "r");
	int found = (file != NULL);
348
	free(res_path);
349 350 351
	if (file) {
		fclose(file);
	}
352 353
	return found;
}
354
#endif
355 356

static void detect_res_prefix(const char* prog) {
357
	char* progpath = NULL;
358
	FILE* writable_file = NULL;
359

360 361 362 363 364 365 366
	if (prog != NULL) {
		progpath = strdup(prog);
		if (strchr(prog, '/') != NULL) {
			progpath[strrchr(prog, '/') - prog + 1] = '\0';
		} else if (strchr(prog, '\\') != NULL) {
			progpath[strrchr(prog, '\\') - prog + 1] = '\0';
		}
367
	}
368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384
#if !defined(BC_TESTER_WINDOWS_PHONE) && !defined(BC_TESTER_WINDOWS_UNIVERSAL) && !defined(__QNX__) && !defined(ANDROID) && !defined(IOS)
	{
		char* prefix = NULL;

		if (file_exists(".")) {
			prefix = strdup(".");
		} else if (file_exists("..")) {
			prefix = strdup("..");
		} else if (progpath) {
			//for autotools, binary is in .libs/ subdirectory
			char * progpath2 = bc_sprintf("%s/../", progpath);
			if (file_exists(progpath)) {
				prefix = strdup(progpath);
			} else if (file_exists(progpath2)) {
				prefix = strdup(progpath2);
			}
			free(progpath2);
385
		}
386

387 388 389 390
		if (bc_tester_resource_dir_prefix != NULL && !file_exists(bc_tester_resource_dir_prefix)) {
			bc_tester_printf(bc_printf_verbosity_error, "Invalid provided resource directory: could not find expected resources in %s.\n", bc_tester_resource_dir_prefix);
			free(bc_tester_resource_dir_prefix);
			bc_tester_resource_dir_prefix = NULL;
391
		}
392

393 394 395 396 397 398 399 400 401 402 403
		if (prefix != NULL) {
			if (bc_tester_resource_dir_prefix == NULL) {
				bc_tester_printf(bc_printf_verbosity_error, "Resource directory set to %s\n", prefix);
				bc_tester_set_resource_dir_prefix(prefix);
			}

			if (bc_tester_writable_dir_prefix == NULL) {
				bc_tester_printf(bc_printf_verbosity_error, "Writable directory set to %s\n", prefix);
				bc_tester_set_writable_dir_prefix(prefix);
			}
			free(prefix);
404
		}
405
	}
406
#endif
407 408 409

	// check that we can write in writable directory
	if (bc_tester_writable_dir_prefix != NULL) {
410 411
		char * writable_file_path = bc_sprintf("%s/%s", bc_tester_writable_dir_prefix, ".bc_tester_utils.tmp");
		writable_file = fopen(writable_file_path, "w");
412 413 414
		if (writable_file) {
			fclose(writable_file);
		}
415
		free(writable_file_path);
416 417 418
	}
	if (bc_tester_resource_dir_prefix == NULL || writable_file == NULL) {
		if (bc_tester_resource_dir_prefix == NULL) {
419 420
			bc_tester_printf(bc_printf_verbosity_error, "Failed to detect resources for %s.\n", prog);
			bc_tester_printf(bc_printf_verbosity_error, "Could not find resource directory in %s! Please try again using option --resource-dir.\n", progpath);
421 422
		}
		if (writable_file == NULL) {
423
			bc_tester_printf(bc_printf_verbosity_error, "Failed to write file in %s. Please try again using option --writable-dir.\n", bc_tester_writable_dir_prefix);
424
		}
425
		abort();
426
	}
427 428 429 430

	if (progpath != NULL) {
		free(progpath);
	}
431 432
}

433
void bc_tester_init(void (*ftester_printf)(int level, const char *format, va_list args), int iverbosity_info, int iverbosity_error) {
434
	tester_printf_va = ftester_printf;
435 436
	bc_printf_verbosity_error = iverbosity_error;
	bc_printf_verbosity_info = iverbosity_info;
437
	bc_tester_writable_dir_prefix = strdup(".");
438 439
}

440 441 442 443 444 445 446 447 448
void bc_tester_set_max_vm(long max_vm_kb) {
#ifdef __linux
	max_vm_kb = max_vm_kb;
	bc_tester_printf(bc_printf_verbosity_info, "Maximum virtual memory space set to %li kilo bytes", max_vm_kb);
#else
	bc_tester_printf(bc_printf_verbosity_error, "Maximum virtual memory space setting is only implemented on Linux.");
#endif
}

449 450 451
int bc_tester_parse_args(int argc, char **argv, int argid)
{
	int i = argid;
452

453 454 455 456 457 458 459 460 461
	if (strcmp(argv[i],"--help")==0){
		return -1;
	} else if (strcmp(argv[i],"--test")==0){
		CHECK_ARG("--test", ++i, argc);
		test_name=argv[i];
	}else if (strcmp(argv[i],"--suite")==0){
		CHECK_ARG("--suite", ++i, argc);
		suite_name=argv[i];
	} else if (strcmp(argv[i],"--list-suites")==0){
462
		bc_tester_list_suites();
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
463
		return 0;
464 465 466
	} else if (strcmp(argv[i],"--list-tests")==0){
		CHECK_ARG("--list-tests", ++i, argc);
		suite_name = argv[i];
467
		bc_tester_list_tests(suite_name);
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
468
		return 0;
469 470 471 472 473 474
	} else if (strcmp(argv[i], "--xml-file") == 0){
		CHECK_ARG("--xml-file", ++i, argc);
		xml_file = argv[i];
		xml_enabled = 1;
	} else if (strcmp(argv[i], "--xml") == 0){
		xml_enabled = 1;
475 476 477
	} else if (strcmp(argv[i], "--max-alloc") == 0) {
		CHECK_ARG("--max-alloc", ++i, argc);
		max_vm_kb = atol(argv[i]);
478 479 480 481 482 483
	} else if (strcmp(argv[i], "--resource-dir") == 0) {
		CHECK_ARG("--resource-dir", ++i, argc);
		bc_tester_resource_dir_prefix = strdup(argv[i]);
	} else if (strcmp(argv[i], "--writable-dir") == 0) {
		CHECK_ARG("--writable-dir", ++i, argc);
		bc_tester_writable_dir_prefix = strdup(argv[i]);
484
	} else {
485
		bc_tester_printf(bc_printf_verbosity_error, "Unknown option \"%s\"\n", argv[i]);
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
486
		return -1;
487 488 489
	}

	if( xml_enabled && (suite_name || test_name) ){
490
		bc_tester_printf(bc_printf_verbosity_error, "Cannot use both XML and specific test suite\n");
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
491
		return -1;
492 493
	}

Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
494 495
	/* returns number of arguments read + 1 */
	return i - argid + 1;
496 497
}

498
int bc_tester_start(const char* prog_name) {
499
	int ret;
500

501 502
	detect_res_prefix(prog_name);

503 504 505
	if (max_vm_kb)
		bc_tester_set_max_vm(max_vm_kb);

506
	if( xml_enabled ){
507
		char * xml_tmp_file = bc_sprintf("%s.tmp", xml_file);
508 509 510 511
		CU_set_output_filename(xml_tmp_file);
		free(xml_tmp_file);
	}

512
	ret = bc_tester_run_tests(suite_name, test_name);
513 514 515

	return ret;
}
516
void bc_tester_add_suite(test_suite_t *suite) {
517 518 519 520 521 522 523 524 525 526
	if (test_suite == NULL) {
		test_suite = (test_suite_t **)malloc(10 * sizeof(test_suite_t *));
	}
	test_suite[nb_test_suites] = suite;
	nb_test_suites++;
	if ((nb_test_suites % 10) == 0) {
		test_suite = (test_suite_t **)realloc(test_suite, (nb_test_suites + 10) * sizeof(test_suite_t *));
	}
}

527
void bc_tester_uninit(void) {
528 529 530
	/* Redisplay list of failed tests on end */
	/*BUG: do not display list of failures on mingw, it crashes mysteriously*/
#if !defined(WIN32) && !defined(_MSC_VER)
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
531 532 533 534
	/* Redisplay list of failed tests on end */
	if (CU_get_number_of_failure_records()){
		CU_basic_show_failures(CU_get_failure_list());
	}
535
#endif
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
536
	CU_cleanup_registry();
537
	/*add missing final newline*/
538
	bc_tester_printf(bc_printf_verbosity_info,"");
Gautier Pelloux-Prayer's avatar
Gautier Pelloux-Prayer committed
539

540
	if( xml_enabled ){
541
		/*create real xml file only if tester did not crash*/
542
		char * xml_tmp_file = bc_sprintf("%s.tmp-Results.xml", xml_file);
543
		rename(xml_tmp_file, xml_file);
544
		free(xml_tmp_file);
545 546
	}

547 548 549 550 551
	if (test_suite != NULL) {
		free(test_suite);
		test_suite = NULL;
		nb_test_suites = 0;
	}
552 553 554 555 556 557 558 559 560

	if (bc_tester_resource_dir_prefix != NULL) {
		free(bc_tester_resource_dir_prefix);
		bc_tester_resource_dir_prefix = NULL;
	}
	if (bc_tester_writable_dir_prefix != NULL) {
		free(bc_tester_writable_dir_prefix);
		bc_tester_writable_dir_prefix = NULL;
	}
561
}
562

563 564
static void bc_tester_set_dir_prefix(char **prefix, const char *name) {
	if (*prefix != NULL) free(*prefix);
565
	*prefix = strdup(name);
566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584
}

const char * bc_tester_get_resource_dir_prefix(void) {
	return bc_tester_resource_dir_prefix;
}

void bc_tester_set_resource_dir_prefix(const char *name) {
	bc_tester_set_dir_prefix(&bc_tester_resource_dir_prefix, name);
}

const char * bc_tester_get_writable_dir_prefix(void) {
	return bc_tester_writable_dir_prefix;
}

void bc_tester_set_writable_dir_prefix(const char *name) {
	bc_tester_set_dir_prefix(&bc_tester_writable_dir_prefix, name);
}

static char * bc_tester_path(const char *prefix, const char *name) {
585
	if (name) {
586 587 588
		return bc_sprintf("%s/%s", prefix, name);
	} else {
		return NULL;
589 590
	}
}
591 592 593 594 595 596 597 598

char * bc_tester_res(const char *name) {
	return bc_tester_path(bc_tester_resource_dir_prefix, name);
}

char * bc_tester_file(const char *name) {
	return bc_tester_path(bc_tester_writable_dir_prefix, name);
}
599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622

char* bc_sprintfva(const char* format, va_list args) {
	/* Guess we need no more than 100 bytes. */
	int n, size = 200;
	char *p,*np;
#ifndef WIN32
	va_list cap;/*copy of our argument list: a va_list cannot be re-used (SIGSEGV on linux 64 bits)*/
#endif
	if ((p = malloc(size)) == NULL)
		return NULL;
	while (1)
	{
		/* Try to print in the allocated space. */
#ifndef WIN32
		va_copy(cap,args);
		n = vsnprintf (p, size, format, cap);
		va_end(cap);
#else
		/*this works on 32 bits, luckily*/
		n = vsnprintf (p, size, format, args);
#endif
		/* If that worked, return the string. */
		if (n > -1 && n < size)
			return p;
623
		//bc_tester_printf(bc_printf_verbosity_error, "Reallocing space.\n");
624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648
		/* Else try again with more space. */
		if (n > -1)	/* glibc 2.1 */
			size = n + 1;	/* precisely what is needed */
		else		/* glibc 2.0 */
			size *= 2;	/* twice the old size */
		if ((np = realloc (p, size)) == NULL)
		{
			free(p);
			return NULL;
		}
		else
		{
			p = np;
		}
	}
}

char* bc_sprintf(const char* format, ...) {
	va_list args;
	char* res;
	va_start(args, format);
	res = bc_sprintfva(format, args);
	va_end (args);
	return res;
}
649 650 651 652 653 654 655 656

const char * bc_tester_current_suite_name() {
	return bc_current_suite_name;
}

const char * bc_tester_current_test_name() {
	return bc_current_test_name;
}