Towards 2.1.40.
[yaz-moved-to-github.git] / src / wrbuf.c
1 /*
2  * Copyright (C) 1995-2005, Index Data ApS
3  * See the file LICENSE for details.
4  *
5  * $Id: wrbuf.c,v 1.12 2006-08-28 12:34:41 adam Exp $
6  */
7
8 /**
9  * \file wrbuf.c
10  * \brief Implements WRBUF (growing buffer)
11  */
12
13 #if HAVE_CONFIG_H
14 #include <config.h>
15 #endif
16
17 #include <stdio.h>
18 #include <stdlib.h>
19 #include <string.h>
20 #include <stdarg.h>
21
22 #include <yaz/wrbuf.h>
23 #include <yaz/yaz-iconv.h>
24
25 WRBUF wrbuf_alloc(void)
26 {
27     WRBUF n;
28
29     if (!(n = (WRBUF)xmalloc(sizeof(*n))))
30         abort();
31     n->buf = 0;
32     n->size = 0;
33     n->pos = 0;
34     return n;
35 }
36
37 void wrbuf_free(WRBUF b, int free_buf)
38 {
39     if (free_buf && b->buf)
40         xfree(b->buf);
41     xfree(b);
42 }
43
44 void wrbuf_rewind(WRBUF b)
45 {
46     b->pos = 0;
47 }
48
49 int wrbuf_grow(WRBUF b, int minsize)
50 {
51     int togrow;
52
53     if (!b->size)
54         togrow = 1024;
55     else
56         togrow = b->size;
57     if (togrow < minsize)
58         togrow = minsize;
59     if (b->size && !(b->buf =(char *)xrealloc(b->buf, b->size += togrow)))
60         abort();
61     else if (!b->size && !(b->buf = (char *)xmalloc(b->size = togrow)))
62         abort();
63     return 0;
64 }
65
66 int wrbuf_write(WRBUF b, const char *buf, int size)
67 {
68     if (size <= 0)
69         return 0;
70     if (b->pos + size >= b->size)
71         wrbuf_grow(b, size);
72     memcpy(b->buf + b->pos, buf, size);
73     b->pos += size;
74     return 0;
75 }
76
77 int wrbuf_puts(WRBUF b, const char *buf)
78 {
79     wrbuf_write(b, buf, strlen(buf)+1);  /* '\0'-terminate as well */
80     (b->pos)--;                          /* don't include '\0' in count */
81     return 0;
82 }
83
84 int wrbuf_puts_replace_char(WRBUF b, const char *buf, 
85                             const char from, const char to)
86 {
87     while(*buf){
88         if (*buf == from)
89             wrbuf_putc(b, to);
90         else
91             wrbuf_putc(b, *buf);
92         buf++;
93     }
94     wrbuf_putc(b, 0);
95     (b->pos)--;                          /* don't include '\0' in count */
96     return 0;
97 }
98
99 void wrbuf_chop_right(WRBUF b)
100 {
101     while (b->pos && b->buf[b->pos-1] == ' ')
102     {
103         (b->pos)--;
104         b->buf[b->pos] = '\0';
105     }
106 }
107
108 int wrbuf_xmlputs(WRBUF b, const char *cp)
109 {
110     return wrbuf_xmlputs_n(b, cp, strlen(cp));
111 }
112
113 int wrbuf_xmlputs_n(WRBUF b, const char *cp, int size)
114 {
115     while (--size >= 0)
116     {
117         /* only TAB,CR,LF of ASCII CTRL are allowed in XML 1.0! */
118         if (*cp >= 0 && *cp <= 31)
119             if (*cp != 9 && *cp != 10 && *cp != 13)
120             {
121                 cp++;  /* we silently ignore (delete) these.. */
122                 continue;
123             }
124         switch(*cp)
125         {
126         case '<':
127             wrbuf_puts(b, "&lt;");
128             break;
129         case '>':
130             wrbuf_puts(b, "&gt;");
131             break;
132         case '&':
133             wrbuf_puts(b, "&amp;");
134             break;
135         case '"':
136             wrbuf_puts(b, "&quot;");
137             break;
138         case '\'':
139             wrbuf_puts(b, "&apos;");
140             break;
141         default:
142             wrbuf_putc(b, *cp);
143         }
144         cp++;
145     }
146     wrbuf_putc(b, 0);
147     (b->pos)--;
148     return 0;
149 }
150
151 void wrbuf_printf(WRBUF b, const char *fmt, ...)
152 {
153     va_list ap;
154     char buf[4096];
155
156     va_start(ap, fmt);
157 #ifdef WIN32
158     _vsnprintf(buf, sizeof(buf)-1, fmt, ap);
159 #else
160 /* !WIN32 */
161 #if HAVE_VSNPRINTF
162     vsnprintf(buf, sizeof(buf)-1, fmt, ap);
163 #else
164     vsprintf(buf, fmt, ap);
165 #endif
166 #endif
167     wrbuf_puts (b, buf);
168
169     va_end(ap);
170 }
171
172 static int wrbuf_iconv_write_x(WRBUF b, yaz_iconv_t cd, const char *buf,
173                                int size, int cdata)
174 {
175     if (cd)
176     {
177         char outbuf[12];
178         size_t inbytesleft = size;
179         const char *inp = buf;
180         while (inbytesleft)
181         {
182             size_t outbytesleft = sizeof(outbuf);
183             char *outp = outbuf;
184             size_t r = yaz_iconv(cd, (char**) &inp,  &inbytesleft,
185                                  &outp, &outbytesleft);
186             if (r == (size_t) (-1))
187             {
188                 int e = yaz_iconv_error(cd);
189                 if (e != YAZ_ICONV_E2BIG)
190                     break;
191             }
192             if (cdata)
193                 wrbuf_xmlputs_n(b, outbuf, outp - outbuf);
194             else
195                 wrbuf_write(b, outbuf, outp - outbuf);
196         }
197     }
198     else
199     {
200         if (cdata)
201             wrbuf_xmlputs_n(b, buf, size);
202         else
203             wrbuf_write(b, buf, size);
204     }
205     return wrbuf_len(b);
206 }
207
208 int wrbuf_iconv_write(WRBUF b, yaz_iconv_t cd, const char *buf, int size)
209 {
210     return wrbuf_iconv_write_x(b, cd, buf, size, 0);
211 }
212
213 int wrbuf_iconv_puts(WRBUF b, yaz_iconv_t cd, const char *strz)
214 {
215     return wrbuf_iconv_write(b, cd, strz, strlen(strz));
216 }
217
218 int wrbuf_iconv_putchar(WRBUF b, yaz_iconv_t cd, int ch)
219 {
220     char buf[1];
221     buf[0] = ch;
222     return wrbuf_iconv_write(b, cd, buf, 1);
223 }
224
225 int wrbuf_iconv_write_cdata(WRBUF b, yaz_iconv_t cd, const char *buf, int size)
226 {
227     return wrbuf_iconv_write_x(b, cd, buf, size, 1);
228 }
229
230 /*
231  * Local variables:
232  * c-basic-offset: 4
233  * indent-tabs-mode: nil
234  * End:
235  * vim: shiftwidth=4 tabstop=8 expandtab
236  */
237