summaryrefslogtreecommitdiff
path: root/include/genalloc.h
blob: 351fb98694986fe17ef3777b17638877b46c1114 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
#ifndef GENALLOC_H
#define GENALLOC_H

/*
 * Revision 20210307, Erwin Hoffmann
 * -
 */

#include <sys/types.h>

#include "alloc.h"

/*
  GEN_ALLOC demystified:

  GEN_ALLOC generates a list of self-defined types (structs) in an
  allocated contiguous heap chunk while copying the content of the
  entire field members or appending the existing field.
  GEN_ALLOC types care of currently used and/or allocated bytes of field.

  Macros:
    GEN_ALLOC_ready    (ta,type,field,len,a,i,n,x,base,ta_ready)
    GEN_ALLOC_readyplus(ta,type,field,len,a,i,n,x,base,ta_rplus)
    GEN_ALLOC_append   (ta,type,field,len,a,i,n,x,base,ta_rplus,ta_append)

   0. ta: 'type alloc' - typedef'ed struct name (aka ipalloc et al.)
   1. type: defined struct (used for size information)
   2. field: declared public name of struct
   2. len: used length of string
   4. a: allocated size
   5. i: current allocated size for member x
   6. n: bytes to allocate; in 'ready' mode: +size of one entry;
                            in 'readyplus' mode: +size of +used size of one entry
   7. x: local name (alias to field name)
   8. base: size of single entry
   9. ta_ready/ta_readyplus (operation)
  10. ta_append (operation)
*/

/* file: gen_alloc.h */
#define GEN_ALLOC_typedef(ta, type, field, len, a) \
  typedef struct ta {                              \
    type *field;                                   \
    unsigned int len;                              \
    unsigned int a;                                \
  } ta;

/* file: gen_allocdefs.h   (deprecated) */
// used in: ipalloc, prioq, qmail-remote, qmail-inject, token822
#define GEN_ALLOC_ready(ta, type, field, len, a, i, n, x, base, ta_ready)         \
  int ta_ready(ta *x, unsigned int n)                                             \
  {                                                                               \
    unsigned int i;                                                               \
    if (x->field) {                                                               \
      i = x->a;                                                                   \
      if (n > i) {                                                                \
        x->a = base + n + (n >> 3);                                               \
        if (alloc_re(&x->field, i * sizeof(type), x->a * sizeof(type))) return 1; \
        x->a = i;                                                                 \
        return 0;                                                                 \
      }                                                                           \
      return 1;                                                                   \
    }                                                                             \
    x->len = 0;                                                                   \
    return !!(x->field = (type *)alloc((x->a = n) * sizeof(type)));               \
  }

#define GEN_ALLOC_readyplus(ta, type, field, len, a, i, n, x, base, ta_rplus)     \
  int ta_rplus(ta *x, unsigned int n)                                             \
  {                                                                               \
    unsigned int i;                                                               \
    if (x->field) {                                                               \
      i = x->a;                                                                   \
      n += x->len;                                                                \
      if (n > i) {                                                                \
        x->a = base + n + (n >> 3);                                               \
        if (alloc_re(&x->field, i * sizeof(type), x->a * sizeof(type))) return 1; \
        x->a = i;                                                                 \
        return 0;                                                                 \
      }                                                                           \
      return 1;                                                                   \
    }                                                                             \
    x->len = 0;                                                                   \
    return !!(x->field = (type *)alloc((x->a = n) * sizeof(type)));               \
  }

#define GEN_ALLOC_append(ta, type, field, len, a, i, n, x, base, ta_rplus, ta_append) \
  int ta_append(ta *x, type *i)                                                       \
  {                                                                                   \
    if (!ta_rplus(x, 1)) return 0;                                                    \
    x->field[x->len++] = *i;                                                          \
    return 1;                                                                         \
  }

#endif