xref: /freebsd/lib/libc/iconv/citrus_memstream.c (revision ad30f8e79bd1007cc2476e491bd21b4f5e389e0a)
1*ad30f8e7SGabor Kovesdan /* $FreeBSD$ */
2*ad30f8e7SGabor Kovesdan /* $NetBSD: citrus_memstream.c,v 1.4 2009/02/03 05:02:12 lukem Exp $ */
3*ad30f8e7SGabor Kovesdan 
4*ad30f8e7SGabor Kovesdan /*-
5*ad30f8e7SGabor Kovesdan  * Copyright (c)2003 Citrus Project,
6*ad30f8e7SGabor Kovesdan  * All rights reserved.
7*ad30f8e7SGabor Kovesdan  *
8*ad30f8e7SGabor Kovesdan  * Redistribution and use in source and binary forms, with or without
9*ad30f8e7SGabor Kovesdan  * modification, are permitted provided that the following conditions
10*ad30f8e7SGabor Kovesdan  * are met:
11*ad30f8e7SGabor Kovesdan  * 1. Redistributions of source code must retain the above copyright
12*ad30f8e7SGabor Kovesdan  *    notice, this list of conditions and the following disclaimer.
13*ad30f8e7SGabor Kovesdan  * 2. Redistributions in binary form must reproduce the above copyright
14*ad30f8e7SGabor Kovesdan  *    notice, this list of conditions and the following disclaimer in the
15*ad30f8e7SGabor Kovesdan  *    documentation and/or other materials provided with the distribution.
16*ad30f8e7SGabor Kovesdan  *
17*ad30f8e7SGabor Kovesdan  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18*ad30f8e7SGabor Kovesdan  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19*ad30f8e7SGabor Kovesdan  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20*ad30f8e7SGabor Kovesdan  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21*ad30f8e7SGabor Kovesdan  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*ad30f8e7SGabor Kovesdan  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23*ad30f8e7SGabor Kovesdan  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24*ad30f8e7SGabor Kovesdan  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25*ad30f8e7SGabor Kovesdan  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26*ad30f8e7SGabor Kovesdan  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27*ad30f8e7SGabor Kovesdan  * SUCH DAMAGE.
28*ad30f8e7SGabor Kovesdan  */
29*ad30f8e7SGabor Kovesdan 
30*ad30f8e7SGabor Kovesdan #include <sys/cdefs.h>
31*ad30f8e7SGabor Kovesdan 
32*ad30f8e7SGabor Kovesdan #include <assert.h>
33*ad30f8e7SGabor Kovesdan #include <stdio.h>
34*ad30f8e7SGabor Kovesdan #include <stdlib.h>
35*ad30f8e7SGabor Kovesdan #include <string.h>
36*ad30f8e7SGabor Kovesdan 
37*ad30f8e7SGabor Kovesdan #include "citrus_namespace.h"
38*ad30f8e7SGabor Kovesdan #include "citrus_region.h"
39*ad30f8e7SGabor Kovesdan #include "citrus_memstream.h"
40*ad30f8e7SGabor Kovesdan #include "citrus_bcs.h"
41*ad30f8e7SGabor Kovesdan 
42*ad30f8e7SGabor Kovesdan const char *
43*ad30f8e7SGabor Kovesdan _citrus_memory_stream_getln(struct _citrus_memory_stream * __restrict ms,
44*ad30f8e7SGabor Kovesdan     size_t * __restrict rlen)
45*ad30f8e7SGabor Kovesdan {
46*ad30f8e7SGabor Kovesdan 	const uint8_t *h, *p;
47*ad30f8e7SGabor Kovesdan 	size_t ret;
48*ad30f8e7SGabor Kovesdan 	int i;
49*ad30f8e7SGabor Kovesdan 
50*ad30f8e7SGabor Kovesdan 	if (ms->ms_pos>=_region_size(&ms->ms_region))
51*ad30f8e7SGabor Kovesdan 		return (NULL);
52*ad30f8e7SGabor Kovesdan 
53*ad30f8e7SGabor Kovesdan 	h = p = (uint8_t *)_region_offset(&ms->ms_region, ms->ms_pos);
54*ad30f8e7SGabor Kovesdan 	ret = 0;
55*ad30f8e7SGabor Kovesdan 	for (i = _region_size(&ms->ms_region) - ms->ms_pos; i > 0; i--) {
56*ad30f8e7SGabor Kovesdan 		ret++;
57*ad30f8e7SGabor Kovesdan 		if (_bcs_iseol(*p))
58*ad30f8e7SGabor Kovesdan 			break;
59*ad30f8e7SGabor Kovesdan 		p++;
60*ad30f8e7SGabor Kovesdan 	}
61*ad30f8e7SGabor Kovesdan 
62*ad30f8e7SGabor Kovesdan 	ms->ms_pos += ret;
63*ad30f8e7SGabor Kovesdan 	*rlen = ret;
64*ad30f8e7SGabor Kovesdan 	return ((const char *)h);
65*ad30f8e7SGabor Kovesdan }
66*ad30f8e7SGabor Kovesdan 
67*ad30f8e7SGabor Kovesdan #define T_COMM	'#'
68*ad30f8e7SGabor Kovesdan 
69*ad30f8e7SGabor Kovesdan const char *
70*ad30f8e7SGabor Kovesdan _citrus_memory_stream_matchline(struct _citrus_memory_stream * __restrict ms,
71*ad30f8e7SGabor Kovesdan     const char * __restrict key, size_t * __restrict rlen, int iscasesensitive)
72*ad30f8e7SGabor Kovesdan {
73*ad30f8e7SGabor Kovesdan 	const char *p, *q;
74*ad30f8e7SGabor Kovesdan 	size_t keylen, len;
75*ad30f8e7SGabor Kovesdan 
76*ad30f8e7SGabor Kovesdan 	keylen = strlen(key);
77*ad30f8e7SGabor Kovesdan 	for(;;) {
78*ad30f8e7SGabor Kovesdan 		p = _citrus_memory_stream_getln(ms, &len);
79*ad30f8e7SGabor Kovesdan 		if (p == NULL)
80*ad30f8e7SGabor Kovesdan 			return (NULL);
81*ad30f8e7SGabor Kovesdan 
82*ad30f8e7SGabor Kovesdan 		/* ignore comment */
83*ad30f8e7SGabor Kovesdan 		q = memchr(p, T_COMM, len);
84*ad30f8e7SGabor Kovesdan 		if (q) {
85*ad30f8e7SGabor Kovesdan 			len = q - p;
86*ad30f8e7SGabor Kovesdan 		}
87*ad30f8e7SGabor Kovesdan 		/* ignore trailing white space and newline */
88*ad30f8e7SGabor Kovesdan 		_bcs_trunc_rws_len(p, &len);
89*ad30f8e7SGabor Kovesdan 		if (len == 0)
90*ad30f8e7SGabor Kovesdan 			continue; /* ignore null line */
91*ad30f8e7SGabor Kovesdan 
92*ad30f8e7SGabor Kovesdan 		/* skip white spaces at the head of the line */
93*ad30f8e7SGabor Kovesdan 		p = _bcs_skip_ws_len(p, &len);
94*ad30f8e7SGabor Kovesdan 		q = _bcs_skip_nonws_len(p, &len);
95*ad30f8e7SGabor Kovesdan 
96*ad30f8e7SGabor Kovesdan 		if ((size_t)(q - p) == keylen) {
97*ad30f8e7SGabor Kovesdan 			if (iscasesensitive) {
98*ad30f8e7SGabor Kovesdan 				if (memcmp(key, p, keylen) == 0)
99*ad30f8e7SGabor Kovesdan 					break; /* match */
100*ad30f8e7SGabor Kovesdan 			} else {
101*ad30f8e7SGabor Kovesdan 				if (_bcs_strncasecmp(key, p, keylen) == 0)
102*ad30f8e7SGabor Kovesdan 					break; /* match */
103*ad30f8e7SGabor Kovesdan 			}
104*ad30f8e7SGabor Kovesdan 		}
105*ad30f8e7SGabor Kovesdan 	}
106*ad30f8e7SGabor Kovesdan 
107*ad30f8e7SGabor Kovesdan 	p = _bcs_skip_ws_len(q, &len);
108*ad30f8e7SGabor Kovesdan 	*rlen = len;
109*ad30f8e7SGabor Kovesdan 
110*ad30f8e7SGabor Kovesdan 	return (p);
111*ad30f8e7SGabor Kovesdan }
112*ad30f8e7SGabor Kovesdan 
113*ad30f8e7SGabor Kovesdan void *
114*ad30f8e7SGabor Kovesdan _citrus_memory_stream_chr(struct _citrus_memory_stream *ms,
115*ad30f8e7SGabor Kovesdan     struct _citrus_region *r, char ch)
116*ad30f8e7SGabor Kovesdan {
117*ad30f8e7SGabor Kovesdan 	void *chr, *head;
118*ad30f8e7SGabor Kovesdan 	size_t sz;
119*ad30f8e7SGabor Kovesdan 
120*ad30f8e7SGabor Kovesdan 	if (ms->ms_pos >= _region_size(&ms->ms_region))
121*ad30f8e7SGabor Kovesdan 		return (NULL);
122*ad30f8e7SGabor Kovesdan 
123*ad30f8e7SGabor Kovesdan 	head = _region_offset(&ms->ms_region, ms->ms_pos);
124*ad30f8e7SGabor Kovesdan 	chr = memchr(head, ch, _memstream_remainder(ms));
125*ad30f8e7SGabor Kovesdan 	if (chr == NULL) {
126*ad30f8e7SGabor Kovesdan 		_region_init(r, head, _memstream_remainder(ms));
127*ad30f8e7SGabor Kovesdan 		ms->ms_pos = _region_size(&ms->ms_region);
128*ad30f8e7SGabor Kovesdan 		return (NULL);
129*ad30f8e7SGabor Kovesdan 	}
130*ad30f8e7SGabor Kovesdan 	sz = (char *)chr - (char *)head;
131*ad30f8e7SGabor Kovesdan 
132*ad30f8e7SGabor Kovesdan 	_region_init(r, head, sz);
133*ad30f8e7SGabor Kovesdan 	ms->ms_pos += sz + 1;
134*ad30f8e7SGabor Kovesdan 
135*ad30f8e7SGabor Kovesdan 	return (chr);
136*ad30f8e7SGabor Kovesdan }
137*ad30f8e7SGabor Kovesdan 
138*ad30f8e7SGabor Kovesdan void
139*ad30f8e7SGabor Kovesdan _citrus_memory_stream_skip_ws(struct _citrus_memory_stream *ms)
140*ad30f8e7SGabor Kovesdan {
141*ad30f8e7SGabor Kovesdan 	int ch;
142*ad30f8e7SGabor Kovesdan 
143*ad30f8e7SGabor Kovesdan 	while ((ch = _memstream_peek(ms)) != EOF) {
144*ad30f8e7SGabor Kovesdan 		if (!_bcs_isspace(ch))
145*ad30f8e7SGabor Kovesdan 			break;
146*ad30f8e7SGabor Kovesdan 		_memstream_getc(ms);
147*ad30f8e7SGabor Kovesdan 	}
148*ad30f8e7SGabor Kovesdan }
149