aboutsummaryrefslogtreecommitdiff
path: root/src/mempool.h
blob: bca1e31ca931d3cf1fd1ce69041a6a57d85b0a5c (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
#ifndef MEMPOOL_H
#define MEMPOOL_H

/* To generate use MEMPOOL_GENERATE(id, type, cap), where functions
 *   _mempool_allocate() and _mempool_free(type *obj) will be prefixed with <id>
 *   and can be used to allocate objects of type <type> with <cap> objects in
 *   each block
 */

#define for_each_block(id, b, head)                     \
    for(struct id##_block *b = (head), *next = NULL;    \
        b && (next = b->next, 1); b = next)
#define get_last_block(b) \
    while((b)->next != NULL) (b) = (b)->next

#define MEMPOOL_GENERATE(id, type, cap)                                  \
                                                                         \
    struct id##_block {                                                  \
        union id##_chunk {                                               \
            type obj;                                                    \
            union id##_chunk *next;                                      \
        } chunks[cap];                                                   \
                                                                         \
        struct id##_block *next;                                         \
    };                                                                   \
                                                                         \
    static union  id##_chunk *id##_ap = NULL;                            \
    static struct id##_block id##_block;                                 \
    static size_t id##_allocations = 0;                                  \
                                                                         \
    static inline void *_##id##_mempool_init_block(struct id##_block *b) \
    {                                                                    \
        b->next = NULL;                                                  \
        for(size_t i = 0; i < cap; i++)                                  \
            b->chunks[i].next = &b->chunks[i+1];                         \
        b->chunks[cap-1].next = NULL;                                    \
                                                                         \
        return b->chunks;                                                \
    }                                                                    \
                                                                         \
    static inline void *id##_mempool_allocate(void)                      \
    {                                                                    \
        if(!id##_ap) {                                                   \
            struct id##_block *new_block = &(id##_block);                \
            if((id##_allocations) > 0) {                                 \
                new_block = malloc(sizeof(*new_block));                  \
                struct id##_block *last_block = &(id##_block);           \
                get_last_block(last_block);                              \
                last_block->next = new_block;                            \
            }                                                            \
            id##_ap = _##id##_mempool_init_block(new_block);             \
        }                                                                \
                                                                         \
        id##_allocations++;                                              \
                                                                         \
        void *ptr = &(id##_ap)->obj;                                     \
        id##_ap = (id##_ap)->next;                                       \
        return ptr;                                                      \
    }                                                                    \
                                                                         \
    static inline void id##_mempool_free(void *ptr)                      \
    {                                                                    \
        ((union id##_chunk *)ptr)->next = id##_ap;                       \
        id##_ap = (union id##_chunk *)ptr;                               \
                                                                         \
        if(--id##_allocations == 0) {                                    \
            for_each_block(id, b, (id##_block).next)                     \
                free(b);                                                 \
            id##_ap = id##_block.chunks;                                 \
        }                                                                \
    }                                                                    \

#endif