Workshop o mikrokontrolérech na SKSP 2024.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

282 lines
5.8 KiB

2 months ago
/*
* UCW Library -- Thread Pools and Work Queues
*
* (c) 2006 Martin Mares <mj@ucw.cz>
*
* This software may be freely distributed and used according to the terms
* of the GNU Lesser General Public License.
*/
#include <ucw/lib.h>
#include <ucw/threads.h>
#include <ucw/workqueue.h>
#include <ucw/heap.h>
static void *
worker_thread_init(void *arg)
{
struct worker_thread *t = arg;
struct worker_pool *pool = t->pool;
if (pool->init_thread)
pool->init_thread(t);
sem_post(pool->init_cleanup_sem);
for (;;)
{
struct work *w = raw_queue_get(&pool->requests);
w->go(t, w);
raw_queue_put(&w->reply_to->finished, w);
}
return NULL;
}
static void
worker_thread_signal_finish(struct worker_thread *t, struct work *w UNUSED)
{
if (t->pool->cleanup_thread)
t->pool->cleanup_thread(t);
sem_post(t->pool->init_cleanup_sem);
pthread_exit(NULL);
}
void
worker_pool_init(struct worker_pool *p)
{
clist_init(&p->worker_threads);
raw_queue_init(&p->requests);
p->init_cleanup_sem = sem_alloc();
pthread_attr_t attr;
if (pthread_attr_init(&attr) < 0 ||
pthread_attr_setstacksize(&attr, p->stack_size ? : ucwlib_thread_stack_size) < 0)
ASSERT(0);
for (uint i=0; i < p->num_threads; i++)
{
struct worker_thread *t = (p->new_thread ? p->new_thread() : xmalloc(sizeof(*t)));
t->pool = p;
t->id = i;
int err = pthread_create(&t->thread, &attr, worker_thread_init, t);
if (err)
die("Unable to create thread: %m");
clist_add_tail(&p->worker_threads, &t->n);
sem_wait(p->init_cleanup_sem);
}
pthread_attr_destroy(&attr);
}
void
worker_pool_cleanup(struct worker_pool *p)
{
for (uint i=0; i < p->num_threads; i++)
{
struct work w = {
.go = worker_thread_signal_finish
};
raw_queue_put(&p->requests, &w);
sem_wait(p->init_cleanup_sem);
}
struct worker_thread *tmp;
CLIST_FOR_EACH_DELSAFE(struct worker_thread *, t, p->worker_threads, tmp)
{
int err = pthread_join(t->thread, NULL);
ASSERT(!err);
if (p->free_thread)
p->free_thread(t);
else
xfree(t);
}
raw_queue_cleanup(&p->requests);
sem_free(p->init_cleanup_sem);
}
void
raw_queue_init(struct raw_queue *q)
{
pthread_mutex_init(&q->queue_mutex, NULL);
clist_init(&q->pri0_queue);
q->queue_sem = sem_alloc();
q->pri_heap = NULL;
q->heap_cnt = q->heap_max = 0;
}
void
raw_queue_cleanup(struct raw_queue *q)
{
ASSERT(clist_empty(&q->pri0_queue));
ASSERT(!q->heap_cnt);
xfree(q->pri_heap);
sem_free(q->queue_sem);
pthread_mutex_destroy(&q->queue_mutex);
}
#define PRI_LESS(x,y) ((x)->priority > (y)->priority)
void
raw_queue_put(struct raw_queue *q, struct work *w)
{
pthread_mutex_lock(&q->queue_mutex);
if (!w->priority)
clist_add_tail(&q->pri0_queue, &w->n);
else
{
if (unlikely(q->heap_cnt >= q->heap_max))
{
struct work **old_heap = q->pri_heap;
q->heap_max = (q->heap_max ? 2*q->heap_max : 16);
q->pri_heap = xrealloc(old_heap, (q->heap_max + 1) * sizeof(struct work *));
}
struct work **heap = q->pri_heap;
HEAP_INSERT(struct work *, heap, q->heap_cnt, PRI_LESS, HEAP_SWAP, w);
}
pthread_mutex_unlock(&q->queue_mutex);
sem_post(q->queue_sem);
}
static inline struct work *
raw_queue_do_get(struct raw_queue *q)
{
pthread_mutex_lock(&q->queue_mutex);
struct work *w;
if (!q->heap_cnt)
{
w = clist_head(&q->pri0_queue);
ASSERT(w);
clist_remove(&w->n);
}
else
{
struct work **heap = q->pri_heap;
w = heap[1];
HEAP_DELETE_MIN(struct work *, heap, q->heap_cnt, PRI_LESS, HEAP_SWAP);
}
pthread_mutex_unlock(&q->queue_mutex);
return w;
}
struct work *
raw_queue_get(struct raw_queue *q)
{
sem_wait(q->queue_sem);
return raw_queue_do_get(q);
}
struct work *
raw_queue_try_get(struct raw_queue *q)
{
if (!sem_trywait(q->queue_sem))
return raw_queue_do_get(q);
else
return NULL;
}
void
work_queue_init(struct worker_pool *p, struct work_queue *q)
{
q->pool = p;
q->nr_running = 0;
raw_queue_init(&q->finished);
}
void
work_queue_cleanup(struct work_queue *q)
{
ASSERT(!q->nr_running);
raw_queue_cleanup(&q->finished);
}
void
work_submit(struct work_queue *q, struct work *w)
{
ASSERT(w->go);
w->reply_to = q;
raw_queue_put(&q->pool->requests, w);
q->nr_running++;
}
static struct work *
work_do_wait(struct work_queue *q, int try)
{
if (!q->nr_running)
return NULL;
struct work *w = (try ? raw_queue_try_get : raw_queue_get)(&q->finished);
if (!w)
return NULL;
q->nr_running--;
return w;
}
struct work *
work_wait(struct work_queue *q)
{
return work_do_wait(q, 0);
}
struct work *
work_try_wait(struct work_queue *q)
{
return work_do_wait(q, 1);
}
#ifdef TEST
#include <unistd.h>
static void wt_init(struct worker_thread *t)
{
msg(L_INFO, "INIT %d", t->id);
}
static void wt_cleanup(struct worker_thread *t)
{
msg(L_INFO, "CLEANUP %d", t->id);
}
struct w {
struct work w;
uint id;
};
static void go(struct worker_thread *t, struct work *w)
{
msg(L_INFO, "GO %d: request %d (pri %d)", t->id, ((struct w *)w)->id, w->priority);
usleep(1);
}
int main(void)
{
struct worker_pool pool = {
.num_threads = 10,
.stack_size = 65536,
.init_thread = wt_init,
.cleanup_thread = wt_cleanup
};
worker_pool_init(&pool);
struct work_queue q;
work_queue_init(&pool, &q);
for (uint i=0; i<500; i++)
{
struct w *w = xmalloc_zero(sizeof(*w));
w->w.go = go;
w->w.priority = (i < 250 ? i : 0);
w->id = i;
work_submit(&q, &w->w);
msg(L_INFO, "Submitted request %d (pri %d)", w->id, w->w.priority);
}
struct w *w;
while (w = (struct w *) work_wait(&q))
msg(L_INFO, "Finished request %d", w->id);
work_queue_cleanup(&q);
worker_pool_cleanup(&pool);
return 0;
}
#endif