1a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#ifndef DYNARRAY_H
2a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#define DYNARRAY_H
3a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
4a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#include <stddef.h>
5a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
6a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* simple dynamic array of pointers */
7a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnertypedef struct {
8a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    int count;
9a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    int capacity;
10a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    void** items;
11a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner} dynarray_t;
12a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
13a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#define DYNARRAY_INITIALIZER  { 0, 0, NULL }
14a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
15a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid dynarray_init( dynarray_t *a );
16a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid dynarray_done( dynarray_t *a );
17a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
18a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid dynarray_append( dynarray_t *a, void* item );
19a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
20a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* Used to iterate over a dynarray_t
21a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _array :: pointer to the array
22a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _item_type :: type of objects pointed to by the array
23a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _item      :: name of a local variable defined within the loop
24a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner *               with type '_item_type'
25a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _stmnt     :: C statement that will be executed in each iteration.
26a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner *
27a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * You case use 'break' and 'continue' within _stmnt
28a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner *
29a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * This macro is only intended for simple uses. I.e. do not add or
30a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * remove items from the array during iteration.
31a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner */
32a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#define DYNARRAY_FOREACH_TYPE(_array,_item_type,_item,_stmnt) \
33a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    do { \
34a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner        int _nn_##__LINE__ = 0; \
35a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner        for (;_nn_##__LINE__ < (_array)->count; ++ _nn_##__LINE__) { \
36a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner            _item_type _item = (_item_type)(_array)->items[_nn_##__LINE__]; \
37a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner            _stmnt; \
38a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner        } \
39a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    } while (0)
40a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
41a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#define DYNARRAY_FOREACH(_array,_item,_stmnt) \
42a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    DYNARRAY_FOREACH_TYPE(_array,void *,_item,_stmnt)
43a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
44a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* Simple dynamic string arrays
45a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner *
46a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * NOTE: A strlist_t owns the strings it references.
47a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner */
48a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnertypedef dynarray_t  strlist_t;
49a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
50a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#define  STRLIST_INITIALIZER  DYNARRAY_INITIALIZER
51a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
52a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* Used to iterate over a strlist_t
53a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _list   :: pointer to strlist_t object
54a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _string :: name of local variable name defined within the loop with
55a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner *            type 'char*'
56a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * _stmnt  :: C statement executed in each iteration
57a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner *
58a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * This macro is only intended for simple uses. Do not add or remove items
59a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * to/from the list during iteration.
60a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner */
61a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#define  STRLIST_FOREACH(_list,_string,_stmnt) \
62a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner    DYNARRAY_FOREACH_TYPE(_list,char *,_string,_stmnt)
63a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
64a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid strlist_init( strlist_t *list );
65a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
66a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* note: strlist_done will free all the strings owned by the list */
67a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid strlist_done( strlist_t *list );
68a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
69a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* append a new string made of the first 'slen' characters from 'str'
70a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner * followed by a trailing zero.
71a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner */
72a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid strlist_append_b( strlist_t *list, const void* str, size_t  slen );
73a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
74a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* append the copy of a given input string to a strlist_t */
75a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid strlist_append_dup( strlist_t *list, const char *str);
76a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
77a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner/* sort the strings in a given list (using strcmp) */
78a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turnervoid strlist_sort( strlist_t *list );
79a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner
80a8d1afb3e91b043fcd303a40ca4ac9293bbd2781David 'Digit' Turner#endif /* DYNARRAY_H */