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