summaryrefslogtreecommitdiffhomepage
path: root/src/nxt_work_queue.h
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/nxt_work_queue.h137
1 files changed, 137 insertions, 0 deletions
diff --git a/src/nxt_work_queue.h b/src/nxt_work_queue.h
new file mode 100644
index 00000000..2b168b00
--- /dev/null
+++ b/src/nxt_work_queue.h
@@ -0,0 +1,137 @@
+
+/*
+ * Copyright (C) Igor Sysoev
+ * Copyright (C) NGINX, Inc.
+ */
+
+#ifndef _NXT_WORK_QUEUE_H_INCLUDED_
+#define _NXT_WORK_QUEUE_H_INCLUDED_
+
+
+/*
+ * A work handler with just the obj and data arguments instead
+ * of pointer to a possibly large a work struct allows to call
+ * the handler not only via a work queue but also directly.
+ * The only obj argument is enough for the most cases expect the
+ * source filters, so the data argument has been introduced and
+ * is used where appropriate.
+ */
+typedef void (*nxt_work_handler_t)(nxt_thread_t *thr, void *obj, void *data);
+
+typedef struct nxt_work_s nxt_work_t;
+
+
+struct nxt_work_s {
+ nxt_work_t *next;
+ nxt_work_handler_t handler;
+ void *obj;
+ void *data;
+ nxt_log_t *log;
+};
+
+
+typedef struct nxt_work_queue_chunk_s nxt_work_queue_chunk_t;
+
+struct nxt_work_queue_chunk_s {
+ nxt_work_queue_chunk_t *next;
+ nxt_work_t work;
+};
+
+
+typedef struct {
+ nxt_work_t *next;
+ nxt_work_t *spare;
+ nxt_work_queue_chunk_t *chunk;
+ size_t chunk_size;
+} nxt_work_queue_cache_t;
+
+
+typedef struct nxt_work_queue_s nxt_work_queue_t;
+
+struct nxt_work_queue_s {
+ nxt_work_t *head;
+ nxt_work_t *tail;
+ nxt_work_queue_t *next;
+#if (NXT_DEBUG)
+ const char *name;
+#endif
+};
+
+
+typedef struct {
+ nxt_work_queue_t *head;
+ nxt_work_queue_t *tail;
+ nxt_work_queue_t main;
+ nxt_work_queue_t last;
+ nxt_work_queue_cache_t cache;
+} nxt_thread_work_queue_t;
+
+
+typedef struct {
+ nxt_thread_spinlock_t lock;
+ nxt_work_t *head;
+ nxt_work_t *tail;
+ nxt_work_queue_cache_t cache;
+} nxt_locked_work_queue_t;
+
+
+NXT_EXPORT void nxt_thread_work_queue_create(nxt_thread_t *thr,
+ size_t chunk_size);
+NXT_EXPORT void nxt_thread_work_queue_destroy(nxt_thread_t *thr);
+NXT_EXPORT void nxt_thread_work_queue_add(nxt_thread_t *thr,
+ nxt_work_queue_t *wq,
+ nxt_work_handler_t handler, void *obj, void *data, nxt_log_t *log);
+NXT_EXPORT void nxt_thread_work_queue_push(nxt_thread_t *thr,
+ nxt_work_queue_t *wq, nxt_work_handler_t handler, void *obj, void *data,
+ nxt_log_t *log);
+NXT_EXPORT void nxt_work_queue_attach(nxt_thread_t *thr, nxt_work_queue_t *wq);
+NXT_EXPORT nxt_work_handler_t nxt_thread_work_queue_pop(nxt_thread_t *thr,
+ void **obj, void **data, nxt_log_t **log);
+NXT_EXPORT void nxt_thread_work_queue_drop(nxt_thread_t *thr, void *data);
+
+
+#define \
+nxt_thread_current_work_queue_add(thr, handler, obj, data, log) \
+ do { \
+ nxt_thread_t *_thr = thr; \
+ \
+ nxt_thread_work_queue_add(_thr, _thr->work_queue.head, \
+ handler, obj, data, log); \
+ } while (0)
+
+
+NXT_EXPORT void nxt_work_queue_destroy(nxt_work_queue_t *wq);
+
+
+#if (NXT_DEBUG)
+
+#define \
+nxt_work_queue_name(_wq, _name) \
+ (_wq)->name = _name
+
+#else
+
+#define \
+nxt_work_queue_name(_wq, _name)
+
+#endif
+
+
+NXT_EXPORT void nxt_thread_last_work_queue_add(nxt_thread_t *thr,
+ nxt_work_handler_t handler, void *obj, void *data, nxt_log_t *log);
+NXT_EXPORT nxt_work_handler_t nxt_thread_last_work_queue_pop(nxt_thread_t *thr,
+ void **obj, void **data, nxt_log_t **log);
+
+
+NXT_EXPORT void nxt_locked_work_queue_create(nxt_locked_work_queue_t *lwq,
+ size_t chunk_size);
+NXT_EXPORT void nxt_locked_work_queue_destroy(nxt_locked_work_queue_t *lwq);
+NXT_EXPORT void nxt_locked_work_queue_add(nxt_locked_work_queue_t *lwq,
+ nxt_work_handler_t handler, void *obj, void *data, nxt_log_t *log);
+NXT_EXPORT nxt_work_handler_t nxt_locked_work_queue_pop(
+ nxt_locked_work_queue_t *lwq, void **obj, void **data, nxt_log_t **log);
+NXT_EXPORT void nxt_locked_work_queue_move(nxt_thread_t *thr,
+ nxt_locked_work_queue_t *lwq, nxt_work_queue_t *wq);
+
+
+#endif /* _NXT_WORK_QUEUE_H_INCLUDED_ */