Coverage Report

Created: 2026-06-02 06:36

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/Zend/zend_signal.c
Line
Count
Source
1
/*
2
  +----------------------------------------------------------------------+
3
  | Zend Signal Handling                                                 |
4
  +----------------------------------------------------------------------+
5
  | Copyright © The PHP Group and Contributors.                          |
6
  +----------------------------------------------------------------------+
7
  | This source file is subject to the Modified BSD License that is      |
8
  | bundled with this package in the file LICENSE, and is available      |
9
  | through the World Wide Web at <https://www.php.net/license/>.        |
10
  |                                                                      |
11
  | SPDX-License-Identifier: BSD-3-Clause                                |
12
  +----------------------------------------------------------------------+
13
  | Authors: Lucas Nealan <lucas@php.net>                                |
14
  |          Arnaud Le Blanc <lbarnaud@php.net>                          |
15
  +----------------------------------------------------------------------+
16
17
   This software was contributed to PHP by Facebook Inc. in 2008.
18
19
   Future revisions and derivatives of this source code must acknowledge
20
   Facebook Inc. as the original contributor of this module by leaving
21
   this note intact in the source code.
22
23
   All other licensing and usage conditions are those of the PHP Group and
24
   Contributors.
25
 */
26
27
#ifndef _GNU_SOURCE
28
# define _GNU_SOURCE
29
#endif
30
#include <string.h>
31
32
#include "zend.h"
33
#include "zend_globals.h"
34
#include <signal.h>
35
36
#ifdef HAVE_UNISTD_H
37
#include <unistd.h>
38
#endif
39
40
#ifdef ZEND_SIGNALS
41
42
#include "zend_signal.h"
43
44
#ifdef ZTS
45
ZEND_API int zend_signal_globals_id;
46
ZEND_API size_t zend_signal_globals_offset;
47
#else
48
ZEND_API zend_signal_globals_t zend_signal_globals;
49
#endif /* not ZTS */
50
51
#define SIGNAL_BEGIN_CRITICAL() \
52
0
  sigset_t oldmask; \
53
0
  zend_sigprocmask(SIG_BLOCK, &global_sigmask, &oldmask);
54
#define SIGNAL_END_CRITICAL() \
55
0
  zend_sigprocmask(SIG_SETMASK, &oldmask, NULL);
56
57
#ifdef ZTS
58
# define zend_sigprocmask(signo, set, oldset) tsrm_sigmask((signo), (set), (oldset))
59
#else
60
1.99k
# define zend_sigprocmask(signo, set, oldset) sigprocmask((signo), (set), (oldset))
61
#endif
62
63
static void zend_signal_handler(int signo, siginfo_t *siginfo, void *context);
64
static zend_result zend_signal_register(int signo, void (*handler)(int, siginfo_t*, void*));
65
66
#if defined(__CYGWIN__) || defined(__PASE__)
67
/* Matches zend_execute_API.c; these platforms don't support ITIMER_PROF. */
68
#define TIMEOUT_SIG SIGALRM
69
#else
70
#define TIMEOUT_SIG SIGPROF
71
#endif
72
73
static const int zend_sigs[] = { TIMEOUT_SIG, SIGHUP, SIGINT, SIGQUIT, SIGTERM, SIGUSR1, SIGUSR2 };
74
75
1.99k
#define SA_FLAGS_MASK ~(SA_NODEFER | SA_RESETHAND)
76
77
/* True globals, written only at process startup */
78
static zend_signal_entry_t global_orig_handlers[NSIG];
79
static sigset_t            global_sigmask;
80
81
/* {{{ zend_signal_handler_defer
82
 *  Blocks signals if in critical section */
83
static void zend_signal_handler_defer(int signo, siginfo_t *siginfo, void *context)
84
0
{
85
0
  int errno_save = errno;
86
0
  zend_signal_queue_t *queue, *qtmp;
87
88
#ifdef ZTS
89
  /* A signal could hit after TSRM shutdown, in this case globals are already freed.
90
   * Or it could be delivered to a thread that didn't execute PHP yet.
91
   * In the latter case we act as if SIGG(active) is false. */
92
  if (tsrm_is_shutdown() || !tsrm_is_managed_thread()) {
93
    /* Forward to default handler handler */
94
    zend_signal_handler(signo, siginfo, context);
95
    return;
96
  }
97
#endif
98
99
0
  if (EXPECTED(SIGG(active))) {
100
0
    if (UNEXPECTED(SIGG(depth) == 0)) { /* try to handle signal */
101
0
      if (UNEXPECTED(SIGG(blocked))) {
102
0
        SIGG(blocked) = 0;
103
0
      }
104
0
      if (EXPECTED(SIGG(running) == 0)) {
105
0
        SIGG(running) = 1;
106
0
        zend_signal_handler(signo, siginfo, context);
107
108
0
        queue = SIGG(phead);
109
0
        SIGG(phead) = NULL;
110
111
0
        while (queue) {
112
0
          zend_signal_handler(queue->zend_signal.signo, queue->zend_signal.siginfo, queue->zend_signal.context);
113
0
          qtmp = queue->next;
114
0
          queue->next = SIGG(pavail);
115
0
          queue->zend_signal.signo = 0;
116
0
          SIGG(pavail) = queue;
117
0
          queue = qtmp;
118
0
        }
119
0
        SIGG(running) = 0;
120
0
      }
121
0
    } else { /* delay signal handling */
122
0
      SIGG(blocked) = 1; /* signal is blocked */
123
124
0
      if ((queue = SIGG(pavail))) { /* if none available it's simply forgotten */
125
0
        SIGG(pavail) = queue->next;
126
0
        queue->zend_signal.signo = signo;
127
0
        queue->zend_signal.siginfo = siginfo;
128
0
        queue->zend_signal.context = context;
129
0
        queue->next = NULL;
130
131
0
        if (SIGG(phead) && SIGG(ptail)) {
132
0
          SIGG(ptail)->next = queue;
133
0
        } else {
134
0
          SIGG(phead) = queue;
135
0
        }
136
0
        SIGG(ptail) = queue;
137
0
      }
138
0
#if ZEND_DEBUG
139
0
      else { /* this may not be safe to do, but could work and be useful */
140
0
        zend_output_debug_string(0, "zend_signal: not enough queue storage, lost signal (%d)", signo);
141
0
      }
142
0
#endif
143
0
    }
144
0
  } else {
145
    /* need to just run handler if we're inactive and getting a signal */
146
0
    zend_signal_handler(signo, siginfo, context);
147
0
  }
148
149
0
  errno = errno_save;
150
0
} /* }}} */
151
152
/* {{{ zend_signal_handler_unblock
153
 * Handle deferred signal from HANDLE_UNBLOCK_ALARMS */
154
ZEND_API void zend_signal_handler_unblock(void)
155
0
{
156
0
  zend_signal_queue_t *queue;
157
0
  zend_signal_t zend_signal;
158
159
0
  if (EXPECTED(SIGG(active))) {
160
0
    SIGNAL_BEGIN_CRITICAL(); /* procmask to protect handler_defer as if it were called by the kernel */
161
0
    queue = SIGG(phead);
162
0
    SIGG(phead) = queue->next;
163
0
    zend_signal = queue->zend_signal;
164
0
    queue->next = SIGG(pavail);
165
0
    queue->zend_signal.signo = 0;
166
0
    SIGG(pavail) = queue;
167
168
0
    zend_signal_handler_defer(zend_signal.signo, zend_signal.siginfo, zend_signal.context);
169
0
    SIGNAL_END_CRITICAL();
170
0
  }
171
0
}
172
/* }}} */
173
174
/* {{{ zend_signal_handler
175
 *  Call the previously registered handler for a signal
176
 */
177
static void zend_signal_handler(int signo, siginfo_t *siginfo, void *context)
178
0
{
179
0
  int errno_save = errno;
180
0
  struct sigaction sa;
181
0
  sigset_t sigset;
182
0
  zend_signal_entry_t p_sig;
183
#ifdef ZTS
184
  if (tsrm_is_shutdown() || !tsrm_is_managed_thread()) {
185
    p_sig = global_orig_handlers[signo-1];
186
  } else
187
#endif
188
0
  p_sig = SIGG(handlers)[signo-1];
189
190
0
  if (p_sig.handler == SIG_DFL) { /* raise default handler */
191
0
    if (sigaction(signo, NULL, &sa) == 0) {
192
0
      sa.sa_handler = SIG_DFL;
193
0
      sigemptyset(&sa.sa_mask);
194
195
0
      sigemptyset(&sigset);
196
0
      sigaddset(&sigset, signo);
197
198
0
      if (sigaction(signo, &sa, NULL) == 0) {
199
        /* throw away any blocked signals */
200
0
        zend_sigprocmask(SIG_UNBLOCK, &sigset, NULL);
201
#ifdef ZTS
202
# define RAISE_ERROR "raise() failed\n"
203
        if (raise(signo) != 0) {
204
          /* On some systems raise() fails with errno 3: No such process */
205
          kill(getpid(), signo);
206
        }
207
#else
208
0
        kill(getpid(), signo);
209
0
#endif
210
0
      }
211
0
    }
212
0
  } else if (p_sig.handler != SIG_IGN) {
213
0
    if (p_sig.flags & SA_SIGINFO) {
214
0
      if (p_sig.flags & SA_RESETHAND) {
215
0
        SIGG(handlers)[signo-1].flags   = 0;
216
0
        SIGG(handlers)[signo-1].handler = SIG_DFL;
217
0
      }
218
0
      (*(void (*)(int, siginfo_t*, void*))p_sig.handler)(signo, siginfo, context);
219
0
    } else {
220
0
      (*(void (*)(int))p_sig.handler)(signo);
221
0
    }
222
0
  }
223
224
0
  errno = errno_save;
225
0
} /* }}} */
226
227
/* {{{ zend_sigaction
228
 *  Register a signal handler that will be deferred in critical sections */
229
ZEND_API void zend_sigaction(int signo, const struct sigaction *act, struct sigaction *oldact)
230
1.99k
{
231
1.99k
  struct sigaction sa;
232
1.99k
  sigset_t sigset;
233
234
1.99k
  if (oldact != NULL) {
235
0
    oldact->sa_flags   = SIGG(handlers)[signo-1].flags;
236
0
    oldact->sa_handler = (void *) SIGG(handlers)[signo-1].handler;
237
0
    oldact->sa_mask    = global_sigmask;
238
0
  }
239
1.99k
  if (act != NULL) {
240
1.99k
    SIGG(handlers)[signo-1].flags = act->sa_flags;
241
1.99k
    if (act->sa_flags & SA_SIGINFO) {
242
0
      SIGG(handlers)[signo-1].handler = (void *) act->sa_sigaction;
243
1.99k
    } else {
244
1.99k
      SIGG(handlers)[signo-1].handler = (void *) act->sa_handler;
245
1.99k
    }
246
247
1.99k
    memset(&sa, 0, sizeof(sa));
248
1.99k
    if (SIGG(handlers)[signo-1].handler == (void *) SIG_IGN) {
249
0
      sa.sa_sigaction = (void *) SIG_IGN;
250
1.99k
    } else {
251
1.99k
      sa.sa_flags     = SA_ONSTACK | SA_SIGINFO | (act->sa_flags & SA_FLAGS_MASK);
252
1.99k
      sa.sa_sigaction = zend_signal_handler_defer;
253
1.99k
      sa.sa_mask      = global_sigmask;
254
1.99k
    }
255
256
1.99k
    if (sigaction(signo, &sa, NULL) < 0) {
257
0
      zend_error_noreturn(E_ERROR, "Error installing signal handler for %d", signo);
258
0
    }
259
260
    /* unsure this signal is not blocked */
261
1.99k
    sigemptyset(&sigset);
262
1.99k
    sigaddset(&sigset, signo);
263
1.99k
    zend_sigprocmask(SIG_UNBLOCK, &sigset, NULL);
264
1.99k
  }
265
1.99k
}
266
/* }}} */
267
268
/* {{{ zend_signal
269
 *  Register a signal handler that will be deferred in critical sections */
270
ZEND_API void zend_signal(int signo, void (*handler)(int))
271
1.99k
{
272
1.99k
  struct sigaction sa;
273
274
1.99k
  memset(&sa, 0, sizeof(sa));
275
1.99k
  sa.sa_flags   = 0;
276
1.99k
  sa.sa_handler = handler;
277
1.99k
  sa.sa_mask    = global_sigmask;
278
279
1.99k
  zend_sigaction(signo, &sa, NULL);
280
1.99k
}
281
/* }}} */
282
283
/* {{{ zend_signal_register
284
 *  Set a handler for a signal we want to defer.
285
 *  Previously set handler must have been saved before.
286
 */
287
static zend_result zend_signal_register(int signo, void (*handler)(int, siginfo_t*, void*))
288
0
{
289
0
  struct sigaction sa;
290
291
0
  if (sigaction(signo, NULL, &sa) == 0) {
292
0
    if ((sa.sa_flags & SA_SIGINFO) && sa.sa_sigaction == handler) {
293
0
      return FAILURE;
294
0
    }
295
296
0
    SIGG(handlers)[signo-1].flags = sa.sa_flags;
297
0
    if (sa.sa_flags & SA_SIGINFO) {
298
0
      SIGG(handlers)[signo-1].handler = (void *)sa.sa_sigaction;
299
0
    } else {
300
0
      SIGG(handlers)[signo-1].handler = (void *)sa.sa_handler;
301
0
    }
302
303
0
    sa.sa_flags     = SA_SIGINFO; /* we'll use a siginfo handler */
304
0
    sa.sa_sigaction = handler;
305
0
    sa.sa_mask      = global_sigmask;
306
307
0
    if (sigaction(signo, &sa, NULL) < 0) {
308
0
      zend_error_noreturn(E_ERROR, "Error installing signal handler for %d", signo);
309
0
    }
310
311
0
    return SUCCESS;
312
0
  }
313
0
  return FAILURE;
314
0
} /* }}} */
315
316
/* {{{ zend_signal_activate
317
 *  Install our signal handlers, per request */
318
void zend_signal_activate(void)
319
1.99k
{
320
1.99k
  size_t x;
321
322
1.99k
  memcpy(&SIGG(handlers), &global_orig_handlers, sizeof(global_orig_handlers));
323
324
1.99k
  if (SIGG(reset)) {
325
0
    for (x = 0; x < sizeof(zend_sigs) / sizeof(*zend_sigs); x++) {
326
0
      zend_signal_register(zend_sigs[x], zend_signal_handler_defer);
327
0
    }
328
0
  }
329
330
1.99k
  SIGG(active) = 1;
331
1.99k
  SIGG(depth)  = 0;
332
1.99k
} /* }}} */
333
334
/* {{{ zend_signal_deactivate */
335
void zend_signal_deactivate(void)
336
1.99k
{
337
1.99k
  if (SIGG(check)) {
338
0
    size_t x;
339
0
    struct sigaction sa;
340
341
0
    if (SIGG(depth) != 0) {
342
0
      zend_error(E_CORE_WARNING, "zend_signal: shutdown with non-zero blocking depth (%d)", SIGG(depth));
343
0
    }
344
345
    /* did anyone steal our installed handler */
346
0
    for (x = 0; x < sizeof(zend_sigs) / sizeof(*zend_sigs); x++) {
347
0
      sigaction(zend_sigs[x], NULL, &sa);
348
0
      if (sa.sa_sigaction != zend_signal_handler_defer &&
349
0
          sa.sa_sigaction != (void *) SIG_IGN) {
350
0
        zend_error(E_CORE_WARNING, "zend_signal: handler was replaced for signal (%d) after startup", zend_sigs[x]);
351
0
      }
352
0
    }
353
0
  }
354
355
  /* After active=0 is set, signal handlers will be called directly and other
356
   * state that is reset below will not be accessed. */
357
1.99k
  *((volatile int *) &SIGG(active)) = 0;
358
359
1.99k
  SIGG(running) = 0;
360
1.99k
  SIGG(blocked) = 0;
361
1.99k
  SIGG(depth) = 0;
362
363
  /* If there are any queued signals because of a missed unblock, drop them. */
364
1.99k
  if (SIGG(phead) && SIGG(ptail)) {
365
0
    SIGG(ptail)->next = SIGG(pavail);
366
0
    SIGG(pavail) = SIGG(phead);
367
0
    SIGG(phead) = NULL;
368
0
    SIGG(ptail) = NULL;
369
0
  }
370
1.99k
}
371
/* }}} */
372
373
static void zend_signal_globals_ctor(zend_signal_globals_t *zend_signal_globals) /* {{{ */
374
0
{
375
0
  size_t x;
376
377
0
  memset(zend_signal_globals, 0, sizeof(*zend_signal_globals));
378
0
  zend_signal_globals->reset = 1;
379
380
0
  for (x = 0; x < sizeof(zend_signal_globals->pstorage) / sizeof(*zend_signal_globals->pstorage); ++x) {
381
0
    zend_signal_queue_t *queue = &zend_signal_globals->pstorage[x];
382
0
    queue->zend_signal.signo = 0;
383
0
    queue->next = zend_signal_globals->pavail;
384
0
    zend_signal_globals->pavail = queue;
385
0
  }
386
0
}
387
/* }}} */
388
389
void zend_signal_init(void) /* {{{ */
390
0
{
391
0
  int signo;
392
0
  struct sigaction sa;
393
394
  /* Save previously registered signal handlers into orig_handlers */
395
0
  memset(&global_orig_handlers, 0, sizeof(global_orig_handlers));
396
0
  for (signo = 1; signo < NSIG; ++signo) {
397
0
    if (sigaction(signo, NULL, &sa) == 0) {
398
0
      global_orig_handlers[signo-1].flags = sa.sa_flags;
399
0
      if (sa.sa_flags & SA_SIGINFO) {
400
0
        global_orig_handlers[signo-1].handler = (void *) sa.sa_sigaction;
401
0
      } else {
402
0
        global_orig_handlers[signo-1].handler = (void *) sa.sa_handler;
403
0
      }
404
0
    }
405
0
  }
406
0
}
407
/* }}} */
408
409
/* {{{ zend_signal_startup
410
 * alloc zend signal globals */
411
ZEND_API void zend_signal_startup(void)
412
0
{
413
414
#ifdef ZTS
415
  ts_allocate_fast_id(&zend_signal_globals_id, &zend_signal_globals_offset, sizeof(zend_signal_globals_t), (ts_allocate_ctor) zend_signal_globals_ctor, NULL);
416
#else
417
0
  zend_signal_globals_ctor(&zend_signal_globals);
418
0
#endif
419
420
  /* Used to block signals during execution of signal handlers */
421
0
  sigfillset(&global_sigmask);
422
0
  sigdelset(&global_sigmask, SIGILL);
423
0
  sigdelset(&global_sigmask, SIGABRT);
424
0
  sigdelset(&global_sigmask, SIGFPE);
425
0
  sigdelset(&global_sigmask, SIGKILL);
426
0
  sigdelset(&global_sigmask, SIGSEGV);
427
0
  sigdelset(&global_sigmask, SIGCONT);
428
0
  sigdelset(&global_sigmask, SIGSTOP);
429
0
  sigdelset(&global_sigmask, SIGTSTP);
430
0
  sigdelset(&global_sigmask, SIGTTIN);
431
0
  sigdelset(&global_sigmask, SIGTTOU);
432
0
#ifdef SIGBUS
433
0
  sigdelset(&global_sigmask, SIGBUS);
434
0
#endif
435
0
#ifdef SIGSYS
436
0
  sigdelset(&global_sigmask, SIGSYS);
437
0
#endif
438
0
#ifdef SIGTRAP
439
0
  sigdelset(&global_sigmask, SIGTRAP);
440
0
#endif
441
442
0
  zend_signal_init();
443
0
}
444
/* }}} */
445
446
447
#endif /* ZEND_SIGNALS */