|
Szabolcs Nagy |
814c6f |
#include <stdlib.h>
|
|
Szabolcs Nagy |
814c6f |
#include <string.h>
|
|
Szabolcs Nagy |
814c6f |
#include <errno.h>
|
|
Szabolcs Nagy |
814c6f |
#include <signal.h>
|
|
Szabolcs Nagy |
814c6f |
#include <time.h>
|
|
Szabolcs Nagy |
814c6f |
#include <sys/types.h>
|
|
Szabolcs Nagy |
814c6f |
#include <sys/wait.h>
|
|
Szabolcs Nagy |
814c6f |
#include <sys/time.h>
|
|
Szabolcs Nagy |
814c6f |
#include <sys/resource.h>
|
|
Szabolcs Nagy |
814c6f |
#include <unistd.h>
|
|
Szabolcs Nagy |
814c6f |
#include "test.h"
|
|
Szabolcs Nagy |
814c6f |
|
|
Szabolcs Nagy |
814c6f |
static void handler(int s)
|
|
Szabolcs Nagy |
814c6f |
{
|
|
Szabolcs Nagy |
814c6f |
}
|
|
Szabolcs Nagy |
814c6f |
|
|
Szabolcs Nagy |
7ef8af |
static int start(char *wrap, char *argv[])
|
|
Szabolcs Nagy |
814c6f |
{
|
|
Szabolcs Nagy |
814c6f |
int pid;
|
|
Szabolcs Nagy |
814c6f |
|
|
Szabolcs Nagy |
814c6f |
pid = fork();
|
|
Szabolcs Nagy |
814c6f |
if (pid == 0) {
|
|
Szabolcs Nagy |
157cc4 |
t_setrlim(RLIMIT_STACK, 100*1024);
|
|
Szabolcs Nagy |
7ef8af |
if (*wrap) {
|
|
Szabolcs Nagy |
7ef8af |
argv--;
|
|
Szabolcs Nagy |
7ef8af |
argv[0] = wrap;
|
|
Szabolcs Nagy |
7ef8af |
}
|
|
Szabolcs Nagy |
ea83e5 |
execv(argv[0], argv);
|
|
Szabolcs Nagy |
814c6f |
t_error("%s exec failed: %s\n", argv[0], strerror(errno));
|
|
Szabolcs Nagy |
814c6f |
exit(1);
|
|
Szabolcs Nagy |
814c6f |
}
|
|
Szabolcs Nagy |
814c6f |
return pid;
|
|
Szabolcs Nagy |
814c6f |
}
|
|
Szabolcs Nagy |
814c6f |
|
|
Szabolcs Nagy |
7ef8af |
static void usage(char *argv[])
|
|
Szabolcs Nagy |
7ef8af |
{
|
|
Szabolcs Nagy |
7ef8af |
t_error("usage: %s [-t timeoutsec] [-w wrapcmd] cmd [args..]\n", argv[0]);
|
|
Szabolcs Nagy |
7ef8af |
exit(-1);
|
|
Szabolcs Nagy |
7ef8af |
}
|
|
Szabolcs Nagy |
7ef8af |
|
|
Szabolcs Nagy |
814c6f |
int main(int argc, char *argv[])
|
|
Szabolcs Nagy |
814c6f |
{
|
|
Szabolcs Nagy |
7ef8af |
char *wrap = "";
|
|
Szabolcs Nagy |
7ef8af |
int timeoutsec = 5;
|
|
Szabolcs Nagy |
7ef8af |
int timeout = 0;
|
|
Szabolcs Nagy |
814c6f |
int status;
|
|
Szabolcs Nagy |
814c6f |
sigset_t set;
|
|
Szabolcs Nagy |
7ef8af |
int opt;
|
|
Szabolcs Nagy |
814c6f |
int pid;
|
|
Szabolcs Nagy |
814c6f |
|
|
Szabolcs Nagy |
7ef8af |
while ((opt = getopt(argc, argv, "w:t:")) != -1) {
|
|
Szabolcs Nagy |
7ef8af |
switch (opt) {
|
|
Szabolcs Nagy |
7ef8af |
case 'w':
|
|
Szabolcs Nagy |
7ef8af |
wrap = optarg;
|
|
Szabolcs Nagy |
7ef8af |
break;
|
|
Szabolcs Nagy |
7ef8af |
case 't':
|
|
Szabolcs Nagy |
7ef8af |
timeoutsec = atoi(optarg);
|
|
Szabolcs Nagy |
7ef8af |
break;
|
|
Szabolcs Nagy |
7ef8af |
default:
|
|
Szabolcs Nagy |
7ef8af |
usage(argv);
|
|
Szabolcs Nagy |
7ef8af |
}
|
|
Szabolcs Nagy |
814c6f |
}
|
|
Szabolcs Nagy |
7ef8af |
if (optind >= argc)
|
|
Szabolcs Nagy |
7ef8af |
usage(argv);
|
|
Szabolcs Nagy |
7ef8af |
argv += optind;
|
|
Szabolcs Nagy |
814c6f |
sigemptyset(&set);
|
|
Szabolcs Nagy |
814c6f |
sigaddset(&set, SIGCHLD);
|
|
Szabolcs Nagy |
814c6f |
sigprocmask(SIG_BLOCK, &set, 0);
|
|
Szabolcs Nagy |
814c6f |
signal(SIGCHLD, handler);
|
|
Szabolcs Nagy |
7ef8af |
pid = start(wrap, argv);
|
|
Szabolcs Nagy |
7ef8af |
if (pid == -1) {
|
|
Szabolcs Nagy |
7ef8af |
t_error("%s fork failed: %s\n", argv[0], strerror(errno));
|
|
Szabolcs Nagy |
7ef8af |
t_printf("FAIL %s [internal]\n", argv[0]);
|
|
Szabolcs Nagy |
7ef8af |
return -1;
|
|
Szabolcs Nagy |
7ef8af |
}
|
|
Szabolcs Nagy |
7ef8af |
if (sigtimedwait(&set, 0, &(struct timespec){timeoutsec,0}) == -1) {
|
|
Szabolcs Nagy |
814c6f |
if (errno == EAGAIN)
|
|
Szabolcs Nagy |
814c6f |
timeout = 1;
|
|
Szabolcs Nagy |
1291ba |
else
|
|
Szabolcs Nagy |
1291ba |
t_error("%s sigtimedwait failed: %s\n", argv[0], strerror(errno));
|
|
Szabolcs Nagy |
814c6f |
if (kill(pid, SIGKILL) == -1)
|
|
Szabolcs Nagy |
814c6f |
t_error("%s kill failed: %s\n", argv[0], strerror(errno));
|
|
Szabolcs Nagy |
814c6f |
}
|
|
Szabolcs Nagy |
814c6f |
if (waitpid(pid, &status, 0) != pid) {
|
|
Szabolcs Nagy |
814c6f |
t_error("%s waitpid failed: %s\n", argv[0], strerror(errno));
|
|
Szabolcs Nagy |
7ef8af |
t_printf("FAIL %s [internal]\n", argv[0]);
|
|
Szabolcs Nagy |
814c6f |
return -1;
|
|
Szabolcs Nagy |
814c6f |
}
|
|
Szabolcs Nagy |
814c6f |
if (WIFEXITED(status)) {
|
|
Szabolcs Nagy |
814c6f |
if (WEXITSTATUS(status) == 0)
|
|
Szabolcs Nagy |
29ee9a |
return t_status;
|
|
Szabolcs Nagy |
814c6f |
t_printf("FAIL %s [status %d]\n", argv[0], WEXITSTATUS(status));
|
|
Szabolcs Nagy |
814c6f |
} else if (timeout) {
|
|
Szabolcs Nagy |
814c6f |
t_printf("FAIL %s [timed out]\n", argv[0]);
|
|
Szabolcs Nagy |
814c6f |
} else if (WIFSIGNALED(status)) {
|
|
Szabolcs Nagy |
814c6f |
t_printf("FAIL %s [signal %s]\n", argv[0], strsignal(WTERMSIG(status)));
|
|
Szabolcs Nagy |
814c6f |
} else
|
|
Szabolcs Nagy |
814c6f |
t_printf("FAIL %s [unknown]\n", argv[0]);
|
|
Szabolcs Nagy |
814c6f |
return 1;
|
|
Szabolcs Nagy |
814c6f |
}
|