xref: /minix3/external/bsd/tmux/dist/compat/imsg-buffer.c (revision 0a6a1f1d05b60e214de2f05a7310ddd1f0e590e7)
1*0a6a1f1dSLionel Sambuc /* Id */
2eda6f593SDavid van Moolenbroek /*	$OpenBSD: imsg-buffer.c,v 1.3 2010/05/26 13:56:07 nicm Exp $	*/
3eda6f593SDavid van Moolenbroek 
4eda6f593SDavid van Moolenbroek /*
5eda6f593SDavid van Moolenbroek  * Copyright (c) 2003, 2004 Henning Brauer <henning@openbsd.org>
6eda6f593SDavid van Moolenbroek  *
7eda6f593SDavid van Moolenbroek  * Permission to use, copy, modify, and distribute this software for any
8eda6f593SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9eda6f593SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10eda6f593SDavid van Moolenbroek  *
11eda6f593SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
12eda6f593SDavid van Moolenbroek  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
13eda6f593SDavid van Moolenbroek  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
14eda6f593SDavid van Moolenbroek  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
15eda6f593SDavid van Moolenbroek  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
16eda6f593SDavid van Moolenbroek  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
17eda6f593SDavid van Moolenbroek  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18eda6f593SDavid van Moolenbroek  */
19eda6f593SDavid van Moolenbroek 
20eda6f593SDavid van Moolenbroek #include <sys/param.h>
21eda6f593SDavid van Moolenbroek #include <sys/socket.h>
22eda6f593SDavid van Moolenbroek #include <sys/uio.h>
23eda6f593SDavid van Moolenbroek 
24eda6f593SDavid van Moolenbroek #include <errno.h>
25eda6f593SDavid van Moolenbroek #include <stdlib.h>
26eda6f593SDavid van Moolenbroek #include <string.h>
27*0a6a1f1dSLionel Sambuc #include <strings.h>
28eda6f593SDavid van Moolenbroek #include <unistd.h>
29eda6f593SDavid van Moolenbroek 
30eda6f593SDavid van Moolenbroek #include "tmux.h"
31eda6f593SDavid van Moolenbroek 
32eda6f593SDavid van Moolenbroek int	ibuf_realloc(struct ibuf *, size_t);
33eda6f593SDavid van Moolenbroek void	ibuf_enqueue(struct msgbuf *, struct ibuf *);
34eda6f593SDavid van Moolenbroek void	ibuf_dequeue(struct msgbuf *, struct ibuf *);
35eda6f593SDavid van Moolenbroek 
36eda6f593SDavid van Moolenbroek struct ibuf *
ibuf_open(size_t len)37eda6f593SDavid van Moolenbroek ibuf_open(size_t len)
38eda6f593SDavid van Moolenbroek {
39eda6f593SDavid van Moolenbroek 	struct ibuf	*buf;
40eda6f593SDavid van Moolenbroek 
41eda6f593SDavid van Moolenbroek 	if ((buf = calloc(1, sizeof(struct ibuf))) == NULL)
42eda6f593SDavid van Moolenbroek 		return (NULL);
43eda6f593SDavid van Moolenbroek 	if ((buf->buf = malloc(len)) == NULL) {
44eda6f593SDavid van Moolenbroek 		free(buf);
45eda6f593SDavid van Moolenbroek 		return (NULL);
46eda6f593SDavid van Moolenbroek 	}
47eda6f593SDavid van Moolenbroek 	buf->size = buf->max = len;
48eda6f593SDavid van Moolenbroek 	buf->fd = -1;
49eda6f593SDavid van Moolenbroek 
50eda6f593SDavid van Moolenbroek 	return (buf);
51eda6f593SDavid van Moolenbroek }
52eda6f593SDavid van Moolenbroek 
53eda6f593SDavid van Moolenbroek struct ibuf *
ibuf_dynamic(size_t len,size_t max)54eda6f593SDavid van Moolenbroek ibuf_dynamic(size_t len, size_t max)
55eda6f593SDavid van Moolenbroek {
56eda6f593SDavid van Moolenbroek 	struct ibuf	*buf;
57eda6f593SDavid van Moolenbroek 
58eda6f593SDavid van Moolenbroek 	if (max < len)
59eda6f593SDavid van Moolenbroek 		return (NULL);
60eda6f593SDavid van Moolenbroek 
61eda6f593SDavid van Moolenbroek 	if ((buf = ibuf_open(len)) == NULL)
62eda6f593SDavid van Moolenbroek 		return (NULL);
63eda6f593SDavid van Moolenbroek 
64eda6f593SDavid van Moolenbroek 	if (max > 0)
65eda6f593SDavid van Moolenbroek 		buf->max = max;
66eda6f593SDavid van Moolenbroek 
67eda6f593SDavid van Moolenbroek 	return (buf);
68eda6f593SDavid van Moolenbroek }
69eda6f593SDavid van Moolenbroek 
70eda6f593SDavid van Moolenbroek int
ibuf_realloc(struct ibuf * buf,size_t len)71eda6f593SDavid van Moolenbroek ibuf_realloc(struct ibuf *buf, size_t len)
72eda6f593SDavid van Moolenbroek {
73eda6f593SDavid van Moolenbroek 	u_char	*b;
74eda6f593SDavid van Moolenbroek 
75eda6f593SDavid van Moolenbroek 	/* on static buffers max is eq size and so the following fails */
76eda6f593SDavid van Moolenbroek 	if (buf->wpos + len > buf->max) {
77eda6f593SDavid van Moolenbroek 		errno = ENOMEM;
78eda6f593SDavid van Moolenbroek 		return (-1);
79eda6f593SDavid van Moolenbroek 	}
80eda6f593SDavid van Moolenbroek 
81eda6f593SDavid van Moolenbroek 	b = realloc(buf->buf, buf->wpos + len);
82eda6f593SDavid van Moolenbroek 	if (b == NULL)
83eda6f593SDavid van Moolenbroek 		return (-1);
84eda6f593SDavid van Moolenbroek 	buf->buf = b;
85eda6f593SDavid van Moolenbroek 	buf->size = buf->wpos + len;
86eda6f593SDavid van Moolenbroek 
87eda6f593SDavid van Moolenbroek 	return (0);
88eda6f593SDavid van Moolenbroek }
89eda6f593SDavid van Moolenbroek 
90eda6f593SDavid van Moolenbroek int
ibuf_add(struct ibuf * buf,const void * data,size_t len)91eda6f593SDavid van Moolenbroek ibuf_add(struct ibuf *buf, const void *data, size_t len)
92eda6f593SDavid van Moolenbroek {
93eda6f593SDavid van Moolenbroek 	if (buf->wpos + len > buf->size)
94eda6f593SDavid van Moolenbroek 		if (ibuf_realloc(buf, len) == -1)
95eda6f593SDavid van Moolenbroek 			return (-1);
96eda6f593SDavid van Moolenbroek 
97eda6f593SDavid van Moolenbroek 	memcpy(buf->buf + buf->wpos, data, len);
98eda6f593SDavid van Moolenbroek 	buf->wpos += len;
99eda6f593SDavid van Moolenbroek 	return (0);
100eda6f593SDavid van Moolenbroek }
101eda6f593SDavid van Moolenbroek 
102eda6f593SDavid van Moolenbroek void *
ibuf_reserve(struct ibuf * buf,size_t len)103eda6f593SDavid van Moolenbroek ibuf_reserve(struct ibuf *buf, size_t len)
104eda6f593SDavid van Moolenbroek {
105eda6f593SDavid van Moolenbroek 	void	*b;
106eda6f593SDavid van Moolenbroek 
107eda6f593SDavid van Moolenbroek 	if (buf->wpos + len > buf->size)
108eda6f593SDavid van Moolenbroek 		if (ibuf_realloc(buf, len) == -1)
109eda6f593SDavid van Moolenbroek 			return (NULL);
110eda6f593SDavid van Moolenbroek 
111eda6f593SDavid van Moolenbroek 	b = buf->buf + buf->wpos;
112eda6f593SDavid van Moolenbroek 	buf->wpos += len;
113eda6f593SDavid van Moolenbroek 	return (b);
114eda6f593SDavid van Moolenbroek }
115eda6f593SDavid van Moolenbroek 
116eda6f593SDavid van Moolenbroek void *
ibuf_seek(struct ibuf * buf,size_t pos,size_t len)117eda6f593SDavid van Moolenbroek ibuf_seek(struct ibuf *buf, size_t pos, size_t len)
118eda6f593SDavid van Moolenbroek {
119eda6f593SDavid van Moolenbroek 	/* only allowed to seek in already written parts */
120eda6f593SDavid van Moolenbroek 	if (pos + len > buf->wpos)
121eda6f593SDavid van Moolenbroek 		return (NULL);
122eda6f593SDavid van Moolenbroek 
123eda6f593SDavid van Moolenbroek 	return (buf->buf + pos);
124eda6f593SDavid van Moolenbroek }
125eda6f593SDavid van Moolenbroek 
126eda6f593SDavid van Moolenbroek size_t
ibuf_size(struct ibuf * buf)127eda6f593SDavid van Moolenbroek ibuf_size(struct ibuf *buf)
128eda6f593SDavid van Moolenbroek {
129eda6f593SDavid van Moolenbroek 	return (buf->wpos);
130eda6f593SDavid van Moolenbroek }
131eda6f593SDavid van Moolenbroek 
132eda6f593SDavid van Moolenbroek size_t
ibuf_left(struct ibuf * buf)133eda6f593SDavid van Moolenbroek ibuf_left(struct ibuf *buf)
134eda6f593SDavid van Moolenbroek {
135eda6f593SDavid van Moolenbroek 	return (buf->max - buf->wpos);
136eda6f593SDavid van Moolenbroek }
137eda6f593SDavid van Moolenbroek 
138eda6f593SDavid van Moolenbroek void
ibuf_close(struct msgbuf * msgbuf,struct ibuf * buf)139eda6f593SDavid van Moolenbroek ibuf_close(struct msgbuf *msgbuf, struct ibuf *buf)
140eda6f593SDavid van Moolenbroek {
141eda6f593SDavid van Moolenbroek 	ibuf_enqueue(msgbuf, buf);
142eda6f593SDavid van Moolenbroek }
143eda6f593SDavid van Moolenbroek 
144eda6f593SDavid van Moolenbroek int
ibuf_write(struct msgbuf * msgbuf)145eda6f593SDavid van Moolenbroek ibuf_write(struct msgbuf *msgbuf)
146eda6f593SDavid van Moolenbroek {
147eda6f593SDavid van Moolenbroek 	struct iovec	 iov[IOV_MAX];
148eda6f593SDavid van Moolenbroek 	struct ibuf	*buf;
149eda6f593SDavid van Moolenbroek 	unsigned int	 i = 0;
150eda6f593SDavid van Moolenbroek 	ssize_t	n;
151eda6f593SDavid van Moolenbroek 
152eda6f593SDavid van Moolenbroek 	bzero(&iov, sizeof(iov));
153eda6f593SDavid van Moolenbroek 	TAILQ_FOREACH(buf, &msgbuf->bufs, entry) {
154eda6f593SDavid van Moolenbroek 		if (i >= IOV_MAX)
155eda6f593SDavid van Moolenbroek 			break;
156eda6f593SDavid van Moolenbroek 		iov[i].iov_base = buf->buf + buf->rpos;
157eda6f593SDavid van Moolenbroek 		iov[i].iov_len = buf->wpos - buf->rpos;
158eda6f593SDavid van Moolenbroek 		i++;
159eda6f593SDavid van Moolenbroek 	}
160eda6f593SDavid van Moolenbroek 
161eda6f593SDavid van Moolenbroek 	if ((n = writev(msgbuf->fd, iov, i)) == -1) {
162eda6f593SDavid van Moolenbroek 		if (errno == EAGAIN || errno == ENOBUFS ||
163eda6f593SDavid van Moolenbroek 		    errno == EINTR)	/* try later */
164eda6f593SDavid van Moolenbroek 			return (0);
165eda6f593SDavid van Moolenbroek 		else
166eda6f593SDavid van Moolenbroek 			return (-1);
167eda6f593SDavid van Moolenbroek 	}
168eda6f593SDavid van Moolenbroek 
169eda6f593SDavid van Moolenbroek 	if (n == 0) {			/* connection closed */
170eda6f593SDavid van Moolenbroek 		errno = 0;
171eda6f593SDavid van Moolenbroek 		return (-2);
172eda6f593SDavid van Moolenbroek 	}
173eda6f593SDavid van Moolenbroek 
174eda6f593SDavid van Moolenbroek 	msgbuf_drain(msgbuf, n);
175eda6f593SDavid van Moolenbroek 
176eda6f593SDavid van Moolenbroek 	return (0);
177eda6f593SDavid van Moolenbroek }
178eda6f593SDavid van Moolenbroek 
179eda6f593SDavid van Moolenbroek void
ibuf_free(struct ibuf * buf)180eda6f593SDavid van Moolenbroek ibuf_free(struct ibuf *buf)
181eda6f593SDavid van Moolenbroek {
182eda6f593SDavid van Moolenbroek 	free(buf->buf);
183eda6f593SDavid van Moolenbroek 	free(buf);
184eda6f593SDavid van Moolenbroek }
185eda6f593SDavid van Moolenbroek 
186eda6f593SDavid van Moolenbroek void
msgbuf_init(struct msgbuf * msgbuf)187eda6f593SDavid van Moolenbroek msgbuf_init(struct msgbuf *msgbuf)
188eda6f593SDavid van Moolenbroek {
189eda6f593SDavid van Moolenbroek 	msgbuf->queued = 0;
190eda6f593SDavid van Moolenbroek 	msgbuf->fd = -1;
191eda6f593SDavid van Moolenbroek 	TAILQ_INIT(&msgbuf->bufs);
192eda6f593SDavid van Moolenbroek }
193eda6f593SDavid van Moolenbroek 
194eda6f593SDavid van Moolenbroek void
msgbuf_drain(struct msgbuf * msgbuf,size_t n)195eda6f593SDavid van Moolenbroek msgbuf_drain(struct msgbuf *msgbuf, size_t n)
196eda6f593SDavid van Moolenbroek {
197eda6f593SDavid van Moolenbroek 	struct ibuf	*buf, *next;
198eda6f593SDavid van Moolenbroek 
199eda6f593SDavid van Moolenbroek 	for (buf = TAILQ_FIRST(&msgbuf->bufs); buf != NULL && n > 0;
200eda6f593SDavid van Moolenbroek 	    buf = next) {
201eda6f593SDavid van Moolenbroek 		next = TAILQ_NEXT(buf, entry);
202eda6f593SDavid van Moolenbroek 		if (buf->rpos + n >= buf->wpos) {
203eda6f593SDavid van Moolenbroek 			n -= buf->wpos - buf->rpos;
204eda6f593SDavid van Moolenbroek 			ibuf_dequeue(msgbuf, buf);
205eda6f593SDavid van Moolenbroek 		} else {
206eda6f593SDavid van Moolenbroek 			buf->rpos += n;
207eda6f593SDavid van Moolenbroek 			n = 0;
208eda6f593SDavid van Moolenbroek 		}
209eda6f593SDavid van Moolenbroek 	}
210eda6f593SDavid van Moolenbroek }
211eda6f593SDavid van Moolenbroek 
212eda6f593SDavid van Moolenbroek void
msgbuf_clear(struct msgbuf * msgbuf)213eda6f593SDavid van Moolenbroek msgbuf_clear(struct msgbuf *msgbuf)
214eda6f593SDavid van Moolenbroek {
215eda6f593SDavid van Moolenbroek 	struct ibuf	*buf;
216eda6f593SDavid van Moolenbroek 
217eda6f593SDavid van Moolenbroek 	while ((buf = TAILQ_FIRST(&msgbuf->bufs)) != NULL)
218eda6f593SDavid van Moolenbroek 		ibuf_dequeue(msgbuf, buf);
219eda6f593SDavid van Moolenbroek }
220eda6f593SDavid van Moolenbroek 
221eda6f593SDavid van Moolenbroek int
msgbuf_write(struct msgbuf * msgbuf)222eda6f593SDavid van Moolenbroek msgbuf_write(struct msgbuf *msgbuf)
223eda6f593SDavid van Moolenbroek {
224eda6f593SDavid van Moolenbroek 	struct iovec	 iov[IOV_MAX];
225eda6f593SDavid van Moolenbroek 	struct ibuf	*buf;
226eda6f593SDavid van Moolenbroek 	unsigned int	 i = 0;
227eda6f593SDavid van Moolenbroek 	ssize_t		 n;
228eda6f593SDavid van Moolenbroek 	struct msghdr	 msg;
229eda6f593SDavid van Moolenbroek 	struct cmsghdr	*cmsg;
230eda6f593SDavid van Moolenbroek 	union {
231eda6f593SDavid van Moolenbroek 		struct cmsghdr	hdr;
232eda6f593SDavid van Moolenbroek 		char		buf[CMSG_SPACE(sizeof(int))];
233eda6f593SDavid van Moolenbroek 	} cmsgbuf;
234eda6f593SDavid van Moolenbroek 
235eda6f593SDavid van Moolenbroek 	bzero(&iov, sizeof(iov));
236eda6f593SDavid van Moolenbroek 	bzero(&msg, sizeof(msg));
237eda6f593SDavid van Moolenbroek 	TAILQ_FOREACH(buf, &msgbuf->bufs, entry) {
238eda6f593SDavid van Moolenbroek 		if (i >= IOV_MAX)
239eda6f593SDavid van Moolenbroek 			break;
240eda6f593SDavid van Moolenbroek 		iov[i].iov_base = buf->buf + buf->rpos;
241eda6f593SDavid van Moolenbroek 		iov[i].iov_len = buf->wpos - buf->rpos;
242eda6f593SDavid van Moolenbroek 		i++;
243eda6f593SDavid van Moolenbroek 		if (buf->fd != -1)
244eda6f593SDavid van Moolenbroek 			break;
245eda6f593SDavid van Moolenbroek 	}
246eda6f593SDavid van Moolenbroek 
247eda6f593SDavid van Moolenbroek 	msg.msg_iov = iov;
248eda6f593SDavid van Moolenbroek 	msg.msg_iovlen = i;
249eda6f593SDavid van Moolenbroek 
250eda6f593SDavid van Moolenbroek 	if (buf != NULL && buf->fd != -1) {
251eda6f593SDavid van Moolenbroek 		msg.msg_control = (caddr_t)&cmsgbuf.buf;
252eda6f593SDavid van Moolenbroek 		msg.msg_controllen = CMSG_SPACE(sizeof(int));
253eda6f593SDavid van Moolenbroek 		cmsg = CMSG_FIRSTHDR(&msg);
254eda6f593SDavid van Moolenbroek 		cmsg->cmsg_len = CMSG_LEN(sizeof(int));
255eda6f593SDavid van Moolenbroek 		cmsg->cmsg_level = SOL_SOCKET;
256eda6f593SDavid van Moolenbroek 		cmsg->cmsg_type = SCM_RIGHTS;
257eda6f593SDavid van Moolenbroek 		*(int *)CMSG_DATA(cmsg) = buf->fd;
258eda6f593SDavid van Moolenbroek 	}
259eda6f593SDavid van Moolenbroek 
260eda6f593SDavid van Moolenbroek 	if ((n = sendmsg(msgbuf->fd, &msg, 0)) == -1) {
261eda6f593SDavid van Moolenbroek 		if (errno == EAGAIN || errno == ENOBUFS ||
262eda6f593SDavid van Moolenbroek 		    errno == EINTR)	/* try later */
263eda6f593SDavid van Moolenbroek 			return (0);
264eda6f593SDavid van Moolenbroek 		else
265eda6f593SDavid van Moolenbroek 			return (-1);
266eda6f593SDavid van Moolenbroek 	}
267eda6f593SDavid van Moolenbroek 
268eda6f593SDavid van Moolenbroek 	if (n == 0) {			/* connection closed */
269eda6f593SDavid van Moolenbroek 		errno = 0;
270eda6f593SDavid van Moolenbroek 		return (-2);
271eda6f593SDavid van Moolenbroek 	}
272eda6f593SDavid van Moolenbroek 
273eda6f593SDavid van Moolenbroek 	/*
274eda6f593SDavid van Moolenbroek 	 * assumption: fd got sent if sendmsg sent anything
275eda6f593SDavid van Moolenbroek 	 * this works because fds are passed one at a time
276eda6f593SDavid van Moolenbroek 	 */
277eda6f593SDavid van Moolenbroek 	if (buf != NULL && buf->fd != -1) {
278eda6f593SDavid van Moolenbroek 		close(buf->fd);
279eda6f593SDavid van Moolenbroek 		buf->fd = -1;
280eda6f593SDavid van Moolenbroek 	}
281eda6f593SDavid van Moolenbroek 
282eda6f593SDavid van Moolenbroek 	msgbuf_drain(msgbuf, n);
283eda6f593SDavid van Moolenbroek 
284eda6f593SDavid van Moolenbroek 	return (0);
285eda6f593SDavid van Moolenbroek }
286eda6f593SDavid van Moolenbroek 
287eda6f593SDavid van Moolenbroek void
ibuf_enqueue(struct msgbuf * msgbuf,struct ibuf * buf)288eda6f593SDavid van Moolenbroek ibuf_enqueue(struct msgbuf *msgbuf, struct ibuf *buf)
289eda6f593SDavid van Moolenbroek {
290eda6f593SDavid van Moolenbroek 	TAILQ_INSERT_TAIL(&msgbuf->bufs, buf, entry);
291eda6f593SDavid van Moolenbroek 	msgbuf->queued++;
292eda6f593SDavid van Moolenbroek }
293eda6f593SDavid van Moolenbroek 
294eda6f593SDavid van Moolenbroek void
ibuf_dequeue(struct msgbuf * msgbuf,struct ibuf * buf)295eda6f593SDavid van Moolenbroek ibuf_dequeue(struct msgbuf *msgbuf, struct ibuf *buf)
296eda6f593SDavid van Moolenbroek {
297eda6f593SDavid van Moolenbroek 	TAILQ_REMOVE(&msgbuf->bufs, buf, entry);
298eda6f593SDavid van Moolenbroek 
299eda6f593SDavid van Moolenbroek 	if (buf->fd != -1)
300eda6f593SDavid van Moolenbroek 		close(buf->fd);
301eda6f593SDavid van Moolenbroek 
302eda6f593SDavid van Moolenbroek 	msgbuf->queued--;
303eda6f593SDavid van Moolenbroek 	ibuf_free(buf);
304eda6f593SDavid van Moolenbroek }
305