// 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 #include #include #include #include #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; }