| /* 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); |