• source navigation  • diff markup  • identifier search  • freetext search  • 

Sources/libubox/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;
118         uint8_t pending_cb;
119 
120         enum read_blocked_reason read_blocked;
121 };
122 
123 struct ustream_fd {
124         struct ustream stream;
125         struct uloop_fd fd;
126 };
127 
128 struct ustream_buf {
129         struct ustream_buf *next;
130 
131         char *data;
132         char *tail;
133         char *end;
134 
135         char head[];
136 };
137 
138 /* ustream_fd_init: create a file descriptor ustream (uses uloop) */
139 void ustream_fd_init(struct ustream_fd *s, int fd);
140 
141 /* ustream_free: free all buffers and data associated with a ustream */
142 void ustream_free(struct ustream *s);
143 
144 /* ustream_consume: remove data from the head of the read buffer */
145 void ustream_consume(struct ustream *s, int len);
146 
147 /*
148  * ustream_read: read and consume data in read buffer into caller-specified
149  * area.  Return length of data read.
150  */
151 int ustream_read(struct ustream *s, char *buf, int buflen);
152 /* ustream_write: add data to the write buffer */
153 int ustream_write(struct ustream *s, const char *buf, int len, bool more);
154 int ustream_printf(struct ustream *s, const char *format, ...)
155         __attribute__ ((format (printf, 2, 3)));
156 int ustream_vprintf(struct ustream *s, const char *format, va_list arg)
157         __attribute__ ((format (printf, 2, 0)));
158 
159 /* ustream_get_read_buf: get a pointer to the next read buffer data */
160 char *ustream_get_read_buf(struct ustream *s, int *buflen);
161 
162 /*
163  * ustream_set_read_blocked: set read blocked state
164  *
165  * if set, the ustream will no longer fetch pending data.
166  */
167 void ustream_set_read_blocked(struct ustream *s, bool set);
168 
169 static inline bool ustream_read_blocked(struct ustream *s)
170 {
171         return !!(s->read_blocked & READ_BLOCKED_USER);
172 }
173 
174 static inline int ustream_pending_data(struct ustream *s, bool write)
175 {
176         struct ustream_buf_list *b = write ? &s->w : &s->r;
177         return b->data_bytes;
178 }
179 
180 static inline bool ustream_read_buf_full(struct ustream *s)
181 {
182         struct ustream_buf *buf = s->r.data_tail;
183         return buf && buf->data == buf->head && buf->tail == buf->end &&
184                s->r.buffers == s->r.max_buffers;
185 }
186 
187 /*** --- functions only used by ustream implementations --- ***/
188 
189 /* ustream_init_defaults: fill default callbacks and options */
190 void ustream_init_defaults(struct ustream *s);
191 
192 /*
193  * ustream_reserve: allocate rx buffer space
194  *
195  * len: hint for how much space is needed (not guaranteed to be met)
196  * maxlen: pointer to where the actual buffer size is going to be stored
197  */
198 char *ustream_reserve(struct ustream *s, int len, int *maxlen);
199 
200 /* ustream_fill_read: mark rx buffer space as filled */
201 void ustream_fill_read(struct ustream *s, int len);
202 
203 /*
204  * ustream_write_pending: attempt to write more data from write buffers
205  * returns true if all write buffers have been emptied.
206  */
207 bool ustream_write_pending(struct ustream *s);
208 
209 static inline void ustream_state_change(struct ustream *s)
210 {
211         uloop_timeout_set(&s->state_change, 0);
212 }
213 
214 static inline bool ustream_poll(struct ustream *s)
215 {
216         if (!s->poll)
217                 return false;
218 
219         return s->poll(s);
220 }
221 
222 #endif
223 

This page was automatically generated by LXR 0.3.1.  •  OpenWrt