From 66733cef1cf1af2f2610f1cb816a9e39caf01202 Mon Sep 17 00:00:00 2001
From: Přemysl Eric Janouch
Date: Fri, 3 Jan 2025 07:50:50 +0100
Subject: WIP
---
tools/wdye/wdye.c | 177 +++++++++++++++++++++++++++++++++++++++++++++---------
1 file changed, 148 insertions(+), 29 deletions(-)
diff --git a/tools/wdye/wdye.c b/tools/wdye/wdye.c
index cb3c97c..47e485b 100644
--- a/tools/wdye/wdye.c
+++ b/tools/wdye/wdye.c
@@ -16,18 +16,141 @@
*
*/
-#define LIBERTY_WANT_POLLER
-
#define PROGRAM_NAME "wdye"
#define PROGRAM_VERSION "1"
+
+#define LIBERTY_WANT_POLLER
#include "../../liberty.c"
#include
#include
#include
+#include
+#if defined SOLARIS
+#include
+#endif
+
#include
+// --- Pseudoterminal ----------------------------------------------------------
+// This is largely taken from Advanced Programming in the UNIX® Environment,
+// just without a bunch of bugs.
+
+static int
+ptym_open (char **pts_name)
+{
+ int fdm = -1, err = 0;
+ if ((fdm = posix_openpt (O_RDWR | O_NOCTTY)) < 0)
+ return -1;
+ if (grantpt (fdm) < 0
+ || unlockpt (fdm) < 0)
+ goto errout;
+
+ char *ptr = NULL;
+ if ((ptr = ptsname (fdm)) == NULL)
+ goto errout;
+
+ cstr_set (pts_name, xstrdup (ptr));
+ return fdm;
+
+errout:
+ err = errno;
+ xclose (fdm);
+ errno = err;
+ return -1;
+}
+
+static int
+ptys_open (const char *pts_name)
+{
+ int fds = -1;
+#if defined SOLARIS
+ int err = 0, setup = 0;
+#endif
+ if ((fds = open (pts_name, O_RDWR)) < 0)
+ return -1;
+#if defined SOLARIS
+ if ((setup = ioctl (fds, I_FIND, "ldterm")) < 0)
+ goto errout;
+ if (setup == 0)
+ {
+ if (ioctl (fds, I_PUSH, "ptem") < 0
+ || ioctl (fds, I_PUSH, "ldterm") < 0)
+ goto errout;
+
+ if (ioctl (fds, I_PUSH, "ttcompat") < 0)
+ {
+errout:
+ err = errno;
+ xclose (fds);
+ errno = err;
+ return -1;
+ }
+ }
+#endif
+ return fds;
+}
+
+static pid_t
+pty_fork (int *ptrfdm, char **slave_name,
+ const struct termios *slave_termios, const struct winsize *slave_winsize,
+ struct error **e)
+{
+ int fdm = -1, fds = -1;
+
+ char *pts_name = NULL;
+ if ((fdm = ptym_open (&pts_name)) < 0)
+ {
+ error_set (e, "can’t open master pty: %s", strerror (errno));
+ return -1;
+ }
+ if (slave_name != NULL)
+ cstr_set (slave_name, xstrdup (pts_name));
+
+ pid_t pid = fork ();
+ if (pid < 0)
+ {
+ error_set (e, "fork: %s", strerror (errno));
+ xclose (fdm);
+ }
+ else if (pid != 0)
+ *ptrfdm = fdm;
+ else
+ {
+ if (setsid () < 0)
+ exit_fatal ("setsid: %s", strerror (errno));
+ if ((fds = ptys_open (pts_name)) < 0)
+ exit_fatal ("can’t open slave pty: %s", strerror (errno));
+ xclose (fdm);
+
+#if defined BSD
+ if (ioctl (fds, TIOCSCTTY, (char *) 0) < 0)
+ exit_fatal ("TIOCSCTTY: %s", strerror (errno));
+#endif
+
+ if (slave_termios != NULL
+ && tcsetattr (fds, TCSANOW, slave_termios) < 0)
+ exit_fatal ("tcsetattr error on slave pty: %s", strerror (errno));
+ if (slave_winsize != NULL
+ && ioctl (fds, TIOCSWINSZ, slave_winsize) < 0)
+ exit_fatal ("TIOCSWINSZ error on slave pty: %s", strerror (errno));
+
+ if (dup2 (fds, STDIN_FILENO) != STDIN_FILENO)
+ exit_fatal ("dup2 error to stdin");
+ if (dup2 (fds, STDOUT_FILENO) != STDOUT_FILENO)
+ exit_fatal ("dup2 error to stdout");
+ if (dup2 (fds, STDERR_FILENO) != STDERR_FILENO)
+ exit_fatal ("dup2 error to stderr");
+ if (fds != STDIN_FILENO && fds != STDOUT_FILENO && fds != STDERR_FILENO)
+ xclose (fds);
+ }
+ free (pts_name);
+ return pid;
+}
+
+// --- Global state ------------------------------------------------------------
+
static struct
{
lua_State *L; ///< Lua state
@@ -65,8 +188,8 @@ xlua_getfield (lua_State *L, int idx, const char *name,
struct process
{
+ int terminal_fd; ///< Process stdin/stdout/stderr
pid_t pid; ///< Process ID
- // TODO(p): File descriptors.
// TODO(p): Either `TERMINAL *` or read-out values.
};
@@ -83,7 +206,8 @@ static int
xlua_process_gc (lua_State *L)
{
struct process *self = luaL_checkudata (L, 1, XLUA_PROCESS_METATABLE);
- // TODO(p): Clean up file descriptors.
+ if (self->terminal_fd != -1)
+ xclose (self->terminal_fd);
if (self->pid != -1)
kill (self->pid, SIGKILL);
return 0;
@@ -93,7 +217,11 @@ static int
xlua_process_expect (lua_State *L)
{
struct process *self = luaL_checkudata (L, 1, XLUA_PROCESS_METATABLE);
- // TODO(p)
+ // TODO(p): Arguments.
+ // TODO(p): Keep reading input into a buffer,
+ // until there's a match.
+ // - It seems we need to keep the buffer within the process object,
+ // because we can read too much.
return 0;
}
@@ -127,7 +255,7 @@ struct xlua_spawn_context
struct strv envv; ///< Subprocess environment vector
struct strv argv; ///< Subprocess argument vector
- pid_t child; ///< Child process ID
+ struct error *error; ///< Error
};
static struct xlua_spawn_context
@@ -147,8 +275,6 @@ xlua_spawn_context_make (void)
}
self.envv = strv_make ();
self.argv = strv_make ();
-
- self.child = -1;
return self;
}
@@ -159,8 +285,8 @@ xlua_spawn_context_free (struct xlua_spawn_context *self)
strv_free (&self->envv);
strv_free (&self->argv);
- if (self->child != -1)
- kill (self->child, SIGKILL);
+ if (self->error)
+ error_free (self->error);
}
// -0, +0, e
@@ -229,32 +355,25 @@ xlua_spawn_protected (lua_State *L)
if (ctx->argv.len < 1)
return luaL_error (L, "missing argument");
- // Step 4: Spawn the process.
- // Keeping the same process group for simplicity right now.
+ // Step 4: Create a process object.
+ // This will get garbage collected as appropriate on failure.
+ struct process *process = xlua_process_new (L);
- // TODO(p): Allocate a pty, pipes in and out.
- // See Advanced Programming in the UNIX® Environment.
- switch ((ctx->child = fork ()))
+ // Step 5: Spawn the process, which gets a new process group.
+ process->pid =
+ pty_fork (&process->terminal_fd, NULL, NULL, NULL, &ctx->error);
+ if (process->pid < 0)
+ {
+ return luaL_error (L, "failed to spawn %s: %s",
+ ctx->argv.vector[0], ctx->error->message);
+ }
+ if (!process->pid)
{
- case -1:
- // TODO(p): Consider luaL_error().
- print_error ("failed to spawn %s: %s",
- ctx->argv.vector[0], strerror (errno));
- break;
- case 0:
execve (ctx->argv.vector[0], ctx->argv.vector, ctx->envv.vector);
print_error ("failed to spawn %s: %s",
ctx->argv.vector[0], strerror (errno));
_exit (EXIT_FAILURE);
- default:
- break;
}
-
- // Step 5: Return a process object.
- struct process *process = xlua_process_new (L);
- process->pid = ctx->child;
-
- ctx->child = -1;
return 1;
}
--
cgit v1.2.3-70-g09d2