xref: /minix3/external/bsd/tmux/dist/cmd-pipe-pane.c (revision 0a6a1f1d05b60e214de2f05a7310ddd1f0e590e7)
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