1e66f31c5Sopenharmony_ci/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
2e66f31c5Sopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a copy
3e66f31c5Sopenharmony_ci * of this software and associated documentation files (the "Software"), to
4e66f31c5Sopenharmony_ci * deal in the Software without restriction, including without limitation the
5e66f31c5Sopenharmony_ci * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
6e66f31c5Sopenharmony_ci * sell copies of the Software, and to permit persons to whom the Software is
7e66f31c5Sopenharmony_ci * furnished to do so, subject to the following conditions:
8e66f31c5Sopenharmony_ci *
9e66f31c5Sopenharmony_ci * The above copyright notice and this permission notice shall be included in
10e66f31c5Sopenharmony_ci * all copies or substantial portions of the Software.
11e66f31c5Sopenharmony_ci *
12e66f31c5Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
13e66f31c5Sopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
14e66f31c5Sopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
15e66f31c5Sopenharmony_ci * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
16e66f31c5Sopenharmony_ci * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
17e66f31c5Sopenharmony_ci * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
18e66f31c5Sopenharmony_ci * IN THE SOFTWARE.
19e66f31c5Sopenharmony_ci */
20e66f31c5Sopenharmony_ci
21e66f31c5Sopenharmony_ci#include <assert.h>
22e66f31c5Sopenharmony_ci#include <signal.h>
23e66f31c5Sopenharmony_ci
24e66f31c5Sopenharmony_ci#include "uv.h"
25e66f31c5Sopenharmony_ci#include "internal.h"
26e66f31c5Sopenharmony_ci#include "handle-inl.h"
27e66f31c5Sopenharmony_ci#include "req-inl.h"
28e66f31c5Sopenharmony_ci
29e66f31c5Sopenharmony_ci
30e66f31c5Sopenharmony_ciRB_HEAD(uv_signal_tree_s, uv_signal_s);
31e66f31c5Sopenharmony_ci
32e66f31c5Sopenharmony_cistatic struct uv_signal_tree_s uv__signal_tree = RB_INITIALIZER(uv__signal_tree);
33e66f31c5Sopenharmony_cistatic CRITICAL_SECTION uv__signal_lock;
34e66f31c5Sopenharmony_ci
35e66f31c5Sopenharmony_cistatic BOOL WINAPI uv__signal_control_handler(DWORD type);
36e66f31c5Sopenharmony_ci
37e66f31c5Sopenharmony_ciint uv__signal_start(uv_signal_t* handle,
38e66f31c5Sopenharmony_ci                     uv_signal_cb signal_cb,
39e66f31c5Sopenharmony_ci                     int signum,
40e66f31c5Sopenharmony_ci                     int oneshot);
41e66f31c5Sopenharmony_ci
42e66f31c5Sopenharmony_civoid uv__signals_init(void) {
43e66f31c5Sopenharmony_ci  InitializeCriticalSection(&uv__signal_lock);
44e66f31c5Sopenharmony_ci  if (!SetConsoleCtrlHandler(uv__signal_control_handler, TRUE))
45e66f31c5Sopenharmony_ci    abort();
46e66f31c5Sopenharmony_ci}
47e66f31c5Sopenharmony_ci
48e66f31c5Sopenharmony_ci
49e66f31c5Sopenharmony_civoid uv__signal_cleanup(void) {
50e66f31c5Sopenharmony_ci  /* TODO(bnoordhuis) Undo effects of uv_signal_init()? */
51e66f31c5Sopenharmony_ci}
52e66f31c5Sopenharmony_ci
53e66f31c5Sopenharmony_ci
54e66f31c5Sopenharmony_cistatic int uv__signal_compare(uv_signal_t* w1, uv_signal_t* w2) {
55e66f31c5Sopenharmony_ci  /* Compare signums first so all watchers with the same signnum end up
56e66f31c5Sopenharmony_ci   * adjacent. */
57e66f31c5Sopenharmony_ci  if (w1->signum < w2->signum) return -1;
58e66f31c5Sopenharmony_ci  if (w1->signum > w2->signum) return 1;
59e66f31c5Sopenharmony_ci
60e66f31c5Sopenharmony_ci  /* Sort by loop pointer, so we can easily look up the first item after
61e66f31c5Sopenharmony_ci   * { .signum = x, .loop = NULL }. */
62e66f31c5Sopenharmony_ci  if ((uintptr_t) w1->loop < (uintptr_t) w2->loop) return -1;
63e66f31c5Sopenharmony_ci  if ((uintptr_t) w1->loop > (uintptr_t) w2->loop) return 1;
64e66f31c5Sopenharmony_ci
65e66f31c5Sopenharmony_ci  if ((uintptr_t) w1 < (uintptr_t) w2) return -1;
66e66f31c5Sopenharmony_ci  if ((uintptr_t) w1 > (uintptr_t) w2) return 1;
67e66f31c5Sopenharmony_ci
68e66f31c5Sopenharmony_ci  return 0;
69e66f31c5Sopenharmony_ci}
70e66f31c5Sopenharmony_ci
71e66f31c5Sopenharmony_ci
72e66f31c5Sopenharmony_ciRB_GENERATE_STATIC(uv_signal_tree_s, uv_signal_s, tree_entry, uv__signal_compare)
73e66f31c5Sopenharmony_ci
74e66f31c5Sopenharmony_ci
75e66f31c5Sopenharmony_ci/*
76e66f31c5Sopenharmony_ci * Dispatches signal {signum} to all active uv_signal_t watchers in all loops.
77e66f31c5Sopenharmony_ci * Returns 1 if the signal was dispatched to any watcher, or 0 if there were
78e66f31c5Sopenharmony_ci * no active signal watchers observing this signal.
79e66f31c5Sopenharmony_ci */
80e66f31c5Sopenharmony_ciint uv__signal_dispatch(int signum) {
81e66f31c5Sopenharmony_ci  uv_signal_t lookup;
82e66f31c5Sopenharmony_ci  uv_signal_t* handle;
83e66f31c5Sopenharmony_ci  int dispatched;
84e66f31c5Sopenharmony_ci
85e66f31c5Sopenharmony_ci  dispatched = 0;
86e66f31c5Sopenharmony_ci
87e66f31c5Sopenharmony_ci  EnterCriticalSection(&uv__signal_lock);
88e66f31c5Sopenharmony_ci
89e66f31c5Sopenharmony_ci  lookup.signum = signum;
90e66f31c5Sopenharmony_ci  lookup.loop = NULL;
91e66f31c5Sopenharmony_ci
92e66f31c5Sopenharmony_ci  for (handle = RB_NFIND(uv_signal_tree_s, &uv__signal_tree, &lookup);
93e66f31c5Sopenharmony_ci       handle != NULL && handle->signum == signum;
94e66f31c5Sopenharmony_ci       handle = RB_NEXT(uv_signal_tree_s, &uv__signal_tree, handle)) {
95e66f31c5Sopenharmony_ci    unsigned long previous = InterlockedExchange(
96e66f31c5Sopenharmony_ci            (volatile LONG*) &handle->pending_signum, signum);
97e66f31c5Sopenharmony_ci
98e66f31c5Sopenharmony_ci    if (handle->flags & UV_SIGNAL_ONE_SHOT_DISPATCHED)
99e66f31c5Sopenharmony_ci      continue;
100e66f31c5Sopenharmony_ci
101e66f31c5Sopenharmony_ci    if (!previous) {
102e66f31c5Sopenharmony_ci      POST_COMPLETION_FOR_REQ(handle->loop, &handle->signal_req);
103e66f31c5Sopenharmony_ci    }
104e66f31c5Sopenharmony_ci
105e66f31c5Sopenharmony_ci    dispatched = 1;
106e66f31c5Sopenharmony_ci    if (handle->flags & UV_SIGNAL_ONE_SHOT)
107e66f31c5Sopenharmony_ci      handle->flags |= UV_SIGNAL_ONE_SHOT_DISPATCHED;
108e66f31c5Sopenharmony_ci  }
109e66f31c5Sopenharmony_ci
110e66f31c5Sopenharmony_ci  LeaveCriticalSection(&uv__signal_lock);
111e66f31c5Sopenharmony_ci
112e66f31c5Sopenharmony_ci  return dispatched;
113e66f31c5Sopenharmony_ci}
114e66f31c5Sopenharmony_ci
115e66f31c5Sopenharmony_ci
116e66f31c5Sopenharmony_cistatic BOOL WINAPI uv__signal_control_handler(DWORD type) {
117e66f31c5Sopenharmony_ci  switch (type) {
118e66f31c5Sopenharmony_ci    case CTRL_C_EVENT:
119e66f31c5Sopenharmony_ci      return uv__signal_dispatch(SIGINT);
120e66f31c5Sopenharmony_ci
121e66f31c5Sopenharmony_ci    case CTRL_BREAK_EVENT:
122e66f31c5Sopenharmony_ci      return uv__signal_dispatch(SIGBREAK);
123e66f31c5Sopenharmony_ci
124e66f31c5Sopenharmony_ci    case CTRL_CLOSE_EVENT:
125e66f31c5Sopenharmony_ci      if (uv__signal_dispatch(SIGHUP)) {
126e66f31c5Sopenharmony_ci        /* Windows will terminate the process after the control handler
127e66f31c5Sopenharmony_ci         * returns. After that it will just terminate our process. Therefore
128e66f31c5Sopenharmony_ci         * block the signal handler so the main loop has some time to pick up
129e66f31c5Sopenharmony_ci         * the signal and do something for a few seconds. */
130e66f31c5Sopenharmony_ci        Sleep(INFINITE);
131e66f31c5Sopenharmony_ci        return TRUE;
132e66f31c5Sopenharmony_ci      }
133e66f31c5Sopenharmony_ci      return FALSE;
134e66f31c5Sopenharmony_ci
135e66f31c5Sopenharmony_ci    case CTRL_LOGOFF_EVENT:
136e66f31c5Sopenharmony_ci    case CTRL_SHUTDOWN_EVENT:
137e66f31c5Sopenharmony_ci      /* These signals are only sent to services. Services have their own
138e66f31c5Sopenharmony_ci       * notification mechanism, so there's no point in handling these. */
139e66f31c5Sopenharmony_ci
140e66f31c5Sopenharmony_ci    default:
141e66f31c5Sopenharmony_ci      /* We don't handle these. */
142e66f31c5Sopenharmony_ci      return FALSE;
143e66f31c5Sopenharmony_ci  }
144e66f31c5Sopenharmony_ci}
145e66f31c5Sopenharmony_ci
146e66f31c5Sopenharmony_ci
147e66f31c5Sopenharmony_ciint uv_signal_init(uv_loop_t* loop, uv_signal_t* handle) {
148e66f31c5Sopenharmony_ci  uv__handle_init(loop, (uv_handle_t*) handle, UV_SIGNAL);
149e66f31c5Sopenharmony_ci  handle->pending_signum = 0;
150e66f31c5Sopenharmony_ci  handle->signum = 0;
151e66f31c5Sopenharmony_ci  handle->signal_cb = NULL;
152e66f31c5Sopenharmony_ci
153e66f31c5Sopenharmony_ci  UV_REQ_INIT(&handle->signal_req, UV_SIGNAL_REQ);
154e66f31c5Sopenharmony_ci  handle->signal_req.data = handle;
155e66f31c5Sopenharmony_ci
156e66f31c5Sopenharmony_ci  return 0;
157e66f31c5Sopenharmony_ci}
158e66f31c5Sopenharmony_ci
159e66f31c5Sopenharmony_ci
160e66f31c5Sopenharmony_ciint uv_signal_stop(uv_signal_t* handle) {
161e66f31c5Sopenharmony_ci  uv_signal_t* removed_handle;
162e66f31c5Sopenharmony_ci
163e66f31c5Sopenharmony_ci  /* If the watcher wasn't started, this is a no-op. */
164e66f31c5Sopenharmony_ci  if (handle->signum == 0)
165e66f31c5Sopenharmony_ci    return 0;
166e66f31c5Sopenharmony_ci
167e66f31c5Sopenharmony_ci  EnterCriticalSection(&uv__signal_lock);
168e66f31c5Sopenharmony_ci
169e66f31c5Sopenharmony_ci  removed_handle = RB_REMOVE(uv_signal_tree_s, &uv__signal_tree, handle);
170e66f31c5Sopenharmony_ci  assert(removed_handle == handle);
171e66f31c5Sopenharmony_ci
172e66f31c5Sopenharmony_ci  LeaveCriticalSection(&uv__signal_lock);
173e66f31c5Sopenharmony_ci
174e66f31c5Sopenharmony_ci  handle->signum = 0;
175e66f31c5Sopenharmony_ci  uv__handle_stop(handle);
176e66f31c5Sopenharmony_ci
177e66f31c5Sopenharmony_ci  return 0;
178e66f31c5Sopenharmony_ci}
179e66f31c5Sopenharmony_ci
180e66f31c5Sopenharmony_ci
181e66f31c5Sopenharmony_ciint uv_signal_start(uv_signal_t* handle, uv_signal_cb signal_cb, int signum) {
182e66f31c5Sopenharmony_ci  return uv__signal_start(handle, signal_cb, signum, 0);
183e66f31c5Sopenharmony_ci}
184e66f31c5Sopenharmony_ci
185e66f31c5Sopenharmony_ci
186e66f31c5Sopenharmony_ciint uv_signal_start_oneshot(uv_signal_t* handle,
187e66f31c5Sopenharmony_ci                            uv_signal_cb signal_cb,
188e66f31c5Sopenharmony_ci                            int signum) {
189e66f31c5Sopenharmony_ci  return uv__signal_start(handle, signal_cb, signum, 1);
190e66f31c5Sopenharmony_ci}
191e66f31c5Sopenharmony_ci
192e66f31c5Sopenharmony_ci
193e66f31c5Sopenharmony_ciint uv__signal_start(uv_signal_t* handle,
194e66f31c5Sopenharmony_ci                            uv_signal_cb signal_cb,
195e66f31c5Sopenharmony_ci                            int signum,
196e66f31c5Sopenharmony_ci                            int oneshot) {
197e66f31c5Sopenharmony_ci  /* Test for invalid signal values. */
198e66f31c5Sopenharmony_ci  if (signum <= 0 || signum >= NSIG)
199e66f31c5Sopenharmony_ci    return UV_EINVAL;
200e66f31c5Sopenharmony_ci
201e66f31c5Sopenharmony_ci  /* Short circuit: if the signal watcher is already watching {signum} don't go
202e66f31c5Sopenharmony_ci   * through the process of deregistering and registering the handler.
203e66f31c5Sopenharmony_ci   * Additionally, this avoids pending signals getting lost in the (small) time
204e66f31c5Sopenharmony_ci   * frame that handle->signum == 0. */
205e66f31c5Sopenharmony_ci  if (signum == handle->signum) {
206e66f31c5Sopenharmony_ci    handle->signal_cb = signal_cb;
207e66f31c5Sopenharmony_ci    return 0;
208e66f31c5Sopenharmony_ci  }
209e66f31c5Sopenharmony_ci
210e66f31c5Sopenharmony_ci  /* If the signal handler was already active, stop it first. */
211e66f31c5Sopenharmony_ci  if (handle->signum != 0) {
212e66f31c5Sopenharmony_ci    int r = uv_signal_stop(handle);
213e66f31c5Sopenharmony_ci    /* uv_signal_stop is infallible. */
214e66f31c5Sopenharmony_ci    assert(r == 0);
215e66f31c5Sopenharmony_ci  }
216e66f31c5Sopenharmony_ci
217e66f31c5Sopenharmony_ci  EnterCriticalSection(&uv__signal_lock);
218e66f31c5Sopenharmony_ci
219e66f31c5Sopenharmony_ci  handle->signum = signum;
220e66f31c5Sopenharmony_ci  if (oneshot)
221e66f31c5Sopenharmony_ci    handle->flags |= UV_SIGNAL_ONE_SHOT;
222e66f31c5Sopenharmony_ci
223e66f31c5Sopenharmony_ci  RB_INSERT(uv_signal_tree_s, &uv__signal_tree, handle);
224e66f31c5Sopenharmony_ci
225e66f31c5Sopenharmony_ci  LeaveCriticalSection(&uv__signal_lock);
226e66f31c5Sopenharmony_ci
227e66f31c5Sopenharmony_ci  handle->signal_cb = signal_cb;
228e66f31c5Sopenharmony_ci  uv__handle_start(handle);
229e66f31c5Sopenharmony_ci
230e66f31c5Sopenharmony_ci  return 0;
231e66f31c5Sopenharmony_ci}
232e66f31c5Sopenharmony_ci
233e66f31c5Sopenharmony_ci
234e66f31c5Sopenharmony_civoid uv__process_signal_req(uv_loop_t* loop, uv_signal_t* handle,
235e66f31c5Sopenharmony_ci    uv_req_t* req) {
236e66f31c5Sopenharmony_ci  long dispatched_signum;
237e66f31c5Sopenharmony_ci
238e66f31c5Sopenharmony_ci  assert(handle->type == UV_SIGNAL);
239e66f31c5Sopenharmony_ci  assert(req->type == UV_SIGNAL_REQ);
240e66f31c5Sopenharmony_ci
241e66f31c5Sopenharmony_ci  dispatched_signum = InterlockedExchange(
242e66f31c5Sopenharmony_ci          (volatile LONG*) &handle->pending_signum, 0);
243e66f31c5Sopenharmony_ci  assert(dispatched_signum != 0);
244e66f31c5Sopenharmony_ci
245e66f31c5Sopenharmony_ci  /* Check if the pending signal equals the signum that we are watching for.
246e66f31c5Sopenharmony_ci   * These can get out of sync when the handler is stopped and restarted while
247e66f31c5Sopenharmony_ci   * the signal_req is pending. */
248e66f31c5Sopenharmony_ci  if (dispatched_signum == handle->signum)
249e66f31c5Sopenharmony_ci    handle->signal_cb(handle, dispatched_signum);
250e66f31c5Sopenharmony_ci
251e66f31c5Sopenharmony_ci  if (handle->flags & UV_SIGNAL_ONE_SHOT)
252e66f31c5Sopenharmony_ci    uv_signal_stop(handle);
253e66f31c5Sopenharmony_ci
254e66f31c5Sopenharmony_ci  if (handle->flags & UV_HANDLE_CLOSING) {
255e66f31c5Sopenharmony_ci    /* When it is closing, it must be stopped at this point. */
256e66f31c5Sopenharmony_ci    assert(handle->signum == 0);
257e66f31c5Sopenharmony_ci    uv__want_endgame(loop, (uv_handle_t*) handle);
258e66f31c5Sopenharmony_ci  }
259e66f31c5Sopenharmony_ci}
260e66f31c5Sopenharmony_ci
261e66f31c5Sopenharmony_ci
262e66f31c5Sopenharmony_civoid uv__signal_close(uv_loop_t* loop, uv_signal_t* handle) {
263e66f31c5Sopenharmony_ci  uv_signal_stop(handle);
264e66f31c5Sopenharmony_ci  uv__handle_closing(handle);
265e66f31c5Sopenharmony_ci
266e66f31c5Sopenharmony_ci  if (handle->pending_signum == 0) {
267e66f31c5Sopenharmony_ci    uv__want_endgame(loop, (uv_handle_t*) handle);
268e66f31c5Sopenharmony_ci  }
269e66f31c5Sopenharmony_ci}
270e66f31c5Sopenharmony_ci
271e66f31c5Sopenharmony_ci
272e66f31c5Sopenharmony_civoid uv__signal_endgame(uv_loop_t* loop, uv_signal_t* handle) {
273e66f31c5Sopenharmony_ci  assert(handle->flags & UV_HANDLE_CLOSING);
274e66f31c5Sopenharmony_ci  assert(!(handle->flags & UV_HANDLE_CLOSED));
275e66f31c5Sopenharmony_ci
276e66f31c5Sopenharmony_ci  assert(handle->signum == 0);
277e66f31c5Sopenharmony_ci  assert(handle->pending_signum == 0);
278e66f31c5Sopenharmony_ci
279e66f31c5Sopenharmony_ci  handle->flags |= UV_HANDLE_CLOSED;
280e66f31c5Sopenharmony_ci
281e66f31c5Sopenharmony_ci  uv__handle_close(handle);
282e66f31c5Sopenharmony_ci}
283