From 74244e5b3ed4a61d99c5fc0967b69e5c9a753456 Mon Sep 17 00:00:00 2001 From: Rich Felker Date: Fri, 30 Aug 2019 16:21:36 -0400 Subject: [PATCH] add posix_spawn [f]chdir file actions these are presently extensions, thus named with _np to match glibc and other implementations that provide them; however they are likely to be standardized in the future without the _np suffix as a result of Austin Group issue 1208. if so, both names will be kept as aliases. --- include/spawn.h | 5 +++++ src/process/fdop.h | 2 ++ src/process/posix_spawn.c | 8 ++++++++ .../posix_spawn_file_actions_addchdir.c | 18 ++++++++++++++++++ .../posix_spawn_file_actions_addfchdir.c | 17 +++++++++++++++++ 5 files changed, 50 insertions(+) create mode 100644 src/process/posix_spawn_file_actions_addchdir.c create mode 100644 src/process/posix_spawn_file_actions_addfchdir.c diff --git a/include/spawn.h b/include/spawn.h index c9bd1939..8eb73e00 100644 --- a/include/spawn.h +++ b/include/spawn.h @@ -71,6 +71,11 @@ int posix_spawn_file_actions_addopen(posix_spawn_file_actions_t *__restrict, int int posix_spawn_file_actions_addclose(posix_spawn_file_actions_t *, int); int posix_spawn_file_actions_adddup2(posix_spawn_file_actions_t *, int, int); +#if defined(_BSD_SOURCE) || defined(_GNU_SOURCE) +int posix_spawn_file_actions_addchdir_np(posix_spawn_file_actions_t *__restrict, const char *__restrict); +int posix_spawn_file_actions_addfchdir_np(posix_spawn_file_actions_t *, int); +#endif + #ifdef __cplusplus } #endif diff --git a/src/process/fdop.h b/src/process/fdop.h index 00b87514..5adf1443 100644 --- a/src/process/fdop.h +++ b/src/process/fdop.h @@ -1,6 +1,8 @@ #define FDOP_CLOSE 1 #define FDOP_DUP2 2 #define FDOP_OPEN 3 +#define FDOP_CHDIR 4 +#define FDOP_FCHDIR 5 struct fdop { struct fdop *next, *prev; diff --git a/src/process/posix_spawn.c b/src/process/posix_spawn.c index 306faa05..29652197 100644 --- a/src/process/posix_spawn.c +++ b/src/process/posix_spawn.c @@ -125,6 +125,14 @@ static int child(void *args_vp) __syscall(SYS_close, fd); } break; + case FDOP_CHDIR: + ret = __syscall(SYS_chdir, op->path); + if (ret<0) goto fail; + break; + case FDOP_FCHDIR: + ret = __syscall(SYS_fchdir, op->fd); + if (ret<0) goto fail; + break; } } } diff --git a/src/process/posix_spawn_file_actions_addchdir.c b/src/process/posix_spawn_file_actions_addchdir.c new file mode 100644 index 00000000..7f2590ae --- /dev/null +++ b/src/process/posix_spawn_file_actions_addchdir.c @@ -0,0 +1,18 @@ +#include +#include +#include +#include +#include "fdop.h" + +int posix_spawn_file_actions_addchdir_np(posix_spawn_file_actions_t *restrict fa, const char *restrict path) +{ + struct fdop *op = malloc(sizeof *op + strlen(path) + 1); + if (!op) return ENOMEM; + op->cmd = FDOP_CHDIR; + op->fd = -1; + strcpy(op->path, path); + if ((op->next = fa->__actions)) op->next->prev = op; + op->prev = 0; + fa->__actions = op; + return 0; +} diff --git a/src/process/posix_spawn_file_actions_addfchdir.c b/src/process/posix_spawn_file_actions_addfchdir.c new file mode 100644 index 00000000..436c683d --- /dev/null +++ b/src/process/posix_spawn_file_actions_addfchdir.c @@ -0,0 +1,17 @@ +#include +#include +#include +#include +#include "fdop.h" + +int posix_spawn_file_actions_addfchdir_np(posix_spawn_file_actions_t *fa, int fd) +{ + struct fdop *op = malloc(sizeof *op); + if (!op) return ENOMEM; + op->cmd = FDOP_FCHDIR; + op->fd = fd; + if ((op->next = fa->__actions)) op->next->prev = op; + op->prev = 0; + fa->__actions = op; + return 0; +}