-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathreaper.c
158 lines (125 loc) · 3.55 KB
/
reaper.c
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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
// SPDX-License-Identifier: LGPL-2.1-or-later
/*
* Copyright (C) 2020-2021 Paul Cercueil <paul@crapouillou.net>
*/
#include "blockcache.h"
#include "debug.h"
#include "lightrec-private.h"
#include "memmanager.h"
#include "slist.h"
#include "reaper.h"
#include <errno.h>
#include <pthread.h>
#include <stdatomic.h>
#include <stdbool.h>
struct reaper_elm {
reap_func_t func;
void *data;
struct slist_elm slist;
};
struct reaper {
struct lightrec_state *state;
pthread_mutex_t mutex;
pthread_cond_t cond;
struct slist_elm reap_list;
bool running;
atomic_uint sem;
};
struct reaper *lightrec_reaper_init(struct lightrec_state *state)
{
struct reaper *reaper;
int ret;
reaper = lightrec_malloc(state, MEM_FOR_LIGHTREC, sizeof(*reaper));
if (!reaper) {
pr_err("Cannot create reaper: Out of memory\n");
return NULL;
}
reaper->state = state;
reaper->running = false;
reaper->sem = 0;
slist_init(&reaper->reap_list);
ret = pthread_mutex_init(&reaper->mutex, NULL);
if (ret) {
pr_err("Cannot init mutex variable: %d\n", ret);
goto err_free_reaper;
}
ret = pthread_cond_init(&reaper->cond, NULL);
if (ret) {
pr_err("Cannot init cond variable: %d\n", ret);
goto err_destroy_mutex;
}
return reaper;
err_destroy_mutex:
pthread_mutex_destroy(&reaper->mutex);
err_free_reaper:
lightrec_free(reaper->state, MEM_FOR_LIGHTREC, sizeof(*reaper), reaper);
return NULL;
}
void lightrec_reaper_destroy(struct reaper *reaper)
{
lightrec_reaper_reap(reaper);
pthread_cond_destroy(&reaper->cond);
pthread_mutex_destroy(&reaper->mutex);
lightrec_free(reaper->state, MEM_FOR_LIGHTREC, sizeof(*reaper), reaper);
}
int lightrec_reaper_add(struct reaper *reaper, reap_func_t f, void *data)
{
struct reaper_elm *reaper_elm;
struct slist_elm *elm;
int ret = 0;
pthread_mutex_lock(&reaper->mutex);
for (elm = reaper->reap_list.next; elm; elm = elm->next) {
reaper_elm = container_of(elm, struct reaper_elm, slist);
if (reaper_elm->data == data)
goto out_unlock;
}
reaper_elm = lightrec_malloc(reaper->state, MEM_FOR_LIGHTREC,
sizeof(*reaper_elm));
if (!reaper_elm) {
pr_err("Cannot add reaper entry: Out of memory\n");
ret = -ENOMEM;
goto out_unlock;
}
reaper_elm->func = f;
reaper_elm->data = data;
slist_append(&reaper->reap_list, &reaper_elm->slist);
out_unlock:
pthread_mutex_unlock(&reaper->mutex);
return ret;
}
static bool lightrec_reaper_can_reap(struct reaper *reaper)
{
return !atomic_load_explicit(&reaper->sem, memory_order_relaxed);
}
void lightrec_reaper_reap(struct reaper *reaper)
{
struct reaper_elm *reaper_elm;
struct slist_elm *elm;
pthread_mutex_lock(&reaper->mutex);
while (lightrec_reaper_can_reap(reaper) &&
!!(elm = slist_first(&reaper->reap_list))) {
slist_remove(&reaper->reap_list, elm);
reaper->running = true;
pthread_mutex_unlock(&reaper->mutex);
reaper_elm = container_of(elm, struct reaper_elm, slist);
(*reaper_elm->func)(reaper->state, reaper_elm->data);
lightrec_free(reaper->state, MEM_FOR_LIGHTREC,
sizeof(*reaper_elm), reaper_elm);
pthread_mutex_lock(&reaper->mutex);
reaper->running = false;
pthread_cond_broadcast(&reaper->cond);
}
pthread_mutex_unlock(&reaper->mutex);
}
void lightrec_reaper_pause(struct reaper *reaper)
{
atomic_fetch_add_explicit(&reaper->sem, 1, memory_order_relaxed);
pthread_mutex_lock(&reaper->mutex);
while (reaper->running)
pthread_cond_wait(&reaper->cond, &reaper->mutex);
pthread_mutex_unlock(&reaper->mutex);
}
void lightrec_reaper_continue(struct reaper *reaper)
{
atomic_fetch_sub_explicit(&reaper->sem, 1, memory_order_relaxed);
}