PageRenderTime 55ms CodeModel.GetById 14ms RepoModel.GetById 0ms app.codeStats 0ms

/Zend/zend_signal.c

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