30#ifndef MUTT_MUTT_ARRAY_H
31#define MUTT_MUTT_ARRAY_H
40#define ARRAY_HEADROOM 25
47#define ARRAY_HEAD(name, type) \
58#define ARRAY_HEAD_INITIALIZER \
65#define ARRAY_INIT(head) \
66 memset((head), 0, sizeof(*(head)))
74#define ARRAY_EMPTY(head) \
87#define ARRAY_SIZE(head) \
95#define ARRAY_CAPACITY(head) \
109#define ARRAY_GET(head, idx) \
110 ((idx >= 0) && ((head)->size > (idx)) ? &(head)->entries[(idx)] : NULL)
123#define ARRAY_SET(head, idx, elem) \
124 (((head)->capacity > (idx) \
126 : ARRAY_RESERVE((head), (idx) + 1)), \
127 ARRAY_SET_NORESERVE((head), (idx), (elem)))
135#define ARRAY_FIRST(head) \
144#define ARRAY_LAST(head) \
145 (ARRAY_EMPTY((head)) \
147 : ARRAY_GET((head), ARRAY_SIZE((head)) - 1))
156#define ARRAY_ADD(head, elem) \
157 (((head)->capacity > (head)->size \
159 : ARRAY_RESERVE((head), (head)->size + 1)), \
160 ARRAY_ADD_NORESERVE((head), (elem)))
172#define ARRAY_SHRINK(head, num) \
173 ((head)->size -= MIN((num), (head)->size))
180#define ARRAY_ELEM_SIZE(head) \
181 (sizeof(*(head)->entries))
189#define ARRAY_RESERVE(head, num) \
190 (((head)->capacity > (num)) ? \
192 ((mutt_mem_reallocarray( \
193 &(head)->entries, (num) + ARRAY_HEADROOM, ARRAY_ELEM_SIZE(head))), \
194 (memset((head)->entries + (head)->capacity, 0, \
195 ((num) + ARRAY_HEADROOM - (head)->capacity) * \
196 ARRAY_ELEM_SIZE(head))), \
197 ((head)->capacity = (num) + ARRAY_HEADROOM)))
204#define ARRAY_FREE(head) \
205 (FREE(&(head)->entries), (head)->size = (head)->capacity = 0)
214#define ARRAY_FOREACH(elem, head) \
215 ARRAY_FOREACH_FROM_TO(elem, (head), 0, (head)->size)
226#define ARRAY_FOREACH_FROM(elem, head, from) \
227 ARRAY_FOREACH_FROM_TO(elem, (head), (from), (head)->size)
238#define ARRAY_FOREACH_TO(elem, head, to) \
239 ARRAY_FOREACH_FROM_TO(elem, (head), 0, (to))
252#define ARRAY_FOREACH_FROM_TO(elem, head, from, to) \
253 for (int ARRAY_FOREACH_IDX_##elem = (from); \
254 (ARRAY_FOREACH_IDX_##elem < (to)) && \
255 (elem = ARRAY_GET((head), ARRAY_FOREACH_IDX_##elem)); \
256 ARRAY_FOREACH_IDX_##elem++)
265#define ARRAY_FOREACH_REVERSE(elem, head) \
266 ARRAY_FOREACH_REVERSE_FROM_TO(elem, (head), (head)->size, 0)
277#define ARRAY_FOREACH_REVERSE_FROM(elem, head, from) \
278 ARRAY_FOREACH_REVERSE_FROM_TO(elem, (head), (from), 0)
289#define ARRAY_FOREACH_REVERSE_TO(elem, head, to) \
290 ARRAY_FOREACH_REVERSE_FROM_TO(elem, (head), (head)->size, (to))
303#define ARRAY_FOREACH_REVERSE_FROM_TO(elem, head, from, to) \
304 for (int ARRAY_FOREACH_IDX_##elem = (from) - 1; \
305 (ARRAY_FOREACH_IDX_##elem >= (to)) && \
306 (elem = ARRAY_GET((head), ARRAY_FOREACH_IDX_##elem)); \
307 ARRAY_FOREACH_IDX_##elem--)
315#define ARRAY_IDX(head, elem) \
316 (elem - (head)->entries)
323#define ARRAY_REMOVE(head, elem) \
324 (memmove((elem), (elem) + 1, \
325 ARRAY_ELEM_SIZE((head)) * \
326 MAX(0, (ARRAY_SIZE((head)) - ARRAY_IDX((head), (elem)) - 1))), \
327 ARRAY_SHRINK((head), 1))
335#define ARRAY_SORT(head, fn, sdata) \
336 ((head)->entries && \
337 (mutt_qsort_r((head)->entries, ARRAY_SIZE(head), ARRAY_ELEM_SIZE(head), (fn), (sdata)), true))
342#define ARRAY_SET_NORESERVE(head, idx, elem) \
343 ((head)->capacity > (idx) \
344 ? (((head)->size = MAX((head)->size, ((idx) + 1))), \
345 ((head)->entries[(idx)] = (elem)), \
350#define __coverity_escape__(x) 0
352#define ARRAY_ADD_NORESERVE(head, elem) \
353 ((head)->capacity > (head)->size \
354 ? (((head)->entries[(head)->size++] = (elem)), \
355 ((void)__coverity_escape__(elem), true)) \
356 : ((void)__coverity_escape__(elem), false))
Memory management wrappers.