summaryrefslogtreecommitdiff
path: root/src/basic/refcnt.h
blob: 40f9a84a22cd22b909df0c5b670365785afa265c (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
/* SPDX-License-Identifier: LGPL-2.1+ */
#pragma once

/* A type-safe atomic refcounter.
 *
 * DO NOT USE THIS UNLESS YOU ACTUALLY CARE ABOUT THREAD SAFETY! */

typedef struct {
        volatile unsigned _value;
} RefCount;

#define REFCNT_GET(r) ((r)._value)
#define REFCNT_INC(r) (__sync_add_and_fetch(&(r)._value, 1))
#define REFCNT_DEC(r) (__sync_sub_and_fetch(&(r)._value, 1))

#define REFCNT_INIT ((RefCount) { ._value = 1 })

#define _DEFINE_ATOMIC_REF_FUNC(type, name, scope)                    \
        scope type *name##_ref(type *p) {                               \
                if (!p)                                                 \
                        return NULL;                                    \
                                                                        \
                assert_se(REFCNT_INC(p->n_ref) >= 2);                   \
                return p;                                               \
        }

#define _DEFINE_ATOMIC_UNREF_FUNC(type, name, free_func, scope)       \
        scope type *name##_unref(type *p) {                             \
                if (!p)                                                 \
                        return NULL;                                    \
                                                                        \
                if (REFCNT_DEC(p->n_ref) > 0)                           \
                        return NULL;                                    \
                                                                        \
                return free_func(p);                                    \
        }

#define DEFINE_ATOMIC_REF_FUNC(type, name)    \
        _DEFINE_ATOMIC_REF_FUNC(type, name,)
#define DEFINE_PUBLIC_ATOMIC_REF_FUNC(type, name)     \
        _DEFINE_ATOMIC_REF_FUNC(type, name, _public_)

#define DEFINE_ATOMIC_UNREF_FUNC(type, name, free_func)       \
        _DEFINE_ATOMIC_UNREF_FUNC(type, name, free_func,)
#define DEFINE_PUBLIC_ATOMIC_UNREF_FUNC(type, name, free_func)        \
        _DEFINE_ATOMIC_UNREF_FUNC(type, name, free_func, _public_)

#define DEFINE_ATOMIC_REF_UNREF_FUNC(type, name, free_func)   \
        DEFINE_ATOMIC_REF_FUNC(type, name);                   \
        DEFINE_ATOMIC_UNREF_FUNC(type, name, free_func);

#define DEFINE_PUBLIC_ATOMIC_REF_UNREF_FUNC(type, name, free_func)   \
        DEFINE_PUBLIC_ATOMIC_REF_FUNC(type, name);                   \
        DEFINE_PUBLIC_ATOMIC_UNREF_FUNC(type, name, free_func);