xref: /minix3/external/bsd/bind/dist/lib/isc/string.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: string.c,v 1.6 2014/12/10 04:37:59 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004-2007, 2011, 2012, 2014  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 1999-2001, 2003  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /*
21*00b67f09SDavid van Moolenbroek  * Copyright (c) 1990, 1993
22*00b67f09SDavid van Moolenbroek  *	The Regents of the University of California.  All rights reserved.
23*00b67f09SDavid van Moolenbroek  *
24*00b67f09SDavid van Moolenbroek  * Redistribution and use in source and binary forms, with or without
25*00b67f09SDavid van Moolenbroek  * modification, are permitted provided that the following conditions
26*00b67f09SDavid van Moolenbroek  * are met:
27*00b67f09SDavid van Moolenbroek  * 1. Redistributions of source code must retain the above copyright
28*00b67f09SDavid van Moolenbroek  *    notice, this list of conditions and the following disclaimer.
29*00b67f09SDavid van Moolenbroek  * 2. Redistributions in binary form must reproduce the above copyright
30*00b67f09SDavid van Moolenbroek  *    notice, this list of conditions and the following disclaimer in the
31*00b67f09SDavid van Moolenbroek  *    documentation and/or other materials provided with the distribution.
32*00b67f09SDavid van Moolenbroek  * 3. Neither the name of the University nor the names of its contributors
33*00b67f09SDavid van Moolenbroek  *    may be used to endorse or promote products derived from this software
34*00b67f09SDavid van Moolenbroek  *    without specific prior written permission.
35*00b67f09SDavid van Moolenbroek  *
36*00b67f09SDavid van Moolenbroek  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
37*00b67f09SDavid van Moolenbroek  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
38*00b67f09SDavid van Moolenbroek  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
39*00b67f09SDavid van Moolenbroek  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
40*00b67f09SDavid van Moolenbroek  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
41*00b67f09SDavid van Moolenbroek  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
42*00b67f09SDavid van Moolenbroek  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
43*00b67f09SDavid van Moolenbroek  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
44*00b67f09SDavid van Moolenbroek  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
45*00b67f09SDavid van Moolenbroek  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
46*00b67f09SDavid van Moolenbroek  * SUCH DAMAGE.
47*00b67f09SDavid van Moolenbroek  */
48*00b67f09SDavid van Moolenbroek 
49*00b67f09SDavid van Moolenbroek /*! \file */
50*00b67f09SDavid van Moolenbroek 
51*00b67f09SDavid van Moolenbroek #include <config.h>
52*00b67f09SDavid van Moolenbroek 
53*00b67f09SDavid van Moolenbroek #include <ctype.h>
54*00b67f09SDavid van Moolenbroek 
55*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
56*00b67f09SDavid van Moolenbroek #include <isc/print.h>
57*00b67f09SDavid van Moolenbroek #include <isc/region.h>
58*00b67f09SDavid van Moolenbroek #include <isc/string.h>
59*00b67f09SDavid van Moolenbroek #include <isc/util.h>
60*00b67f09SDavid van Moolenbroek 
61*00b67f09SDavid van Moolenbroek static char digits[] = "0123456789abcdefghijklmnoprstuvwxyz";
62*00b67f09SDavid van Moolenbroek 
63*00b67f09SDavid van Moolenbroek isc_uint64_t
isc_string_touint64(char * source,char ** end,int base)64*00b67f09SDavid van Moolenbroek isc_string_touint64(char *source, char **end, int base) {
65*00b67f09SDavid van Moolenbroek 	isc_uint64_t tmp;
66*00b67f09SDavid van Moolenbroek 	isc_uint64_t overflow;
67*00b67f09SDavid van Moolenbroek 	char *s = source;
68*00b67f09SDavid van Moolenbroek 	char *o;
69*00b67f09SDavid van Moolenbroek 	char c;
70*00b67f09SDavid van Moolenbroek 
71*00b67f09SDavid van Moolenbroek 	if ((base < 0) || (base == 1) || (base > 36)) {
72*00b67f09SDavid van Moolenbroek 		*end = source;
73*00b67f09SDavid van Moolenbroek 		return (0);
74*00b67f09SDavid van Moolenbroek 	}
75*00b67f09SDavid van Moolenbroek 
76*00b67f09SDavid van Moolenbroek 	while (*s != 0 && isascii(*s&0xff) && isspace(*s&0xff))
77*00b67f09SDavid van Moolenbroek 		s++;
78*00b67f09SDavid van Moolenbroek 	if (*s == '+' /* || *s == '-' */)
79*00b67f09SDavid van Moolenbroek 		s++;
80*00b67f09SDavid van Moolenbroek 	if (base == 0) {
81*00b67f09SDavid van Moolenbroek 		if (*s == '0' && (*(s+1) == 'X' || *(s+1) == 'x')) {
82*00b67f09SDavid van Moolenbroek 			s += 2;
83*00b67f09SDavid van Moolenbroek 			base = 16;
84*00b67f09SDavid van Moolenbroek 		} else if (*s == '0')
85*00b67f09SDavid van Moolenbroek 			base = 8;
86*00b67f09SDavid van Moolenbroek 		else
87*00b67f09SDavid van Moolenbroek 			base = 10;
88*00b67f09SDavid van Moolenbroek 	}
89*00b67f09SDavid van Moolenbroek 	if (*s == 0) {
90*00b67f09SDavid van Moolenbroek 		*end = source;
91*00b67f09SDavid van Moolenbroek 		return (0);
92*00b67f09SDavid van Moolenbroek 	}
93*00b67f09SDavid van Moolenbroek 	overflow = ~0;
94*00b67f09SDavid van Moolenbroek 	overflow /= base;
95*00b67f09SDavid van Moolenbroek 	tmp = 0;
96*00b67f09SDavid van Moolenbroek 
97*00b67f09SDavid van Moolenbroek 	while ((c = *s) != 0) {
98*00b67f09SDavid van Moolenbroek 		c = tolower(c&0xff);
99*00b67f09SDavid van Moolenbroek 		/* end ? */
100*00b67f09SDavid van Moolenbroek 		if ((o = strchr(digits, c)) == NULL) {
101*00b67f09SDavid van Moolenbroek 			*end = s;
102*00b67f09SDavid van Moolenbroek 			return (tmp);
103*00b67f09SDavid van Moolenbroek 		}
104*00b67f09SDavid van Moolenbroek 		/* end ? */
105*00b67f09SDavid van Moolenbroek 		if ((o - digits) >= base) {
106*00b67f09SDavid van Moolenbroek 			*end = s;
107*00b67f09SDavid van Moolenbroek 			return (tmp);
108*00b67f09SDavid van Moolenbroek 		}
109*00b67f09SDavid van Moolenbroek 		/* overflow ? */
110*00b67f09SDavid van Moolenbroek 		if (tmp > overflow) {
111*00b67f09SDavid van Moolenbroek 			*end = source;
112*00b67f09SDavid van Moolenbroek 			return (0);
113*00b67f09SDavid van Moolenbroek 		}
114*00b67f09SDavid van Moolenbroek 		tmp *= base;
115*00b67f09SDavid van Moolenbroek 		/* overflow ? */
116*00b67f09SDavid van Moolenbroek 		if ((tmp + (o - digits)) < tmp) {
117*00b67f09SDavid van Moolenbroek 			*end = source;
118*00b67f09SDavid van Moolenbroek 			return (0);
119*00b67f09SDavid van Moolenbroek 		}
120*00b67f09SDavid van Moolenbroek 		tmp += o - digits;
121*00b67f09SDavid van Moolenbroek 		s++;
122*00b67f09SDavid van Moolenbroek 	}
123*00b67f09SDavid van Moolenbroek 	*end = s;
124*00b67f09SDavid van Moolenbroek 	return (tmp);
125*00b67f09SDavid van Moolenbroek }
126*00b67f09SDavid van Moolenbroek 
127*00b67f09SDavid van Moolenbroek isc_result_t
isc_string_copy(char * target,size_t size,const char * source)128*00b67f09SDavid van Moolenbroek isc_string_copy(char *target, size_t size, const char *source) {
129*00b67f09SDavid van Moolenbroek 	REQUIRE(size > 0U);
130*00b67f09SDavid van Moolenbroek 
131*00b67f09SDavid van Moolenbroek 	if (strlcpy(target, source, size) >= size) {
132*00b67f09SDavid van Moolenbroek 		memset(target, ISC_STRING_MAGIC, size);
133*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOSPACE);
134*00b67f09SDavid van Moolenbroek 	}
135*00b67f09SDavid van Moolenbroek 
136*00b67f09SDavid van Moolenbroek 	ENSURE(strlen(target) < size);
137*00b67f09SDavid van Moolenbroek 
138*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
139*00b67f09SDavid van Moolenbroek }
140*00b67f09SDavid van Moolenbroek 
141*00b67f09SDavid van Moolenbroek void
isc_string_copy_truncate(char * target,size_t size,const char * source)142*00b67f09SDavid van Moolenbroek isc_string_copy_truncate(char *target, size_t size, const char *source) {
143*00b67f09SDavid van Moolenbroek 	REQUIRE(size > 0U);
144*00b67f09SDavid van Moolenbroek 
145*00b67f09SDavid van Moolenbroek 	strlcpy(target, source, size);
146*00b67f09SDavid van Moolenbroek 
147*00b67f09SDavid van Moolenbroek 	ENSURE(strlen(target) < size);
148*00b67f09SDavid van Moolenbroek }
149*00b67f09SDavid van Moolenbroek 
150*00b67f09SDavid van Moolenbroek isc_result_t
isc_string_append(char * target,size_t size,const char * source)151*00b67f09SDavid van Moolenbroek isc_string_append(char *target, size_t size, const char *source) {
152*00b67f09SDavid van Moolenbroek 	REQUIRE(size > 0U);
153*00b67f09SDavid van Moolenbroek 	REQUIRE(strlen(target) < size);
154*00b67f09SDavid van Moolenbroek 
155*00b67f09SDavid van Moolenbroek 	if (strlcat(target, source, size) >= size) {
156*00b67f09SDavid van Moolenbroek 		memset(target, ISC_STRING_MAGIC, size);
157*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOSPACE);
158*00b67f09SDavid van Moolenbroek 	}
159*00b67f09SDavid van Moolenbroek 
160*00b67f09SDavid van Moolenbroek 	ENSURE(strlen(target) < size);
161*00b67f09SDavid van Moolenbroek 
162*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
163*00b67f09SDavid van Moolenbroek }
164*00b67f09SDavid van Moolenbroek 
165*00b67f09SDavid van Moolenbroek void
isc_string_append_truncate(char * target,size_t size,const char * source)166*00b67f09SDavid van Moolenbroek isc_string_append_truncate(char *target, size_t size, const char *source) {
167*00b67f09SDavid van Moolenbroek 	REQUIRE(size > 0U);
168*00b67f09SDavid van Moolenbroek 	REQUIRE(strlen(target) < size);
169*00b67f09SDavid van Moolenbroek 
170*00b67f09SDavid van Moolenbroek 	strlcat(target, source, size);
171*00b67f09SDavid van Moolenbroek 
172*00b67f09SDavid van Moolenbroek 	ENSURE(strlen(target) < size);
173*00b67f09SDavid van Moolenbroek }
174*00b67f09SDavid van Moolenbroek 
175*00b67f09SDavid van Moolenbroek isc_result_t
isc_string_printf(char * target,size_t size,const char * format,...)176*00b67f09SDavid van Moolenbroek isc_string_printf(char *target, size_t size, const char *format, ...) {
177*00b67f09SDavid van Moolenbroek 	va_list args;
178*00b67f09SDavid van Moolenbroek 	size_t n;
179*00b67f09SDavid van Moolenbroek 
180*00b67f09SDavid van Moolenbroek 	REQUIRE(size > 0U);
181*00b67f09SDavid van Moolenbroek 
182*00b67f09SDavid van Moolenbroek 	va_start(args, format);
183*00b67f09SDavid van Moolenbroek 	n = vsnprintf(target, size, format, args);
184*00b67f09SDavid van Moolenbroek 	va_end(args);
185*00b67f09SDavid van Moolenbroek 
186*00b67f09SDavid van Moolenbroek 	if (n >= size) {
187*00b67f09SDavid van Moolenbroek 		memset(target, ISC_STRING_MAGIC, size);
188*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOSPACE);
189*00b67f09SDavid van Moolenbroek 	}
190*00b67f09SDavid van Moolenbroek 
191*00b67f09SDavid van Moolenbroek 	ENSURE(strlen(target) < size);
192*00b67f09SDavid van Moolenbroek 
193*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
194*00b67f09SDavid van Moolenbroek }
195*00b67f09SDavid van Moolenbroek 
196*00b67f09SDavid van Moolenbroek void
isc_string_printf_truncate(char * target,size_t size,const char * format,...)197*00b67f09SDavid van Moolenbroek isc_string_printf_truncate(char *target, size_t size, const char *format, ...)
198*00b67f09SDavid van Moolenbroek {
199*00b67f09SDavid van Moolenbroek 	va_list args;
200*00b67f09SDavid van Moolenbroek 
201*00b67f09SDavid van Moolenbroek 	REQUIRE(size > 0U);
202*00b67f09SDavid van Moolenbroek 
203*00b67f09SDavid van Moolenbroek 	va_start(args, format);
204*00b67f09SDavid van Moolenbroek 	/* check return code? */
205*00b67f09SDavid van Moolenbroek 	(void)vsnprintf(target, size, format, args);
206*00b67f09SDavid van Moolenbroek 	va_end(args);
207*00b67f09SDavid van Moolenbroek 
208*00b67f09SDavid van Moolenbroek 	ENSURE(strlen(target) < size);
209*00b67f09SDavid van Moolenbroek }
210*00b67f09SDavid van Moolenbroek 
211*00b67f09SDavid van Moolenbroek char *
isc_string_regiondup(isc_mem_t * mctx,const isc_region_t * source)212*00b67f09SDavid van Moolenbroek isc_string_regiondup(isc_mem_t *mctx, const isc_region_t *source) {
213*00b67f09SDavid van Moolenbroek 	char *target;
214*00b67f09SDavid van Moolenbroek 
215*00b67f09SDavid van Moolenbroek 	REQUIRE(mctx != NULL);
216*00b67f09SDavid van Moolenbroek 	REQUIRE(source != NULL);
217*00b67f09SDavid van Moolenbroek 
218*00b67f09SDavid van Moolenbroek 	target = (char *) isc_mem_allocate(mctx, source->length + 1);
219*00b67f09SDavid van Moolenbroek 	if (target != NULL) {
220*00b67f09SDavid van Moolenbroek 		memmove(source->base, target, source->length);
221*00b67f09SDavid van Moolenbroek 		target[source->length] = '\0';
222*00b67f09SDavid van Moolenbroek 	}
223*00b67f09SDavid van Moolenbroek 
224*00b67f09SDavid van Moolenbroek 	return (target);
225*00b67f09SDavid van Moolenbroek }
226*00b67f09SDavid van Moolenbroek 
227*00b67f09SDavid van Moolenbroek char *
isc_string_separate(char ** stringp,const char * delim)228*00b67f09SDavid van Moolenbroek isc_string_separate(char **stringp, const char *delim) {
229*00b67f09SDavid van Moolenbroek 	char *string = *stringp;
230*00b67f09SDavid van Moolenbroek 	char *s;
231*00b67f09SDavid van Moolenbroek 	const char *d;
232*00b67f09SDavid van Moolenbroek 	char sc, dc;
233*00b67f09SDavid van Moolenbroek 
234*00b67f09SDavid van Moolenbroek 	if (string == NULL)
235*00b67f09SDavid van Moolenbroek 		return (NULL);
236*00b67f09SDavid van Moolenbroek 
237*00b67f09SDavid van Moolenbroek 	for (s = string; (sc = *s) != '\0'; s++)
238*00b67f09SDavid van Moolenbroek 		for (d = delim; (dc = *d) != '\0'; d++)
239*00b67f09SDavid van Moolenbroek 			if (sc == dc) {
240*00b67f09SDavid van Moolenbroek 				*s++ = '\0';
241*00b67f09SDavid van Moolenbroek 				*stringp = s;
242*00b67f09SDavid van Moolenbroek 				return (string);
243*00b67f09SDavid van Moolenbroek 			}
244*00b67f09SDavid van Moolenbroek 	*stringp = NULL;
245*00b67f09SDavid van Moolenbroek 	return (string);
246*00b67f09SDavid van Moolenbroek }
247*00b67f09SDavid van Moolenbroek 
248*00b67f09SDavid van Moolenbroek size_t
isc_string_strlcpy(char * dst,const char * src,size_t size)249*00b67f09SDavid van Moolenbroek isc_string_strlcpy(char *dst, const char *src, size_t size)
250*00b67f09SDavid van Moolenbroek {
251*00b67f09SDavid van Moolenbroek 	char *d = dst;
252*00b67f09SDavid van Moolenbroek 	const char *s = src;
253*00b67f09SDavid van Moolenbroek 	size_t n = size;
254*00b67f09SDavid van Moolenbroek 
255*00b67f09SDavid van Moolenbroek 	/* Copy as many bytes as will fit */
256*00b67f09SDavid van Moolenbroek 	if (n != 0U && --n != 0U) {
257*00b67f09SDavid van Moolenbroek 		do {
258*00b67f09SDavid van Moolenbroek 			if ((*d++ = *s++) == 0)
259*00b67f09SDavid van Moolenbroek 				break;
260*00b67f09SDavid van Moolenbroek 		} while (--n != 0U);
261*00b67f09SDavid van Moolenbroek 	}
262*00b67f09SDavid van Moolenbroek 
263*00b67f09SDavid van Moolenbroek 	/* Not enough room in dst, add NUL and traverse rest of src */
264*00b67f09SDavid van Moolenbroek 	if (n == 0U) {
265*00b67f09SDavid van Moolenbroek 		if (size != 0U)
266*00b67f09SDavid van Moolenbroek 			*d = '\0';		/* NUL-terminate dst */
267*00b67f09SDavid van Moolenbroek 		while (*s++)
268*00b67f09SDavid van Moolenbroek 			;
269*00b67f09SDavid van Moolenbroek 	}
270*00b67f09SDavid van Moolenbroek 
271*00b67f09SDavid van Moolenbroek 	return(s - src - 1);	/* count does not include NUL */
272*00b67f09SDavid van Moolenbroek }
273*00b67f09SDavid van Moolenbroek 
274*00b67f09SDavid van Moolenbroek size_t
isc_string_strlcat(char * dst,const char * src,size_t size)275*00b67f09SDavid van Moolenbroek isc_string_strlcat(char *dst, const char *src, size_t size)
276*00b67f09SDavid van Moolenbroek {
277*00b67f09SDavid van Moolenbroek 	char *d = dst;
278*00b67f09SDavid van Moolenbroek 	const char *s = src;
279*00b67f09SDavid van Moolenbroek 	size_t n = size;
280*00b67f09SDavid van Moolenbroek 	size_t dlen;
281*00b67f09SDavid van Moolenbroek 
282*00b67f09SDavid van Moolenbroek 	/* Find the end of dst and adjust bytes left but don't go past end */
283*00b67f09SDavid van Moolenbroek 	while (n-- != 0U && *d != '\0')
284*00b67f09SDavid van Moolenbroek 		d++;
285*00b67f09SDavid van Moolenbroek 	dlen = d - dst;
286*00b67f09SDavid van Moolenbroek 	n = size - dlen;
287*00b67f09SDavid van Moolenbroek 
288*00b67f09SDavid van Moolenbroek 	if (n == 0U)
289*00b67f09SDavid van Moolenbroek 		return(dlen + strlen(s));
290*00b67f09SDavid van Moolenbroek 	while (*s != '\0') {
291*00b67f09SDavid van Moolenbroek 		if (n != 1U) {
292*00b67f09SDavid van Moolenbroek 			*d++ = *s;
293*00b67f09SDavid van Moolenbroek 			n--;
294*00b67f09SDavid van Moolenbroek 		}
295*00b67f09SDavid van Moolenbroek 		s++;
296*00b67f09SDavid van Moolenbroek 	}
297*00b67f09SDavid van Moolenbroek 	*d = '\0';
298*00b67f09SDavid van Moolenbroek 
299*00b67f09SDavid van Moolenbroek 	return(dlen + (s - src));	/* count does not include NUL */
300*00b67f09SDavid van Moolenbroek }
301*00b67f09SDavid van Moolenbroek 
302*00b67f09SDavid van Moolenbroek char *
isc_string_strcasestr(const char * str,const char * search)303*00b67f09SDavid van Moolenbroek isc_string_strcasestr(const char *str, const char *search) {
304*00b67f09SDavid van Moolenbroek 	char c, sc, *s;
305*00b67f09SDavid van Moolenbroek 	size_t len;
306*00b67f09SDavid van Moolenbroek 
307*00b67f09SDavid van Moolenbroek 	if ((c = *search++) != 0) {
308*00b67f09SDavid van Moolenbroek 		c = tolower((unsigned char) c);
309*00b67f09SDavid van Moolenbroek 		len = strlen(search);
310*00b67f09SDavid van Moolenbroek 		do {
311*00b67f09SDavid van Moolenbroek 			do {
312*00b67f09SDavid van Moolenbroek 				if ((sc = *str++) == 0)
313*00b67f09SDavid van Moolenbroek 					return (NULL);
314*00b67f09SDavid van Moolenbroek 			} while ((char) tolower((unsigned char) sc) != c);
315*00b67f09SDavid van Moolenbroek 		} while (strncasecmp(str, search, len) != 0);
316*00b67f09SDavid van Moolenbroek 		str--;
317*00b67f09SDavid van Moolenbroek 	}
318*00b67f09SDavid van Moolenbroek 	DE_CONST(str, s);
319*00b67f09SDavid van Moolenbroek 	return (s);
320*00b67f09SDavid van Moolenbroek 
321*00b67f09SDavid van Moolenbroek }
322