libcstl
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Modules
array.h
Go to the documentation of this file.
1/*!
2 * @file
3 */
4
5#ifndef CSTL_ARRAY_H
6#define CSTL_ARRAY_H
7
8#include "cstl/memory.h"
9
10#include <sys/types.h>
11
12/*!
13 * @defgroup array Array
14 * @ingroup highlevel
15 * @brief Dynamically-allocated, fixed size array
16 */
17/*!
18 * @addtogroup array
19 * @{
20 */
21
22/*!
23 * @brief The array object
24 *
25 * The object manages an allocated or externally supplied array
26 * of elements and refers to all or a slice thereof. The memory
27 * is managed such that its lifetime is preserved so long as any
28 * array object still refers to it.
29 */
30typedef struct
31{
32 /*! @privatesection */
34 /* measured in number of elements */
35 size_t off, len;
37
38/*!
39 * @brief Initialize an array object at the time of declaration
40 *
41 * @param[in] NAME The name of the array object being initialized
42 */
43#define CSTL_ARRAY_INITIALIZER(NAME) \
44 { \
45 .ptr = CSTL_SHARED_PTR_INITIALIZER(NAME.ptr), \
46 .off = 0, \
47 .len = 0, \
48 }
49/*!
50 * @brief Declare (and initialize) an array object at compile time
51 *
52 * @param[in] NAME The name of the array object being declared
53 */
54#define DECLARE_CSTL_ARRAY(NAME) \
55 cstl_array_t NAME = \
56 CSTL_ARRAY_INITIALIZER(NAME)
57
58/*!
59 * @brief Initialize a previously declared/allocated array object
60 *
61 * @param[in] a A pointer to an uninitialized array object
62 */
63static inline void cstl_array_init(cstl_array_t * const a)
64{
65 cstl_shared_ptr_init(&a->ptr);
66 a->off = a->len = 0;
67}
68
69/*!
70 * @brief Get the number of elements in the array
71 *
72 * @param[in] a A pointer to an array object
73 *
74 * @return The number of elements in the array
75 */
76static inline size_t cstl_array_size(const cstl_array_t * const a)
77{
78 return a->len;
79}
80
81/*!
82 * @name Danger Zone
83 * @brief Functions requiring extra careful handling
84 *
85 * Callers can use cstl_array_set() to use an externally allocated array,
86 * but the caller must cstl_array_release() that array. Otherwise, the
87 * code will try to return the array to the heap when the last reference
88 * to it goes away.
89 *
90 * @{
91 */
92
93/*!
94 * @brief Manage an externally allocated array
95 *
96 * @param[in,out] a A pointer to an initialized array object
97 * @param[in] buf A pointer to an array containing @p nm elements,
98 * each of @p sz bytes
99 * @param[in] nm The number of elements in the underlying array
100 * @param[in] sz The size of each element in the underlying array
101 */
102void cstl_array_set(cstl_array_t * const a, void * buf, size_t nm, size_t sz);
103
104/*!
105 * @brief Release an externally allocated array
106 *
107 * If the underlying array was not externally allocated, this function
108 * has no effect. Furthermore, there must be no other references to the
109 * underlying array from other cstl_array_t objects for this function to
110 * succeed. The underlying array will not be released while other references
111 * still exist.
112 *
113 * @param[in] a A pointer to an array object
114 * @param[out] buf A pointer into which to return the externally allocated
115 * array. The array is returned on success or set to NULL
116 * on failure. This parameter may be NULL
117 */
118void cstl_array_release(cstl_array_t * const a, void ** buf);
119
120/*!
121 * @}
122 */
123
124/*!
125 * @brief Allocate an array to be managed
126 *
127 * @param[in,out] a A pointer to an initialized array object
128 * @param[in] nm The number of elements in the underlying array
129 * @param[in] sz The size of each element in the underlying array
130 *
131 * A failed allocation will leave the array object in an initialized state
132 */
133void cstl_array_alloc(cstl_array_t * a, size_t nm, size_t sz);
134
135/*!
136 * @brief Drop a reference to memory managed by this object
137 *
138 * If this object is the last reference to the underlying memory, that
139 * memory will be freed.
140 *
141 * @param[in,out] a A pointer to an array object
142 */
143static inline void cstl_array_reset(cstl_array_t * const a)
144{
145 cstl_shared_ptr_reset(&a->ptr);
146 a->off = a->len = 0;
147}
148
149/*!
150 * @brief Return a pointer to the underlying array
151 *
152 * @param[in] a A pointer to the array object
153 *
154 * @return A pointer to the underlying array
155 * @retval NULL The object does not manage an array
156 */
157const void * cstl_array_data_const(const cstl_array_t * a);
158/*! @copydoc cstl_array_data_const() */
159static inline void * cstl_array_data(cstl_array_t * const a)
160{
161 return (void *)cstl_array_data_const(a);
162}
163
164/*!
165 * @brief Return a pointer to an element in the array
166 *
167 * @param[in] a A pointer to an array object
168 * @param[in] i The index of the sought element in the array
169 *
170 * The function will abort() if the index is out of bounds.
171 *
172 * @return A pointer to the sought element
173 */
174const void * cstl_array_at_const(const cstl_array_t * a, size_t i);
175/*! @copydoc cstl_array_at_const() */
176static inline void * cstl_array_at(cstl_array_t * const a, const size_t i)
177{
178 return (void *)cstl_array_at_const(a, i);
179}
180
181/*!
182 * @brief Create an array object referring to a slice of another
183 *
184 * @param[in] a A pointer to an array object
185 * @param[in] beg The index in @p a at which the slice should begin
186 * @param[in] end The index in @p a at which the slice should end
187 * @param[in,out] s A pointer to an array object. @p s and @p a may point
188 * to the same object
189 *
190 * If end < beg or if the slice would exceed the bounds of the array,
191 * the function abort()s. Otherwise the slice refers to the elements
192 * indicated by [beg, end).
193 */
195 cstl_array_t * a, size_t beg, size_t end, cstl_array_t * s);
196
197/*!
198 * @brief Create an array object referring to the entire underlying array
199 *
200 * @param[in] a A pointer to an array object
201 * @param[in,out] s A pointer to an array object. @p s and @p a may
202 * point to the same object
203 *
204 * The resulting array object will refer to the entirety of the underlying
205 * array, as originally allocated.
206 */
208
209/*!
210 * @name Raw Array Functions
211 * @{
212 */
213
214/*!
215 * @brief Reverse the contents of an array
216 *
217 * @param[in,out] arr A pointer to the first element in an array
218 * @param[in] count The number of elements in the array
219 * @param[in] size The size of each element in the array
220 * @param[in] swap A pointer to a function to swap elements in the array
221 * @param tmp A pointer to scratch space to be used by the swap function
222 */
224 void * arr, size_t count, size_t size,
225 cstl_swap_func_t * swap, void * tmp);
226
227/*!
228 * @brief Perform a binary search of the array
229 *
230 * @param[in,out] arr A pointer to the first element in an array
231 * @param[in] count The number of elements in the array
232 * @param[in] size The size of each element in the array
233 * @param[in] ex A pointer to the element to be found
234 * @param[in] cmp A pointer to a function to compare elements
235 * @param[in] priv A pointer to be passed to the comparison function
236 *
237 * The array must be sorted, or the behavior is undefined
238 *
239 * @return The index of the sought element
240 * @retval -1 if the sought value is not found
241 */
243 const void * arr, size_t count, size_t size,
244 const void * ex, cstl_compare_func_t * cmp, void * priv);
245
246/*!
247 * @brief Perform a linear search of the array
248 *
249 * @param[in,out] arr A pointer to the first element in an array
250 * @param[in] count The number of elements in the array
251 * @param[in] size The size of each element in the array
252 * @param[in] ex A pointer to the element to be found
253 * @param[in] cmp A pointer to a function to compare elements
254 * @param[in] priv A pointer to be passed to the comparison function
255 *
256 * @return The index of the sought element
257 * @retval -1 if the sought value is not found
258 */
259ssize_t cstl_raw_array_find(
260 const void * arr, size_t count, size_t size,
261 const void * ex, cstl_compare_func_t * cmp, void * priv);
262
263/*!
264 * @brief Sort the array using the specified algorithm
265 *
266 * @param[in,out] arr A pointer to the first element in an array
267 * @param[in] count The number of elements in the array
268 * @param[in] size The size of each element in the array
269 * @param[in] cmp A pointer to a function to compare elements
270 * @param[in] priv A pointer to be passed to the comparison function
271 * @param[in] swap A pointer to a function to swap elements in the array
272 * @param tmp A pointer to scratch space to be used by the swap function
273 * @param[in] algo The algorithm to use to sort the array
274 */
276 void * arr, size_t count, size_t size,
277 cstl_compare_func_t * cmp, void * priv,
278 cstl_swap_func_t * swap, void * tmp,
280
281/*!
282 * @}
283 */
284
285/*!
286 * @}
287 */
288
289#endif
cstl_sort_algorithm_t
Enumeration indicating the desired sort algorithm.
Definition common.h:24
void cstl_swap_func_t(void *a, void *b, void *t, size_t len)
Type of function called to swap two objects.
Definition common.h:118
int cstl_compare_func_t(const void *obj1, const void *obj2, void *priv)
Function type for comparing (in)equality of two objects.
Definition common.h:49
void cstl_array_alloc(cstl_array_t *a, size_t nm, size_t sz)
Allocate an array to be managed.
Definition array.c:376
ssize_t cstl_raw_array_find(const void *arr, size_t count, size_t size, const void *ex, cstl_compare_func_t *cmp, void *priv)
Perform a linear search of the array.
Definition array.c:54
void cstl_raw_array_sort(void *arr, size_t count, size_t size, cstl_compare_func_t *cmp, void *priv, cstl_swap_func_t *swap, void *tmp, cstl_sort_algorithm_t algo)
Sort the array using the specified algorithm.
Definition array.c:340
ssize_t cstl_raw_array_search(const void *arr, size_t count, size_t size, const void *ex, cstl_compare_func_t *cmp, void *priv)
Perform a binary search of the array.
Definition array.c:30
void cstl_array_set(cstl_array_t *const a, void *buf, size_t nm, size_t sz)
Manage an externally allocated array.
Definition array.c:395
static size_t cstl_array_size(const cstl_array_t *const a)
Get the number of elements in the array.
Definition array.h:76
void cstl_array_release(cstl_array_t *const a, void **buf)
Release an externally allocated array.
Definition array.c:409
static void * cstl_array_data(cstl_array_t *const a)
Return a pointer to the underlying array.
Definition array.h:159
const void * cstl_array_at_const(const cstl_array_t *a, size_t i)
Return a pointer to an element in the array.
Definition array.c:437
const void * cstl_array_data_const(const cstl_array_t *a)
Return a pointer to the underlying array.
Definition array.c:427
static void cstl_array_reset(cstl_array_t *const a)
Drop a reference to memory managed by this object.
Definition array.h:143
void cstl_array_slice(cstl_array_t *a, size_t beg, size_t end, cstl_array_t *s)
Create an array object referring to a slice of another.
Definition array.c:449
void cstl_array_unslice(cstl_array_t *s, cstl_array_t *a)
Create an array object referring to the entire underlying array.
Definition array.c:469
static void * cstl_array_at(cstl_array_t *const a, const size_t i)
Return a pointer to an element in the array.
Definition array.h:176
void cstl_raw_array_reverse(void *arr, size_t count, size_t size, cstl_swap_func_t *swap, void *tmp)
Reverse the contents of an array.
Definition array.c:15
static void cstl_array_init(cstl_array_t *const a)
Initialize a previously declared/allocated array object.
Definition array.h:63
void cstl_shared_ptr_reset(cstl_shared_ptr_t *sp)
Stop managing the underlying memory via this object.
Definition memory.c:128
static void cstl_shared_ptr_init(cstl_shared_ptr_t *const sp)
Initialize a shared pointer object.
Definition memory.h:408
The array object.
Definition array.h:31
The shared pointer object.
Definition memory.h:396