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