Blob Blame History Raw
#define _GNU_SOURCE
#include <unistd.h>
#include <sys/wait.h>
#include <errno.h>
#include <string.h>
#include "test.h"

#define TEST(c, ...) ( (c) || (t_error(#c " failed: " __VA_ARGS__),0) )

static int w(pid_t pid)
{
	int r, s;
	r = waitpid(pid, &s, 0);
	if (r == -1)
		t_error("waitpid failed: %s\n", strerror(errno));
	else if (r != pid)
		t_error("child pid was %d, waitpid returned %d\n", pid, r);
	else
		return s;
	return -1;
}

static void test_exit(int code)
{
	pid_t pid;
	if((pid = vfork()) == 0) {
		_exit(code);
		t_error("exit failed: %s\n", strerror(errno));
	}
	if (pid == -1) {
		t_error("vfork failed: %s\n", strerror(errno));
		return;
	}
	int r = w(pid);
	TEST(WIFEXITED(r), "child terminated abnormally\n");
	TEST(WEXITSTATUS(r) == code, "child exited with %d, expected %d\n", WEXITSTATUS(r), code);
}

static int sh(const char *cmd)
{
	pid_t pid;
	if((pid = vfork()) == 0) {
		execl("/bin/sh", "/bin/sh", "-c", cmd, (char*)0);
		t_error("execl failed: %s\n", strerror(errno));
		_exit(1);
	}
	if (pid == -1) {
		t_error("vfork failed: %s\n", strerror(errno));
		return -1;
	}
	return w(pid);
}

static void test_shell_exit(const char *cmd, int code)
{
	int r = sh(cmd);
	TEST(WIFEXITED(r), "child terminated abnormally\n");
	TEST(WEXITSTATUS(r) == code, "child exited with %d, expected %d\n", WEXITSTATUS(r), code);
}

static void test_shell_kill(const char *cmd, int sig)
{
	int r = sh(cmd);
	TEST(WIFSIGNALED(r), "child did not get killed\n");
	TEST(WTERMSIG(r) == sig, "child is killed by %d, expected %d\n", WTERMSIG(r), sig);
}

int main() {
	test_exit(0);
	test_exit(1);
	test_shell_exit("exit 0", 0);
	test_shell_exit("exit 1", 1);
	test_shell_kill("kill -9 $$", 9);
	return t_status;
}