summaryrefslogtreecommitdiffstats
path: root/cmd-pipe-pane.c
diff options
context:
space:
mode:
authorThomas Adam <thomas@xteddy.org>2018-01-17 11:26:10 +0000
committerThomas Adam <thomas@xteddy.org>2018-01-17 11:26:10 +0000
commit84ddc72744d88f399103ea350c28d5fb45e28b96 (patch)
tree76a62ad185f8a274f4243a2365fd01801e421b8e /cmd-pipe-pane.c
parent58e9d12f23be4ecc7ab6879e8d19239b430f6553 (diff)
parent75842bfe66d983580dddf11b676445d5b9fa9f8a (diff)
Merge branch 'obsd-master'
Diffstat (limited to 'cmd-pipe-pane.c')
-rw-r--r--cmd-pipe-pane.c74
1 files changed, 56 insertions, 18 deletions
diff --git a/cmd-pipe-pane.c b/cmd-pipe-pane.c
index 9c2290d6..7683df31 100644
--- a/cmd-pipe-pane.c
+++ b/cmd-pipe-pane.c
@@ -35,6 +35,7 @@
static enum cmd_retval cmd_pipe_pane_exec(struct cmd *, struct cmdq_item *);
+static void cmd_pipe_pane_read_callback(struct bufferevent *, void *);
static void cmd_pipe_pane_write_callback(struct bufferevent *, void *);
static void cmd_pipe_pane_error_callback(struct bufferevent *, short, void *);
@@ -42,8 +43,8 @@ const struct cmd_entry cmd_pipe_pane_entry = {
.name = "pipe-pane",
.alias = "pipep",
- .args = { "ot:", 0, 1 },
- .usage = "[-o] " CMD_TARGET_PANE_USAGE " [command]",
+ .args = { "IOot:", 0, 1 },
+ .usage = "[-IOo] " CMD_TARGET_PANE_USAGE " [command]",
.target = { 't', CMD_FIND_PANE, 0 },
@@ -60,7 +61,7 @@ cmd_pipe_pane_exec(struct cmd *self, struct cmdq_item *item)
struct session *s = item->target.s;
struct winlink *wl = item->target.wl;
char *cmd;
- int old_fd, pipe_fd[2], null_fd;
+ int old_fd, pipe_fd[2], null_fd, in, out;
struct format_tree *ft;
sigset_t set, oldset;
@@ -90,6 +91,15 @@ cmd_pipe_pane_exec(struct cmd *self, struct cmdq_item *item)
if (args_has(self->args, 'o') && old_fd != -1)
return (CMD_RETURN_NORMAL);
+ /* What do we want to do? Neither -I or -O is -O. */
+ if (args_has(self->args, 'I')) {
+ in = 1;
+ out = args_has(self->args, 'O');
+ } else {
+ in = 0;
+ out = 1;
+ }
+
/* Open the new pipe. */
if (socketpair(AF_UNIX, SOCK_STREAM, PF_UNSPEC, pipe_fd) != 0) {
cmdq_error(item, "socketpair error: %s", strerror(errno));
@@ -118,19 +128,25 @@ cmd_pipe_pane_exec(struct cmd *self, struct cmdq_item *item)
sigprocmask(SIG_SETMASK, &oldset, NULL);
close(pipe_fd[0]);
- if (dup2(pipe_fd[1], STDIN_FILENO) == -1)
- _exit(1);
- if (pipe_fd[1] != STDIN_FILENO)
- close(pipe_fd[1]);
-
null_fd = open(_PATH_DEVNULL, O_WRONLY, 0);
- if (dup2(null_fd, STDOUT_FILENO) == -1)
- _exit(1);
+ if (out) {
+ if (dup2(pipe_fd[1], STDIN_FILENO) == -1)
+ _exit(1);
+ } else {
+ if (dup2(null_fd, STDIN_FILENO) == -1)
+ _exit(1);
+ }
+ if (in) {
+ if (dup2(pipe_fd[1], STDOUT_FILENO) == -1)
+ _exit(1);
+ if (pipe_fd[1] != STDOUT_FILENO)
+ close(pipe_fd[1]);
+ } else {
+ if (dup2(null_fd, STDOUT_FILENO) == -1)
+ _exit(1);
+ }
if (dup2(null_fd, STDERR_FILENO) == -1)
_exit(1);
- if (null_fd != STDOUT_FILENO && null_fd != STDERR_FILENO)
- close(null_fd);
-
closefrom(STDERR_FILENO + 1);
execl(_PATH_BSHELL, "sh", "-c", cmd, (char *) NULL);
@@ -143,12 +159,16 @@ cmd_pipe_pane_exec(struct cmd *self, struct cmdq_item *item)
wp->pipe_fd = pipe_fd[0];
wp->pipe_off = EVBUFFER_LENGTH(wp->event->input);
- wp->pipe_event = bufferevent_new(wp->pipe_fd, NULL,
- cmd_pipe_pane_write_callback, cmd_pipe_pane_error_callback,
- wp);
- bufferevent_enable(wp->pipe_event, EV_WRITE);
-
setblocking(wp->pipe_fd, 0);
+ wp->pipe_event = bufferevent_new(wp->pipe_fd,
+ cmd_pipe_pane_read_callback,
+ cmd_pipe_pane_write_callback,
+ cmd_pipe_pane_error_callback,
+ wp);
+ if (out)
+ bufferevent_enable(wp->pipe_event, EV_WRITE);
+ if (in)
+ bufferevent_enable(wp->pipe_event, EV_READ);
free(cmd);
return (CMD_RETURN_NORMAL);
@@ -156,11 +176,29 @@ cmd_pipe_pane_exec(struct cmd *self, struct cmdq_item *item)
}
static void
+cmd_pipe_pane_read_callback(__unused struct bufferevent *bufev, void *data)
+{
+ struct window_pane *wp = data;
+ struct evbuffer *evb = wp->pipe_event->input;
+ size_t available;
+
+ available = EVBUFFER_LENGTH(evb);
+ log_debug("%%%u pipe read %zu", wp->id, available);
+
+ bufferevent_write(wp->event, EVBUFFER_DATA(evb), available);
+ evbuffer_drain(evb, available);
+
+ if (window_pane_destroy_ready(wp))
+ server_destroy_pane(wp, 1);
+}
+
+static void
cmd_pipe_pane_write_callback(__unused struct bufferevent *bufev, void *data)
{
struct window_pane *wp = data;
log_debug("%%%u pipe empty", wp->id);
+
if (window_pane_destroy_ready(wp))
server_destroy_pane(wp, 1);
}