-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathtpool.c
201 lines (180 loc) · 6.11 KB
/
tpool.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
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
#define _GNU_SOURCE
#include <stdio.h>
#include <sys/sysinfo.h>
#include <pthread.h>
#include "tpool.h"
#include "tsc.h"
#include "os.h"
static void *__thread_wrapper(void *arg)
{
struct kitem *i;
int total_enqueued = 0;
struct tpool *t = (struct tpool*)arg;
os_thread_init();
while(1) {
i = NULL;
spin_pdr_lock(&t->lock);
if(t->stats.active_threads < t->nprocs)
i = kqueue_dequeue_item(t->q);
if(i)
__sync_fetch_and_add(&t->stats.active_threads, 1);
else
total_enqueued = t->q->qstats.total_enqueued;
spin_pdr_unlock(&t->lock);
if(i) {
__sync_fetch_and_add(&t->stats.active_threads_sum, t->stats.active_threads);
__sync_fetch_and_add(&t->stats.active_threads_samples, 1);
uint64_t beg = read_tsc();
t->func(t->q, i);
uint64_t end = read_tsc();
__sync_fetch_and_add(&t->stats.active_threads, -1);
__sync_fetch_and_add(&t->stats.processing_time_sum, (end - beg));
__sync_fetch_and_add(&t->stats.items_processed, 1);
}
else {
futex_wait(&t->q->qstats.total_enqueued, total_enqueued);
}
if(t->size > t->new_size) {
__sync_fetch_and_add(&t->size, -1);
return NULL;
}
}
}
static int create_threads(struct tpool *t, int num)
{
int created = 0;
pthread_t thread;
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setstacksize(&attr, t->stacksize);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
for(int i=0; i<num; i++) {
if(pthread_create(&thread, &attr, __thread_wrapper, t) == 0)
created++;
else
break;
}
return created;
}
int tpool_init(struct tpool *t, int size, struct kqueue *q,
void (*func)(struct kqueue *, struct kitem *), size_t stacksize)
{
t->size = 0;
t->new_size = 0;
t->q = q;
t->func = func;
t->nprocs = get_nprocs();
spin_pdr_init(&t->lock);
t->stats.active_threads = 0;
t->stats.blocked_threads = 0;
t->stats.active_threads_sum = 0;
t->stats.blocked_threads_sum = 0;
t->stats.active_threads_samples = 0;
t->stats.blocked_threads_samples = 0;
t->stats.items_processed = 0;
t->stats.processing_time_sum = 0;
t->stacksize = stacksize;
t->size = create_threads(t, size);
t->new_size = t->size;
return t->size;
}
void tpool_resize(struct tpool *t, int size)
{
int new_size = *((volatile int *)(&(t->new_size)));
do {
while(t->size != new_size) {
pthread_yield();
cpu_relax();
new_size = *((volatile int *)(&(t->new_size)));
}
} while(!__sync_bool_compare_and_swap(&t->new_size, new_size, size));
if(t->new_size < 1 || t->new_size == t->size)
return;
if(t->new_size > t->size) {
t->size += create_threads(t, t->new_size - t->size);
}
else {
futex_wake(&t->q->qstats.total_enqueued, INT_MAX);
while(t->size != t->new_size) {
pthread_yield();
cpu_relax();
}
}
}
void tpool_wake(struct tpool *t, int count)
{
if(t->stats.active_threads < t->nprocs)
futex_wake(&t->q->qstats.total_enqueued, count);
}
struct tpool_stats tpool_get_stats(struct tpool *t)
{
struct tpool_stats s = t->stats;
return s;
}
void tpool_inform_blocking(struct tpool *t)
{
__sync_fetch_and_add(&t->stats.active_threads, -1);
__sync_fetch_and_add(&t->stats.blocked_threads, 1);
__sync_fetch_and_add(&t->stats.blocked_threads_sum, t->stats.blocked_threads);
__sync_fetch_and_add(&t->stats.blocked_threads_samples, 1);
tpool_wake(t, 1);
}
void tpool_inform_unblocked(struct tpool *t)
{
__sync_fetch_and_add(&t->stats.blocked_threads, -1);
__sync_fetch_and_add(&t->stats.active_threads, 1);
}
int tpool_get_items_processed(struct tpool_stats *prev,
struct tpool_stats *curr)
{
return curr->items_processed - prev->items_processed;
}
double tpool_get_average_active_threads(struct tpool_stats *prev,
struct tpool_stats *curr)
{
int active_threads_samples = curr->active_threads_samples - prev->active_threads_samples;
double active_threads_sum = curr->active_threads_sum - prev->active_threads_sum;
return active_threads_samples ? (double)active_threads_sum/active_threads_samples : 0;
}
double tpool_get_average_blocked_threads(struct tpool_stats *prev,
struct tpool_stats *curr)
{
int blocked_threads_samples = curr->blocked_threads_samples - prev->blocked_threads_samples;
uint64_t blocked_threads_sum = curr->blocked_threads_sum - prev->blocked_threads_sum;
return blocked_threads_samples ? (double)blocked_threads_sum/blocked_threads_samples : 0;
}
double tpool_get_average_processing_time(struct tpool_stats *prev,
struct tpool_stats *curr)
{
int items_processed = tpool_get_items_processed(prev, curr);
uint64_t processing_time_sum = curr->processing_time_sum - prev->processing_time_sum;
return items_processed ? (double)processing_time_sum/items_processed : 0;
}
void tpool_print_items_processed(char *prefix,
struct tpool_stats *prev,
struct tpool_stats *curr)
{
int items_processed = tpool_get_items_processed(prev, curr);
printf("%sTotal items processed: %d\n", prefix, items_processed);
}
void tpool_print_average_active_threads(char *prefix,
struct tpool_stats *prev,
struct tpool_stats *curr)
{
double average = tpool_get_average_active_threads(prev, curr);
printf("%sAverage active threads: %lf\n", prefix, average);
}
void tpool_print_average_blocked_threads(char *prefix,
struct tpool_stats *prev,
struct tpool_stats *curr)
{
double average = tpool_get_average_blocked_threads(prev, curr);
printf("%sAverage blocked threads: %lf\n", prefix, average);
}
void tpool_print_average_processing_time(char *prefix,
struct tpool_stats *prev,
struct tpool_stats *curr)
{
double average = tpool_get_average_processing_time(prev, curr);
printf("%sAverage item processing time: %lf\n", prefix, average);
}