1*0a6a1f1dSLionel Sambuc /* Id */
2eda6f593SDavid van Moolenbroek
3eda6f593SDavid van Moolenbroek /*
4eda6f593SDavid van Moolenbroek * Copyright (c) 2009 Nicholas Marriott <nicm@users.sourceforge.net>
5eda6f593SDavid van Moolenbroek *
6eda6f593SDavid van Moolenbroek * Permission to use, copy, modify, and distribute this software for any
7eda6f593SDavid van Moolenbroek * purpose with or without fee is hereby granted, provided that the above
8eda6f593SDavid van Moolenbroek * copyright notice and this permission notice appear in all copies.
9eda6f593SDavid van Moolenbroek *
10eda6f593SDavid van Moolenbroek * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11eda6f593SDavid van Moolenbroek * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12eda6f593SDavid van Moolenbroek * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13eda6f593SDavid van Moolenbroek * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14eda6f593SDavid van Moolenbroek * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
15eda6f593SDavid van Moolenbroek * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
16eda6f593SDavid van Moolenbroek * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17eda6f593SDavid van Moolenbroek */
18eda6f593SDavid van Moolenbroek
19eda6f593SDavid van Moolenbroek #include <sys/types.h>
20eda6f593SDavid van Moolenbroek #include <sys/socket.h>
21eda6f593SDavid van Moolenbroek
22eda6f593SDavid van Moolenbroek #include <errno.h>
23eda6f593SDavid van Moolenbroek #include <fcntl.h>
24eda6f593SDavid van Moolenbroek #include <string.h>
25eda6f593SDavid van Moolenbroek #include <time.h>
26eda6f593SDavid van Moolenbroek #include <unistd.h>
27eda6f593SDavid van Moolenbroek
28eda6f593SDavid van Moolenbroek #include "tmux.h"
29eda6f593SDavid van Moolenbroek
30eda6f593SDavid van Moolenbroek /*
31eda6f593SDavid van Moolenbroek * Open pipe to redirect pane output. If already open, close first.
32eda6f593SDavid van Moolenbroek */
33eda6f593SDavid van Moolenbroek
34*0a6a1f1dSLionel Sambuc enum cmd_retval cmd_pipe_pane_exec(struct cmd *, struct cmd_q *);
35eda6f593SDavid van Moolenbroek
36eda6f593SDavid van Moolenbroek void cmd_pipe_pane_error_callback(struct bufferevent *, short, void *);
37eda6f593SDavid van Moolenbroek
38eda6f593SDavid van Moolenbroek const struct cmd_entry cmd_pipe_pane_entry = {
39eda6f593SDavid van Moolenbroek "pipe-pane", "pipep",
40eda6f593SDavid van Moolenbroek "ot:", 0, 1,
41*0a6a1f1dSLionel Sambuc "[-o] " CMD_TARGET_PANE_USAGE " [command]",
42eda6f593SDavid van Moolenbroek 0,
43eda6f593SDavid van Moolenbroek NULL,
44eda6f593SDavid van Moolenbroek cmd_pipe_pane_exec
45eda6f593SDavid van Moolenbroek };
46eda6f593SDavid van Moolenbroek
47*0a6a1f1dSLionel Sambuc enum cmd_retval
cmd_pipe_pane_exec(struct cmd * self,struct cmd_q * cmdq)48*0a6a1f1dSLionel Sambuc cmd_pipe_pane_exec(struct cmd *self, struct cmd_q *cmdq)
49eda6f593SDavid van Moolenbroek {
50eda6f593SDavid van Moolenbroek struct args *args = self->args;
51eda6f593SDavid van Moolenbroek struct client *c;
52eda6f593SDavid van Moolenbroek struct window_pane *wp;
53eda6f593SDavid van Moolenbroek char *command;
54eda6f593SDavid van Moolenbroek int old_fd, pipe_fd[2], null_fd;
55eda6f593SDavid van Moolenbroek
56*0a6a1f1dSLionel Sambuc if (cmd_find_pane(cmdq, args_get(args, 't'), NULL, &wp) == NULL)
57*0a6a1f1dSLionel Sambuc return (CMD_RETURN_ERROR);
58*0a6a1f1dSLionel Sambuc c = cmd_find_client(cmdq, NULL, 1);
59eda6f593SDavid van Moolenbroek
60eda6f593SDavid van Moolenbroek /* Destroy the old pipe. */
61eda6f593SDavid van Moolenbroek old_fd = wp->pipe_fd;
62eda6f593SDavid van Moolenbroek if (wp->pipe_fd != -1) {
63eda6f593SDavid van Moolenbroek bufferevent_free(wp->pipe_event);
64eda6f593SDavid van Moolenbroek close(wp->pipe_fd);
65eda6f593SDavid van Moolenbroek wp->pipe_fd = -1;
66eda6f593SDavid van Moolenbroek }
67eda6f593SDavid van Moolenbroek
68eda6f593SDavid van Moolenbroek /* If no pipe command, that is enough. */
69eda6f593SDavid van Moolenbroek if (args->argc == 0 || *args->argv[0] == '\0')
70*0a6a1f1dSLionel Sambuc return (CMD_RETURN_NORMAL);
71eda6f593SDavid van Moolenbroek
72eda6f593SDavid van Moolenbroek /*
73eda6f593SDavid van Moolenbroek * With -o, only open the new pipe if there was no previous one. This
74eda6f593SDavid van Moolenbroek * allows a pipe to be toggled with a single key, for example:
75eda6f593SDavid van Moolenbroek *
76eda6f593SDavid van Moolenbroek * bind ^p pipep -o 'cat >>~/output'
77eda6f593SDavid van Moolenbroek */
78eda6f593SDavid van Moolenbroek if (args_has(self->args, 'o') && old_fd != -1)
79*0a6a1f1dSLionel Sambuc return (CMD_RETURN_NORMAL);
80eda6f593SDavid van Moolenbroek
81eda6f593SDavid van Moolenbroek /* Open the new pipe. */
82eda6f593SDavid van Moolenbroek if (socketpair(AF_UNIX, SOCK_STREAM, PF_UNSPEC, pipe_fd) != 0) {
83*0a6a1f1dSLionel Sambuc cmdq_error(cmdq, "socketpair error: %s", strerror(errno));
84*0a6a1f1dSLionel Sambuc return (CMD_RETURN_ERROR);
85eda6f593SDavid van Moolenbroek }
86eda6f593SDavid van Moolenbroek
87eda6f593SDavid van Moolenbroek /* Fork the child. */
88eda6f593SDavid van Moolenbroek switch (fork()) {
89eda6f593SDavid van Moolenbroek case -1:
90*0a6a1f1dSLionel Sambuc cmdq_error(cmdq, "fork error: %s", strerror(errno));
91*0a6a1f1dSLionel Sambuc return (CMD_RETURN_ERROR);
92eda6f593SDavid van Moolenbroek case 0:
93eda6f593SDavid van Moolenbroek /* Child process. */
94eda6f593SDavid van Moolenbroek close(pipe_fd[0]);
95eda6f593SDavid van Moolenbroek clear_signals(1);
96eda6f593SDavid van Moolenbroek
97eda6f593SDavid van Moolenbroek if (dup2(pipe_fd[1], STDIN_FILENO) == -1)
98eda6f593SDavid van Moolenbroek _exit(1);
99eda6f593SDavid van Moolenbroek if (pipe_fd[1] != STDIN_FILENO)
100eda6f593SDavid van Moolenbroek close(pipe_fd[1]);
101eda6f593SDavid van Moolenbroek
102eda6f593SDavid van Moolenbroek null_fd = open(_PATH_DEVNULL, O_WRONLY, 0);
103eda6f593SDavid van Moolenbroek if (dup2(null_fd, STDOUT_FILENO) == -1)
104eda6f593SDavid van Moolenbroek _exit(1);
105eda6f593SDavid van Moolenbroek if (dup2(null_fd, STDERR_FILENO) == -1)
106eda6f593SDavid van Moolenbroek _exit(1);
107eda6f593SDavid van Moolenbroek if (null_fd != STDOUT_FILENO && null_fd != STDERR_FILENO)
108eda6f593SDavid van Moolenbroek close(null_fd);
109eda6f593SDavid van Moolenbroek
110eda6f593SDavid van Moolenbroek closefrom(STDERR_FILENO + 1);
111eda6f593SDavid van Moolenbroek
112eda6f593SDavid van Moolenbroek command = status_replace(
113eda6f593SDavid van Moolenbroek c, NULL, NULL, NULL, args->argv[0], time(NULL), 0);
114eda6f593SDavid van Moolenbroek execl(_PATH_BSHELL, "sh", "-c", command, (char *) NULL);
115eda6f593SDavid van Moolenbroek _exit(1);
116eda6f593SDavid van Moolenbroek default:
117eda6f593SDavid van Moolenbroek /* Parent process. */
118eda6f593SDavid van Moolenbroek close(pipe_fd[1]);
119eda6f593SDavid van Moolenbroek
120eda6f593SDavid van Moolenbroek wp->pipe_fd = pipe_fd[0];
121eda6f593SDavid van Moolenbroek wp->pipe_off = EVBUFFER_LENGTH(wp->event->input);
122eda6f593SDavid van Moolenbroek
123eda6f593SDavid van Moolenbroek wp->pipe_event = bufferevent_new(wp->pipe_fd,
124eda6f593SDavid van Moolenbroek NULL, NULL, cmd_pipe_pane_error_callback, wp);
125eda6f593SDavid van Moolenbroek bufferevent_enable(wp->pipe_event, EV_WRITE);
126eda6f593SDavid van Moolenbroek
127eda6f593SDavid van Moolenbroek setblocking(wp->pipe_fd, 0);
128*0a6a1f1dSLionel Sambuc return (CMD_RETURN_NORMAL);
129eda6f593SDavid van Moolenbroek }
130eda6f593SDavid van Moolenbroek }
131eda6f593SDavid van Moolenbroek
132eda6f593SDavid van Moolenbroek void
cmd_pipe_pane_error_callback(unused struct bufferevent * bufev,unused short what,void * data)133eda6f593SDavid van Moolenbroek cmd_pipe_pane_error_callback(
134eda6f593SDavid van Moolenbroek unused struct bufferevent *bufev, unused short what, void *data)
135eda6f593SDavid van Moolenbroek {
136eda6f593SDavid van Moolenbroek struct window_pane *wp = data;
137eda6f593SDavid van Moolenbroek
138eda6f593SDavid van Moolenbroek bufferevent_free(wp->pipe_event);
139eda6f593SDavid van Moolenbroek close(wp->pipe_fd);
140eda6f593SDavid van Moolenbroek wp->pipe_fd = -1;
141eda6f593SDavid van Moolenbroek }
142