diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2021-06-15 21:03:51 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2021-06-15 21:03:51 -0700 |
commit | 5376b586a03b83a65f4cd40e7d982c9b8019c90a (patch) | |
tree | 83a267966a447e7fa2c6e428ee9addc2801478c3 /stream.c | |
parent | ca118fac3a97a6cd6b9342d301b3b06c515ad5f1 (diff) | |
download | txr-5376b586a03b83a65f4cd40e7d982c9b8019c90a.tar.gz txr-5376b586a03b83a65f4cd40e7d982c9b8019c90a.tar.bz2 txr-5376b586a03b83a65f4cd40e7d982c9b8019c90a.zip |
subprocesses: move fds_swizzle to child process.
In all cases in which we control fork and exec, we should be
doing the fds_swizzle setup in the child process. This has
several benefits. We do not disturb the file descriptor layout
of the parent. We don't have to set up a catch to do the
cleanup. We don't have to do the clean-up in the child either;
but just let it terminate.
* stream.c (struct save_fds): New members subin, subout and
suberr to hold the substitute file descriptors. The reason for
this is so that we can calculate these in the parent process
so that all the error checking is done in the parent and and
carry them over to the child somehow. This structure is the
natural place for that.
(fds_getfd): New static function: just does the job of
fetching and validating the file descriptor from the given
stream: the first part of fds_subst. This will be done in the
parent.
(fds_subst): Now just does the substitution using a given file
descriptor. Done in the child. Some variable renaming here; a
better name for fd_orig is fd_sub, the substitute descriptor.
(fds_prepare): New function: called in the parent process,
it obtains the three file descriptors from the streams.
(fds_swizzle): Just do fd_subst with the ready-made file
descriptors, not dealing with the streams. Called in the
child.
(open_command): Add fds_prepare call that is now needed.
(open_subprocess, run): Get rid of catch frame. Just call
fds_prepare where fds_swizzle was previously called, and only
call fds_swizzle in the child process.
Diffstat (limited to 'stream.c')
-rw-r--r-- | stream.c | 70 |
1 files changed, 41 insertions, 29 deletions
@@ -4234,6 +4234,9 @@ struct save_fds { volatile int in; volatile int out; volatile int err; + volatile int subin; + volatile int subout; + volatile int suberr; }; #define FDS_IN 1 @@ -4242,27 +4245,32 @@ struct save_fds { static void fds_init(struct save_fds *fds) { - fds->in = fds->out = fds->err = -1; + fds->in = fds->out = fds->err = fds->subin = fds->subin = fds->subin = -1; } -static int fds_subst(val stream, int fd_std, val self) +static int fds_getfd(val stream, val self) { val sfd = stream_fd(stream); - int fd_orig = if3(integerp(sfd), c_num(sfd, self), INT_MIN); + int fd_sub = if3(integerp(sfd), c_num(sfd, self), INT_MIN); - if (fd_orig == INT_MIN) + if (fd_sub == INT_MIN) uw_throwf(file_error_s, lit("~a: (fileno ~s) is ~s, which is unusable"), self, stream, sfd, nao); - if (fd_orig == fd_std) + return fd_sub; +} + +static int fds_subst(int fd_sub, int fd_std, val self) +{ + if (fd_sub == fd_std) return -1; { int fd_dup = dup(fd_std); if (fd_dup != -1) { - dup2(fd_orig, fd_std); + dup2(fd_sub, fd_std); return fd_dup; } @@ -4271,16 +4279,28 @@ static int fds_subst(val stream, int fd_std, val self) } } +static void fds_prepare(struct save_fds *fds, int flags, val self) +{ + if ((flags & FDS_IN) != 0) + fds->subin = fds_getfd(std_input, self); + + if ((flags & FDS_OUT) != 0) + fds->subout = fds_getfd(std_output, self); + + if ((flags & FDS_ERR) != 0) + fds->suberr = fds_getfd(std_error, self); +} + static void fds_swizzle(struct save_fds *fds, int flags, val self) { if ((flags & FDS_IN) != 0) - fds->in = fds_subst(std_input, STDIN_FILENO, self); + fds->in = fds_subst(fds->subin, STDIN_FILENO, self); if ((flags & FDS_OUT) != 0) - fds->out = fds_subst(std_output, STDOUT_FILENO, self); + fds->out = fds_subst(fds->subout, STDOUT_FILENO, self); if ((flags & FDS_ERR) != 0) - fds->err = fds_subst(std_error, STDERR_FILENO, self); + fds->err = fds_subst(fds->suberr, STDERR_FILENO, self); } static void fds_restore(struct save_fds *fds) @@ -4301,7 +4321,6 @@ static void fds_restore(struct save_fds *fds) } } - val open_command(val path, val mode_str) { val self = lit("open-command"); @@ -4310,12 +4329,15 @@ val open_command(val path, val mode_str) int input = m.read != 0; struct save_fds sfds; FILE *f = 0; + int fds_flags = (input ? FDS_IN : FDS_OUT) | FDS_ERR; fds_init(&sfds); uw_simple_catch_begin; - fds_swizzle(&sfds, (input ? FDS_IN : FDS_OUT) | FDS_ERR, self); + fds_prepare(&sfds, fds_flags, self); + + fds_swizzle(&sfds, fds_flags, self); f = w_popen(c_str(path), c_str(mode)); @@ -4348,6 +4370,7 @@ static val open_subprocess(val name, val mode_str, val args, val fun) int i, nargs; struct save_fds sfds; val ret = nil; + int fds_flags = (input ? FDS_IN : FDS_OUT) | FDS_ERR; args = default_null_arg(args); fun = default_null_arg(fun); @@ -4358,9 +4381,7 @@ static val open_subprocess(val name, val mode_str, val args, val fun) fds_init(&sfds); - uw_simple_catch_begin; - - fds_swizzle(&sfds, (input ? FDS_IN : FDS_OUT) | FDS_ERR, self); + fds_prepare(&sfds, fds_flags, self); if (nargs < 0 || nargs == INT_MAX) uw_throwf(error_s, lit("~a: argument list overflow"), self, nao); @@ -4397,6 +4418,8 @@ static val open_subprocess(val name, val mode_str, val args, val fun) } if (pid == 0) { + fds_swizzle(&sfds, fds_flags, self); + if (input) { dup2(fd[1], STDOUT_FILENO); if (fd[1] != STDOUT_FILENO) /* You never know */ @@ -4482,12 +4505,6 @@ static val open_subprocess(val name, val mode_str, val args, val fun) ret = set_mode_props(m, make_pipevp_stream(f, name, pid)); } - uw_unwind { - fds_restore(&sfds); - } - - uw_catch_end; - return ret; } @@ -4631,6 +4648,8 @@ static val run(val command, val args) uw_simple_catch_begin; + fds_prepare(&sfds, FDS_IN | FDS_OUT | FDS_ERR, self); + fds_swizzle(&sfds, FDS_IN | FDS_OUT | FDS_ERR, self); if (nargs < 0 || nargs == INT_MAX) @@ -4694,9 +4713,7 @@ static val run(val name, val args) fds_init(&sfds); - uw_simple_catch_begin; - - fds_swizzle(&sfds, FDS_IN | FDS_OUT | FDS_ERR, self); + fds_prepare(&sfds, FDS_IN | FDS_OUT | FDS_ERR, self); pid = fork(); @@ -4709,6 +4726,7 @@ static val run(val name, val args) } if (pid == 0) { + fds_swizzle(&sfds, FDS_IN | FDS_OUT | FDS_ERR, self); execvp(argv[0], argv); _exit(errno); } else { @@ -4731,12 +4749,6 @@ static val run(val name, val args) } out: - uw_unwind { - fds_restore(&sfds); - } - - uw_catch_end; - return ret; } |