538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore/*
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * This file and its contents are supplied under the terms of the
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * Common Development and Distribution License ("CDDL"), version 1.0.
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * You may only use this file in accordance with the terms of version
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * 1.0 of the CDDL.
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore *
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * A full copy of the text of the CDDL should have accompanied this
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * source. A copy of the CDDL is also available via the Internet at
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * http://www.illumos.org/license/CDDL.
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore */
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore/*
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore * Copyright 2015 Garrett D'Amore <garrett@damore.org>
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore */
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore/*
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore * Common handling for test programs.
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore */
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#include <stdio.h>
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#include <stdlib.h>
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#include <stdarg.h>
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#include <string.h>
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#include <errno.h>
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#include <pthread.h>
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#include <ctype.h>
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#include <unistd.h>
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#include <sys/param.h>
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#include "test_common.h"
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorestatic int debug = 0;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorestatic int force = 0;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorestatic pthread_mutex_t lk;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amorestatic int passes;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amorestatic int tests;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorestruct test {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore char *name;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore int ntids;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore pthread_t *tids;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore int fails;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore void *arg;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore void (*func)(test_t t, void *);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore};
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorevoid
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_set_debug(void)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore debug++;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorevoid
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_set_force(void)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore force++;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_t
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_start(const char *format, ...)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_list args;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_t t;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore char *s;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t = calloc(1, sizeof (*t));
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_start(args, format);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) vasprintf(&s, format, args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_end(args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) asprintf(&t->name, "%s (%s)", s, ARCH);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore free(s);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_lock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST STARTING %s:\n", t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) fflush(stdout);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_unlock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#ifdef LINT
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore /* We inject references to make avoid name unused warnings */
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_run(0, NULL, NULL, NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_debugf(t, NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_failed(t, NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_passed(t);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_set_debug();
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_set_force();
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_summary();
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) test_load_config(t, NULL, NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore#endif
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore tests++;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore return (t);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorevoid
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_failed(test_t t, const char *format, ...)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_list args;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_lock(&lk);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (t == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) printf("FAILURE: ");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore va_start(args, format);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) vprintf(format, args);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore va_end(args);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) printf("\n");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) fflush(stdout);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) pthread_mutex_unlock(&lk);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore return;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (force || (t->ntids > 0)) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST FAILING %s: ", t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore } else {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST FAILED %s: ", t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_start(args, format);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) vprintf(format, args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_end(args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("\n");
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) fflush(stdout);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_unlock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->fails++;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (!force) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (t->ntids > 0) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore pthread_exit(NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore } else {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) exit(EXIT_FAILURE);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorevoid
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_passed(test_t t)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (t == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore return;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (t->ntids > 0) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (debug) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_lock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST PASSING: %s\n", t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_unlock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore return;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_lock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (t->fails == 0) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore passes++;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST PASS: %s\n", t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore } else {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST FAILED: %d failures\n", t->fails);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) fflush(stdout);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_unlock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore free(t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (t->tids) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore free(t->tids);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore free(t);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amorevoid
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amoretest_summary(void)
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore{
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (passes == tests) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) printf("TEST SUMMARY: %d / %d (ok)\n", passes, tests);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore } else {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) printf("TEST SUMMARY: %d / %d (%d failing)\n",
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore passes, tests, tests - passes);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore}
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorevoid
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_debugf(test_t t, const char *format, ...)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_list args;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore if (!debug)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore return;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_lock(&lk);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (t) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) printf("TEST DEBUG %s: ", t->name);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore } else {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) printf("TEST DEBUG: ");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_start(args, format);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) vprintf(format, args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_end(args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("\n");
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) fflush(stdout);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_unlock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorestatic void *
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_thr_one(void *arg)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_t t = arg;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->func(t, t->arg);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore return (NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amorevoid
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amoretest_run(int nthr, void (*func)(test_t, void *), void *arg,
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore const char *tname, ...)
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore{
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_t t;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore char *s;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_list args;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t = calloc(1, sizeof (*t));
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->ntids = nthr;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->tids = calloc(nthr, sizeof (pthread_t));
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->func = func;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->arg = arg;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_start(args, tname);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) vasprintf(&s, tname, args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore va_end(args);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) asprintf(&t->name, "%s (%s)", s, ARCH);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore free(s);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_lock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) printf("TEST STARTING %s:\n", t->name);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) fflush(stdout);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_mutex_unlock(&lk);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_debugf(t, "running %d threads", nthr);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore for (int i = 0; i < nthr; i++) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_debugf(t, "started thread %d", i);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_create(&t->tids[i], NULL, test_thr_one, t);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore for (int i = 0; i < nthr; i++) {
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore (void) pthread_join(t->tids[i], NULL);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_debugf(t, "thread %d joined", i);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore t->ntids--;
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore }
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore test_passed(t);
538aa54d819fa7751ca82bcc30d4ed8c57ec2ef2Garrett D'Amore}
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amorevoid
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amoretest_trim(char **ptr)
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore{
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char *p = *ptr;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore while (isspace(*p)) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore p++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore *ptr = p;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore p += strlen(p);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore while ((--p >= *ptr) && (isspace(*p))) {
fc2512cfb727d49529d8ed99164db871f4829b73Robert Mustacchi *p = '\0';
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore}
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#define MAXCB 20
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore#define MAXFIELD 20
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amoreint
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amoretest_load_config(test_t t, const char *fname, ...)
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore{
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore va_list va;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore const char *keyws[MAXCB];
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_cfg_func_t callbs[MAXCB];
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char *fields[MAXFIELD];
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore int nfields;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore FILE *cfg;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char line[1024];
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char buf[1024];
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore int done;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char *ptr;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char *tok;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char *err;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore int lineno;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore int rv;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore int found;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char path[MAXPATHLEN];
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore int i;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore va_start(va, fname);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore for (i = 0; i < MAXCB; i++) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore keyws[i] = (const char *)va_arg(va, const char *);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (keyws[i] == NULL)
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore break;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore callbs[i] = (test_cfg_func_t)va_arg(va, test_cfg_func_t);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore va_end(va);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (i == MAXCB) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_debugf(t, "too many arguments to function >= %d", MAXCB);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore found = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (access(fname, F_OK) == 0) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore found++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (!found && fname[0] != '/') {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore char *stf = getenv("STF_SUITE");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (stf == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore stf = "../..";
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) snprintf(path, sizeof (path), "%s/cfg/%s", stf, fname);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (access(path, F_OK) == 0) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore fname = path;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore found++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore } else {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) snprintf(path, sizeof (path), "cfg/%s", fname);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (access(path, F_OK) == 0) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore fname = path;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore found++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if ((cfg = fopen(fname, "r")) == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_failed(t, "open(%s): %s", fname, strerror(errno));
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore return (-1);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore line[0] = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore done = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore lineno = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore while (!done) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore lineno++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (fgets(buf, sizeof (buf), cfg) == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore done++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore } else {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) strtok(buf, "\n");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if ((*buf != 0) && (buf[strlen(buf)-1] == '\\')) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore /*
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore * Continuation. This isn't quite right,
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore * as it doesn't allow for a "\" at the
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore * end of line (no escaping).
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore */
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore buf[strlen(buf)-1] = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) strlcat(line, buf, sizeof (line));
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore continue;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) strlcat(line, buf, sizeof (line));
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore /* got a line */
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore ptr = line;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_trim(&ptr);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore /* skip comments and empty lines */
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (ptr[0] == 0 || ptr[0] == '#') {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore line[0] = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore continue;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore tok = strsep(&ptr, "|");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (tok == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore break;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_trim(&tok);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore for (nfields = 0; nfields < MAXFIELD; nfields++) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore fields[nfields] = strsep(&ptr, "|");
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (fields[nfields] == NULL) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore break;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_trim(&fields[nfields]);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore found = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore rv = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore for (int i = 0; keyws[i] != NULL; i++) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (strcmp(tok, keyws[i]) == 0) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore found++;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore err = NULL;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore rv = callbs[i](fields, nfields, &err);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (!found) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore rv = -1;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore err = NULL;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) asprintf(&err, "unknown keyword %s", tok);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (rv != 0) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore if (err) {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_failed(t, "%s:%d: %s", fname,
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore lineno, err);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore free(err);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore } else {
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore test_failed(t, "%s:%d: unknown error",
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore fname, lineno);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) fclose(cfg);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore return (rv);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore line[0] = 0;
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore }
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore (void) fclose(cfg);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore return (0);
de572d98af8238405c5d1292a788b1a85b0c68ebGarrett D'Amore}