xref: /freebsd/contrib/bmake/str.h (revision 357378bbdedf24ce2b90e9bd831af4a9db3ec70a)
1 /*	$NetBSD: str.h,v 1.20 2024/07/07 07:50:57 rillig Exp $	*/
2 
3 /*
4  Copyright (c) 2021 Roland Illig <rillig@NetBSD.org>
5  All rights reserved.
6 
7  Redistribution and use in source and binary forms, with or without
8  modification, are permitted provided that the following conditions
9  are met:
10 
11  1. Redistributions of source code must retain the above copyright
12     notice, this list of conditions and the following disclaimer.
13  2. Redistributions in binary form must reproduce the above copyright
14     notice, this list of conditions and the following disclaimer in the
15     documentation and/or other materials provided with the distribution.
16 
17  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18  "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19  TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20  PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS
21  BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27  POSSIBILITY OF SUCH DAMAGE.
28  */
29 
30 
31 /*
32  * Memory-efficient string handling.
33  */
34 
35 
36 /* A read-only string that may need to be freed after use. */
37 typedef struct FStr {
38 	const char *str;
39 	void *freeIt;
40 } FStr;
41 
42 /* A read-only range of a character array, NOT null-terminated. */
43 typedef struct Substring {
44 	const char *start;
45 	const char *end;
46 } Substring;
47 
48 /*
49  * Builds a string, only allocating memory if the string is different from the
50  * expected string.
51  */
52 typedef struct LazyBuf {
53 	char *data;
54 	size_t len;
55 	size_t cap;
56 	const char *expected;
57 } LazyBuf;
58 
59 /* The result of splitting a string into words. */
60 typedef struct Words {
61 	char **words;
62 	size_t len;
63 	void *freeIt;
64 } Words;
65 
66 /* The result of splitting a string into words. */
67 typedef struct SubstringWords {
68 	Substring *words;
69 	size_t len;
70 	void *freeIt;
71 } SubstringWords;
72 
73 typedef struct StrMatchResult {
74 	const char *error;
75 	bool matched;
76 } StrMatchResult;
77 
78 
79 /* Return a string that is the sole owner of str. */
80 MAKE_INLINE FStr
81 FStr_InitOwn(char *str)
82 {
83 	FStr fstr;
84 	fstr.str = str;
85 	fstr.freeIt = str;
86 	return fstr;
87 }
88 
89 /* Return a string that refers to the shared str. */
90 MAKE_INLINE FStr
91 FStr_InitRefer(const char *str)
92 {
93 	FStr fstr;
94 	fstr.str = str;
95 	fstr.freeIt = NULL;
96 	return fstr;
97 }
98 
99 MAKE_INLINE void
100 FStr_Done(FStr *fstr)
101 {
102 	free(fstr->freeIt);
103 #ifdef CLEANUP
104 	fstr->str = NULL;
105 	fstr->freeIt = NULL;
106 #endif
107 }
108 
109 
110 MAKE_STATIC Substring
111 Substring_Init(const char *start, const char *end)
112 {
113 	Substring sub;
114 
115 	sub.start = start;
116 	sub.end = end;
117 	return sub;
118 }
119 
120 MAKE_INLINE Substring
121 Substring_InitStr(const char *str)
122 {
123 	return Substring_Init(str, str + strlen(str));
124 }
125 
126 MAKE_STATIC size_t
127 Substring_Length(Substring sub)
128 {
129 	return (size_t)(sub.end - sub.start);
130 }
131 
132 MAKE_STATIC bool
133 Substring_IsEmpty(Substring sub)
134 {
135 	return sub.start == sub.end;
136 }
137 
138 MAKE_INLINE bool
139 Substring_Equals(Substring sub, const char *str)
140 {
141 	size_t len = strlen(str);
142 	return Substring_Length(sub) == len &&
143 	       memcmp(sub.start, str, len) == 0;
144 }
145 
146 MAKE_INLINE bool
147 Substring_Eq(Substring sub, Substring str)
148 {
149 	size_t len = Substring_Length(sub);
150 	return len == Substring_Length(str) &&
151 	       memcmp(sub.start, str.start, len) == 0;
152 }
153 
154 MAKE_STATIC bool
155 Substring_HasPrefix(Substring sub, Substring prefix)
156 {
157 	return Substring_Length(sub) >= Substring_Length(prefix) &&
158 	       memcmp(sub.start, prefix.start, Substring_Length(prefix)) == 0;
159 }
160 
161 MAKE_STATIC bool
162 Substring_HasSuffix(Substring sub, Substring suffix)
163 {
164 	size_t suffixLen = Substring_Length(suffix);
165 	return Substring_Length(sub) >= suffixLen &&
166 	       memcmp(sub.end - suffixLen, suffix.start, suffixLen) == 0;
167 }
168 
169 /* Returns an independent, null-terminated copy of the substring. */
170 MAKE_STATIC FStr
171 Substring_Str(Substring sub)
172 {
173 	if (Substring_IsEmpty(sub))
174 		return FStr_InitRefer("");
175 	return FStr_InitOwn(bmake_strsedup(sub.start, sub.end));
176 }
177 
178 MAKE_STATIC const char *
179 Substring_SkipFirst(Substring sub, char ch)
180 {
181 	const char *p;
182 
183 	for (p = sub.start; p != sub.end; p++)
184 		if (*p == ch)
185 			return p + 1;
186 	return sub.start;
187 }
188 
189 MAKE_STATIC const char *
190 Substring_FindLast(Substring sub, char ch)
191 {
192 	const char *p;
193 
194 	for (p = sub.end; p != sub.start; p--)
195 		if (p[-1] == ch)
196 			return p - 1;
197 	return NULL;
198 }
199 
200 MAKE_STATIC Substring
201 Substring_Dirname(Substring pathname)
202 {
203 	const char *p;
204 
205 	for (p = pathname.end; p != pathname.start; p--)
206 		if (p[-1] == '/')
207 			return Substring_Init(pathname.start, p - 1);
208 	return Substring_InitStr(".");
209 }
210 
211 MAKE_STATIC Substring
212 Substring_Basename(Substring pathname)
213 {
214 	const char *p;
215 
216 	for (p = pathname.end; p != pathname.start; p--)
217 		if (p[-1] == '/')
218 			return Substring_Init(p, pathname.end);
219 	return pathname;
220 }
221 
222 
223 MAKE_STATIC void
224 LazyBuf_Init(LazyBuf *buf, const char *expected)
225 {
226 	buf->data = NULL;
227 	buf->len = 0;
228 	buf->cap = 0;
229 	buf->expected = expected;
230 }
231 
232 MAKE_INLINE void
233 LazyBuf_Done(LazyBuf *buf)
234 {
235 	free(buf->data);
236 }
237 
238 MAKE_STATIC void
239 LazyBuf_Add(LazyBuf *buf, char ch)
240 {
241 
242 	if (buf->data != NULL) {
243 		if (buf->len == buf->cap) {
244 			buf->cap *= 2;
245 			buf->data = bmake_realloc(buf->data, buf->cap);
246 		}
247 		buf->data[buf->len++] = ch;
248 
249 	} else if (ch == buf->expected[buf->len]) {
250 		buf->len++;
251 		return;
252 
253 	} else {
254 		buf->cap = buf->len + 16;
255 		buf->data = bmake_malloc(buf->cap);
256 		memcpy(buf->data, buf->expected, buf->len);
257 		buf->data[buf->len++] = ch;
258 	}
259 }
260 
261 MAKE_STATIC void
262 LazyBuf_AddStr(LazyBuf *buf, const char *str)
263 {
264 	const char *p;
265 
266 	for (p = str; *p != '\0'; p++)
267 		LazyBuf_Add(buf, *p);
268 }
269 
270 MAKE_INLINE void
271 LazyBuf_AddSubstring(LazyBuf *buf, Substring sub)
272 {
273 	const char *p;
274 
275 	for (p = sub.start; p != sub.end; p++)
276 		LazyBuf_Add(buf, *p);
277 }
278 
279 MAKE_STATIC Substring
280 LazyBuf_Get(const LazyBuf *buf)
281 {
282 	const char *start = buf->data != NULL ? buf->data : buf->expected;
283 	return Substring_Init(start, start + buf->len);
284 }
285 
286 /*
287  * Returns the content of the buffer as a newly allocated string.
288  *
289  * See LazyBuf_Get to avoid unnecessary memory allocations.
290  */
291 MAKE_STATIC FStr
292 LazyBuf_DoneGet(LazyBuf *buf)
293 {
294 	if (buf->data != NULL) {
295 		LazyBuf_Add(buf, '\0');
296 		return FStr_InitOwn(buf->data);
297 	}
298 	return Substring_Str(LazyBuf_Get(buf));
299 }
300 
301 
302 Words Str_Words(const char *, bool);
303 
304 MAKE_INLINE void
305 Words_Free(Words w)
306 {
307 	free(w.words);
308 	free(w.freeIt);
309 }
310 
311 
312 SubstringWords Substring_Words(const char *, bool);
313 
314 MAKE_INLINE void
315 SubstringWords_Init(SubstringWords *w)
316 {
317 	w->words = NULL;
318 	w->len = 0;
319 	w->freeIt = NULL;
320 }
321 
322 MAKE_INLINE void
323 SubstringWords_Free(SubstringWords w)
324 {
325 	free(w.words);
326 	free(w.freeIt);
327 }
328 
329 
330 char *str_concat2(const char *, const char *);
331 char *str_concat3(const char *, const char *, const char *);
332 
333 StrMatchResult Str_Match(const char *, const char *);
334 
335 void Str_Intern_Init(void);
336 #ifdef CLEANUP
337 void Str_Intern_End(void);
338 #endif
339 const char *Str_Intern(const char *);
340