|
160
|
1 /* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
|
|
|
2 *
|
|
|
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:
|
|
|
9 *
|
|
|
10 * The above copyright notice and this permission notice shall be included in
|
|
|
11 * all copies or substantial portions of the Software.
|
|
|
12 *
|
|
|
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
|
|
|
19 * IN THE SOFTWARE.
|
|
|
20 */
|
|
|
21
|
|
|
22 #include "uv.h"
|
|
|
23 #include "uv/tree.h"
|
|
|
24 #include "internal.h"
|
|
|
25 #include "heap-inl.h"
|
|
|
26 #include <stdlib.h>
|
|
|
27 #include <string.h>
|
|
|
28 #include <unistd.h>
|
|
|
29
|
|
|
30 int uv_loop_init(uv_loop_t* loop) {
|
|
|
31 uv__loop_internal_fields_t* lfields;
|
|
|
32 void* saved_data;
|
|
|
33 int err;
|
|
|
34
|
|
|
35 saved_data = loop->data;
|
|
|
36 memset(loop, 0, sizeof(*loop));
|
|
|
37 loop->data = saved_data;
|
|
|
38
|
|
|
39 lfields = uv__calloc(1, sizeof(*lfields));
|
|
|
40 if (lfields == NULL)
|
|
|
41 return UV_ENOMEM;
|
|
|
42 loop->internal_fields = lfields;
|
|
|
43
|
|
|
44 err = uv_mutex_init(&lfields->loop_metrics.lock);
|
|
|
45 if (err)
|
|
|
46 goto fail_metrics_mutex_init;
|
|
|
47 memset(&lfields->loop_metrics.metrics,
|
|
|
48 0,
|
|
|
49 sizeof(lfields->loop_metrics.metrics));
|
|
|
50
|
|
|
51 heap_init((struct heap*) &loop->timer_heap);
|
|
|
52 uv__queue_init(&loop->wq);
|
|
|
53 uv__queue_init(&loop->idle_handles);
|
|
|
54 uv__queue_init(&loop->async_handles);
|
|
|
55 uv__queue_init(&loop->check_handles);
|
|
|
56 uv__queue_init(&loop->prepare_handles);
|
|
|
57 uv__queue_init(&loop->handle_queue);
|
|
|
58
|
|
|
59 loop->active_handles = 0;
|
|
|
60 loop->active_reqs.count = 0;
|
|
|
61 loop->nfds = 0;
|
|
|
62 loop->watchers = NULL;
|
|
|
63 loop->nwatchers = 0;
|
|
|
64 uv__queue_init(&loop->pending_queue);
|
|
|
65 uv__queue_init(&loop->watcher_queue);
|
|
|
66
|
|
|
67 loop->closing_handles = NULL;
|
|
|
68 uv__update_time(loop);
|
|
|
69 loop->async_io_watcher.fd = -1;
|
|
|
70 loop->async_wfd = -1;
|
|
|
71 loop->signal_pipefd[0] = -1;
|
|
|
72 loop->signal_pipefd[1] = -1;
|
|
|
73 loop->backend_fd = -1;
|
|
|
74 loop->emfile_fd = -1;
|
|
|
75
|
|
|
76 loop->timer_counter = 0;
|
|
|
77 loop->stop_flag = 0;
|
|
|
78
|
|
|
79 err = uv__platform_loop_init(loop);
|
|
|
80 if (err)
|
|
|
81 goto fail_platform_init;
|
|
|
82
|
|
|
83 uv__signal_global_once_init();
|
|
|
84 err = uv__process_init(loop);
|
|
|
85 if (err)
|
|
|
86 goto fail_signal_init;
|
|
|
87 uv__queue_init(&loop->process_handles);
|
|
|
88
|
|
|
89 err = uv_rwlock_init(&loop->cloexec_lock);
|
|
|
90 if (err)
|
|
|
91 goto fail_rwlock_init;
|
|
|
92
|
|
|
93 err = uv_mutex_init(&loop->wq_mutex);
|
|
|
94 if (err)
|
|
|
95 goto fail_mutex_init;
|
|
|
96
|
|
|
97 err = uv_async_init(loop, &loop->wq_async, uv__work_done);
|
|
|
98 if (err)
|
|
|
99 goto fail_async_init;
|
|
|
100
|
|
|
101 uv__handle_unref(&loop->wq_async);
|
|
|
102 loop->wq_async.flags |= UV_HANDLE_INTERNAL;
|
|
|
103
|
|
|
104 return 0;
|
|
|
105
|
|
|
106 fail_async_init:
|
|
|
107 uv_mutex_destroy(&loop->wq_mutex);
|
|
|
108
|
|
|
109 fail_mutex_init:
|
|
|
110 uv_rwlock_destroy(&loop->cloexec_lock);
|
|
|
111
|
|
|
112 fail_rwlock_init:
|
|
|
113 uv__signal_loop_cleanup(loop);
|
|
|
114
|
|
|
115 fail_signal_init:
|
|
|
116 uv__platform_loop_delete(loop);
|
|
|
117
|
|
|
118 if (loop->backend_fd != -1) {
|
|
|
119 uv__close(loop->backend_fd);
|
|
|
120 loop->backend_fd = -1;
|
|
|
121 }
|
|
|
122
|
|
|
123 fail_platform_init:
|
|
|
124 uv_mutex_destroy(&lfields->loop_metrics.lock);
|
|
|
125
|
|
|
126 fail_metrics_mutex_init:
|
|
|
127 uv__free(lfields);
|
|
|
128 loop->internal_fields = NULL;
|
|
|
129
|
|
|
130 uv__free(loop->watchers);
|
|
|
131 loop->nwatchers = 0;
|
|
|
132 return err;
|
|
|
133 }
|
|
|
134
|
|
|
135
|
|
|
136 int uv_loop_fork(uv_loop_t* loop) {
|
|
|
137 int err;
|
|
|
138 unsigned int i;
|
|
|
139 uv__io_t* w;
|
|
|
140
|
|
|
141 err = uv__io_fork(loop);
|
|
|
142 if (err)
|
|
|
143 return err;
|
|
|
144
|
|
|
145 err = uv__async_fork(loop);
|
|
|
146 if (err)
|
|
|
147 return err;
|
|
|
148
|
|
|
149 err = uv__signal_loop_fork(loop);
|
|
|
150 if (err)
|
|
|
151 return err;
|
|
|
152
|
|
|
153 /* Rearm all the watchers that aren't re-queued by the above. */
|
|
|
154 for (i = 0; i < loop->nwatchers; i++) {
|
|
|
155 w = loop->watchers[i];
|
|
|
156 if (w == NULL)
|
|
|
157 continue;
|
|
|
158
|
|
|
159 if (w->pevents != 0 && uv__queue_empty(&w->watcher_queue)) {
|
|
|
160 w->events = 0; /* Force re-registration in uv__io_poll. */
|
|
|
161 uv__queue_insert_tail(&loop->watcher_queue, &w->watcher_queue);
|
|
|
162 }
|
|
|
163 }
|
|
|
164
|
|
|
165 return 0;
|
|
|
166 }
|
|
|
167
|
|
|
168
|
|
|
169 void uv__loop_close(uv_loop_t* loop) {
|
|
|
170 uv__loop_internal_fields_t* lfields;
|
|
|
171
|
|
|
172 uv__signal_loop_cleanup(loop);
|
|
|
173 uv__platform_loop_delete(loop);
|
|
|
174 uv__async_stop(loop);
|
|
|
175
|
|
|
176 if (loop->emfile_fd != -1) {
|
|
|
177 uv__close(loop->emfile_fd);
|
|
|
178 loop->emfile_fd = -1;
|
|
|
179 }
|
|
|
180
|
|
|
181 if (loop->backend_fd != -1) {
|
|
|
182 uv__close(loop->backend_fd);
|
|
|
183 loop->backend_fd = -1;
|
|
|
184 }
|
|
|
185
|
|
|
186 uv_mutex_lock(&loop->wq_mutex);
|
|
|
187 assert(uv__queue_empty(&loop->wq) && "thread pool work queue not empty!");
|
|
|
188 assert(!uv__has_active_reqs(loop));
|
|
|
189 uv_mutex_unlock(&loop->wq_mutex);
|
|
|
190 uv_mutex_destroy(&loop->wq_mutex);
|
|
|
191
|
|
|
192 /*
|
|
|
193 * Note that all thread pool stuff is finished at this point and
|
|
|
194 * it is safe to just destroy rw lock
|
|
|
195 */
|
|
|
196 uv_rwlock_destroy(&loop->cloexec_lock);
|
|
|
197
|
|
|
198 #if 0
|
|
|
199 assert(uv__queue_empty(&loop->pending_queue));
|
|
|
200 assert(uv__queue_empty(&loop->watcher_queue));
|
|
|
201 assert(loop->nfds == 0);
|
|
|
202 #endif
|
|
|
203
|
|
|
204 uv__free(loop->watchers);
|
|
|
205 loop->watchers = NULL;
|
|
|
206 loop->nwatchers = 0;
|
|
|
207
|
|
|
208 lfields = uv__get_internal_fields(loop);
|
|
|
209 uv_mutex_destroy(&lfields->loop_metrics.lock);
|
|
|
210 uv__free(lfields);
|
|
|
211 loop->internal_fields = NULL;
|
|
|
212 }
|
|
|
213
|
|
|
214
|
|
|
215 int uv__loop_configure(uv_loop_t* loop, uv_loop_option option, va_list ap) {
|
|
|
216 uv__loop_internal_fields_t* lfields;
|
|
|
217
|
|
|
218 lfields = uv__get_internal_fields(loop);
|
|
|
219 if (option == UV_METRICS_IDLE_TIME) {
|
|
|
220 lfields->flags |= UV_METRICS_IDLE_TIME;
|
|
|
221 return 0;
|
|
|
222 }
|
|
|
223
|
|
|
224 #if defined(__linux__)
|
|
|
225 if (option == UV_LOOP_USE_IO_URING_SQPOLL) {
|
|
|
226 loop->flags |= UV_LOOP_ENABLE_IO_URING_SQPOLL;
|
|
|
227 return 0;
|
|
|
228 }
|
|
|
229 #endif
|
|
|
230
|
|
|
231
|
|
|
232 if (option != UV_LOOP_BLOCK_SIGNAL)
|
|
|
233 return UV_ENOSYS;
|
|
|
234
|
|
|
235 if (va_arg(ap, int) != SIGPROF)
|
|
|
236 return UV_EINVAL;
|
|
|
237
|
|
|
238 loop->flags |= UV_LOOP_BLOCK_SIGPROF;
|
|
|
239 return 0;
|
|
|
240 }
|