1 /* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
3 * Permission is hereby granted, free of charge, to any person obtaining a copy
4 * of this software and associated documentation files (the "Software"), to
5 * deal in the Software without restriction, including without limitation the
6 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
7 * sell copies of the Software, and to permit persons to whom the Software is
8 * furnished to do so, subject to the following conditions:
10 * The above copyright notice and this permission notice shall be included in
11 * all copies or substantial portions of the Software.
13 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
18 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
32 #include "handle-inl.h"
36 /* The only event loop we support right now */
37 static uv_loop_t uv_default_loop_;
39 /* uv_once intialization guards */
40 static uv_once_t uv_init_guard_ = UV_ONCE_INIT;
41 static uv_once_t uv_default_loop_init_guard_ = UV_ONCE_INIT;
44 static void uv__crt_invalid_parameter_handler(const wchar_t* expression,
45 const wchar_t* function, const wchar_t * file, unsigned int line,
51 static void uv_init(void) {
52 /* Tell Windows that we will handle critical errors. */
53 SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOGPFAULTERRORBOX |
54 SEM_NOOPENFILEERRORBOX);
56 /* Tell the CRT to not exit the application when an invalid parameter is */
57 /* passed. The main issue is that invalid FDs will trigger this behavior. */
58 #ifdef _WRITE_ABORT_MSG
59 _set_invalid_parameter_handler(uv__crt_invalid_parameter_handler);
62 /* Fetch winapi function pointers. This must be done first because other */
63 /* intialization code might need these function pointers to be loaded. */
66 /* Initialize winsock */
72 /* Initialize signal stuff */
75 /* Initialize console */
78 /* Initialize utilities */
83 static void uv_loop_init(uv_loop_t* loop) {
84 /* Create an I/O completion port */
85 loop->iocp = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 1);
86 if (loop->iocp == NULL) {
87 uv_fatal_error(GetLastError(), "CreateIoCompletionPort");
90 /* To prevent uninitialized memory access, loop->time must be intialized */
91 /* to zero before calling uv_update_time for the first time. */
95 ngx_queue_init(&loop->handle_queue);
96 ngx_queue_init(&loop->active_reqs);
97 loop->active_handles = 0;
99 loop->pending_reqs_tail = NULL;
101 loop->endgame_handles = NULL;
103 RB_INIT(&loop->timers);
105 loop->check_handles = NULL;
106 loop->prepare_handles = NULL;
107 loop->idle_handles = NULL;
109 loop->next_prepare_handle = NULL;
110 loop->next_check_handle = NULL;
111 loop->next_idle_handle = NULL;
113 memset(&loop->poll_peer_sockets, 0, sizeof loop->poll_peer_sockets);
115 loop->active_tcp_streams = 0;
116 loop->active_udp_streams = 0;
118 loop->timer_counter = 0;
121 loop->last_err = uv_ok_;
125 static void uv_default_loop_init(void) {
126 /* Initialize libuv itself first */
129 /* Initialize the main loop */
130 uv_loop_init(&uv_default_loop_);
134 void uv__once_init(void) {
135 uv_once(&uv_init_guard_, uv_init);
139 uv_loop_t* uv_default_loop(void) {
140 uv_once(&uv_default_loop_init_guard_, uv_default_loop_init);
141 return &uv_default_loop_;
145 uv_loop_t* uv_loop_new(void) {
148 /* Initialize libuv itself first */
151 loop = (uv_loop_t*)malloc(sizeof(uv_loop_t));
154 uv_fatal_error(ERROR_OUTOFMEMORY, "malloc");
162 void uv_loop_delete(uv_loop_t* loop) {
163 if (loop != &uv_default_loop_) {
165 for (i = 0; i < ARRAY_SIZE(loop->poll_peer_sockets); i++) {
166 SOCKET sock = loop->poll_peer_sockets[i];
167 if (sock != 0 && sock != INVALID_SOCKET) {
177 int uv_backend_fd(const uv_loop_t* loop) {
182 int uv_backend_timeout(const uv_loop_t* loop) {
187 static void uv_poll(uv_loop_t* loop, int block) {
189 DWORD bytes, timeout;
191 OVERLAPPED* overlapped;
195 timeout = uv_get_poll_timeout(loop);
200 success = GetQueuedCompletionStatus(loop->iocp,
207 /* Package was dequeued */
208 req = uv_overlapped_to_req(overlapped);
210 uv_insert_pending_req(loop, req);
212 } else if (GetLastError() != WAIT_TIMEOUT) {
214 uv_fatal_error(GetLastError(), "GetQueuedCompletionStatus");
219 static void uv_poll_ex(uv_loop_t* loop, int block) {
223 OVERLAPPED_ENTRY overlappeds[128];
228 timeout = uv_get_poll_timeout(loop);
233 assert(pGetQueuedCompletionStatusEx);
235 success = pGetQueuedCompletionStatusEx(loop->iocp,
237 ARRAY_SIZE(overlappeds),
242 for (i = 0; i < count; i++) {
243 /* Package was dequeued */
244 req = uv_overlapped_to_req(overlappeds[i].lpOverlapped);
245 uv_insert_pending_req(loop, req);
247 } else if (GetLastError() != WAIT_TIMEOUT) {
249 uv_fatal_error(GetLastError(), "GetQueuedCompletionStatusEx");
254 static int uv__loop_alive(uv_loop_t* loop) {
255 return loop->active_handles > 0 ||
256 !ngx_queue_empty(&loop->active_reqs) ||
257 loop->endgame_handles != NULL;
261 int uv_run(uv_loop_t *loop, uv_run_mode mode) {
263 void (*poll)(uv_loop_t* loop, int block);
265 if (pGetQueuedCompletionStatusEx)
270 if (!uv__loop_alive(loop))
273 r = uv__loop_alive(loop);
274 while (r != 0 && loop->stop_flag == 0) {
275 uv_update_time(loop);
276 uv_process_timers(loop);
278 /* Call idle callbacks if nothing to do. */
279 if (loop->pending_reqs_tail == NULL &&
280 loop->endgame_handles == NULL) {
281 uv_idle_invoke(loop);
284 uv_process_reqs(loop);
285 uv_process_endgames(loop);
287 uv_prepare_invoke(loop);
289 (*poll)(loop, loop->idle_handles == NULL &&
290 loop->pending_reqs_tail == NULL &&
291 loop->endgame_handles == NULL &&
293 (loop->active_handles > 0 ||
294 !ngx_queue_empty(&loop->active_reqs)) &&
295 !(mode & UV_RUN_NOWAIT));
297 uv_check_invoke(loop);
298 r = uv__loop_alive(loop);
299 if (mode & (UV_RUN_ONCE | UV_RUN_NOWAIT))
303 /* The if statement lets the compiler compile it to a conditional store.
304 * Avoids dirtying a cache line.
306 if (loop->stop_flag != 0)