xref: /openbsd-src/usr.sbin/syslogd/ringbuf.c (revision c572cfa710a368c2fa815a9371225c763a87b50c)
1*c572cfa7Sbluhm /* $OpenBSD: ringbuf.c,v 1.10 2016/10/16 22:12:50 bluhm Exp $ */
2c1d2012aSdjm 
3c1d2012aSdjm /*
4c1d2012aSdjm  * Copyright (c) 2004 Damien Miller
5c1d2012aSdjm  *
6c1d2012aSdjm  * Permission to use, copy, modify, and distribute this software for any
7c1d2012aSdjm  * purpose with or without fee is hereby granted, provided that the above
8c1d2012aSdjm  * copyright notice and this permission notice appear in all copies.
9c1d2012aSdjm  *
10c1d2012aSdjm  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11c1d2012aSdjm  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12c1d2012aSdjm  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13c1d2012aSdjm  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14c1d2012aSdjm  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15c1d2012aSdjm  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16c1d2012aSdjm  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17c1d2012aSdjm  */
18c1d2012aSdjm 
19c1d2012aSdjm /*
20c1d2012aSdjm  * Simple ringbuffer for lines of text.
21c1d2012aSdjm  */
22c1d2012aSdjm 
23c1d2012aSdjm #include <stdio.h>
24c1d2012aSdjm #include <stdlib.h>
25c1d2012aSdjm #include <string.h>
26c1d2012aSdjm 
27c1d2012aSdjm #include "syslogd.h"
28c1d2012aSdjm 
29b9fc9a72Sderaadt #define MINIMUM(a, b)	(((a) < (b)) ? (a) : (b))
30b9fc9a72Sderaadt 
31c1d2012aSdjm /* Initialise a ring buffer */
32c1d2012aSdjm struct ringbuf *
ringbuf_init(size_t len)33c1d2012aSdjm ringbuf_init(size_t len)
34c1d2012aSdjm {
35c1d2012aSdjm 	struct ringbuf *ret;
36c1d2012aSdjm 
37c1d2012aSdjm 	if (len == 0 || (ret = malloc(sizeof(*ret))) == NULL)
38c1d2012aSdjm 		return (NULL);
39c1d2012aSdjm 
40c1d2012aSdjm 	if ((ret->buf = malloc(len)) == NULL) {
41c1d2012aSdjm 		free(ret);
42c1d2012aSdjm 		return (NULL);
43c1d2012aSdjm 	}
44c1d2012aSdjm 
45c1d2012aSdjm 	ret->len = len;
46c1d2012aSdjm 	ret->start = ret->end = 0;
47c1d2012aSdjm 
48c1d2012aSdjm 	return (ret);
49c1d2012aSdjm }
50c1d2012aSdjm 
51e735b9b4Sdjm /* Free a ring buffer */
52e735b9b4Sdjm void
ringbuf_free(struct ringbuf * rb)53e735b9b4Sdjm ringbuf_free(struct ringbuf *rb)
54e735b9b4Sdjm {
55e735b9b4Sdjm 	free(rb->buf);
56e735b9b4Sdjm 	free(rb);
57e735b9b4Sdjm }
58e735b9b4Sdjm 
59c1d2012aSdjm /* Clear a ring buffer */
60c1d2012aSdjm void
ringbuf_clear(struct ringbuf * rb)61c1d2012aSdjm ringbuf_clear(struct ringbuf *rb)
62c1d2012aSdjm {
63c1d2012aSdjm 	rb->start = rb->end = 0;
64c1d2012aSdjm }
65c1d2012aSdjm 
66c1d2012aSdjm /* Return the number of bytes used in a ringbuffer */
67c1d2012aSdjm size_t
ringbuf_used(struct ringbuf * rb)68c1d2012aSdjm ringbuf_used(struct ringbuf *rb)
69c1d2012aSdjm {
70c1d2012aSdjm 	return ((rb->len + rb->end - rb->start) % rb->len);
71c1d2012aSdjm }
72c1d2012aSdjm 
73c1d2012aSdjm /*
74c1d2012aSdjm  * Append a line to a ring buffer, will delete lines from start
75c1d2012aSdjm  * of buffer as necessary
76c1d2012aSdjm  */
77c1d2012aSdjm int
ringbuf_append_line(struct ringbuf * rb,char * line)78c1d2012aSdjm ringbuf_append_line(struct ringbuf *rb, char *line)
79c1d2012aSdjm {
80c1d2012aSdjm 	size_t llen, used, copy_len;
8135b96954Sdjm 	int overflow = 0;
82c1d2012aSdjm 
83c1d2012aSdjm 	if (rb == NULL || line == NULL)
84c1d2012aSdjm 		return (-1);
85c1d2012aSdjm 
86c1d2012aSdjm 	llen = strlen(line);
876a1166c6Sdjm 	if (llen == 0)
886a1166c6Sdjm 		return (-1);
896a1166c6Sdjm 
90c1d2012aSdjm 	if (line[llen - 1] != '\n')
91c1d2012aSdjm 		llen++; /* one extra for appended '\n' */
92c1d2012aSdjm 
936a1166c6Sdjm 	if (llen >= rb->len)
94c1d2012aSdjm 		return (-1);
95c1d2012aSdjm 
96c1d2012aSdjm 	/*
97c1d2012aSdjm 	 * If necessary, advance start pointer to make room for appended
98c1d2012aSdjm 	 * string. Ensure that start pointer is at the beginning of a line
99c1d2012aSdjm 	 * once we are done (i.e move to after '\n').
100c1d2012aSdjm 	 */
101c1d2012aSdjm 	used = ringbuf_used(rb);
102c1d2012aSdjm 	if (used + llen >= rb->len) {
103c1d2012aSdjm 		rb->start = (rb->start + used + llen - rb->len) % rb->len;
104c1d2012aSdjm 
105c1d2012aSdjm 		/* Find next '\n' */
106c1d2012aSdjm 		while (rb->buf[rb->start] != '\n')
107c1d2012aSdjm 			rb->start = (rb->start + 1) % rb->len;
108c1d2012aSdjm 		/* Skip it */
109c1d2012aSdjm 		rb->start = (rb->start + 1) % rb->len;
11035b96954Sdjm 
11135b96954Sdjm 		overflow = 1;
112c1d2012aSdjm 	}
113c1d2012aSdjm 
114c1d2012aSdjm 	/*
115f64cf283Savsm 	 * Now append string, starting from last pointer and wrapping if
116c1d2012aSdjm 	 * necessary
117c1d2012aSdjm 	 */
118c1d2012aSdjm 	if (rb->end + llen > rb->len) {
119c1d2012aSdjm 		copy_len = rb->len - rb->end;
120c1d2012aSdjm 		memcpy(rb->buf + rb->end, line, copy_len);
121c1d2012aSdjm 		memcpy(rb->buf, line + copy_len, llen - copy_len - 1);
122c1d2012aSdjm 		rb->buf[llen - copy_len - 1] = '\n';
123c1d2012aSdjm 	} else {
124c1d2012aSdjm 		memcpy(rb->buf + rb->end, line, llen - 1);
125c1d2012aSdjm 		rb->buf[rb->end + llen - 1] = '\n';
126c1d2012aSdjm 	}
127c1d2012aSdjm 
128c1d2012aSdjm 	rb->end = (rb->end + llen) % rb->len;
129c1d2012aSdjm 
13035b96954Sdjm 	return (overflow);
131c1d2012aSdjm }
132c1d2012aSdjm 
133c1d2012aSdjm /*
134c1d2012aSdjm  * Copy and nul-terminate a ringbuffer to a string.
135c1d2012aSdjm  */
136c1d2012aSdjm ssize_t
ringbuf_to_string(char * buf,size_t len,struct ringbuf * rb)137c1d2012aSdjm ringbuf_to_string(char *buf, size_t len, struct ringbuf *rb)
138c1d2012aSdjm {
1399ab071d1Sdjm 	size_t copy_len, n;
140c1d2012aSdjm 
1419ab071d1Sdjm 	if (buf == NULL || rb == NULL || len == 0)
142c1d2012aSdjm 		return (-1);
143c1d2012aSdjm 
144b9fc9a72Sderaadt 	copy_len = MINIMUM(len - 1, ringbuf_used(rb));
145c1d2012aSdjm 
1469ab071d1Sdjm 	if (copy_len == 0)
147c1d2012aSdjm 		return (copy_len);
148c1d2012aSdjm 
149c1d2012aSdjm 	if (rb->start < rb->end)
150c1d2012aSdjm 		memcpy(buf, rb->buf + rb->start, copy_len);
151c1d2012aSdjm 	else {
152c1d2012aSdjm 		/* If the buffer is wrapped, copy each hunk separately */
153c1d2012aSdjm 		n = rb->len - rb->start;
154b9fc9a72Sderaadt 		memcpy(buf, rb->buf + rb->start, MINIMUM(n, copy_len));
1559ab071d1Sdjm 		if (copy_len > n)
156*c572cfa7Sbluhm 			memcpy(buf + n, rb->buf,
157*c572cfa7Sbluhm 			    MINIMUM(rb->end, copy_len - n));
158c1d2012aSdjm 	}
159c1d2012aSdjm 	buf[copy_len] = '\0';
160c1d2012aSdjm 
161c1d2012aSdjm 	return (ringbuf_used(rb));
162c1d2012aSdjm }
163