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