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