2 Copyright (c) 2009 Joseph A. Adams
5 Redistribution and use in source and binary forms, with or without
6 modification, are permitted provided that the following conditions
8 1. Redistributions of source code must retain the above copyright
9 notice, this list of conditions and the following disclaimer.
10 2. Redistributions in binary form must reproduce the above copyright
11 notice, this list of conditions and the following disclaimer in the
12 documentation and/or other materials provided with the distribution.
13 3. The name of the author may not be used to endorse or promote products
14 derived from this software without specific prior written permission.
16 THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 #define ARRAY_USE_TALLOC
37 #ifdef ARRAY_USE_TALLOC
38 #include <ccan/talloc/talloc.h>
41 #ifndef HAVE_ATTRIBUTE_MAY_ALIAS
42 #define HAVE_ATTRIBUTE_MAY_ALIAS 1
45 //Use the array_alias macro to indicate that a pointer has changed but strict aliasing rules are too stupid to know it
46 #if HAVE_ATTRIBUTE_MAY_ALIAS==1
47 #define array_alias(ptr) /* nothing */
48 #define array(type) struct {type *item; size_t size; size_t alloc;} __attribute__((__may_alias__))
50 #define array_alias(ptr) qsort(ptr, 0, 1, array_alias_helper) //hack
51 #define array(type) struct {type *item; size_t size; size_t alloc;}
54 //We call allocator functions directly
55 #ifndef ARRAY_USE_TALLOC
57 #define array_new() {0,0,0}
58 #define array_init(array) do {(array).item=0; (array).size=0; (array).alloc=0;} while(0)
59 #define array_realloc(array, newAlloc) do {(array).item = realloc((array).item, ((array).alloc = (newAlloc))*sizeof(*(array).item));} while(0)
60 #define array_free(array) do {free((array).item);} while(0)
64 //note: the allocations are given an extra byte to prevent free (and thus loss of ctx) on realloc to size 0
66 #define array_new(ctx) {talloc_size(ctx,1), 0,0}
67 #define array_init(array, ctx) do {(array).item=talloc_size(ctx,1); (array).size=0; (array).alloc=0;} while(0)
68 #define array_realloc(array, newAlloc) do {(array).item = talloc_realloc_size(NULL, (array).item, ((array).alloc = (newAlloc))*sizeof(*(array).item) +1);} while(0)
69 #define array_free(array) do {talloc_free((array).item);} while(0)
74 //We call helper functions
75 #define array_resize(array, newSize) do {(array).size = (newSize); if ((array).size > (array).alloc) { array_resize_helper((array_char*)&(array), sizeof(*(array).item)); array_alias(&(array));}} while(0)
76 #define array_resize0(array, newSize) do {array_resize0_helper((array_char*)&(array), sizeof(*(array).item), newSize);} while(0)
77 #define array_prepend_lit(array, stringLiteral) do {array_insert_items_helper((array_char*)&(array), sizeof(*(array).item), 0, stringLiteral, sizeof(stringLiteral)-1, 1); array_alias(&(array)); (array).item[--(array).size] = 0;} while(0)
78 #define array_prepend_string(array, str) do {const char *__str = (str); size_t __len = strlen(__str); array_insert_items_helper((array_char*)&(array), sizeof(*(array).item), 0, __str, __len, 1); array_alias(&(array)); (array).item[--(array).size] = 0;} while(0)
79 #define array_prepend_items(array, items, count) do {array_insert_items_helper((array_char*)&(array), sizeof(*(array).item), 0, items, count, 0); array_alias(&(array));} while(0)
82 //We call other array_* macros
83 #define array_from_c(array, c_array) array_from_items(array, c_array, sizeof(c_array)/sizeof(*(c_array)))
84 #define array_from_lit(array, stringLiteral) do {array_from_items(array, stringLiteral, sizeof(stringLiteral)); (array).size--;} while(0)
85 #define array_from_string(array, str) do {const char *__str = (str); array_from_items(array, __str, strlen(__str)+1); (array).size--;} while(0)
86 #define array_from_items(array, items, count) do {size_t __count = (count); array_resize(array, __count); memcpy((array).item, items, __count*sizeof(*(array).item));} while(0)
87 #define array_append(array, ...) do {array_resize(array, (array).size+1); (array).item[(array).size-1] = (__VA_ARGS__);} while(0)
88 #define array_append_string(array, str) do {const char *__str = (str); array_append_items(array, __str, strlen(__str)+1); (array).size--;} while(0)
89 #define array_append_lit(array, stringLiteral) do {array_append_items(array, stringLiteral, sizeof(stringLiteral)); (array).size--;} while(0)
90 #define array_append_items(array, items, count) do {size_t __count = (count); array_resize(array, (array).size+__count); memcpy((array).item+(array).size-__count, items, __count*sizeof(*(array).item));} while(0)
91 #define array_prepend(array, ...) do {array_resize(array, (array).size+1); memmove((array).item+1, (array).item, ((array).size-1)*sizeof(*(array).item)); *(array).item = (__VA_ARGS__);} while(0)
92 #define array_push(array, ...) array_append(array, __VA_ARGS__)
93 #define array_pop_check(array) ((array).size ? array_pop(array) : NULL)
95 #define array_growalloc(array, newAlloc) do {size_t __newAlloc=(newAlloc); if (__newAlloc > (array).alloc) array_realloc(array, (__newAlloc+63)&~63); } while(0)
96 #if HAVE_STATEMENT_EXPR==1
97 #define array_make_room(array, room) ({size_t newAlloc = (array).size+(room); if ((array).alloc<newAlloc) array_realloc(array, newAlloc); (array).item+(array).size; })
101 //We do just fine by ourselves
102 #define array_pop(array) ((array).item[--(array).size])
106 #define array_appends(array, ...) do {typeof((*(array).item)) __src[] = {__VA_ARGS__}; array_append_items(array, __src, sizeof(__src)/sizeof(*__src));} while(0)
107 #define array_prepends(array, ...) do {typeof((*(array).item)) __src[] = {__VA_ARGS__}; array_prepend_items(array, __src, sizeof(__src)/sizeof(*__src));} while(0)
108 #define array_for(var, array, ...) do {typeof(*(array).item) *var=(void*)(array).item; size_t _r=(array).size, _i=0; for (;_r--;_i++, var++) { __VA_ARGS__ ;} } while(0)
109 #define array_rof(var, array, ...) do {typeof(*(array).item) *var=(void*)(array).item; size_t _i=(array).size, _r=0; var += _i; for (;_i--;_r++) { var--; __VA_ARGS__ ;} } while(0)
113 typedef array(char) array_char;
115 void array_resize_helper(array_char *a, size_t itemSize);
116 void array_resize0_helper(array_char *a, size_t itemSize, size_t newSize);
117 void array_insert_items_helper(array_char *a, size_t itemSize, size_t pos, const void *items, size_t count, size_t tailSize);
118 //Note: there is no array_insert_items yet, but it wouldn't be too hard to add.
120 int array_alias_helper(const void *a, const void *b);
126 array_growalloc(array, newAlloc) sees if the array can currently hold newAlloc items;
127 if not, it increases the alloc to satisfy this requirement, allocating slack
128 space to avoid having to reallocate for every size increment.
130 array_from_string(array, str) copys a string to an array_char.
132 array_push(array, item) pushes an item to the end of the array.
133 array_pop_nocheck(array) pops it back out. Be sure there is at least one item in the array before calling.
134 array_pop(array) does the same as array_pop_nocheck, but returns NULL if there are no more items left in the array.
136 array_make_room(array, room) ensures there's 'room' elements of space after the end of the array, and it returns a pointer to this space.
137 Currently requires HAVE_STATEMENT_EXPR, but I plan to remove this dependency by creating an inline function.
139 The following require HAVE_TYPEOF==1 :
141 array_appends(array, item0, item1...) appends a collection of comma-delimited items to the array.
142 array_prepends(array, item0, item1...) prepends a collection of comma-delimited items to the array.
143 array_for(var, array, commands...) iterates forward through the items in the array using var. var is a pointer to an item.
144 array_rof(var, array, commands...) iterates backward through the items in the array using var. var is a pointer to an item.
149 array_appends(array, 0,1,2,3,4);
150 array_appends(array, -5,-4,-3,-2,-1);
151 array_for(i, array, printf("%d ", *i));
155 array(struct {int n,d;}) fractions;
156 array_appends(fractions, {3,4}, {3,5}, {2,1});
157 array_for(i, fractions, printf("%d/%d\n", i->n, i->d));
158 array_free(fractions);
187 array_append <- array_push
188 array_resize <- array_append
189 array_from_items <- array_from_c, array_from_lit, array_from_string
190 array_append_items <- array_append_string, array_append_lit
191 array_prepend_items <- array_prepends