summaryrefslogtreecommitdiffstats
path: root/sysif.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-04-16 21:34:19 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-04-16 21:34:19 -0700
commite0fa761ad7a76f760fb358f5d5855bdd55eef13c (patch)
tree5d4c615ed10e235b4776a56f6f55d71bbeeb67dd /sysif.c
parent83aea1f74dd3de219de053d9162f8b217b0530ca (diff)
downloadtxr-e0fa761ad7a76f760fb358f5d5855bdd55eef13c.tar.gz
txr-e0fa761ad7a76f760fb358f5d5855bdd55eef13c.tar.bz2
txr-e0fa761ad7a76f760fb358f5d5855bdd55eef13c.zip
Add exec and pipe functions.
* configure: Check for pipe. * sysif.c (exec_wrap, pipe_wrap): New static functions. (sysif_init): Register exec and pipe intrinsics. * txr.1: Documented exec and pipe. * tl.vim, txr.vim: Regenerated.
Diffstat (limited to 'sysif.c')
-rw-r--r--sysif.c36
1 files changed, 36 insertions, 0 deletions
diff --git a/sysif.c b/sysif.c
index 64a453b0..4cbe6f16 100644
--- a/sysif.c
+++ b/sysif.c
@@ -477,6 +477,26 @@ static val dup_wrap(val old, val new)
return num(dup2(c_num(old), c_num(new)));
}
+static val exec_wrap(val file, val args_opt)
+{
+ val args = default_bool_arg(args_opt);
+ int nargs = c_num(length(args)) + 1;
+ char **argv = coerce(char **, chk_malloc((nargs + 1) * sizeof *argv));
+ val iter;
+ int i;
+
+ for (i = 0, iter = cons(file, args); iter; i++, iter = cdr(iter)) {
+ val arg = car(iter);
+ argv[i] = utf8_dup_to(c_str(arg));
+ }
+ argv[i] = 0;
+
+ if (execvp(argv[0], argv) < 0)
+ uw_throwf(file_error_s, lit("execvp ~a: ~a/~s"),
+ file, num(errno), string_utf8(strerror(errno)), nao);
+ uw_throwf(file_error_s, lit("execvp ~a returned"), file, nao);
+}
+
#endif
#if HAVE_SYS_STAT
@@ -523,6 +543,18 @@ val statf(val path)
#endif
}
+#if HAVE_PIPE
+
+static val pipe_wrap(void)
+{
+ int fd[2];
+ if (pipe(fd) < 0)
+ uw_throwf(file_error_s, lit("pipe failed: ~a/~s"),
+ num(errno), string_utf8(strerror(errno)), nao);
+ return cons(num(fd[0]), num(fd[1]));
+}
+
+#endif
void sysif_init(void)
{
@@ -657,6 +689,7 @@ void sysif_init(void)
#if HAVE_FORK_STUFF
reg_fun(intern(lit("fork"), user_package), func_n0(fork_wrap));
reg_fun(intern(lit("wait"), user_package), func_n2o(wait_wrap, 0));
+ reg_fun(intern(lit("exec"), user_package), func_n2o(exec_wrap, 1));
reg_fun(intern(lit("w-ifexited"), user_package), func_n1(wifexited));
reg_fun(intern(lit("w-exitstatus"), user_package), func_n1(wexitstatus));
reg_fun(intern(lit("w-ifsignaled"), user_package), func_n1(wifsignaled));
@@ -680,4 +713,7 @@ void sysif_init(void)
#endif
reg_fun(intern(lit("dupfd"), user_package), func_n2o(dup_wrap, 1));
#endif
+#if HAVE_PIPE
+ reg_fun(intern(lit("pipe"), user_package), func_n0(pipe_wrap));
+#endif
}