fixup isDefaultValueAvailable() and isOptional() for builtins
[hiphop-php.git] / hphp / neo / ulist.c
blob4604f9ce10961989e9bddb0d0c1eede098834aaf
1 /*
2 * Copyright 2001-2004 Brandon Long
3 * All Rights Reserved.
5 * ClearSilver Templating System
7 * This code is made available under the terms of the ClearSilver License.
8 * http://www.clearsilver.net/license.hdf
12 #include "cs_config.h"
14 #include <stdlib.h>
15 #include <string.h>
16 #include <errno.h>
18 #include "neo_misc.h"
19 #include "neo_err.h"
20 #include "ulist.h"
22 #define ULIST_DEFAULT_SIZE 10
24 static NEOERR *check_resize (ULIST *ul, int size)
26 if (size > ul->max)
28 void **new_items;
29 int new_size = 0;
31 new_size = ul->max*2;
32 if (size > new_size)
34 new_size = size + ul->max;
37 new_items = (void **) realloc ((void *)(ul->items), new_size * sizeof(void *));
38 if (new_items == NULL)
40 return nerr_raise(NERR_NOMEM,
41 "Unable to resize ULIST to %d: Out of memory", new_size);
43 ul->items = new_items;
44 ul->max = new_size;
47 return STATUS_OK;
51 NEOERR *uListInit(ULIST **ul, int size, int flags)
53 ULIST *r_ul;
55 *ul = NULL;
56 if (size == 0)
58 size = ULIST_DEFAULT_SIZE;
61 r_ul = (ULIST *) calloc (1, sizeof (ULIST));
62 if (r_ul == NULL)
64 return nerr_raise(NERR_NOMEM, "Unable to create ULIST: Out of memory");
66 r_ul->items = (void **) calloc (size, sizeof(void *));
67 if (r_ul->items == NULL)
69 free (r_ul);
70 return nerr_raise(NERR_NOMEM, "Unable to create ULIST: Out of memory");
73 r_ul->num = 0;
74 r_ul->max = size;
75 r_ul->flags = flags;
76 *ul = r_ul;
78 return STATUS_OK;
81 NEOERR *uListvInit(ULIST **ul, ...)
83 NEOERR *err;
84 va_list ap;
85 void *it;
87 err = uListInit (ul, 0, 0);
88 if (err) return nerr_pass (err);
90 va_start (ap, ul);
92 it = va_arg (ap, void *);
94 while (it)
96 err = uListAppend (*ul, it);
97 if (err)
99 uListDestroy(ul, 0);
100 return nerr_pass (err);
102 it = va_arg (ap, void *);
104 return STATUS_OK;
107 NEOERR *uListAppend (ULIST *ul, void *data)
109 NEOERR *r;
111 r = check_resize (ul, ul->num + 1);
112 if (r != STATUS_OK)
113 return r;
115 ul->items[ul->num] = data;
116 ul->num++;
118 return STATUS_OK;
121 NEOERR *uListPop (ULIST *ul, void **data)
123 if (ul->num == 0)
124 return nerr_raise(NERR_OUTOFRANGE, "uListPop: empty list");
126 *data = ul->items[ul->num - 1];
127 ul->num--;
129 return STATUS_OK;
132 NEOERR *uListInsert (ULIST *ul, int x, void *data)
134 void **start;
135 NEOERR *r;
137 if (x < 0)
138 x = ul->num + x;
140 if (x >= ul->num)
141 return nerr_raise(NERR_OUTOFRANGE, "uListInsert: past end (%d > %d)",
142 x, ul->num);
144 r = check_resize (ul, ul->num + 1);
145 if (r != STATUS_OK)
146 return r;
148 start = &(ul->items[x]);
149 memmove (start + 1, start, (ul->num - x) * sizeof(void *));
150 ul->items[x] = data;
151 ++ul->num;
153 return STATUS_OK;
156 NEOERR *uListDelete (ULIST *ul, int x, void **data)
158 void **start;
160 if (x < 0)
161 x = ul->num + x;
163 if (x >= ul->num)
164 return nerr_raise(NERR_OUTOFRANGE, "uListDelete: past end (%d > %d)",
165 x, ul->num);
167 if (data != NULL)
168 *data = ul->items[x];
170 start = &(ul->items[x]);
171 memmove (start, start+1, (ul->num - x - 1) * sizeof(void *));
172 --ul->num;
174 return STATUS_OK;
177 NEOERR *uListGet (ULIST *ul, int x, void **data)
179 if (x < 0)
180 x = ul->num + x;
182 if (x >= ul->num)
183 return nerr_raise(NERR_OUTOFRANGE, "uListGet: past end (%d > %d)",
184 x, ul->num);
186 if (x < 0)
187 return nerr_raise(NERR_OUTOFRANGE, "uListGet: past beginning (%d < 0)", x);
189 *data = ul->items[x];
191 return STATUS_OK;
194 NEOERR *uListSet (ULIST *ul, int x, void *data)
196 if (x >= ul->num)
197 return nerr_raise(NERR_OUTOFRANGE, "uListSet: past end (%d > %d)",
198 x, ul->num);
200 ul->items[x] = data;
202 return STATUS_OK;
205 NEOERR *uListReverse (ULIST *ul)
207 int i;
209 for (i = 0; i < ul->num/2; ++i) {
210 void *tmp = ul->items[i];
211 ul->items[i] = ul->items[ul->num-1-i];
212 ul->items[ul->num-1-i] = tmp;
215 return STATUS_OK;
218 NEOERR *uListSort (ULIST *ul, int (*compareFunc)(const void *, const void*)) {
219 qsort(ul->items, ul->num, sizeof(void *), compareFunc);
220 return STATUS_OK;
223 void *uListSearch (ULIST *ul, const void *key, int
224 (*compareFunc)(const void *, const void*)) {
225 return bsearch(key, ul->items, ul->num, sizeof(void *), compareFunc);
228 void *uListIn (ULIST *ul, const void *key, int (*compareFunc)(const void *, const void*)) {
229 int i;
231 for (i = 0; i < ul->num; ++i) {
232 if (!compareFunc(key, &ul->items[i])) {
233 return &ul->items[i];
236 return NULL;
239 int uListIndex (ULIST *ul, const void *key, int (*compareFunc)(const void *, const void*)) {
240 void **p = uListIn(ul, key, compareFunc);
241 return p ? (p - ul->items) : -1;
246 NEOERR *uListDestroy (ULIST **ul, int flags)
248 if (flags & ULIST_FREE)
250 return uListDestroyFunc(ul, free);
252 else
254 return uListDestroyFunc(ul, NULL);
258 NEOERR *uListDestroyFunc (ULIST **ul, void (*destroyFunc)(void *))
260 ULIST *r_ul;
262 r_ul = *ul;
264 if (r_ul == NULL)
265 return STATUS_OK;
267 if (destroyFunc != NULL)
269 int x;
270 for (x = 0; x < r_ul->num; x++)
272 (*destroyFunc)(r_ul->items[x]);
275 free (r_ul->items);
276 free (r_ul);
277 *ul = NULL;
279 return STATUS_OK;
282 int uListLength (ULIST *ul)
284 if (ul == NULL) return 0;
285 return ul->num;