8354 sync regcomp(3C) with upstream (fix make catalog)
[unleashed/tickless.git] / usr / src / cmd / svr4pkg / libinst / listmgr.c
blobe73f4144587bc9dec6ff46d29581476437d93f23
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
23 * Copyright 1996 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
28 #include <stdio.h>
29 #include <string.h>
30 #include <stdlib.h>
31 #include <libintl.h>
32 #include "pkglib.h"
35 * This is the module responsible for allocating and maintaining lists that
36 * require allocation of memory. For certain lists, large chunks are
37 * allocated once to contain a large number of entries in each chunk (bl_*
38 * for block list). The other approach involves the augmentation of linked
39 * lists, each entry of which is alloc'd individually.
41 #define ERR_CS_ALLOC "ERROR: Cannot allocate control structure for %s array."
42 #define ERR_MEM_ALLOC "ERROR: Cannot allocate memory for %s array."
44 #define MAX_ARRAYS 50
46 #define ARRAY_END(x) (bl_cs_array[x]->cur_segment->avail_ptr)
47 #define REC_SIZE(x) (bl_cs_array[x]->struct_size)
48 #define EOSEG(x) (bl_cs_array[x]->cur_segment->eoseg_ptr)
49 #define GET_AVAIL(x) (ARRAY_END(x) + REC_SIZE(x))
51 struct alloc_seg {
52 char *seg_ptr; /* ptr to the allocated block */
53 char *avail_ptr; /* ptr to the next available list element */
54 char *eoseg_ptr; /* last byte in the segment */
55 int full; /* segment has no available space */
56 struct alloc_seg *next; /* next record */
59 struct blk_list_cs {
60 int struct_size; /* size of a single list element */
61 int count_per_block; /* number of list elements per block */
62 int block_size; /* just to save time - alloc size */
63 int data_handle; /* list_handle for pointer array */
64 struct alloc_seg *alloc_segs; /* memory pool */
66 struct alloc_seg *cur_segment; /* the current allocated segment */
67 int total_elem; /* total elements stored */
68 int contiguous; /* use realloc to grow */
69 char *desc; /* description of the list */
72 static struct blk_list_cs *bl_cs_array[MAX_ARRAYS];
73 static int next_array_elem;
75 /* Support functions */
76 static int
77 invalid_handle(int list_handle)
79 if (list_handle < 0 || list_handle >= next_array_elem)
80 return (1);
82 return (0);
85 static int
86 invalid_record(int list_handle, int recno)
88 if (invalid_handle(list_handle))
89 return (1);
91 if (recno < 0 || recno > bl_cs_array[list_handle]->total_elem)
92 return (1);
94 return (0);
97 static void
98 free_list(int list_handle)
100 struct blk_list_cs *bl_ptr;
101 struct alloc_seg *segstr_ptr, *nextstr_ptr;
103 /* Make sure this wasn't free'd earlier */
104 if (bl_cs_array[list_handle] == NULL)
105 return;
107 bl_ptr = bl_cs_array[list_handle];
109 /* First free the alloc_seg list. */
110 segstr_ptr = bl_ptr->alloc_segs;
112 if (segstr_ptr) {
113 do {
114 nextstr_ptr = segstr_ptr->next;
116 /* Free the memory block. */
117 free((void *)segstr_ptr->seg_ptr);
119 /* Free the control structure. */
120 free((void *)segstr_ptr);
121 segstr_ptr = nextstr_ptr;
122 } while (segstr_ptr);
125 /* Free the block control structure. */
126 free((void *)bl_ptr->desc);
127 free((void *)bl_ptr);
129 bl_cs_array[list_handle] = NULL;
132 /* Allocate another alloc_seg structure. */
133 static int
134 alloc_next_seg(struct blk_list_cs *bl_ptr)
136 struct alloc_seg *new_alloc_cs;
138 if (bl_ptr->contiguous) {
139 int offset_to_avail, seg_size, new_size;
140 struct alloc_seg *alloc_segment;
142 if (bl_ptr->alloc_segs) {
143 alloc_segment = bl_ptr->alloc_segs;
145 offset_to_avail = (alloc_segment->avail_ptr -
146 alloc_segment->seg_ptr);
147 seg_size = (alloc_segment->eoseg_ptr -
148 alloc_segment->seg_ptr);
149 new_size = (seg_size + bl_ptr->block_size);
150 } else {
151 if ((bl_ptr->alloc_segs =
152 (struct alloc_seg *)calloc(1,
153 sizeof (struct alloc_seg))) == NULL) {
154 logerr(gettext(ERR_CS_ALLOC), (bl_ptr->desc ?
155 bl_ptr->desc : "an unknown"));
156 return (0);
159 alloc_segment = bl_ptr->alloc_segs;
161 offset_to_avail = 0;
162 seg_size = 0;
163 new_size = bl_ptr->block_size;
166 bl_ptr->cur_segment = alloc_segment;
168 if ((alloc_segment->seg_ptr =
169 (char *)realloc((void *)alloc_segment->seg_ptr,
170 (unsigned)new_size)) == NULL) {
171 logerr(gettext(ERR_MEM_ALLOC), (bl_ptr->desc ?
172 bl_ptr->desc : "an unknown"));
173 return (0);
176 alloc_segment->next = NULL;
178 /* reset the status */
179 alloc_segment->full = 0;
181 /* readjust the original pointers */
182 alloc_segment->avail_ptr = alloc_segment->seg_ptr +
183 offset_to_avail;
184 alloc_segment->eoseg_ptr = alloc_segment->seg_ptr + new_size;
186 (void) memset(alloc_segment->avail_ptr, '\000',
187 bl_ptr->block_size);
188 } else {
189 /* Allocate the control structure and link it into the list. */
190 if ((new_alloc_cs = (struct alloc_seg *)malloc(
191 sizeof (struct alloc_seg))) == NULL) {
192 logerr(gettext(ERR_CS_ALLOC), (bl_ptr->desc ?
193 bl_ptr->desc : "an unknown"));
194 return (0);
197 if (bl_ptr->alloc_segs == NULL) {
199 * If this is the first allocation, then initialize
200 * the head pointer and set cur_segment to this first
201 * block of memory.
203 bl_ptr->alloc_segs = new_alloc_cs;
204 } else {
206 * Otherwise, point the current cur_segment to the
207 * next one and then point to the new one.
209 bl_ptr->cur_segment->next = new_alloc_cs;
212 new_alloc_cs->next = NULL;
213 bl_ptr->cur_segment = new_alloc_cs;
215 new_alloc_cs->full = 0;
217 /* Now allocate the block of memory that this controls. */
218 if ((new_alloc_cs->seg_ptr = calloc(bl_ptr->count_per_block,
219 bl_ptr->struct_size)) == NULL) {
220 logerr(gettext(ERR_MEM_ALLOC), (bl_ptr->desc ?
221 bl_ptr->desc : "an unknown"));
222 return (0);
225 new_alloc_cs->avail_ptr = new_alloc_cs->seg_ptr;
226 new_alloc_cs->eoseg_ptr = (new_alloc_cs->seg_ptr +
227 bl_ptr->block_size);
230 return (1);
234 * These first functions (beginning with bl_*) manage simple block lists. The
235 * pointers returned, may get lost if they aren't assigned to an array or
236 * something. While individual records can be obtained by record number, the
237 * process isn't very efficient. Look to the array management section
238 * (ar_*)for an easily administrable list.
242 * Create a block list. Allocate memory for a block list structure and
243 * initialize that structure. This doesn't actually allocate memory for the
244 * list yet, just the controlling data structure. Returns -1 on failure and a
245 * valid block list handle otherwise.
247 * NOTE: At the time of writing, it was not seen as important to recover block
248 * pointers made available with a bl_free() (two of these at most in
249 * pkginstall). If this became important later, we could trade efficiency for
250 * speed by ignoring next_array_elem and actually scanning through the array
251 * for a NULL pointer and then return that.
254 bl_create(int count_per_block, int struct_size, char *desc)
256 struct blk_list_cs *bl_ptr;
257 int retval;
259 if ((bl_cs_array[next_array_elem] =
260 (struct blk_list_cs *)calloc(1, sizeof (struct blk_list_cs))) ==
261 NULL) {
262 logerr(gettext(ERR_CS_ALLOC), (desc ? desc : "an unknown"));
263 return (-1);
266 bl_ptr = bl_cs_array[next_array_elem];
267 retval = next_array_elem++;
269 bl_ptr->data_handle = -1;
270 bl_ptr->struct_size = struct_size;
271 bl_ptr->count_per_block = count_per_block;
272 bl_ptr->block_size = (count_per_block * struct_size);
273 bl_ptr->desc = strdup((desc ? desc : "unknown"));
275 return (retval);
279 * Get the next available entry in the list. This will allocate memory as
280 * required based on the initialization values in bl_create(). Returns a
281 * pointer to the allocated memory segment or NULL if operation was not
282 * possible.
284 char *
285 bl_next_avail(int list_handle)
287 struct blk_list_cs *bl_ptr;
288 char *retval;
290 if (invalid_handle(list_handle))
291 return (NULL);
293 bl_ptr = bl_cs_array[list_handle];
296 * Allocate more memory if none is allocated yet or our last access
297 * filled the allotted segment.
299 if (bl_ptr->cur_segment == NULL || bl_ptr->cur_segment->full)
300 if (!alloc_next_seg(bl_ptr))
301 return (NULL);
303 /* Get the correct pointer. */
304 retval = bl_ptr->cur_segment->avail_ptr;
306 /* Advance it and mark if full. */
307 bl_ptr->cur_segment->avail_ptr += bl_ptr->struct_size;
308 bl_ptr->total_elem++;
310 if (bl_ptr->cur_segment->avail_ptr >= bl_ptr->cur_segment->eoseg_ptr)
311 bl_ptr->cur_segment->full = 1;
313 return (retval);
316 char *
317 bl_get_record(int list_handle, int recno)
319 struct blk_list_cs *bl_ptr;
320 struct alloc_seg *cur_as_ptr;
321 int cur_rec = 0;
323 if (invalid_record(list_handle, recno))
324 return (NULL);
326 bl_ptr = bl_cs_array[list_handle];
328 cur_as_ptr = bl_ptr->alloc_segs;
330 while (recno > (cur_rec + bl_ptr->count_per_block)) {
331 cur_as_ptr = cur_as_ptr->next;
333 if (cur_as_ptr == NULL)
334 return (NULL);
336 cur_rec += bl_ptr->count_per_block;
340 * Now cur_as_ptr points to the allocated segment bearing the
341 * intended record and all we do now is move down that by the
342 * remaining record lengths.
345 return ((char *)cur_as_ptr + ((recno - cur_rec) * bl_ptr->struct_size));
348 void
349 bl_free(int list_handle)
351 int cur_handle;
353 if (list_handle == -1) {
354 for (cur_handle = 0; cur_handle < next_array_elem;
355 cur_handle++) {
356 free_list(cur_handle);
358 } else {
359 if (invalid_handle(list_handle))
360 return;
362 free_list(list_handle);
367 * These are the array management functions. They insert into (and can return
368 * a pointer to) a contiguous list of pointers to stuff. This keeps
369 * everything together in a very handy package and is very similar in
370 * appearance to the arrays created by the old AT&T code. The method for
371 * presenting the interface is entirely different, however.
375 * This constructs, maintains and returns pointers into a growable array of
376 * pointers to structures of the form
377 * struct something *array[n]
378 * The last element in the array is always NULL.
381 ar_create(int count_per_block, int struct_size, char *desc)
383 int data_handle, retval;
384 char ar_desc[60];
385 struct blk_list_cs *array_ptr;
387 if ((data_handle = bl_create(count_per_block, struct_size, desc)) == -1)
388 return (-1);
390 sprintf(ar_desc, "%s pointer", desc);
391 if ((retval = bl_create(count_per_block, sizeof (char *),
392 ar_desc)) == -1)
393 return (-1);
395 array_ptr = bl_cs_array[retval];
397 array_ptr->contiguous = 1;
398 array_ptr->data_handle = data_handle;
400 return (retval);
403 /* Return a pointer to the first element in the array. */
404 char **
405 ar_get_head(int list_handle)
407 if (invalid_handle(list_handle) ||
408 bl_cs_array[list_handle]->alloc_segs == NULL)
409 return (NULL);
411 return ((char **)bl_cs_array[list_handle]->alloc_segs->seg_ptr);
415 * Free up the entry in the array indicated by index, but hold onto it for
416 * future use.
419 ar_delete(int list_handle, int index)
421 char **array;
422 char *deleted_rec;
423 int i;
424 struct blk_list_cs *list_ptr, *data_ptr;
426 if ((array = ar_get_head(list_handle)) == NULL)
427 return (0);
429 if (invalid_record(list_handle, index))
430 return (0);
432 /* Get the pointer to the array control structure. */
433 list_ptr = bl_cs_array[list_handle];
435 if (!(list_ptr->contiguous))
436 return (0); /* This isn't an array. */
438 data_ptr = bl_cs_array[list_ptr->data_handle];
441 * Since this looks just like an array. Record the pointer being
442 * deleted for insertion into the avail list at the end and move all
443 * elements below it up one.
445 deleted_rec = array[index];
447 for (i = index; array[i] != NULL; i++)
448 array[i] = array[i+1];
451 * Now insert the deleted entry into the avails list after the NULL
452 * and adjust the avail_ptr to point to the NULL again.
454 array[i] = deleted_rec;
455 list_ptr->alloc_segs->avail_ptr -= list_ptr->struct_size;
457 /* Adjust other entries in the control structure. */
458 list_ptr->alloc_segs->full = 0;
459 list_ptr->total_elem -= 1;
461 /* Clear the deleted data area. */
462 (void) memset(deleted_rec, '\000', data_ptr->struct_size);
464 return (1);
468 * Return a new pointer to a structure pointer. Find an available element in
469 * the array and point it at an available element in the data pool
470 * constructed of block lists. Allocate new memory as necessary.
472 char **
473 ar_next_avail(int list_handle)
475 struct blk_list_cs *array_ptr;
476 char *data_area, **pointer_area;
478 if (invalid_handle(list_handle) ||
479 !(bl_cs_array[list_handle]->contiguous) ||
480 invalid_handle(bl_cs_array[list_handle]->data_handle))
481 return (NULL);
483 array_ptr = bl_cs_array[list_handle];
486 * First see if an avail has already been allocated (it will be right
487 * after the NULL termination of the array if it exists). Return
488 * that, if found.
490 if ((bl_cs_array[list_handle]->cur_segment != NULL) &&
491 (ARRAY_END(list_handle) + REC_SIZE(list_handle) <
492 EOSEG(list_handle)) &&
493 (*(pointer_area = (char **) GET_AVAIL(list_handle)) != NULL)) {
494 /* We can reclaim a previous deletion. */
495 data_area = *pointer_area;
497 *(char **)(ARRAY_END(list_handle)) = data_area; /* reactivate */
498 *pointer_area-- = NULL; /* new end */
500 array_ptr->cur_segment->avail_ptr += array_ptr->struct_size;
501 array_ptr->total_elem++;
502 } else {
504 * Get the data area first. This is the record we're pointing
505 * to from the array.
507 data_area = bl_next_avail(array_ptr->data_handle);
509 /* Now get the next pointer from the pointer array. */
510 pointer_area = (char **) bl_next_avail(list_handle);
512 *pointer_area = data_area;
515 * The array must be NULL terminated. So, if the block list
516 * structure is full, we have to grow it without resetting
517 * the avail pointer. NOTE: This will only work for a
518 * contiguous list!
520 if (bl_cs_array[list_handle]->alloc_segs->full) {
521 char **old_list_pointer, **new_list_pointer;
524 * First grab the old numbers in case realloc() moves
525 * everything.
527 old_list_pointer = ar_get_head(list_handle);
530 * Now allocate additional contiguous memory, moving
531 * the original block if necessary.
533 if (!alloc_next_seg(array_ptr))
534 return (NULL);
537 * Now determine if everything moved and readjust the
538 * pointer_area if required.
540 new_list_pointer = ar_get_head(list_handle);
542 if (old_list_pointer != new_list_pointer) {
543 pointer_area += (new_list_pointer -
544 old_list_pointer);
549 return (pointer_area);
553 * Relinquish the array back to the memory pool. Note that there is no method
554 * provided to free *all* arrays.
556 void
557 ar_free(int list_handle)
559 if (invalid_handle(list_handle))
560 return;
562 bl_free(bl_cs_array[list_handle]->data_handle);
563 bl_free(list_handle);