xref: /freebsd/contrib/bc/src/vector.c (revision ec0ea6efa1ad229d75c394c1a9b9cac33af2b1d3)
1 /*
2  * *****************************************************************************
3  *
4  * SPDX-License-Identifier: BSD-2-Clause
5  *
6  * Copyright (c) 2018-2021 Gavin D. Howard and contributors.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions are met:
10  *
11  * * Redistributions of source code must retain the above copyright notice, this
12  *   list of conditions and the following disclaimer.
13  *
14  * * Redistributions in binary form must reproduce the above copyright notice,
15  *   this list of conditions and the following disclaimer in the documentation
16  *   and/or other materials provided with the distribution.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28  * POSSIBILITY OF SUCH DAMAGE.
29  *
30  * *****************************************************************************
31  *
32  * Code to manipulate vectors (resizable arrays).
33  *
34  */
35 
36 #include <assert.h>
37 #include <stdlib.h>
38 #include <string.h>
39 #include <stdbool.h>
40 
41 #include <vector.h>
42 #include <lang.h>
43 #include <vm.h>
44 
45 void bc_vec_grow(BcVec *restrict v, size_t n) {
46 
47 	size_t cap, len;
48 	sig_atomic_t lock;
49 
50 	cap = v->cap;
51 	len = v->len + n;
52 
53 	// If this is true, we might overflow.
54 	if (len > SIZE_MAX / 2) cap = len;
55 	else {
56 		// Keep doubling until larger.
57 		while (cap < len) cap += cap;
58 	}
59 
60 	BC_SIG_TRYLOCK(lock);
61 
62 	v->v = bc_vm_realloc(v->v, bc_vm_arraySize(cap, v->size));
63 	v->cap = cap;
64 
65 	BC_SIG_TRYUNLOCK(lock);
66 }
67 
68 void bc_vec_init(BcVec *restrict v, size_t esize, BcDtorType dtor) {
69 
70 	BC_SIG_ASSERT_LOCKED;
71 
72 	assert(v != NULL && esize);
73 
74 	v->v = bc_vm_malloc(bc_vm_arraySize(BC_VEC_START_CAP, esize));
75 
76 	v->size = (BcSize) esize;
77 	v->cap = BC_VEC_START_CAP;
78 	v->len = 0;
79 	v->dtor = (BcSize) dtor;
80 }
81 
82 void bc_vec_expand(BcVec *restrict v, size_t req) {
83 
84 	assert(v != NULL);
85 
86 	// Only expand if necessary.
87 	if (v->cap < req) {
88 
89 		sig_atomic_t lock;
90 
91 		BC_SIG_TRYLOCK(lock);
92 
93 		v->v = bc_vm_realloc(v->v, bc_vm_arraySize(req, v->size));
94 		v->cap = req;
95 
96 		BC_SIG_TRYUNLOCK(lock);
97 	}
98 }
99 
100 void bc_vec_npop(BcVec *restrict v, size_t n) {
101 
102 	sig_atomic_t lock;
103 
104 	assert(v != NULL && n <= v->len);
105 
106 	BC_SIG_TRYLOCK(lock);
107 
108 	if (!v->dtor) v->len -= n;
109 	else {
110 
111 		const BcVecFree d = bc_vec_dtors[v->dtor];
112 		size_t esize = v->size;
113 		size_t len = v->len - n;
114 
115 		// Loop through and manually destruct every element.
116 		while (v->len > len) d(v->v + (esize * --v->len));
117 	}
118 
119 	BC_SIG_TRYUNLOCK(lock);
120 }
121 
122 void bc_vec_npopAt(BcVec *restrict v, size_t n, size_t idx) {
123 
124 	char* ptr, *data;
125 	sig_atomic_t lock;
126 
127 	assert(v != NULL);
128 	assert(idx + n < v->len);
129 
130 	// Grab start and end pointers.
131 	ptr = bc_vec_item(v, idx);
132 	data = bc_vec_item(v, idx + n);
133 
134 	BC_SIG_TRYLOCK(lock);
135 
136 	if (v->dtor) {
137 
138 		size_t i;
139 		const BcVecFree d = bc_vec_dtors[v->dtor];
140 
141 		// Destroy every popped item.
142 		for (i = 0; i < n; ++i) d(bc_vec_item(v, idx + i));
143 	}
144 
145 	v->len -= n;
146 	memmove(ptr, data, (v->len - idx) * v->size);
147 
148 	BC_SIG_TRYUNLOCK(lock);
149 }
150 
151 void bc_vec_npush(BcVec *restrict v, size_t n, const void *data) {
152 
153 	sig_atomic_t lock;
154 	size_t esize;
155 
156 	assert(v != NULL && data != NULL);
157 
158 	BC_SIG_TRYLOCK(lock);
159 
160 	// Grow if necessary.
161 	if (v->len + n > v->cap) bc_vec_grow(v, n);
162 
163 	esize = v->size;
164 
165 	// Copy the elements in.
166 	memcpy(v->v + (esize * v->len), data, esize * n);
167 	v->len += n;
168 
169 	BC_SIG_TRYUNLOCK(lock);
170 }
171 
172 inline void bc_vec_push(BcVec *restrict v, const void *data) {
173 	bc_vec_npush(v, 1, data);
174 }
175 
176 void* bc_vec_pushEmpty(BcVec *restrict v) {
177 
178 	sig_atomic_t lock;
179 	void *ptr;
180 
181 	assert(v != NULL);
182 
183 	BC_SIG_TRYLOCK(lock);
184 
185 	// Grow if necessary.
186 	if (v->len + 1 > v->cap) bc_vec_grow(v, 1);
187 
188 	ptr = v->v + v->size * v->len;
189 	v->len += 1;
190 
191 	BC_SIG_TRYUNLOCK(lock);
192 
193 	return ptr;
194 }
195 
196 inline void bc_vec_pushByte(BcVec *restrict v, uchar data) {
197 	assert(v != NULL && v->size == sizeof(uchar));
198 	bc_vec_npush(v, 1, &data);
199 }
200 
201 void bc_vec_pushIndex(BcVec *restrict v, size_t idx) {
202 
203 	uchar amt, nums[sizeof(size_t) + 1];
204 
205 	assert(v != NULL);
206 	assert(v->size == sizeof(uchar));
207 
208 	// Encode the index.
209 	for (amt = 0; idx; ++amt) {
210 		nums[amt + 1] = (uchar) idx;
211 		idx &= ((size_t) ~(UCHAR_MAX));
212 		idx >>= sizeof(uchar) * CHAR_BIT;
213 	}
214 
215 	nums[0] = amt;
216 
217 	// Push the index onto the vector.
218 	bc_vec_npush(v, amt + 1, nums);
219 }
220 
221 void bc_vec_pushAt(BcVec *restrict v, const void *data, size_t idx) {
222 
223 	assert(v != NULL && data != NULL && idx <= v->len);
224 
225 	BC_SIG_ASSERT_LOCKED;
226 
227 	// Do the easy case.
228 	if (idx == v->len) bc_vec_push(v, data);
229 	else {
230 
231 		char *ptr;
232 		size_t esize;
233 
234 		// Grow if necessary.
235 		if (v->len == v->cap) bc_vec_grow(v, 1);
236 
237 		esize = v->size;
238 
239 		ptr = v->v + esize * idx;
240 
241 		memmove(ptr + esize, ptr, esize * (v->len++ - idx));
242 		memcpy(ptr, data, esize);
243 	}
244 }
245 
246 void bc_vec_string(BcVec *restrict v, size_t len, const char *restrict str) {
247 
248 	sig_atomic_t lock;
249 
250 	assert(v != NULL && v->size == sizeof(char));
251 	assert(!v->dtor);
252 	assert(!v->len || !v->v[v->len - 1]);
253 	assert(v->v != str);
254 
255 	BC_SIG_TRYLOCK(lock);
256 
257 	bc_vec_popAll(v);
258 	bc_vec_expand(v, bc_vm_growSize(len, 1));
259 	memcpy(v->v, str, len);
260 	v->len = len;
261 
262 	bc_vec_pushByte(v, '\0');
263 
264 	BC_SIG_TRYUNLOCK(lock);
265 }
266 
267 void bc_vec_concat(BcVec *restrict v, const char *restrict str) {
268 
269 	sig_atomic_t lock;
270 
271 	assert(v != NULL && v->size == sizeof(char));
272 	assert(!v->dtor);
273 	assert(!v->len || !v->v[v->len - 1]);
274 	assert(v->v != str);
275 
276 	BC_SIG_TRYLOCK(lock);
277 
278 	// If there is already a string, erase its nul byte.
279 	if (v->len) v->len -= 1;
280 
281 	bc_vec_npush(v, strlen(str) + 1, str);
282 
283 	BC_SIG_TRYUNLOCK(lock);
284 }
285 
286 void bc_vec_empty(BcVec *restrict v) {
287 
288 	sig_atomic_t lock;
289 
290 	assert(v != NULL && v->size == sizeof(char));
291 	assert(!v->dtor);
292 
293 	BC_SIG_TRYLOCK(lock);
294 
295 	bc_vec_popAll(v);
296 	bc_vec_pushByte(v, '\0');
297 
298 	BC_SIG_TRYUNLOCK(lock);
299 }
300 
301 #if BC_ENABLE_HISTORY
302 void bc_vec_replaceAt(BcVec *restrict v, size_t idx, const void *data) {
303 
304 	char *ptr;
305 
306 	BC_SIG_ASSERT_LOCKED;
307 
308 	assert(v != NULL);
309 
310 	ptr = bc_vec_item(v, idx);
311 
312 	if (v->dtor) bc_vec_dtors[v->dtor](ptr);
313 
314 	memcpy(ptr, data, v->size);
315 }
316 #endif // BC_ENABLE_HISTORY
317 
318 inline void* bc_vec_item(const BcVec *restrict v, size_t idx) {
319 	assert(v != NULL && v->len && idx < v->len);
320 	return v->v + v->size * idx;
321 }
322 
323 inline void* bc_vec_item_rev(const BcVec *restrict v, size_t idx) {
324 	assert(v != NULL && v->len && idx < v->len);
325 	return v->v + v->size * (v->len - idx - 1);
326 }
327 
328 inline void bc_vec_clear(BcVec *restrict v) {
329 	BC_SIG_ASSERT_LOCKED;
330 	v->v = NULL;
331 	v->len = 0;
332 	v->dtor = BC_DTOR_NONE;
333 }
334 
335 void bc_vec_free(void *vec) {
336 	BcVec *v = (BcVec*) vec;
337 	BC_SIG_ASSERT_LOCKED;
338 	bc_vec_popAll(v);
339 	free(v->v);
340 }
341 
342 #if !BC_ENABLE_LIBRARY
343 
344 /**
345  * Finds a name in a map by binary search. Returns the index where the item
346  * *would* be if it doesn't exist. Callers are responsible for checking that the
347  * item exists at the index.
348  * @param v     The map.
349  * @param name  The name to find.
350  * @return      The index of the item with @a name, or where the item would be
351  *              if it does not exist.
352  */
353 static size_t bc_map_find(const BcVec *restrict v, const char *name) {
354 
355 	size_t low = 0, high = v->len;
356 
357 	while (low < high) {
358 
359 		size_t mid = (low + high) / 2;
360 		const BcId *id = bc_vec_item(v, mid);
361 		int result = strcmp(name, id->name);
362 
363 		if (!result) return mid;
364 		else if (result < 0) high = mid;
365 		else low = mid + 1;
366 	}
367 
368 	return low;
369 }
370 
371 bool bc_map_insert(BcVec *restrict v, const char *name,
372                    size_t idx, size_t *restrict i)
373 {
374 	BcId id;
375 	BcVec *slabs;
376 
377 	BC_SIG_ASSERT_LOCKED;
378 
379 	assert(v != NULL && name != NULL && i != NULL);
380 
381 	*i = bc_map_find(v, name);
382 
383 	assert(*i <= v->len);
384 
385 	if (*i != v->len && !strcmp(name, ((BcId*) bc_vec_item(v, *i))->name))
386 		return false;
387 
388 #if BC_ENABLED
389 	slabs = BC_IS_DC ? &vm.main_slabs : &vm.other_slabs;
390 #else // BC_ENABLED
391 	slabs = &vm.main_slabs;
392 #endif // BC_ENABLED
393 
394 	id.name = bc_slabvec_strdup(slabs, name);
395 	id.idx = idx;
396 
397 	bc_vec_pushAt(v, &id, *i);
398 
399 	return true;
400 }
401 
402 size_t bc_map_index(const BcVec *restrict v, const char *name) {
403 
404 	size_t i;
405 
406 	assert(v != NULL && name != NULL);
407 
408 	i = bc_map_find(v, name);
409 
410 	// If out of range, return invalid.
411 	if (i >= v->len) return BC_VEC_INVALID_IDX;
412 
413 	// Make sure the item exists.
414 	return strcmp(name, ((BcId*) bc_vec_item(v, i))->name) ?
415 	    BC_VEC_INVALID_IDX : i;
416 }
417 
418 #if DC_ENABLED
419 const char* bc_map_name(const BcVec *restrict v, size_t idx) {
420 
421 	size_t i, len = v->len;
422 
423 	for (i = 0; i < len; ++i) {
424 		BcId* id = (BcId*) bc_vec_item(v, i);
425 		if (id->idx == idx) return id->name;
426 	}
427 
428 	BC_UNREACHABLE
429 
430 	return "";
431 }
432 #endif // DC_ENABLED
433 
434 /**
435  * Initializes a single slab.
436  * @param s  The slab to initialize.
437  */
438 static void bc_slab_init(BcSlab *s) {
439 	s->s = bc_vm_malloc(BC_SLAB_SIZE);
440 	s->len = 0;
441 }
442 
443 /**
444  * Adds a string to a slab and returns a pointer to it, or NULL if it could not
445  * be added.
446  * @param s    The slab to add to.
447  * @param str  The string to add.
448  * @param len  The length of the string, including its nul byte.
449  * @return     A pointer to the new string in the slab, or NULL if it could not
450  *             be added.
451  */
452 static char* bc_slab_add(BcSlab *s, const char *str, size_t len) {
453 
454 	char *ptr;
455 
456 	assert(s != NULL);
457 	assert(str != NULL);
458 	assert(len == strlen(str) + 1);
459 
460 	if (s->len + len > BC_SLAB_SIZE) return NULL;
461 
462 	ptr = (char*) (s->s + s->len);
463 
464 	bc_strcpy(ptr, len, str);
465 
466 	s->len += len;
467 
468 	return ptr;
469 }
470 
471 void bc_slab_free(void *slab) {
472 	free(((BcSlab*) slab)->s);
473 }
474 
475 void bc_slabvec_init(BcVec* v) {
476 
477 	BcSlab *slab;
478 
479 	assert(v != NULL);
480 
481 	bc_vec_init(v, sizeof(BcSlab), BC_DTOR_SLAB);
482 
483 	// We always want to have at least one slab.
484 	slab = bc_vec_pushEmpty(v);
485 	bc_slab_init(slab);
486 }
487 
488 char* bc_slabvec_strdup(BcVec *v, const char *str) {
489 
490 	char *s;
491 	size_t len;
492 	BcSlab slab;
493 	BcSlab *slab_ptr;
494 
495 	BC_SIG_ASSERT_LOCKED;
496 
497 	assert(v != NULL && v->len);
498 
499 	assert(str != NULL);
500 
501 	len = strlen(str) + 1;
502 
503 	// If the len is greater than 128, then just allocate it with malloc.
504 	if (BC_UNLIKELY(len >= BC_SLAB_SIZE)) {
505 
506 		// SIZE_MAX is a marker for these standalone allocations.
507 		slab.len = SIZE_MAX;
508 		slab.s = bc_vm_strdup(str);
509 
510 		// Push the standalone slab.
511 		bc_vec_pushAt(v, &slab, v->len - 1);
512 
513 		return slab.s;
514 	}
515 
516 	// Add to a slab.
517 	slab_ptr = bc_vec_top(v);
518 	s = bc_slab_add(slab_ptr, str, len);
519 
520 	// If it couldn't be added, add a slab and try again.
521 	if (BC_UNLIKELY(s == NULL)) {
522 
523 		slab_ptr = bc_vec_pushEmpty(v);
524 		bc_slab_init(slab_ptr);
525 
526 		s = bc_slab_add(slab_ptr, str, len);
527 
528 		assert(s != NULL);
529 	}
530 
531 	return s;
532 }
533 
534 void bc_slabvec_clear(BcVec *v) {
535 
536 	BcSlab *s;
537 	bool again;
538 
539 	// This complicated loop exists because of standalone allocations over 128
540 	// bytes.
541 	do {
542 
543 		// Get the first slab.
544 		s = bc_vec_item(v, 0);
545 
546 		// Either the slab must be valid (not standalone), or there must be
547 		// another slab.
548 		assert(s->len != SIZE_MAX || v->len > 1);
549 
550 		// Do we have to loop again? We do if it's a standalone allocation.
551 		again = (s->len == SIZE_MAX);
552 
553 		// Pop the standalone allocation, not the one after it.
554 		if (again) bc_vec_npopAt(v, 1, 0);
555 
556 	} while(again);
557 
558 	// If we get here, we know that the first slab is a valid slab. We want to
559 	// pop all of the other slabs.
560 	if (v->len > 1) bc_vec_npop(v, v->len - 1);
561 
562 	// Empty the first slab.
563 	s->len = 0;
564 }
565 #endif // !BC_ENABLE_LIBRARY
566 
567 #if BC_DEBUG_CODE
568 
569 void bc_slabvec_print(BcVec *v, const char *func) {
570 
571 	size_t i;
572 	BcSlab *s;
573 
574 	bc_file_printf(&vm.ferr, "%s\n", func);
575 
576 	for (i = 0; i < v->len; ++i) {
577 		s = bc_vec_item(v, i);
578 		bc_file_printf(&vm.ferr, "%zu { s = %zu, len = %zu }\n",
579 		               i, (uintptr_t) s->s, s->len);
580 	}
581 
582 	bc_file_puts(&vm.ferr, bc_flush_none, "\n");
583 	bc_file_flush(&vm.ferr, bc_flush_none);
584 }
585 
586 #endif // BC_DEBUG_CODE
587