aboutsummaryrefslogtreecommitdiffstats
path: root/src/worker.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/worker.c')
-rw-r--r--src/worker.c141
1 files changed, 141 insertions, 0 deletions
diff --git a/src/worker.c b/src/worker.c
new file mode 100644
index 0000000..79bb7b7
--- /dev/null
+++ b/src/worker.c
@@ -0,0 +1,141 @@
+/*
+ Copyright 2007-2016 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include "worker.h"
+
+static LV2_Worker_Status
+jalv_worker_respond(LV2_Worker_Respond_Handle handle,
+ uint32_t size,
+ const void* data)
+{
+ JalvWorker* worker = (JalvWorker*)handle;
+ zix_ring_write(worker->responses, (const char*)&size, sizeof(size));
+ zix_ring_write(worker->responses, (const char*)data, size);
+ return LV2_WORKER_SUCCESS;
+}
+
+static void*
+worker_func(void* data)
+{
+ JalvWorker* worker = (JalvWorker*)data;
+ Jalv* jalv = worker->jalv;
+ void* buf = NULL;
+ while (true) {
+ zix_sem_wait(&worker->sem);
+ if (jalv->exit) {
+ break;
+ }
+
+ uint32_t size = 0;
+ zix_ring_read(worker->requests, (char*)&size, sizeof(size));
+
+ if (!(buf = realloc(buf, size))) {
+ fprintf(stderr, "error: realloc() failed\n");
+ free(buf);
+ return NULL;
+ }
+
+ zix_ring_read(worker->requests, (char*)buf, size);
+
+ zix_sem_wait(&jalv->work_lock);
+ worker->iface->work(
+ jalv->instance->lv2_handle, jalv_worker_respond, worker, size, buf);
+ zix_sem_post(&jalv->work_lock);
+ }
+
+ free(buf);
+ return NULL;
+}
+
+void
+jalv_worker_init(ZIX_UNUSED Jalv* jalv,
+ JalvWorker* worker,
+ const LV2_Worker_Interface* iface,
+ bool threaded)
+{
+ worker->iface = iface;
+ worker->threaded = threaded;
+ if (threaded) {
+ zix_thread_create(&worker->thread, 4096, worker_func, worker);
+ worker->requests = zix_ring_new(4096);
+ zix_ring_mlock(worker->requests);
+ }
+ worker->responses = zix_ring_new(4096);
+ worker->response = malloc(4096);
+ zix_ring_mlock(worker->responses);
+}
+
+void
+jalv_worker_finish(JalvWorker* worker)
+{
+ if (worker->threaded) {
+ zix_sem_post(&worker->sem);
+ zix_thread_join(worker->thread, NULL);
+ }
+}
+
+void
+jalv_worker_destroy(JalvWorker* worker)
+{
+ if (worker->requests) {
+ if (worker->threaded) {
+ zix_ring_free(worker->requests);
+ }
+ zix_ring_free(worker->responses);
+ free(worker->response);
+ }
+}
+
+LV2_Worker_Status
+jalv_worker_schedule(LV2_Worker_Schedule_Handle handle,
+ uint32_t size,
+ const void* data)
+{
+ JalvWorker* worker = (JalvWorker*)handle;
+ Jalv* jalv = worker->jalv;
+ if (worker->threaded) {
+ // Schedule a request to be executed by the worker thread
+ zix_ring_write(worker->requests, (const char*)&size, sizeof(size));
+ zix_ring_write(worker->requests, (const char*)data, size);
+ zix_sem_post(&worker->sem);
+ } else {
+ // Execute work immediately in this thread
+ zix_sem_wait(&jalv->work_lock);
+ worker->iface->work(
+ jalv->instance->lv2_handle, jalv_worker_respond, worker, size, data);
+ zix_sem_post(&jalv->work_lock);
+ }
+ return LV2_WORKER_SUCCESS;
+}
+
+void
+jalv_worker_emit_responses(JalvWorker* worker, LilvInstance* instance)
+{
+ if (worker->responses) {
+ uint32_t read_space = zix_ring_read_space(worker->responses);
+ while (read_space) {
+ uint32_t size = 0;
+ zix_ring_read(worker->responses, (char*)&size, sizeof(size));
+
+ zix_ring_read(worker->responses, (char*)worker->response, size);
+
+ worker->iface->work_response(
+ instance->lv2_handle, size, worker->response);
+
+ read_space -= sizeof(size) + size;
+ }
+ }
+}