unix/fiss

simplifying fork-exec with fork_dup_cd_exec() (e07e59d949f9441cac8fbf8bb4edeb475226441d)
Repositories | LICENSE

commit e07e59d949f9441cac8fbf8bb4edeb475226441d
parent 18905fe60a0155510de380024ba6c54b37a3c3df
Author: Friedel Schön <[email protected]>
Date:   Wed, 10 May 2023 15:16:30 +0200

simplifying fork-exec with fork_dup_cd_exec()

Diffstat:
Minclude/util.h6++++--
Msrc/restart.c15+++------------
Msrc/start.c19++++---------------
Msrc/stop.c15++++-----------
Msrc/util.c20++++++++++++++++++++
5 files changed, 35 insertions(+), 40 deletions(-)

diff --git a/include/util.h b/include/util.h @@ -17,4 +17,6 @@ ssize_t dgetline(int fd, char* line, size_t line_buffer); ssize_t readstr(int fd, char* str); ssize_t writestr(int fd, const char* str); -unsigned int stat_mode(const char* format, ...); -\ No newline at end of file +unsigned int stat_mode(const char* format, ...); + +int fork_dup_cd_exec(int dir, const char* path, int fd0, int fd1, int fd2); +\ No newline at end of file diff --git a/src/restart.c b/src/restart.c @@ -14,18 +14,9 @@ static void do_finish(service_t* s) { struct stat st; if (fstatat(s->dir, "finish", &st, 0) != -1 && st.st_mode & S_IEXEC) { s->state = STATE_FINISHING; - if ((s->pid = fork()) == -1) { - print_error("error: cannot fork process: %s\n"); - } else if (s->pid == 0) { - dup2(null_fd, STDIN_FILENO); - dup2(null_fd, STDOUT_FILENO); - dup2(null_fd, STDERR_FILENO); - - fchdir(s->dir); - - execl("./finish", "./finish", NULL); - print_error("error: cannot execute finish process: %s\n"); - _exit(1); + if ((s->pid = fork_dup_cd_exec(s->dir, "./finish", null_fd, null_fd, null_fd)) == -1) { + print_error("error: cannot execute ./finish: %s\n"); + s->state = STATE_INACTIVE; } } else if (s->fail_count == SV_FAIL_MAX) { s->state = STATE_DEAD; diff --git a/src/start.c b/src/start.c @@ -101,9 +101,7 @@ void service_run(service_t* s) { if (setsid() == -1) print_error("error: cannot setsid: %s\n"); - if (fchdir(s->dir) == -1) - print_error("error: chdir failed: %s\n"); - + fchdir(s->dir); set_pipes(s); char args[SV_ARGUMENTS_MAX][SV_PARAM_FILE_LINE_MAX]; @@ -154,18 +152,9 @@ void service_start(service_t* s, bool* changed) { struct stat st; if (fstatat(s->dir, "setup", &st, 0) != -1 && st.st_mode & S_IXUSR) { s->state = STATE_SETUP; - if ((s->pid = fork()) == -1) { - print_error("error: cannot fork process: %s\n"); - } else if (s->pid == 0) { - dup2(null_fd, STDIN_FILENO); - dup2(null_fd, STDOUT_FILENO); - dup2(null_fd, STDERR_FILENO); - - fchdir(s->dir); - - execl("./setup", "./setup", NULL); - print_error("error: cannot execute setup process: %s\n"); - _exit(1); + if ((s->pid = fork_dup_cd_exec(s->dir, "./setup", null_fd, null_fd, null_fd)) == -1) { + print_error("error: cannot execute ./setup: %s\n"); + s->state = STATE_INACTIVE; } } else { service_run(s); diff --git a/src/stop.c b/src/stop.c @@ -1,4 +1,5 @@ #include "service.h" +#include "util.h" #include <errno.h> #include <limits.h> @@ -24,17 +25,9 @@ void service_stop(service_t* s, bool* changed) { break; case STATE_ACTIVE_BACKGROUND: s->state = STATE_STOPPING; - if ((s->pid = fork()) == -1) { - print_error("error: cannot fork process: %s\n"); - } else if (s->pid == 0) { - dup2(null_fd, STDIN_FILENO); - dup2(null_fd, STDOUT_FILENO); - dup2(null_fd, STDERR_FILENO); - - fchdir(s->dir); - execl("./stop", "./stop", NULL); - print_error("error: cannot execute stop process: %s\n"); - _exit(1); + if ((s->pid = fork_dup_cd_exec(s->dir, "./stop", null_fd, null_fd, null_fd)) == -1) { + print_error("error: cannot execute ./stop: %s\n"); + s->state = STATE_INACTIVE; } if (changed) *changed = true; diff --git a/src/util.c b/src/util.c @@ -1,5 +1,6 @@ #include "util.h" +#include <errno.h> #include <limits.h> #include <stdarg.h> #include <string.h> @@ -54,4 +55,23 @@ unsigned int stat_mode(const char* format, ...) { return st.st_mode; return 0; +} + +int fork_dup_cd_exec(int dir, const char* path, int fd0, int fd1, int fd2) { + pid_t pid; + if ((pid = fork()) == -1) { + print_error("error: cannot fork process: %s\n"); + return -1; + } else if (pid == 0) { + dup2(fd0, STDIN_FILENO); + dup2(fd1, STDOUT_FILENO); + dup2(fd2, STDERR_FILENO); + + fchdir(dir); + + execl(path, path, NULL); + print_error("error: cannot execute stop process: %s\n"); + _exit(1); + } + return pid; } \ No newline at end of file