| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340 |
- /**
- * @file BThreadWork.c
- * @author Ambroz Bizjak <ambrop7@gmail.com>
- *
- * @section LICENSE
- *
- * This file is part of BadVPN.
- *
- * BadVPN is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2
- * as published by the Free Software Foundation.
- *
- * BadVPN is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along
- * with this program; if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
- */
- #include <stdint.h>
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- #include <unistd.h>
- #include <errno.h>
- #endif
- #include <misc/offset.h>
- #include <system/BLog.h>
- #include <generated/blog_channel_BThreadWork.h>
- #include <threadwork/BThreadWork.h>
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- static void * dispatcher_thread (BThreadWorkDispatcher *o)
- {
- while (1) {
- // wait for a work
- ASSERT_FORCE(sem_wait(&o->new_sem) == 0)
-
- // grab the work
- ASSERT_FORCE(pthread_mutex_lock(&o->mutex) == 0)
- if (LinkedList2_IsEmpty(&o->pending_list)) {
- ASSERT_FORCE(pthread_mutex_unlock(&o->mutex) == 0)
- continue;
- }
- BThreadWork *w = UPPER_OBJECT(LinkedList2_GetFirst(&o->pending_list), BThreadWork, list_node);
- ASSERT(w->state == BTHREADWORK_STATE_PENDING)
- LinkedList2_Remove(&o->pending_list, &w->list_node);
- o->running_work = w;
- w->state = BTHREADWORK_STATE_RUNNING;
- ASSERT_FORCE(pthread_mutex_unlock(&o->mutex) == 0)
-
- // do the work
- w->work_func(w->work_func_user);
-
- // release the work
- ASSERT_FORCE(pthread_mutex_lock(&o->mutex) == 0)
- o->running_work = NULL;
- LinkedList2_Append(&o->finished_list, &w->list_node);
- w->state = BTHREADWORK_STATE_FINISHED;
- ASSERT_FORCE(sem_post(&w->finished_sem) == 0)
- ASSERT_FORCE(pthread_mutex_unlock(&o->mutex) == 0)
-
- // write to pipe
- uint8_t b = 0;
- ASSERT_FORCE(write(o->pipe[1], &b, sizeof(b)) == sizeof(b))
- }
- }
- static void pipe_fd_handler (BThreadWorkDispatcher *o, int events)
- {
- ASSERT(o->num_threads > 0)
- DebugObject_Access(&o->d_obj);
-
- // read from pipe
- uint8_t b;
- int res = read(o->pipe[0], &b, sizeof(b));
- if (res < 0) {
- int error = errno;
- ASSERT_FORCE(error == EAGAIN || error == EWOULDBLOCK)
- return;
- }
- ASSERT(res == sizeof(b))
- ASSERT(b == 0)
-
- // grab a finished work
- ASSERT_FORCE(pthread_mutex_lock(&o->mutex) == 0)
- if (LinkedList2_IsEmpty(&o->finished_list)) {
- ASSERT_FORCE(pthread_mutex_unlock(&o->mutex) == 0)
- return;
- }
- BThreadWork *w = UPPER_OBJECT(LinkedList2_GetFirst(&o->finished_list), BThreadWork, list_node);
- ASSERT(w->state == BTHREADWORK_STATE_FINISHED)
- LinkedList2_Remove(&o->finished_list, &w->list_node);
- ASSERT_FORCE(pthread_mutex_unlock(&o->mutex) == 0)
-
- // set state forgotten
- w->state = BTHREADWORK_STATE_FORGOTTEN;
-
- // call handler
- w->handler_done(w->user);
- return;
- }
- #endif
- static void work_job_handler (BThreadWork *o)
- {
- ASSERT(o->d->num_threads == 0)
- DebugObject_Access(&o->d_obj);
-
- // do the work
- o->work_func(o->work_func_user);
-
- // call handler
- o->handler_done(o->user);
- return;
- }
- int BThreadWorkDispatcher_Init (BThreadWorkDispatcher *o, BReactor *reactor, int num_threads_hint)
- {
- // init arguments
- o->reactor = reactor;
-
- // set num threads
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- if (num_threads_hint < 0) {
- o->num_threads = 2;
- } else {
- o->num_threads = num_threads_hint;
- }
- #else
- o->num_threads = 0;
- #endif
-
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
-
- if (o->num_threads > 0) {
- // init pending list
- LinkedList2_Init(&o->pending_list);
-
- // set no running work
- o->running_work = NULL;
-
- // init finished list
- LinkedList2_Init(&o->finished_list);
-
- // init mutex
- if (pthread_mutex_init(&o->mutex, NULL) != 0) {
- BLog(BLOG_ERROR, "pthread_mutex_init failed");
- goto fail0;
- }
-
- // init semaphore
- if (sem_init(&o->new_sem, 0, 0) != 0) {
- BLog(BLOG_ERROR, "sem_init failed");
- goto fail1;
- }
-
- // init pipe
- if (pipe(o->pipe) < 0) {
- BLog(BLOG_ERROR, "pipe failed");
- goto fail2;
- }
-
- // init BFileDescriptor
- BFileDescriptor_Init(&o->bfd, o->pipe[0], (BFileDescriptor_handler)pipe_fd_handler, o);
- if (!BReactor_AddFileDescriptor(o->reactor, &o->bfd)) {
- BLog(BLOG_ERROR, "BReactor_AddFileDescriptor failed");
- goto fail3;
- }
- BReactor_SetFileDescriptorEvents(o->reactor, &o->bfd, BREACTOR_READ);
-
- // init thread
- if (pthread_create(&o->thread, NULL, (void * (*) (void *))dispatcher_thread, o) != 0) {
- BLog(BLOG_ERROR, "pthread_create failed");
- goto fail4;
- }
- }
-
- #endif
-
- DebugObject_Init(&o->d_obj);
- DebugCounter_Init(&o->d_ctr);
- return 1;
-
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- fail4:
- BReactor_RemoveFileDescriptor(o->reactor, &o->bfd);
- fail3:
- ASSERT_FORCE(close(o->pipe[0]) == 0)
- ASSERT_FORCE(close(o->pipe[1]) == 0)
- fail2:
- ASSERT_FORCE(sem_destroy(&o->new_sem) == 0)
- fail1:
- ASSERT_FORCE(pthread_mutex_destroy(&o->mutex) == 0)
- #endif
- fail0:
- return 0;
- }
- void BThreadWorkDispatcher_Free (BThreadWorkDispatcher *o)
- {
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- if (o->num_threads > 0) {
- ASSERT(LinkedList2_IsEmpty(&o->pending_list))
- ASSERT(!o->running_work)
- ASSERT(LinkedList2_IsEmpty(&o->finished_list))
- }
- #endif
- DebugObject_Free(&o->d_obj);
- DebugCounter_Free(&o->d_ctr);
-
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
-
- if (o->num_threads > 0) {
- // stop thread
- ASSERT_FORCE(pthread_cancel(o->thread) == 0)
- void *retval;
- ASSERT_FORCE(pthread_join(o->thread, &retval) == 0)
-
- // free BFileDescriptor
- BReactor_RemoveFileDescriptor(o->reactor, &o->bfd);
-
- // free pipe
- ASSERT_FORCE(close(o->pipe[0]) == 0)
- ASSERT_FORCE(close(o->pipe[1]) == 0)
-
- // free semaphore
- ASSERT_FORCE(sem_destroy(&o->new_sem) == 0)
-
- // free mutex
- ASSERT_FORCE(pthread_mutex_destroy(&o->mutex) == 0)
- }
-
- #endif
- }
- void BThreadWork_Init (BThreadWork *o, BThreadWorkDispatcher *d, BThreadWork_handler_done handler_done, void *user, BThreadWork_work_func work_func, void *work_func_user)
- {
- DebugObject_Access(&d->d_obj);
-
- // init arguments
- o->d = d;
- o->handler_done = handler_done;
- o->user = user;
- o->work_func = work_func;
- o->work_func_user = work_func_user;
-
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- if (d->num_threads > 0) {
- // set state
- o->state = BTHREADWORK_STATE_PENDING;
-
- // init finished semaphore
- ASSERT_FORCE(sem_init(&o->finished_sem, 0, 0) == 0)
-
- // insert to pending list
- ASSERT_FORCE(pthread_mutex_lock(&d->mutex) == 0)
- LinkedList2_Append(&d->pending_list, &o->list_node);
- ASSERT_FORCE(pthread_mutex_unlock(&d->mutex) == 0)
-
- // post to new semaphore
- ASSERT_FORCE(sem_post(&d->new_sem) == 0)
- } else {
- #endif
- // schedule job
- BPending_Init(&o->job, BReactor_PendingGroup(d->reactor), (BPending_handler)work_job_handler, o);
- BPending_Set(&o->job);
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- }
- #endif
-
- DebugObject_Init(&o->d_obj);
- DebugCounter_Increment(&d->d_ctr);
- }
- void BThreadWork_Free (BThreadWork *o)
- {
- BThreadWorkDispatcher *d = o->d;
- DebugObject_Free(&o->d_obj);
- DebugCounter_Decrement(&d->d_ctr);
-
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- if (d->num_threads > 0) {
- ASSERT_FORCE(pthread_mutex_lock(&d->mutex) == 0)
-
- switch (o->state) {
- case BTHREADWORK_STATE_PENDING: {
- BLog(BLOG_DEBUG, "remove pending work");
-
- // remove from pending list
- LinkedList2_Remove(&d->pending_list, &o->list_node);
- } break;
-
- case BTHREADWORK_STATE_RUNNING: {
- BLog(BLOG_DEBUG, "remove running work");
-
- // wait for the work to finish running
- ASSERT_FORCE(pthread_mutex_unlock(&d->mutex) == 0)
- ASSERT_FORCE(sem_wait(&o->finished_sem) == 0)
- ASSERT_FORCE(pthread_mutex_lock(&d->mutex) == 0)
-
- ASSERT(o->state == BTHREADWORK_STATE_FINISHED)
-
- // remove from finished list
- LinkedList2_Remove(&d->finished_list, &o->list_node);
- } break;
-
- case BTHREADWORK_STATE_FINISHED: {
- BLog(BLOG_DEBUG, "remove finished work");
-
- // remove from finished list
- LinkedList2_Remove(&d->finished_list, &o->list_node);
- } break;
-
- case BTHREADWORK_STATE_FORGOTTEN: {
- BLog(BLOG_DEBUG, "remove forgotten work");
- } break;
-
- default:
- ASSERT(0);
- }
-
- ASSERT_FORCE(pthread_mutex_unlock(&d->mutex) == 0)
-
- // free finished semaphore
- ASSERT_FORCE(sem_destroy(&o->finished_sem) == 0)
- } else {
- #endif
- BPending_Free(&o->job);
- #ifdef BADVPN_THREADWORK_USE_PTHREAD
- }
- #endif
- }
|