Blame


1 2bc5ec6b 2023-12-19 op // This is free and unencumbered software released into the public domain.
2 2bc5ec6b 2023-12-19 op //
3 2bc5ec6b 2023-12-19 op // Anyone is free to copy, modify, publish, use, compile, sell, or
4 2bc5ec6b 2023-12-19 op // distribute this software, either in source code form or as a compiled
5 2bc5ec6b 2023-12-19 op // binary, for any purpose, commercial or non-commercial, and by any
6 2bc5ec6b 2023-12-19 op // means.
7 2bc5ec6b 2023-12-19 op //
8 2bc5ec6b 2023-12-19 op // In jurisdictions that recognize copyright laws, the author or authors
9 2bc5ec6b 2023-12-19 op // of this software dedicate any and all copyright interest in the
10 2bc5ec6b 2023-12-19 op // software to the public domain. We make this dedication for the benefit
11 2bc5ec6b 2023-12-19 op // of the public at large and to the detriment of our heirs and
12 2bc5ec6b 2023-12-19 op // successors. We intend this dedication to be an overt act of
13 2bc5ec6b 2023-12-19 op // relinquishment in perpetuity of all present and future rights to this
14 2bc5ec6b 2023-12-19 op // software under copyright law.
15 2bc5ec6b 2023-12-19 op //
16 2bc5ec6b 2023-12-19 op // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17 2bc5ec6b 2023-12-19 op // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18 2bc5ec6b 2023-12-19 op // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
19 2bc5ec6b 2023-12-19 op // IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
20 2bc5ec6b 2023-12-19 op // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
21 2bc5ec6b 2023-12-19 op // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
22 2bc5ec6b 2023-12-19 op // OTHER DEALINGS IN THE SOFTWARE.
23 2bc5ec6b 2023-12-19 op
24 2bc5ec6b 2023-12-19 op // XXX would be nice to eventually have multiple backends for poll,
25 2bc5ec6b 2023-12-19 op // kqueue, epoll.
26 2bc5ec6b 2023-12-19 op
27 2bc5ec6b 2023-12-19 op use errors;
28 2bc5ec6b 2023-12-19 op use io;
29 2bc5ec6b 2023-12-19 op use unix;
30 2bc5ec6b 2023-12-19 op use unix::poll;
31 2bc5ec6b 2023-12-19 op use unix::signal;
32 2bc5ec6b 2023-12-19 op
33 2bc5ec6b 2023-12-19 op export type event = enum i16 {
34 2bc5ec6b 2023-12-19 op READ,
35 2bc5ec6b 2023-12-19 op WRITE,
36 2bc5ec6b 2023-12-19 op };
37 2bc5ec6b 2023-12-19 op
38 f2f593e9 2023-12-19 op export def READ = event::READ;
39 f2f593e9 2023-12-19 op export def WRITE = event::WRITE;
40 2bc5ec6b 2023-12-19 op
41 2bc5ec6b 2023-12-19 op type cb = struct {
42 01312690 2023-12-19 op cb: *fn(io::file, event, nullable *opaque) void,
43 01312690 2023-12-19 op data: nullable *opaque,
44 2bc5ec6b 2023-12-19 op };
45 2bc5ec6b 2023-12-19 op
46 e9feffee 2023-12-19 op type queue = struct {
47 2bc5ec6b 2023-12-19 op pfds: []poll::pollfd,
48 2bc5ec6b 2023-12-19 op cbs: []cb,
49 e9feffee 2023-12-19 op };
50 e9feffee 2023-12-19 op
51 094f9ac0 2023-12-19 op // XXX we should hide the poll::error and return 'just' an error.
52 094f9ac0 2023-12-19 op // maybe.
53 094f9ac0 2023-12-19 op export type evloop = struct {
54 094f9ac0 2023-12-19 op add: *fn(*evloop, io::file, event, *fn(io::file, event, nullable *opaque) void, nullable *opaque) void,
55 094f9ac0 2023-12-19 op del: *fn(*evloop, io::file) void,
56 094f9ac0 2023-12-19 op loop: *fn(*evloop) (void | poll::error),
57 7a04e249 2023-12-19 op loopbreak: *fn(*evloop) void,
58 094f9ac0 2023-12-19 op };
59 094f9ac0 2023-12-19 op
60 e9feffee 2023-12-19 op type base = struct {
61 094f9ac0 2023-12-19 op evloop,
62 e9feffee 2023-12-19 op working: queue,
63 e9feffee 2023-12-19 op wip: queue,
64 ae2955a0 2023-12-19 op sigpipe: (void | (io::file, io::file)),
65 ae2955a0 2023-12-19 op sigcb: (void | *fn(signal::sig, nullable *opaque) void),
66 ae2955a0 2023-12-19 op sigdata: nullable *opaque,
67 2bc5ec6b 2023-12-19 op stop: bool, // signal to stop
68 2bc5ec6b 2023-12-19 op // TODO: timeout
69 2bc5ec6b 2023-12-19 op };
70 2bc5ec6b 2023-12-19 op
71 094f9ac0 2023-12-19 op export fn new() *evloop = alloc(base {
72 094f9ac0 2023-12-19 op add = &evadd,
73 094f9ac0 2023-12-19 op del = &evdel,
74 094f9ac0 2023-12-19 op loop = &loop,
75 7a04e249 2023-12-19 op loopbreak = &evloopbreak,
76 094f9ac0 2023-12-19 op sigpipe = void,
77 094f9ac0 2023-12-19 op sigcb = void,
78 094f9ac0 2023-12-19 op ...
79 094f9ac0 2023-12-19 op });
80 094f9ac0 2023-12-19 op
81 2bc5ec6b 2023-12-19 op fn ev2poll(ev: event) poll::event = switch (ev) {
82 2bc5ec6b 2023-12-19 op case event::READ => yield poll::event::POLLIN;
83 2bc5ec6b 2023-12-19 op case event::WRITE => yield poll::event::POLLOUT;
84 2bc5ec6b 2023-12-19 op };
85 2bc5ec6b 2023-12-19 op
86 2bc5ec6b 2023-12-19 op fn poll2ev(ev: i16) event = {
87 2bc5ec6b 2023-12-19 op if ((ev & poll::event::POLLOUT) != 0) {
88 2bc5ec6b 2023-12-19 op return event::WRITE;
89 2bc5ec6b 2023-12-19 op };
90 2bc5ec6b 2023-12-19 op return event::READ;
91 2bc5ec6b 2023-12-19 op };
92 2bc5ec6b 2023-12-19 op
93 1b9c9f8a 2023-12-19 op export fn evadd(
94 1b9c9f8a 2023-12-19 op evloop: *evloop,
95 1b9c9f8a 2023-12-19 op fd: io::file,
96 a3fc3b94 2023-12-19 op ev: event,
97 a3fc3b94 2023-12-19 op f: *fn(io::file, event, nullable *opaque) void,
98 1b9c9f8a 2023-12-19 op data: nullable *opaque
99 1b9c9f8a 2023-12-19 op ) void = {
100 094f9ac0 2023-12-19 op const b = evloop: *base;
101 562f76f0 2023-12-21 op
102 562f76f0 2023-12-21 op for (let i = 0z; i < len(b.wip.pfds); i += 1) {
103 562f76f0 2023-12-21 op if (b.wip.pfds[i].fd != fd)
104 562f76f0 2023-12-21 op continue;
105 562f76f0 2023-12-21 op b.wip.pfds[i].events = ev2poll(ev);
106 562f76f0 2023-12-21 op b.wip.cbs[i] = cb { cb = f, data = data };
107 562f76f0 2023-12-21 op return;
108 562f76f0 2023-12-21 op };
109 562f76f0 2023-12-21 op
110 e9feffee 2023-12-19 op append(b.wip.pfds, poll::pollfd {
111 2bc5ec6b 2023-12-19 op fd = fd,
112 2bc5ec6b 2023-12-19 op events = ev2poll(ev),
113 2bc5ec6b 2023-12-19 op ...
114 2bc5ec6b 2023-12-19 op });
115 e9feffee 2023-12-19 op append(b.wip.cbs, cb {
116 2bc5ec6b 2023-12-19 op cb = f,
117 2bc5ec6b 2023-12-19 op data = data,
118 2bc5ec6b 2023-12-19 op });
119 2bc5ec6b 2023-12-19 op };
120 2bc5ec6b 2023-12-19 op
121 094f9ac0 2023-12-19 op export fn evdel(evloop: *evloop, fd: io::file) void = {
122 094f9ac0 2023-12-19 op const b = evloop: *base;
123 e9feffee 2023-12-19 op for (let i = 0z; i < len(b.wip.pfds); i += 1) {
124 e9feffee 2023-12-19 op if (b.wip.pfds[i].fd != fd) {
125 2bc5ec6b 2023-12-19 op continue;
126 2bc5ec6b 2023-12-19 op };
127 e9feffee 2023-12-19 op delete(b.wip.pfds[i]);
128 e9feffee 2023-12-19 op delete(b.wip.cbs[i]);
129 2bc5ec6b 2023-12-19 op return;
130 2bc5ec6b 2023-12-19 op };
131 2bc5ec6b 2023-12-19 op };
132 2bc5ec6b 2023-12-19 op
133 094f9ac0 2023-12-19 op fn prepare_queue(b: *base) void = {
134 6b358d51 2023-12-19 op delete(b.working.pfds[..]);
135 6b358d51 2023-12-19 op delete(b.working.cbs[..]);
136 e9feffee 2023-12-19 op
137 710a6fd8 2023-12-21 op for (let i = 0z; i < len(b.wip.pfds); i += 1) {
138 710a6fd8 2023-12-21 op append(b.working.pfds, b.wip.pfds[i]);
139 710a6fd8 2023-12-21 op append(b.working.cbs, b.wip.cbs[i]);
140 e9feffee 2023-12-19 op };
141 e9feffee 2023-12-19 op };
142 e9feffee 2023-12-19 op
143 094f9ac0 2023-12-19 op export fn loop(evloop: *evloop) (void | poll::error) = {
144 094f9ac0 2023-12-19 op const b = evloop: *base;
145 094f9ac0 2023-12-19 op
146 2bc5ec6b 2023-12-19 op const mask = poll::event::POLLIN | poll::event::POLLOUT |
147 2bc5ec6b 2023-12-19 op poll::event::POLLHUP;
148 2bc5ec6b 2023-12-19 op
149 2bc5ec6b 2023-12-19 op for (!b.stop) {
150 094f9ac0 2023-12-19 op prepare_queue(b);
151 e9feffee 2023-12-19 op
152 e9feffee 2023-12-19 op let q = b.working;
153 e9feffee 2023-12-19 op
154 ae2955a0 2023-12-19 op let n = match (poll::poll(q.pfds, poll::INDEF)) {
155 ae2955a0 2023-12-19 op case let err: errors::interrupted =>
156 ae2955a0 2023-12-19 op continue;
157 ae2955a0 2023-12-19 op case let err: errors::error =>
158 ae2955a0 2023-12-19 op return err;
159 ae2955a0 2023-12-19 op case let n: uint => yield n;
160 ae2955a0 2023-12-19 op };
161 ae2955a0 2023-12-19 op
162 e9feffee 2023-12-19 op for (let i = 0z; i < len(q.pfds); i += 1) {
163 0e544c93 2023-12-21 op // XXX check for POLLNVAL ?
164 0e544c93 2023-12-21 op
165 e9feffee 2023-12-19 op if ((q.pfds[i].revents & mask) == 0) {
166 2bc5ec6b 2023-12-19 op continue;
167 2bc5ec6b 2023-12-19 op };
168 e9feffee 2023-12-19 op q.cbs[i].cb(q.pfds[i].fd, poll2ev(q.pfds[i].revents),
169 e9feffee 2023-12-19 op q.cbs[i].data);
170 2bc5ec6b 2023-12-19 op };
171 2bc5ec6b 2023-12-19 op };
172 2bc5ec6b 2023-12-19 op };
173 2bc5ec6b 2023-12-19 op
174 7a04e249 2023-12-19 op export fn evloopbreak(evloop: *evloop) void = {
175 7a04e249 2023-12-19 op const b = evloop: *base;
176 7a04e249 2023-12-19 op b.stop = true;
177 7a04e249 2023-12-19 op };