Welcome to mirror list, hosted at ThFree Co, Russian Federation.

git.openwrt.org/project/libubox.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFelix Fietkau <nbd@openwrt.org>2012-10-22 01:00:06 +0400
committerFelix Fietkau <nbd@openwrt.org>2012-10-22 01:00:06 +0400
commit768a69b3cedfebde10825847e42f35ed4aee1856 (patch)
treebae7cb3e6fc95b78987df3ffacca24ee2d8102c0 /ustream.h
parentae3bcf84e1e6099635c67cf79daddf0929e73823 (diff)
add ustream, an api for stream buffer management
Diffstat (limited to 'ustream.h')
-rw-r--r--ustream.h183
1 files changed, 183 insertions, 0 deletions
diff --git a/ustream.h b/ustream.h
new file mode 100644
index 0000000..c9e23e8
--- /dev/null
+++ b/ustream.h
@@ -0,0 +1,183 @@
+/*
+ * ustream - library for stream buffer management
+ *
+ * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
+ *
+ * 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.
+ *
+ * THE 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.
+ */
+
+#ifndef __USTREAM_H
+#define __USTREAM_H
+
+#include "uloop.h"
+
+struct ustream;
+struct ustream_buf;
+
+enum read_blocked_reason {
+ READ_BLOCKED_USER = (1 << 0),
+ READ_BLOCKED_FULL = (1 << 1),
+};
+
+struct ustream_buf_list {
+ struct ustream_buf *head;
+ struct ustream_buf *data_tail;
+ struct ustream_buf *tail;
+
+ int (*alloc)(struct ustream *s, struct ustream_buf_list *l);
+
+ int data_bytes;
+
+ int min_buffers;
+ int max_buffers;
+ int buffer_len;
+
+ int buffers;
+};
+
+struct ustream {
+ struct ustream_buf_list r, w;
+ struct uloop_timeout state_change;
+
+ /*
+ * notify_read:
+ * called by the ustream core to notify that new data is available
+ * for reading.
+ * must not free the ustream from this callback
+ */
+ void (*notify_read)(struct ustream *s, int bytes_new);
+
+ /*
+ * notify_write: (optional)
+ * called by the ustream core to notify that some buffered data has
+ * been written to the stream.
+ * must not free the ustream from this callback
+ */
+ void (*notify_write)(struct ustream *s, int bytes);
+
+ /*
+ * notify_state:
+ * called by the ustream implementation to notify that the read
+ * side of the stream is closed (eof is set) or there was a write
+ * error (write_error is set).
+ * will be called again after the write buffer has been emptied when
+ * the read side has hit EOF.
+ */
+ void (*notify_state)(struct ustream *s);
+
+ /*
+ * write:
+ * must be defined by ustream implementation, accepts new write data.
+ * 'more' is used to indicate that a subsequent call will provide more
+ * data (useful for aggregating writes)
+ * returns the number of bytes accepted, or -1 if no more writes can
+ * be accepted (link error)
+ */
+ int (*write)(struct ustream *s, const char *buf, int len, bool more);
+
+ /*
+ * free: (optional)
+ * defined by ustream implementation, tears down the ustream and frees data
+ */
+ void (*free)(struct ustream *s);
+
+ /*
+ * set_read_blocked: (optional)
+ * defined by ustream implementation, called when the read_blocked flag
+ * changes
+ */
+ void (*set_read_blocked)(struct ustream *s);
+
+ /*
+ * ustream user should set this if the input stream is expected
+ * to contain string data. the core will keep all data 0-terminated.
+ */
+ bool string_data;
+ bool write_error;
+ bool eof, eof_write_done;
+
+ enum read_blocked_reason read_blocked;
+};
+
+struct ustream_fd {
+ struct ustream stream;
+ struct uloop_fd fd;
+};
+
+struct ustream_buf {
+ struct ustream_buf *next;
+
+ char *data;
+ char *tail;
+ char *end;
+
+ char head[];
+};
+
+/* ustream_fd_init: create a file descriptor ustream (uses uloop) */
+void ustream_fd_init(struct ustream_fd *s, int fd);
+
+/* ustream_free: free all buffers and data associated with a ustream */
+void ustream_free(struct ustream *s);
+
+/* ustream_consume: remove data from the head of the read buffer */
+void ustream_consume(struct ustream *s, int len);
+
+/* ustream_write: add data to the write buffer */
+int ustream_write(struct ustream *s, const char *buf, int len, bool more);
+int ustream_printf(struct ustream *s, const char *format, ...);
+int ustream_vprintf(struct ustream *s, const char *format, va_list arg);
+
+/* ustream_get_read_buf: get a pointer to the next read buffer data */
+char *ustream_get_read_buf(struct ustream *s, int *buflen);
+
+/*
+ * ustream_set_read_blocked: set read blocked state
+ *
+ * if set, the ustream will no longer fetch pending data.
+ */
+void ustream_set_read_blocked(struct ustream *s, bool set);
+
+static inline bool ustream_read_blocked(struct ustream *s)
+{
+ return !!(s->read_blocked & READ_BLOCKED_USER);
+}
+
+/*** --- functions only used by ustream implementations --- ***/
+
+/* ustream_init_defaults: fill default callbacks and options */
+void ustream_init_defaults(struct ustream *s);
+
+/*
+ * ustream_reserve: allocate rx buffer space
+ *
+ * len: hint for how much space is needed (not guaranteed to be met)
+ * maxlen: pointer to where the actual buffer size is going to be stored
+ */
+char *ustream_reserve(struct ustream *s, int len, int *maxlen);
+
+/* ustream_fill_read: mark rx buffer space as filled */
+void ustream_fill_read(struct ustream *s, int len);
+
+/*
+ * ustream_write_pending: attempt to write more data from write buffers
+ * returns true if all write buffers have been emptied.
+ */
+bool ustream_write_pending(struct ustream *s);
+
+static inline void ustream_state_change(struct ustream *s)
+{
+ uloop_timeout_set(&s->state_change, 0);
+}
+
+#endif