aboutsummaryrefslogtreecommitdiff
path: root/sys/dev/acpi/uacpi/shareable.c
blob: b42660a7ccee2e7d0b8d5e0bd4e8c40b16e21ab1 (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
#include <uacpi/internal/shareable.h>
#include <uacpi/internal/stdlib.h>
#include <uacpi/platform/atomic.h>

#ifndef UACPI_BAREBONES_MODE

#define BUGGED_REFCOUNT 0xFFFFFFFF

void uacpi_shareable_init(uacpi_handle handle)
{
    struct uacpi_shareable *shareable = handle;
    shareable->reference_count = 1;
}

uacpi_bool uacpi_bugged_shareable(uacpi_handle handle)
{
    struct uacpi_shareable *shareable = handle;

    if (uacpi_unlikely(shareable->reference_count == 0))
        uacpi_make_shareable_bugged(shareable);

    return uacpi_atomic_load32(&shareable->reference_count) == BUGGED_REFCOUNT;
}

void uacpi_make_shareable_bugged(uacpi_handle handle)
{
    struct uacpi_shareable *shareable = handle;
    uacpi_atomic_store32(&shareable->reference_count, BUGGED_REFCOUNT);
}

uacpi_u32 uacpi_shareable_ref(uacpi_handle handle)
{
    struct uacpi_shareable *shareable = handle;

    if (uacpi_unlikely(uacpi_bugged_shareable(shareable)))
        return BUGGED_REFCOUNT;

    return uacpi_atomic_inc32(&shareable->reference_count) - 1;
}

uacpi_u32 uacpi_shareable_unref(uacpi_handle handle)
{
    struct uacpi_shareable *shareable = handle;

    if (uacpi_unlikely(uacpi_bugged_shareable(shareable)))
        return BUGGED_REFCOUNT;

    return uacpi_atomic_dec32(&shareable->reference_count) + 1;
}

void uacpi_shareable_unref_and_delete_if_last(
    uacpi_handle handle, void (*do_free)(uacpi_handle)
)
{
    if (handle == UACPI_NULL)
        return;

    if (uacpi_unlikely(uacpi_bugged_shareable(handle)))
        return;

    if (uacpi_shareable_unref(handle) == 1)
        do_free(handle);
}

uacpi_u32 uacpi_shareable_refcount(uacpi_handle handle)
{
    struct uacpi_shareable *shareable = handle;
    return uacpi_atomic_load32(&shareable->reference_count);
}

#endif // !UACPI_BAREBONES_MODE