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
|
#ifndef MACRO_H
#define MACRO_H
/*
* NOTE This file uses __VA_OPT__. This is not standard compliant.
*/
/*
* Token paste
*/
#define TP(a, b) a ## b
#define TP3(a, b, c) a ## b ## c
#define TP4(a, b, c, d) a ## b ## c ## d
#define TP5(a, b, c, d, e) a ## b ## c ## d ## e
#define TP6(a, b, c, d, e, f) a ## b ## c ## d ## e ## f
/*
* Get length of __VA_ARGS__
* Borrowed fram:
* https://stackoverflow.com/a/35986932
*/
#define VA_ARGS_NUM_PRIV(P1, P2, P3, P4, P5, P6, Pn, ...) Pn
#define VA_ARGS_NUM(...) VA_ARGS_NUM_PRIV(-1, ## __VA_ARGS__, 5, 4, 3, 2, 1, 0)
/*
* Templetize
* TODO actuall documentatino
* ᐸElementTᐳ
*/
#define TEMPL(name, T) TP4(name, \U00001438 , T, \U00001433 )
#define TEMPL_N(name, T, argcount) TP6(name, \U00001438 , T, _, argcount, \U00001433 )
/* Constructor type name */
#define __INIT_T(T, C) TEMPL_N(init, T, C)
/* Returns full type of constructor */
#define INIT_F(T, ...) \
int __INIT_T(T, VA_ARGS_NUM(__VA_ARGS__)) (T* this __VA_OPT__(,) __VA_ARGS__)
/* Call the constructor of an object */
#define INIT(T, N, ...) \
__INIT_T(T, VA_ARGS_NUM(__VA_ARGS__)) (N __VA_OPT__(,) __VA_ARGS__)
/* Allocate a new object on the HEAP */
#define NEW(T, N, ...) \
T* N = malloc(sizeof(*N)); \
INIT(T, N, __VA_ARGS__);
#define RENEW(T, N, ...) do { \
N = malloc(sizeof(*N)); \
INIT(T, N, __VA_ARGS__); \
} while (0)
/* Allocate a new object on the STACK */
#define SNEW(T, N, ...) \
T N; \
INIT(T, & N, __VA_ARGS__);
/* Destructor for type */
#define FREE(T) TEMPL(free, T)
/* Call destructor for type, and free object */
#define FFREE(T, N) do { FREE(T)(N); free(N); } while (0)
/* Declare destructor */
#define FREE_F(T) int FREE(T) (T* this)
#define DEEP_COPY(T) TEMPL(deep_copy , T)
#define RESOLVE(T) TEMPL(resolve , T)
#define APPEND(T) TEMPL(append , T)
#define SIZE(T) TEMPL(size , T)
#define EMPTY(T) TEMPL(empty , T)
#define PUSH(T) TEMPL(push , T)
#define GET(T) TEMPL(get , T)
#endif /* MACRO_H */
|