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