2 * ustream - library for stream buffer management
4 * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
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.
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.
24 static void ustream_fd_set_uloop(struct ustream
*s
, bool write
)
26 struct ustream_fd
*sf
= container_of(s
, struct ustream_fd
, stream
);
27 struct ustream_buf
*buf
;
28 unsigned int flags
= ULOOP_EDGE_TRIGGER
;
30 if (!s
->read_blocked
&& !s
->eof
)
34 if (write
|| (buf
&& s
->w
.data_bytes
&& !s
->write_error
))
37 uloop_fd_add(&sf
->fd
, flags
);
39 if (flags
& ULOOP_READ
)
40 sf
->fd
.cb(&sf
->fd
, ULOOP_READ
);
43 static void ustream_fd_set_read_blocked(struct ustream
*s
)
45 ustream_fd_set_uloop(s
, false);
48 static void ustream_fd_read_pending(struct ustream_fd
*sf
, bool *more
)
50 struct ustream
*s
= &sf
->stream
;
56 buf
= ustream_reserve(s
, 1, &buflen
);
60 len
= read(sf
->fd
.fd
, buf
, buflen
);
74 ustream_fill_read(s
, len
);
79 static int ustream_fd_write(struct ustream
*s
, const char *buf
, int buflen
, bool more
)
81 struct ustream_fd
*sf
= container_of(s
, struct ustream_fd
, stream
);
88 len
= write(sf
->fd
.fd
, buf
, buflen
);
96 if (errno
== EAGAIN
|| errno
== EWOULDBLOCK
)
100 if (len
>= 0 && len
< buflen
)
101 ustream_fd_set_uloop(s
, true);
106 static bool __ustream_fd_poll(struct ustream_fd
*sf
, unsigned int events
)
108 struct ustream
*s
= &sf
->stream
;
109 struct uloop_fd
*fd
= &sf
->fd
;
112 if (events
& ULOOP_READ
)
113 ustream_fd_read_pending(sf
, &more
);
115 if (events
& ULOOP_WRITE
) {
116 if (!ustream_write_pending(s
))
117 ustream_fd_set_uloop(s
, false);
120 if (!s
->eof
&& fd
->eof
) {
122 ustream_fd_set_uloop(s
, false);
123 ustream_state_change(s
);
129 static bool ustream_fd_poll(struct ustream
*s
)
131 struct ustream_fd
*sf
= container_of(s
, struct ustream_fd
, stream
);
133 return __ustream_fd_poll(sf
, ULOOP_READ
| ULOOP_WRITE
);
136 static void ustream_uloop_cb(struct uloop_fd
*fd
, unsigned int events
)
138 struct ustream_fd
*sf
= container_of(fd
, struct ustream_fd
, fd
);
140 __ustream_fd_poll(sf
, events
);
143 static void ustream_fd_free(struct ustream
*s
)
145 struct ustream_fd
*sf
= container_of(s
, struct ustream_fd
, stream
);
147 uloop_fd_delete(&sf
->fd
);
150 void ustream_fd_init(struct ustream_fd
*sf
, int fd
)
152 struct ustream
*s
= &sf
->stream
;
154 ustream_init_defaults(s
);
157 sf
->fd
.cb
= ustream_uloop_cb
;
158 s
->set_read_blocked
= ustream_fd_set_read_blocked
;
159 s
->write
= ustream_fd_write
;
160 s
->free
= ustream_fd_free
;
161 s
->poll
= ustream_fd_poll
;
162 ustream_fd_set_uloop(s
, false);