aboutsummaryrefslogtreecommitdiffstats
path: root/Source/charon/utils/tester.c
diff options
context:
space:
mode:
Diffstat (limited to 'Source/charon/utils/tester.c')
-rw-r--r--Source/charon/utils/tester.c206
1 files changed, 206 insertions, 0 deletions
diff --git a/Source/charon/utils/tester.c b/Source/charon/utils/tester.c
new file mode 100644
index 000000000..ce2c7fff1
--- /dev/null
+++ b/Source/charon/utils/tester.c
@@ -0,0 +1,206 @@
+/**
+ * @file tester.c
+ *
+ * @brief Test module for automatic testing
+ *
+ */
+
+/*
+ * Copyright (C) 2005 Jan Hutter, Martin Willi
+ * Hochschule fuer Technik Rapperswil
+ *
+ * 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. See <http://www.fsf.org/copyleft/gpl.txt>.
+ *
+ * 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.
+ */
+
+
+#include <stdlib.h>
+#include <string.h>
+#include <pthread.h>
+#include <sys/time.h>
+
+#include "tester.h"
+
+#include "allocator.h"
+#include "linked_list.h"
+#include "../queues/job_queue.h"
+
+/**
+ * @brief Private Variables and Functions of tester class
+ *
+ */
+typedef struct private_tester_s private_tester_t;
+
+struct private_tester_s {
+ tester_t public;
+
+
+ /* Private functions */
+ void (*run_test) (tester_t *tester, void (*test_function) (tester_t * tester), char * test_name);
+
+
+ /* Private values */
+ FILE* output;
+ int tests_count;
+ int failed_tests_count;
+ int failed_asserts_count;
+ bool display_succeeded_asserts;
+ pthread_mutex_t mutex;
+};
+
+/*
+ * Implementation of function perform_tests
+ */
+static status_t perform_tests(tester_t *tester,test_t **tests)
+{
+ private_tester_t *this =(private_tester_t*) tester;
+ int current_test = 0;
+ fprintf(this->output,"\nStart testing...\n\n");
+ fprintf(this->output,"_____________________________________________________________________\n");
+ fprintf(this->output,"Testname | running time\n");
+ fprintf(this->output,"_______________________________________________________|_____________\n");
+
+ while (tests[current_test] != NULL)
+ {
+ this->run_test(tester,tests[current_test]->test_function,tests[current_test]->test_name);
+ current_test++;
+ }
+ fprintf(this->output,"=====================================================================\n");
+ fprintf(this->output,"End testing. %d of %d tests succeeded\n",this->tests_count - this->failed_tests_count,this->tests_count);
+ fprintf(this->output,"=====================================================================\n");
+ return SUCCESS;
+}
+
+/*
+ * Implementation of function perform_test
+ */
+static status_t perform_test(tester_t *tester, test_t *test)
+{
+ test_t *tests[] = {test, NULL};
+ return (perform_tests(tester,tests));
+}
+
+/**
+ * Returns the difference of to timeval structs in microseconds
+ *
+ * @param end_time end time
+ * @param start_time start time
+ *
+ * @warning this function is also defined in the event queue
+ * in later improvements, this function can be added to a general
+ * class type!
+ *
+ * @return difference in microseconds
+ */
+static long time_difference(struct timeval *end_time, struct timeval *start_time)
+{
+ long seconds, microseconds;
+
+ seconds = (end_time->tv_sec - start_time->tv_sec);
+ microseconds = (end_time->tv_usec - start_time->tv_usec);
+ return ((seconds * 1000000) + microseconds);
+}
+
+
+/**
+ * Implementation of function run_test
+ */
+static void run_test(tester_t *tester, void (*test_function) (tester_t * tester), char * test_name)
+{
+ struct timeval start_time, end_time;
+ long timediff;
+ private_tester_t *this = (private_tester_t *) tester;
+ this->tests_count++;
+ this->failed_asserts_count = 0;
+ fprintf(this->output,"%-55s", test_name);
+ gettimeofday(&start_time,NULL);
+ test_function(tester);
+ gettimeofday(&end_time,NULL);
+ timediff = time_difference(&end_time, &start_time);
+
+ if (this->failed_asserts_count > 0)
+ {
+ fprintf(this->output,"FAILED: %-47s|%10ld us\n",test_name,timediff);
+ }else
+ {
+ fprintf(this->output,"|%10ld us\n",timediff);
+ }
+ if (this->failed_asserts_count > 0)
+ {
+ this->failed_tests_count++;
+ }
+}
+
+/**
+ * Implementation of function assert_true
+ */
+static void assert_true(tester_t *tester, bool to_be_true,char * assert_name)
+{
+ private_tester_t *this = (private_tester_t *) tester;
+
+ if (assert_name == NULL)
+ {
+ assert_name = "unknown";
+ }
+
+ pthread_mutex_lock(&(this->mutex));
+ if (!to_be_true)
+ {
+ this->failed_asserts_count++;
+ fprintf(this->output," check '%s' failed!\n", assert_name);
+ }else
+ {
+ if (this->display_succeeded_asserts)
+ {
+ fprintf(this->output," check '%s' succeeded\n", assert_name);
+ }
+ }
+ pthread_mutex_unlock(&(this->mutex));
+}
+
+/**
+ * Implementation of function assert_false
+ */
+static void assert_false(tester_t *tester, bool to_be_false,char * assert_name)
+{
+ tester->assert_true(tester,(!to_be_false),assert_name);
+}
+
+/**
+ * Implements the destroy function
+ */
+static status_t destroy(tester_t *tester)
+{
+ private_tester_t *this = (private_tester_t*) tester;
+ pthread_mutex_destroy(&(this->mutex));
+ allocator_free(this);
+ return SUCCESS;
+}
+
+tester_t *tester_create(FILE *output, bool display_succeeded_asserts)
+{
+ private_tester_t *this = allocator_alloc_thing(private_tester_t);
+
+ this->public.destroy = destroy;
+ this->public.perform_tests = perform_tests;
+ this->public.perform_test = perform_test;
+ this->public.assert_true = assert_true;
+ this->public.assert_false = assert_false;
+
+
+ this->run_test = run_test;
+ this->display_succeeded_asserts = display_succeeded_asserts;
+ this->failed_tests_count = 0;
+ this->tests_count = 0;
+ this->output = output;
+ pthread_mutex_init(&(this->mutex),NULL);
+
+ return &(this->public);
+}