FreeRDP
Loading...
Searching...
No Matches
pollset.c
1#ifndef _WIN32
2#include <errno.h>
3
4#include "pollset.h"
5#include <winpr/handle.h>
6#include <winpr/sysinfo.h>
7#include <winpr/assert.h>
8#include "../log.h"
9
10#if defined(__EMSCRIPTEN__)
11#include <emscripten.h>
12#endif
13
14#define TAG WINPR_TAG("sync.pollset")
15
16#ifdef WINPR_HAVE_POLL_H
17static INT16 handle_mode_to_pollevent(ULONG mode)
18{
19 INT16 event = 0;
20
21 if (mode & WINPR_FD_READ)
22 event |= POLLIN;
23
24 if (mode & WINPR_FD_WRITE)
25 event |= POLLOUT;
26
27 return event;
28}
29#endif
30
31BOOL pollset_init(WINPR_POLL_SET* set, size_t nhandles)
32{
33 WINPR_ASSERT(set);
34#ifdef WINPR_HAVE_POLL_H
35 if (nhandles > MAXIMUM_WAIT_OBJECTS)
36 {
37 set->isStatic = FALSE;
38 set->pollset = calloc(nhandles, sizeof(*set->pollset));
39 if (!set->pollset)
40 return FALSE;
41 }
42 else
43 {
44 set->pollset = set->staticSet;
45 set->isStatic = TRUE;
46 }
47#else
48 set->fdIndex = calloc(nhandles, sizeof(*set->fdIndex));
49 if (!set->fdIndex)
50 return FALSE;
51
52 FD_ZERO(&set->rset_base);
53 FD_ZERO(&set->rset);
54 FD_ZERO(&set->wset_base);
55 FD_ZERO(&set->wset);
56 set->maxFd = 0;
57 set->nread = set->nwrite = 0;
58#endif
59
60 set->size = nhandles;
61 set->fillIndex = 0;
62 return TRUE;
63}
64
65void pollset_uninit(WINPR_POLL_SET* set)
66{
67 WINPR_ASSERT(set);
68#ifdef WINPR_HAVE_POLL_H
69 if (!set->isStatic)
70 free(set->pollset);
71#else
72 free(set->fdIndex);
73#endif
74}
75
76void pollset_reset(WINPR_POLL_SET* set)
77{
78 WINPR_ASSERT(set);
79#ifndef WINPR_HAVE_POLL_H
80 FD_ZERO(&set->rset_base);
81 FD_ZERO(&set->wset_base);
82 set->maxFd = 0;
83 set->nread = set->nwrite = 0;
84#endif
85 set->fillIndex = 0;
86}
87
88BOOL pollset_add(WINPR_POLL_SET* set, int fd, ULONG mode)
89{
90 WINPR_ASSERT(set);
91#ifdef WINPR_HAVE_POLL_H
92 struct pollfd* item = NULL;
93 if (set->fillIndex == set->size)
94 return FALSE;
95
96 item = &set->pollset[set->fillIndex];
97 item->fd = fd;
98 item->revents = 0;
99 item->events = handle_mode_to_pollevent(mode);
100#else
101 FdIndex* fdIndex = &set->fdIndex[set->fillIndex];
102 if (mode & WINPR_FD_READ)
103 {
104 FD_SET(fd, &set->rset_base);
105 set->nread++;
106 }
107
108 if (mode & WINPR_FD_WRITE)
109 {
110 FD_SET(fd, &set->wset_base);
111 set->nwrite++;
112 }
113
114 if (fd > set->maxFd)
115 set->maxFd = fd;
116
117 fdIndex->fd = fd;
118 fdIndex->mode = mode;
119#endif
120 set->fillIndex++;
121 return TRUE;
122}
123
124int pollset_poll(WINPR_POLL_SET* set, DWORD dwMilliseconds)
125{
126 WINPR_ASSERT(set);
127 int ret = 0;
128 UINT64 dueTime = 0;
129 UINT64 now = 0;
130
131 now = GetTickCount64();
132 if (dwMilliseconds == INFINITE)
133 dueTime = 0xFFFFFFFFFFFFFFFF;
134 else
135 dueTime = now + dwMilliseconds;
136
137#ifdef WINPR_HAVE_POLL_H
138 int timeout = 0;
139
140 do
141 {
142 if (dwMilliseconds == INFINITE)
143 timeout = -1;
144 else
145 timeout = (int)(dueTime - now);
146
147 ret = poll(set->pollset, set->fillIndex, timeout);
148 if (ret >= 0)
149 {
150#if defined(__EMSCRIPTEN__)
151 /* Yield in emscripten so event handlers can be processed */
152 if (ret == 0)
153 emscripten_sleep(0);
154#endif
155 return ret;
156 }
157
158 if (errno != EINTR)
159 return -1;
160
161 now = GetTickCount64();
162 } while (now < dueTime);
163
164#else
165 do
166 {
167 struct timeval staticTimeout;
168 struct timeval* timeout;
169
170 fd_set* rset = NULL;
171 fd_set* wset = NULL;
172
173 if (dwMilliseconds == INFINITE)
174 {
175 timeout = NULL;
176 }
177 else
178 {
179 long waitTime = (long)(dueTime - now);
180
181 timeout = &staticTimeout;
182 timeout->tv_sec = waitTime / 1000;
183 timeout->tv_usec = (waitTime % 1000) * 1000;
184 }
185
186 if (set->nread)
187 {
188 rset = &set->rset;
189 memcpy(rset, &set->rset_base, sizeof(*rset));
190 }
191
192 if (set->nwrite)
193 {
194 wset = &set->wset;
195 memcpy(wset, &set->wset_base, sizeof(*wset));
196 }
197
198 ret = select(set->maxFd + 1, rset, wset, NULL, timeout);
199 if (ret >= 0)
200 return ret;
201
202 if (errno != EINTR)
203 return -1;
204
205 now = GetTickCount64();
206
207 } while (now < dueTime);
208
209 FD_ZERO(&set->rset);
210 FD_ZERO(&set->wset);
211#endif
212
213 return 0; /* timeout */
214}
215
216BOOL pollset_isSignaled(WINPR_POLL_SET* set, size_t idx)
217{
218 WINPR_ASSERT(set);
219
220 if (idx > set->fillIndex)
221 {
222 WLog_ERR(TAG, "index=%d out of pollset(fillIndex=%" PRIuz ")", idx, set->fillIndex);
223 return FALSE;
224 }
225
226#ifdef WINPR_HAVE_POLL_H
227 return !!(set->pollset[idx].revents & set->pollset[idx].events);
228#else
229 FdIndex* fdIndex = &set->fdIndex[idx];
230 if (fdIndex->fd < 0)
231 return FALSE;
232
233 if ((fdIndex->mode & WINPR_FD_READ) && FD_ISSET(fdIndex->fd, &set->rset))
234 return TRUE;
235
236 if ((fdIndex->mode & WINPR_FD_WRITE) && FD_ISSET(fdIndex->fd, &set->wset))
237 return TRUE;
238
239 return FALSE;
240#endif
241}
242
243BOOL pollset_isReadSignaled(WINPR_POLL_SET* set, size_t idx)
244{
245 WINPR_ASSERT(set);
246
247 if (idx > set->fillIndex)
248 {
249 WLog_ERR(TAG, "index=%d out of pollset(fillIndex=%" PRIuz ")", idx, set->fillIndex);
250 return FALSE;
251 }
252
253#ifdef WINPR_HAVE_POLL_H
254 return !!(set->pollset[idx].revents & POLLIN);
255#else
256 FdIndex* fdIndex = &set->fdIndex[idx];
257 if (fdIndex->fd < 0)
258 return FALSE;
259
260 return FD_ISSET(fdIndex->fd, &set->rset);
261#endif
262}
263
264BOOL pollset_isWriteSignaled(WINPR_POLL_SET* set, size_t idx)
265{
266 WINPR_ASSERT(set);
267
268 if (idx > set->fillIndex)
269 {
270 WLog_ERR(TAG, "index=%d out of pollset(fillIndex=%" PRIuz ")", idx, set->fillIndex);
271 return FALSE;
272 }
273
274#ifdef WINPR_HAVE_POLL_H
275 return !!(set->pollset[idx].revents & POLLOUT);
276#else
277 FdIndex* fdIndex = &set->fdIndex[idx];
278 if (fdIndex->fd < 0)
279 return FALSE;
280
281 return FD_ISSET(fdIndex->fd, &set->wset);
282#endif
283}
284
285#endif