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