vlist: add static initializer macros
[project/libubox.git] / ustream.h
1 /*
2  * ustream - library for stream buffer management
3  *
4  * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
5  *
6  * Permission to use, copy, modify, and/or distribute this software for any
7  * purpose with or without fee is hereby granted, provided that the above
8  * copyright notice and this permission notice appear in all copies.
9  *
10  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17  */
18
19 #ifndef __USTREAM_H
20 #define __USTREAM_H
21
22 #include <stdarg.h>
23 #include "uloop.h"
24
25 struct ustream;
26 struct ustream_buf;
27
28 enum read_blocked_reason {
29         READ_BLOCKED_USER = (1 << 0),
30         READ_BLOCKED_FULL = (1 << 1),
31 };
32
33 struct ustream_buf_list {
34         struct ustream_buf *head;
35         struct ustream_buf *data_tail;
36         struct ustream_buf *tail;
37
38         int (*alloc)(struct ustream *s, struct ustream_buf_list *l);
39
40         int data_bytes;
41
42         int min_buffers;
43         int max_buffers;
44         int buffer_len;
45
46         int buffers;
47 };
48
49 struct ustream {
50         struct ustream_buf_list r, w;
51         struct uloop_timeout state_change;
52         struct ustream *next;
53
54         /*
55          * notify_read: (optional)
56          * called by the ustream core to notify that new data is available
57          * for reading.
58          * must not free the ustream from this callback
59          */
60         void (*notify_read)(struct ustream *s, int bytes_new);
61
62         /*
63          * notify_write: (optional)
64          * called by the ustream core to notify that some buffered data has
65          * been written to the stream.
66          * must not free the ustream from this callback
67          */
68         void (*notify_write)(struct ustream *s, int bytes);
69
70         /*
71          * notify_state: (optional)
72          * called by the ustream implementation to notify that the read
73          * side of the stream is closed (eof is set) or there was a write
74          * error (write_error is set).
75          * will be called again after the write buffer has been emptied when
76          * the read side has hit EOF.
77          */
78         void (*notify_state)(struct ustream *s);
79
80         /*
81          * write:
82          * must be defined by ustream implementation, accepts new write data.
83          * 'more' is used to indicate that a subsequent call will provide more
84          * data (useful for aggregating writes)
85          * returns the number of bytes accepted, or -1 if no more writes can
86          * be accepted (link error)
87          */
88         int (*write)(struct ustream *s, const char *buf, int len, bool more);
89
90         /*
91          * free: (optional)
92          * defined by ustream implementation, tears down the ustream and frees data
93          */
94         void (*free)(struct ustream *s);
95
96         /*
97          * set_read_blocked: (optional)
98          * defined by ustream implementation, called when the read_blocked flag
99          * changes
100          */
101         void (*set_read_blocked)(struct ustream *s);
102
103         /*
104          * poll: (optional)
105          * defined by the upstream implementation, called to request polling for
106          * available data.
107          * returns true if data was fetched.
108          */
109         bool (*poll)(struct ustream *s);
110
111         /*
112          * ustream user should set this if the input stream is expected
113          * to contain string data. the core will keep all data 0-terminated.
114          */
115         bool string_data;
116         bool write_error;
117         bool eof, eof_write_done;
118
119         enum read_blocked_reason read_blocked;
120 };
121
122 struct ustream_fd {
123         struct ustream stream;
124         struct uloop_fd fd;
125 };
126
127 struct ustream_buf {
128         struct ustream_buf *next;
129
130         char *data;
131         char *tail;
132         char *end;
133
134         char head[];
135 };
136
137 /* ustream_fd_init: create a file descriptor ustream (uses uloop) */
138 void ustream_fd_init(struct ustream_fd *s, int fd);
139
140 /* ustream_free: free all buffers and data associated with a ustream */
141 void ustream_free(struct ustream *s);
142
143 /* ustream_consume: remove data from the head of the read buffer */
144 void ustream_consume(struct ustream *s, int len);
145
146 /* ustream_write: add data to the write buffer */
147 int ustream_write(struct ustream *s, const char *buf, int len, bool more);
148 int ustream_printf(struct ustream *s, const char *format, ...);
149 int ustream_vprintf(struct ustream *s, const char *format, va_list arg);
150
151 /* ustream_get_read_buf: get a pointer to the next read buffer data */
152 char *ustream_get_read_buf(struct ustream *s, int *buflen);
153
154 /*
155  * ustream_set_read_blocked: set read blocked state
156  *
157  * if set, the ustream will no longer fetch pending data.
158  */
159 void ustream_set_read_blocked(struct ustream *s, bool set);
160
161 static inline bool ustream_read_blocked(struct ustream *s)
162 {
163         return !!(s->read_blocked & READ_BLOCKED_USER);
164 }
165
166 static inline int ustream_pending_data(struct ustream *s, bool write)
167 {
168         struct ustream_buf_list *b = write ? &s->w : &s->r;
169         return b->data_bytes;
170 }
171
172 static inline bool ustream_read_buf_full(struct ustream *s)
173 {
174         struct ustream_buf *buf = s->r.data_tail;
175         return buf && buf->data == buf->head && buf->tail == buf->end &&
176                s->r.buffers == s->r.max_buffers;
177 }
178
179 /*** --- functions only used by ustream implementations --- ***/
180
181 /* ustream_init_defaults: fill default callbacks and options */
182 void ustream_init_defaults(struct ustream *s);
183
184 /*
185  * ustream_reserve: allocate rx buffer space
186  *
187  * len: hint for how much space is needed (not guaranteed to be met)
188  * maxlen: pointer to where the actual buffer size is going to be stored
189  */
190 char *ustream_reserve(struct ustream *s, int len, int *maxlen);
191
192 /* ustream_fill_read: mark rx buffer space as filled */
193 void ustream_fill_read(struct ustream *s, int len);
194
195 /*
196  * ustream_write_pending: attempt to write more data from write buffers
197  * returns true if all write buffers have been emptied.
198  */
199 bool ustream_write_pending(struct ustream *s);
200
201 static inline void ustream_state_change(struct ustream *s)
202 {
203         uloop_timeout_set(&s->state_change, 0);
204 }
205
206 static inline bool ustream_poll(struct ustream *s)
207 {
208         if (!s->poll)
209                 return false;
210
211         return s->poll(s);
212 }
213
214 #endif