xref: /freebsd/contrib/lib9p/sbuf/sbuf.c (revision 134e17798c9af53632b372348ab828e75e65bf46)
1*134e1779SJakub Wojciech Klama /*
2*134e1779SJakub Wojciech Klama  * Copyright 2016 Jakub Klama <jceel@FreeBSD.org>
3*134e1779SJakub Wojciech Klama  * All rights reserved
4*134e1779SJakub Wojciech Klama  *
5*134e1779SJakub Wojciech Klama  * Redistribution and use in source and binary forms, with or without
6*134e1779SJakub Wojciech Klama  * modification, are permitted providing that the following conditions
7*134e1779SJakub Wojciech Klama  * are met:
8*134e1779SJakub Wojciech Klama  * 1. Redistributions of source code must retain the above copyright
9*134e1779SJakub Wojciech Klama  *    notice, this list of conditions and the following disclaimer.
10*134e1779SJakub Wojciech Klama  * 2. Redistributions in binary form must reproduce the above copyright
11*134e1779SJakub Wojciech Klama  *    notice, this list of conditions and the following disclaimer in the
12*134e1779SJakub Wojciech Klama  *    documentation and/or other materials provided with the distribution.
13*134e1779SJakub Wojciech Klama  *
14*134e1779SJakub Wojciech Klama  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
15*134e1779SJakub Wojciech Klama  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
16*134e1779SJakub Wojciech Klama  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17*134e1779SJakub Wojciech Klama  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
18*134e1779SJakub Wojciech Klama  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19*134e1779SJakub Wojciech Klama  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20*134e1779SJakub Wojciech Klama  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21*134e1779SJakub Wojciech Klama  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
22*134e1779SJakub Wojciech Klama  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
23*134e1779SJakub Wojciech Klama  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
24*134e1779SJakub Wojciech Klama  * POSSIBILITY OF SUCH DAMAGE.
25*134e1779SJakub Wojciech Klama  *
26*134e1779SJakub Wojciech Klama  */
27*134e1779SJakub Wojciech Klama 
28*134e1779SJakub Wojciech Klama /*
29*134e1779SJakub Wojciech Klama  * Minimal libsbuf reimplementation for Mac OS X.
30*134e1779SJakub Wojciech Klama  */
31*134e1779SJakub Wojciech Klama 
32*134e1779SJakub Wojciech Klama #include <stdlib.h>
33*134e1779SJakub Wojciech Klama #include <unistd.h>
34*134e1779SJakub Wojciech Klama #include <string.h>
35*134e1779SJakub Wojciech Klama #include <stdarg.h>
36*134e1779SJakub Wojciech Klama #include <stdio.h>
37*134e1779SJakub Wojciech Klama #include "sbuf.h"
38*134e1779SJakub Wojciech Klama 
39*134e1779SJakub Wojciech Klama #define	SBUF_INITIAL_SIZE	128
40*134e1779SJakub Wojciech Klama 
41*134e1779SJakub Wojciech Klama struct sbuf *
sbuf_new_auto()42*134e1779SJakub Wojciech Klama sbuf_new_auto()
43*134e1779SJakub Wojciech Klama {
44*134e1779SJakub Wojciech Klama 	struct sbuf *s;
45*134e1779SJakub Wojciech Klama 
46*134e1779SJakub Wojciech Klama 	s = malloc(sizeof(struct sbuf));
47*134e1779SJakub Wojciech Klama 	s->s_buf = calloc(1, SBUF_INITIAL_SIZE + 1);
48*134e1779SJakub Wojciech Klama 	s->s_capacity = s->s_buf != NULL ? SBUF_INITIAL_SIZE : 0;
49*134e1779SJakub Wojciech Klama 	s->s_size = 0;
50*134e1779SJakub Wojciech Klama 
51*134e1779SJakub Wojciech Klama 	return (s);
52*134e1779SJakub Wojciech Klama }
53*134e1779SJakub Wojciech Klama 
54*134e1779SJakub Wojciech Klama int
sbuf_cat(struct sbuf * s,const char * str)55*134e1779SJakub Wojciech Klama sbuf_cat(struct sbuf *s, const char *str)
56*134e1779SJakub Wojciech Klama {
57*134e1779SJakub Wojciech Klama 	int req = (int)strlen(str);
58*134e1779SJakub Wojciech Klama 
59*134e1779SJakub Wojciech Klama 	if (s->s_size + req >= s->s_capacity) {
60*134e1779SJakub Wojciech Klama 		s->s_capacity = s->s_size + req + 1;
61*134e1779SJakub Wojciech Klama 		s->s_buf = realloc(s->s_buf, (size_t)s->s_capacity);
62*134e1779SJakub Wojciech Klama 	}
63*134e1779SJakub Wojciech Klama 	if (s->s_buf == NULL)
64*134e1779SJakub Wojciech Klama 		return (-1);
65*134e1779SJakub Wojciech Klama 
66*134e1779SJakub Wojciech Klama 	strcpy(s->s_buf + s->s_size, str);
67*134e1779SJakub Wojciech Klama 	s->s_size += req;
68*134e1779SJakub Wojciech Klama 
69*134e1779SJakub Wojciech Klama 	return (0);
70*134e1779SJakub Wojciech Klama }
71*134e1779SJakub Wojciech Klama 
72*134e1779SJakub Wojciech Klama int
sbuf_printf(struct sbuf * s,const char * fmt,...)73*134e1779SJakub Wojciech Klama sbuf_printf(struct sbuf *s, const char *fmt, ...)
74*134e1779SJakub Wojciech Klama {
75*134e1779SJakub Wojciech Klama 	int ret;
76*134e1779SJakub Wojciech Klama 	va_list ap;
77*134e1779SJakub Wojciech Klama 
78*134e1779SJakub Wojciech Klama 	va_start(ap, fmt);
79*134e1779SJakub Wojciech Klama 	ret = sbuf_vprintf(s, fmt, ap);
80*134e1779SJakub Wojciech Klama 	va_end(ap);
81*134e1779SJakub Wojciech Klama 
82*134e1779SJakub Wojciech Klama 	return (ret);
83*134e1779SJakub Wojciech Klama }
84*134e1779SJakub Wojciech Klama 
85*134e1779SJakub Wojciech Klama int
sbuf_vprintf(struct sbuf * s,const char * fmt,va_list args)86*134e1779SJakub Wojciech Klama sbuf_vprintf(struct sbuf *s, const char *fmt, va_list args)
87*134e1779SJakub Wojciech Klama {
88*134e1779SJakub Wojciech Klama 	va_list copy;
89*134e1779SJakub Wojciech Klama 	int req;
90*134e1779SJakub Wojciech Klama 
91*134e1779SJakub Wojciech Klama 	va_copy(copy, args);
92*134e1779SJakub Wojciech Klama 	req = vsnprintf(NULL, 0, fmt, copy);
93*134e1779SJakub Wojciech Klama 	va_end(copy);
94*134e1779SJakub Wojciech Klama 
95*134e1779SJakub Wojciech Klama 	if (s->s_size + req >= s->s_capacity) {
96*134e1779SJakub Wojciech Klama 		s->s_capacity = s->s_size + req + 1;
97*134e1779SJakub Wojciech Klama 		s->s_buf = realloc(s->s_buf, (size_t)s->s_capacity);
98*134e1779SJakub Wojciech Klama 	}
99*134e1779SJakub Wojciech Klama 	if (s->s_buf == NULL)
100*134e1779SJakub Wojciech Klama 		return (-1);
101*134e1779SJakub Wojciech Klama 
102*134e1779SJakub Wojciech Klama 	req = vsnprintf(s->s_buf + s->s_size, req + 1, fmt, args);
103*134e1779SJakub Wojciech Klama 	s->s_size += req;
104*134e1779SJakub Wojciech Klama 
105*134e1779SJakub Wojciech Klama 	return (0);
106*134e1779SJakub Wojciech Klama }
107*134e1779SJakub Wojciech Klama 
108*134e1779SJakub Wojciech Klama char *
sbuf_data(struct sbuf * s)109*134e1779SJakub Wojciech Klama sbuf_data(struct sbuf *s)
110*134e1779SJakub Wojciech Klama {
111*134e1779SJakub Wojciech Klama 	return (s->s_buf);
112*134e1779SJakub Wojciech Klama }
113*134e1779SJakub Wojciech Klama 
114*134e1779SJakub Wojciech Klama int
sbuf_finish(struct sbuf * s)115*134e1779SJakub Wojciech Klama sbuf_finish(struct sbuf *s)
116*134e1779SJakub Wojciech Klama {
117*134e1779SJakub Wojciech Klama 	if (s->s_buf != NULL)
118*134e1779SJakub Wojciech Klama 		s->s_buf[s->s_size] = '\0';
119*134e1779SJakub Wojciech Klama 	return (0);
120*134e1779SJakub Wojciech Klama }
121*134e1779SJakub Wojciech Klama 
122*134e1779SJakub Wojciech Klama void
sbuf_delete(struct sbuf * s)123*134e1779SJakub Wojciech Klama sbuf_delete(struct sbuf *s)
124*134e1779SJakub Wojciech Klama {
125*134e1779SJakub Wojciech Klama 	free(s->s_buf);
126*134e1779SJakub Wojciech Klama 	free(s);
127*134e1779SJakub Wojciech Klama }
128