diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2021-09-26 10:17:00 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2021-09-26 10:17:00 -0700 |
commit | da59e8ce3fc5a35b3d22dd971ff96f498f655a98 (patch) | |
tree | c29c855e17d545e71d657e5630cc6ee364f7f6d8 /sysif.c | |
parent | ba75167d1e7c609b829978b3f4a748ce3cf69ad9 (diff) | |
download | txr-da59e8ce3fc5a35b3d22dd971ff96f498f655a98.tar.gz txr-da59e8ce3fc5a35b3d22dd971ff96f498f655a98.tar.bz2 txr-da59e8ce3fc5a35b3d22dd971ff96f498f655a98.zip |
New variable: *child-env*.
This specifies the environment to be used for executing
programs.
* stream.c (open_subprocess, run): Check *child-env* variable and if
other than t, then install the environment before execvp.
In the spawn-based version of run, we save and restore the
environment around the spawn call, if *child-env* is in
effect.
* sysif.c (child_env_s): New symbol variable.
(exec_wrap): If *child-env* is other than t, then save the
environment in a list, and install the specified environment
before calling execvp. If that function returns, restore the
environbment.
* sysif.h (child_env_s): Declared.
(child_env): New macro.
* tests/018/process.tl: New tests.
* txr.1: Documented.
* stdlib/doc-syms.tl: Updated.
Diffstat (limited to 'sysif.c')
-rw-r--r-- | sysif.c | 20 |
1 files changed, 18 insertions, 2 deletions
@@ -128,6 +128,8 @@ val atime_s, mtime_s, ctime_s; val atime_nsec_s, mtime_nsec_s, ctime_nsec_s; val path_s, dir_s, dirent_s; +val child_env_s; + #if HAVE_PWUID || HAVE_GRGID val passwd_s; #endif @@ -1173,7 +1175,9 @@ val exec_wrap(val file, val args_opt) self, nao), convert(char **, 0)), coerce(char **, chk_xalloc(nargs + 1, sizeof *argv, self))); val iter; - int i; + val ch_env = child_env; + val save_env = nil; + int res, i; for (i = 0, iter = cons(file, args); iter; i++, iter = cdr(iter)) { val arg = car(iter); @@ -1181,7 +1185,17 @@ val exec_wrap(val file, val args_opt) } argv[i] = 0; - if (execvp(argv[0], argv) < 0) + if (ch_env != t) { + save_env = env(); + replace_env(ch_env); + } + + res = execvp(argv[0], argv); + + if (ch_env != t) + replace_env(save_env); + + if (res < 0) uw_ethrowf(process_error_s, lit("~s ~a: ~d/~s"), self, file, num(errno), errno_to_str(errno), nao); uw_throwf(process_error_s, lit("~s ~a returned"), self, file, nao); @@ -2588,6 +2602,7 @@ void sysif_init(void) len_s = intern(lit("len"), user_package); pid_s = intern(lit("pid"), user_package); #endif + child_env_s = intern(lit("*child-env*"), user_package); dir_cls = cobj_register(dir_s); @@ -2989,6 +3004,7 @@ void sysif_init(void) reg_fun(intern(lit("setenv"), user_package), func_n3o(setenv_wrap, 2)); reg_fun(intern(lit("unsetenv"), user_package), func_n1(unsetenv_wrap)); #endif + reg_var(child_env_s, t); #if HAVE_GETEUID reg_fun(intern(lit("getuid"), user_package), func_n0(getuid_wrap)); |