1 // This is free and unencumbered software released into the public domain.
3 // Anyone is free to copy, modify, publish, use, compile, sell, or
4 // distribute this software, either in source code form or as a compiled
5 // binary, for any purpose, commercial or non-commercial, and by any
8 // In jurisdictions that recognize copyright laws, the author or authors
9 // of this software dedicate any and all copyright interest in the
10 // software to the public domain. We make this dedication for the benefit
11 // of the public at large and to the detriment of our heirs and
12 // successors. We intend this dedication to be an overt act of
13 // relinquishment in perpetuity of all present and future rights to this
14 // software under copyright law.
16 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
19 // IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
20 // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
21 // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
22 // OTHER DEALINGS IN THE SOFTWARE.
24 // XXX would be nice to eventually have multiple backends for poll,
33 export type event = enum i16 {
38 export def READ = event::READ;
39 export def WRITE = event::WRITE;
42 cb: *fn(io::file, event, nullable *opaque) void,
43 data: nullable *opaque,
51 // XXX we should hide the poll::error and return 'just' an error.
53 export type evloop = struct {
54 add: *fn(*evloop, io::file, event, *fn(io::file, event, nullable *opaque) void, nullable *opaque) void,
55 del: *fn(*evloop, io::file) void,
56 loop: *fn(*evloop) (void | poll::error),
57 loopbreak: *fn(*evloop) void,
64 sigpipe: ((io::file, io::file) | void),
66 stop: bool, // signal to stop
74 loopbreak = &evloopbreak,
80 let global = &_default;
82 export fn new() *evloop = alloc(base {
86 loopbreak = &evloopbreak,
92 fn ev2poll(ev: event) poll::event = switch (ev) {
93 case event::READ => yield poll::event::POLLIN;
94 case event::WRITE => yield poll::event::POLLOUT;
97 fn poll2ev(ev: i16) event = {
98 if ((ev & poll::event::POLLOUT) != 0) {
104 export fn add(fd: io::file, ev: event, f: *fn(io::file, event, nullable *opaque) void, data: nullable *opaque) void = {
105 global.add(global, fd, ev, f, data);
108 export fn evadd(evloop: *evloop, fd: io::file, ev: event, f: *fn(io::file, event, nullable *opaque) void, data: nullable *opaque) void = {
109 const b = evloop: *base;
110 append(b.wip.pfds, poll::pollfd {
112 events = ev2poll(ev),
115 append(b.wip.cbs, cb {
121 export fn del(fd: io::file) void = {
122 global.del(global, fd);
125 export fn evdel(evloop: *evloop, fd: io::file) void = {
126 const b = evloop: *base;
127 for (let i = 0z; i < len(b.wip.pfds); i += 1) {
128 if (b.wip.pfds[i].fd != fd) {
131 delete(b.wip.pfds[i]);
132 delete(b.wip.cbs[i]);
137 fn prepare_queue(b: *base) void = {
140 delete(b.working.pfds[..]);
141 delete(b.working.cbs[..]);
143 for (let i = 0z; i < len(wip.pfds); i += 1) {
144 append(b.working.pfds, wip.pfds[i]);
145 append(b.working.cbs, wip.cbs[i]);
149 export fn mainloop() (void | poll::error) = global.loop(global);
151 export fn loop(evloop: *evloop) (void | poll::error) = {
152 const b = evloop: *base;
154 const mask = poll::event::POLLIN | poll::event::POLLOUT |
155 poll::event::POLLHUP;
162 let n = poll::poll(q.pfds, poll::INDEF)?;
163 for (let i = 0z; i < len(q.pfds); i += 1) {
164 if ((q.pfds[i].revents & mask) == 0) {
167 q.cbs[i].cb(q.pfds[i].fd, poll2ev(q.pfds[i].revents),
173 export fn loopbreak() void = {
174 global.loopbreak(global);
177 export fn evloopbreak(evloop: *evloop) void = {
178 const b = evloop: *base;