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