Annotation of freem/src/sighnd.c, revision 1.4
1.1 snw 1: /*
1.4 ! snw 2: * $Id: sighnd.c,v 1.3 2025/03/09 19:50:47 snw Exp $
1.1 snw 3: * FreeM signal handlers
4: *
5: *
1.2 snw 6: * Author: Serena Willis <snw@coherent-logic.com>
1.1 snw 7: * Copyright (C) 1998 MUG Deutschland
1.3 snw 8: * Copyright (C) 2020, 2025 Coherent Logic Development LLC
1.1 snw 9: *
10: *
11: * This file is part of FreeM.
12: *
13: * FreeM is free software: you can redistribute it and/or modify
14: * it under the terms of the GNU Affero Public License as published by
15: * the Free Software Foundation, either version 3 of the License, or
16: * (at your option) any later version.
17: *
18: * FreeM is distributed in the hope that it will be useful,
19: * but WITHOUT ANY WARRANTY; without even the implied warranty of
20: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21: * GNU Affero Public License for more details.
22: *
23: * You should have received a copy of the GNU Affero Public License
24: * along with FreeM. If not, see <https://www.gnu.org/licenses/>.
25: *
1.4 ! snw 26: * $Log: sighnd.c,v $
! 27: * Revision 1.3 2025/03/09 19:50:47 snw
! 28: * Second phase of REUSE compliance and header reformat
! 29: *
1.3 snw 30: *
31: * SPDX-FileCopyrightText: (C) 2025 Coherent Logic Development LLC
32: * SPDX-License-Identifier: AGPL-3.0-or-later
1.1 snw 33: **/
34:
35: #include <stddef.h>
36: #include <stdlib.h>
37: #include <setjmp.h>
38: #include <signal.h>
39: #include <unistd.h>
40: #include <stdio.h>
41:
42: #include <sys/types.h>
43: #include <sys/wait.h>
44:
1.4 ! snw 45: #if !defined(__APPLE__) && !defined(__gnu_hurd__) && !defined(EMSCRIPTEN)
! 46: # if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__AMIGA)
! 47: # include <termios.h>
! 48: # if !defined(__AMIGA)
! 49: # define TCGETA TIOCGETA
! 50: # define TCSETA TIOCSETA
! 51: # endif
! 52: # define termio termios
! 53: # else
! 54: # if !defined(MSDOS)
! 55: # include <termio.h>
! 56: # endif
! 57: # endif
! 58: #else
! 59: # include <termios.h>
! 60: #endif
! 61:
1.1 snw 62: #include <sys/ioctl.h>
63:
64: #include "mpsdef.h"
65: #include "transact.h"
66: #include "init.h"
67: #include "events.h"
68: #include "jobtab.h"
69: #include "shmmgr.h"
70:
1.4 ! snw 71:
! 72:
1.1 snw 73: void m_log (int, const char *);
74: int pending_signal_type = -1;
75:
76: void sig_attach(int sig, void *handler)
77: {
78: struct sigaction act;
79:
80: act.sa_handler = handler;
81: #if !defined(__AMIGA)
82: sigaction (sig, &act, NULL);
83: #else
84: //TODO: fill in for m68k-amigaos
85: #endif
86: }
87:
88:
89: void sig_init(void)
90: {
91: /* signals stuff */
92: sig_attach (SIGINT, &onintr); /* set_up INTERRUPT */
93: sig_attach (SIGQUIT, &onquit); /* set_up ZBREAK */
94: sig_attach (SIGTERM, &onkill); /* catch kill signal */
95:
96: #if !defined(__CYGWIN__) && !defined(MSDOS)
97: sig_attach (SIGIOT, &onbus); /* catch IOT error */
98: #endif/*__CYGWIN__*/
99:
100: #ifndef LINUX
101: sig_attach (SIGEMT, &onbus); /* catch EMT error */
102: #endif/*LINUX*/
103:
104: #if !defined(MSDOS)
105: sig_attach (SIGWINCH, &onwinch);
106: #endif
107:
108: sig_attach (SIGUSR1, &oncld); /* catch son dies signal */
109: sig_attach (SIGHUP, &onhup); /* catch hangup */
110:
111: sig_attach (SIGUSR2, &onipc); /* catch IPC signal */
112: sig_attach (SIGFPE, &onfpe); /* catch floating pt except */
113: }
114:
115:
116: #if !defined(MSDOS)
117: void onwinch (void)
118: {
119: struct winsize ws;
120:
121: /* restore handler */
122: sig_attach (SIGWINCH, &onwinch);
123:
124: ioctl (STDIN_FILENO, TIOCGWINSZ, &ws);
125:
126: n_lines = ws.ws_row;
127: n_columns = ws.ws_col;
128:
129: if (evt_async_enabled) {
130: pending_signal_type = SIGWINCH;
131: merr_raise (ASYNC);
132: }
133:
134: return;
135: }
136: #endif
137:
138: void onintr (void)
139: {
140: sig_attach (SIGINT, &onintr); /* restore handler */
141:
142: //printf ("\r\nSIGINT codptr = '%s'\r\n", codptr);
143:
144: if (first_process) {
145: job_request_stop (pid);
146: }
147: else {
148: if (shm_config->hdr->maintenance_mode == 1) {
149:
150: job_slot_t *s = job_get (pid);
151:
152:
153: if ((s->flags & JFLG_FMADM) != JFLG_FMADM) {
154:
155: fprintf (stderr, "\r\n***ENVIRONMENT IN MAINTENANCE MODE***\r\n");
156:
157: while (shm_config->hdr->maintenance_mode == 1) {
158: sleep (1);
159: }
160:
161: return;
162:
163: }
164:
165: }
166: }
167:
168:
169: if (breakon) {
170: merr_raise (INRPT);
171: inrpt_after_async = TRUE;
172: if (forsw) sigint_in_for = TRUE;
173: }
174: else {
175: zcc = TRUE;
176: }
177:
178: if (evt_async_enabled) {
179: pending_signal_type = SIGINT;
180: }
181:
182: return;
183: } /* end of onintr */
184:
185: void onfpe (void)
186: {
187: sig_attach (SIGFPE, &onfpe); /* restore handler */
188:
189: if (evt_async_enabled) {
190: pending_signal_type = SIGFPE;
191: }
192:
193: merr_raise (MXNUM);
194: return;
195: } /* end of onfpe */
196:
197: void onquit (void)
198: {
199:
200: if (run_daemon == TRUE) {
201: job_request_stop (pid);
202: }
203:
204: sig_attach (SIGQUIT, &onquit); /* restore handler */
205:
206: if (zbreakon && (merr () == OK)) ierr = OK - CTRLB;
207:
208: if (evt_async_enabled) {
209: pending_signal_type = SIGQUIT;
210: }
211:
212: return;
213: } /* end of onquit */
214:
215: void onkill (void)
216: {
217: int n = 0;
218:
219: if (run_daemon == TRUE) {
220: job_request_stop (pid);
221: }
222:
223: #if !defined(AMIGA68K)
224: if (direct_mode == TRUE) {
225: set_io (UNIX);
226: fprintf (stderr, "\n\nFreeM process %d caught SIGTERM\n", pid);
227: set_io (MUMPS);
228: }
229: #endif
230:
231:
232: sig_attach (SIGTERM, &onkill); /* restore handler */
233:
234: if (killerflag == FALSE) return; /* ignore that signal */
235:
236: /* if there exists an error trap, process as an error */
237: /* otherwise terminate the job */
238:
239: if (DSM2err) { /* DSM V.2 error trapping */
240:
241:
242:
243: if (ztrap[NESTLEVLS + 1][0] != EOL) {
244: merr_raise (KILLER);
245: return;
246: }
247:
248:
249: }
250: else {
251:
252:
253: while (n >= 0) {
254: if (ztrap[n--][0] != EOL) {
255: merr_raise (KILLER);
256: return;
257: }
258: }
259:
260:
261: }
262:
263: cleanup ();
264: if (father) kill (father, SIGUSR1); /* advertise death to parent */
265:
266: exit (1); /* terminate mumps */
267: } /* end of onkill() */
268:
269: void onhup (void)
270: {
271:
272: int n = nstx;
273:
274:
275: if (run_daemon == TRUE) {
276: fprintf (stderr, "freem: daemon received SIGHUP\r\n");
277: m_log (1, "freem: daemon received SIGHUP");
278:
279: sig_attach (SIGHUP, &onhup); /* restore handler */
280:
281: return;
282: }
283:
284: sig_attach (SIGHUP, &onhup); /* restore handler */
285:
286: if (huperflag == FALSE) return; /* ignore that signal */
287:
288: /* if there exists an error trap, process as an error */
289: /* otherwise terminate the job */
290:
291: if (DSM2err) { /* DSM V.2 error trapping */
292:
293: if (ztrap[NESTLEVLS + 1][0] != EOL) {
294: merr_raise (HUPER);
295: return;
296: }
297:
298:
299: }
300: else {
301:
302: while (n >= 0) {
303: if (ztrap[n--][0] != EOL) {
304: merr_raise (HUPER);
305: return;
306: }
307: }
308:
309: }
310:
311: cleanup ();
312:
313: if (father) kill (father, SIGUSR1); /* advertise death to parent */
314:
315: exit (1); /* terminate mumps */
316:
317: } /* end of onhup() */
318:
319: void onbus (void)
320: {
321: cleanup ();
322:
323: printf ("\012\015BUS ERROR, SEGMENTATION VIOLATION\012\015");
324:
325: if (father) kill (father, SIGUSR1); /* advertise death to parent */
326:
327: exit (1); /* terminate mumps */
328: } /* end of onbus() */
329:
330: /* under XENIX processes started with JOB hang around as zombies */
331: /* if they HALT before the parent process, unless the parent process */
332: /* waits for his child to terminate. to solve the problem, the child */
333: /* sends a signal to his parent to avoid an unattended funeral which */
334: /* inevitably would result in a living dead sucking up cpu time */
335: void oncld (void)
336: {
337: int status;
338:
339: /* ignore signal while as we're here */
340: sig_attach (SIGUSR1, SIG_IGN);
341:
342: wait (&status); /* wait for report from child */
343:
344: sig_attach (SIGUSR1, &oncld);/* restore handler */
345:
346: return;
347: } /* end of oncld() */
348:
349: void onipc (void)
350: {
351: /* restore handler */
352: sig_attach (SIGUSR2, &onipc);
353:
354: ipc_pending = 1;
355:
356: return;
357: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>