Blob Blame History Raw
// commit: 19e35c500bd2b5e6146e42705ab9b69c155a2006 2011-02-17
// commit: 187fe29d5b89644b68cade75a34257a1c32a75f6 2011-02-17
// non-standard musl specific behaviour
// daemon should not fork in case of failure of chdir or open, but
// since setsid and fork may still fail after fork this behaviour
// is not very useful
#define _DEFAULT_SOURCE 1
#define _BSD_SOURCE 1
#include <string.h>
#include <errno.h>
#include <sys/wait.h>
#include <fcntl.h>
#include <unistd.h>
#include "test.h"

int daemon(int, int);

int main(void)
{
	int r, pid, fd[2], fdout, s;
	char c;

	r = pipe(fd);
	if (r == -1) {
		t_error("pipe failed: %s\n", strerror(errno));
		return 1;
	}
	fdout = dup(1);
	if (fdout == -1) {
		t_error("dup(1) failed: %s\n", strerror(errno));
		return 1;
	}
	r = fork();
	if (r == -1) {
		t_error("fork failed: %s\n", strerror(errno));
		return 1;
	}

	if (r == 0) {
		/* exhausting all fds makes open("/dev/null") fail in daemon */
		t_fdfill();
		pid = getpid();
		errno = 0;
		r = daemon(0, 0);
		if (dup2(fdout,1) == -1) {
			write(fdout, "ERROR:\n", 7);
			t_error("failed to dup pipe fd for communicating results: %s\n", strerror(errno));
		}
		if (r != -1)
			t_error("daemon should have failed\n");
		if (errno != EMFILE)
			t_error("daemon should have failed with %d [EMFILE] got %d [%s]\n", EMFILE, errno, strerror(errno));
		if (getpid() != pid || getppid() == 1)
			t_error("daemon forked despite failure: ppid is %d, pid is %d, old pid is %d\n",
				getppid(), getpid(), pid);
		if (write(fd[1], "1" + !t_status, 1) != 1)
			t_error("write failed: %s\n", strerror(errno));
		return t_status;
	}
	close(fd[1]);
	if (waitpid(r, &s, 0) != r)
		t_error("waitpid failed: %s\n", strerror(errno));
	else if (!WIFEXITED(s))
		t_error("child exited abnormally (signal %d)\n", WIFSIGNALED(s) ? WTERMSIG(s) : 0);
	else if (WEXITSTATUS(s))
		t_error("child exited with %d\n", WEXITSTATUS(s));
	r = read(fd[0], &c, 1);
	if (r == -1)
		t_error("read failed: %s\n", strerror(errno));
	else if (r == 0)
		t_error("read failed: child did not send its exit status\n");
	else if (c != 0)
		t_error("child failed\n");

	return t_status;
}