diff options
| author | Laurent Bercot <ska-skaware@skarnet.org> | 2025-08-19 15:31:17 +0000 |
|---|---|---|
| committer | Laurent Bercot <ska@appnovation.com> | 2025-08-19 15:31:17 +0000 |
| commit | 982dbebfa6703c2ea5c3dd65dfaf46657fac8aa8 (patch) | |
| tree | f5e14205c9de088e850580f689a55983c80c01ef /src/libenvexec/env_dump4.c | |
| parent | 4e30b6899578bae9343bacf8840a80bf1ada0f2a (diff) | |
| download | skalibs-982dbebfa6703c2ea5c3dd65dfaf46657fac8aa8.tar.gz | |
Add qsortr, env_dump4
Signed-off-by: Laurent Bercot <ska@appnovation.com>
Diffstat (limited to 'src/libenvexec/env_dump4.c')
| -rw-r--r-- | src/libenvexec/env_dump4.c | 55 |
1 files changed, 55 insertions, 0 deletions
diff --git a/src/libenvexec/env_dump4.c b/src/libenvexec/env_dump4.c new file mode 100644 index 0000000..b59c858 --- /dev/null +++ b/src/libenvexec/env_dump4.c @@ -0,0 +1,55 @@ +/* ISC license. */ + +#include <skalibs/nonposix.h> +#include <unistd.h> +#include <sys/stat.h> +#include <string.h> +#include <errno.h> +#include <stdlib.h> +#include <stdio.h> + +#include <skalibs/bytestr.h> +#include <skalibs/env.h> +#include <skalibs/djbunix.h> + + /* XXX: breaks layering, but really openat() should be supported everywhere */ +#include <skalibs/unix-transactional.h> + +#define SUFFIX ":envdump:XXXXXX" + +int env_dump4 (char const *dir, mode_t mode, char const *const *envp, int nochomp) +{ + int fd ; + size_t dirlen = strlen(dir) ; + char tmpdir[dirlen + sizeof(SUFFIX)] ; + memcpy(tmpdir, dir, dirlen) ; + memcpy(tmpdir + dirlen, SUFFIX, sizeof(SUFFIX)) ; + if (!mkdtemp(tmpdir)) return 0 ; + fd = open_read(tmpdir) ; + if (fd == -1) goto err ; + for (; *envp ; envp++) + { + size_t len = str_chr(*envp, '=') ; + size_t vallen = strlen(*envp + len + 1) ; + char fn[len + 1 + !nochomp] ; + memcpy(fn, *envp, len) ; + if (nochomp) fn[len] = 0 ; + else { fn[len] = '\n' ; fn[len+1] = 0 ; } + len = openwritenclose_at(fd, fn, *envp + len + 1 + !nochomp, vallen) ; + if (len < vallen) goto cerr ; + } + fd_close(fd) ; + if (chmod(tmpdir, mode) == -1) goto err ; + if (rename(tmpdir, dir) == -1) goto err ; + return 1 ; + + cerr: + fd_close(fd) ; + err: + { + int e = errno ; + rm_rf(tmpdir) ; + errno = e ; + } + return 0 ; +} |
