kernel - Fix races created by a comedy of circumstansces (3)
[dragonfly.git] / contrib / less / signal.c
blob37a9793be49c03b78ed71d7c74269c4f6aa7d07b
1 /*
2 * Copyright (C) 1984-2015 Mark Nudelman
4 * You may distribute under the terms of either the GNU General Public
5 * License or the Less License, as specified in the README file.
7 * For more information, see the README file.
8 */
12 * Routines dealing with signals.
14 * A signal usually merely causes a bit to be set in the "signals" word.
15 * At some convenient time, the mainline code checks to see if any
16 * signals need processing by calling psignal().
17 * If we happen to be reading from a file [in iread()] at the time
18 * the signal is received, we call intread to interrupt the iread.
21 #include "less.h"
22 #include <signal.h>
25 * "sigs" contains bits indicating signals which need to be processed.
27 public int sigs;
29 extern int sc_width, sc_height;
30 extern int screen_trashed;
31 extern int lnloop;
32 extern int linenums;
33 extern int wscroll;
34 extern int reading;
35 extern int quit_on_intr;
36 extern long jump_sline_fraction;
39 * Interrupt signal handler.
41 /* ARGSUSED*/
42 static RETSIGTYPE
43 u_interrupt(type)
44 int type;
46 bell();
47 #if OS2
48 LSIGNAL(SIGINT, SIG_ACK);
49 #endif
50 LSIGNAL(SIGINT, u_interrupt);
51 sigs |= S_INTERRUPT;
52 #if MSDOS_COMPILER==DJGPPC
54 * If a keyboard has been hit, it must be Ctrl-C
55 * (as opposed to Ctrl-Break), so consume it.
56 * (Otherwise, Less will beep when it sees Ctrl-C from keyboard.)
58 if (kbhit())
59 getkey();
60 #endif
61 if (reading)
62 intread(); /* May longjmp */
65 #ifdef SIGTSTP
67 * "Stop" (^Z) signal handler.
69 /* ARGSUSED*/
70 static RETSIGTYPE
71 stop(type)
72 int type;
74 LSIGNAL(SIGTSTP, stop);
75 sigs |= S_STOP;
76 if (reading)
77 intread();
79 #endif
81 #ifdef SIGWINCH
83 * "Window" change handler
85 /* ARGSUSED*/
86 public RETSIGTYPE
87 winch(type)
88 int type;
90 LSIGNAL(SIGWINCH, winch);
91 sigs |= S_WINCH;
92 if (reading)
93 intread();
95 #else
96 #ifdef SIGWIND
98 * "Window" change handler
100 /* ARGSUSED*/
101 public RETSIGTYPE
102 winch(type)
103 int type;
105 LSIGNAL(SIGWIND, winch);
106 sigs |= S_WINCH;
107 if (reading)
108 intread();
110 #endif
111 #endif
113 #if MSDOS_COMPILER==WIN32C
115 * Handle CTRL-C and CTRL-BREAK keys.
117 #include "windows.h"
119 static BOOL WINAPI
120 wbreak_handler(dwCtrlType)
121 DWORD dwCtrlType;
123 switch (dwCtrlType)
125 case CTRL_C_EVENT:
126 case CTRL_BREAK_EVENT:
127 sigs |= S_INTERRUPT;
128 return (TRUE);
129 default:
130 break;
132 return (FALSE);
134 #endif
137 * Set up the signal handlers.
139 public void
140 init_signals(on)
141 int on;
143 if (on)
146 * Set signal handlers.
148 (void) LSIGNAL(SIGINT, u_interrupt);
149 #if MSDOS_COMPILER==WIN32C
150 SetConsoleCtrlHandler(wbreak_handler, TRUE);
151 #endif
152 #ifdef SIGTSTP
153 (void) LSIGNAL(SIGTSTP, stop);
154 #endif
155 #ifdef SIGWINCH
156 (void) LSIGNAL(SIGWINCH, winch);
157 #endif
158 #ifdef SIGWIND
159 (void) LSIGNAL(SIGWIND, winch);
160 #endif
161 #ifdef SIGQUIT
162 (void) LSIGNAL(SIGQUIT, SIG_IGN);
163 #endif
164 } else
167 * Restore signals to defaults.
169 (void) LSIGNAL(SIGINT, SIG_DFL);
170 #if MSDOS_COMPILER==WIN32C
171 SetConsoleCtrlHandler(wbreak_handler, FALSE);
172 #endif
173 #ifdef SIGTSTP
174 (void) LSIGNAL(SIGTSTP, SIG_DFL);
175 #endif
176 #ifdef SIGWINCH
177 (void) LSIGNAL(SIGWINCH, SIG_IGN);
178 #endif
179 #ifdef SIGWIND
180 (void) LSIGNAL(SIGWIND, SIG_IGN);
181 #endif
182 #ifdef SIGQUIT
183 (void) LSIGNAL(SIGQUIT, SIG_DFL);
184 #endif
189 * Process any signals we have received.
190 * A received signal cause a bit to be set in "sigs".
192 public void
193 psignals()
195 register int tsignals;
197 if ((tsignals = sigs) == 0)
198 return;
199 sigs = 0;
201 #ifdef SIGTSTP
202 if (tsignals & S_STOP)
205 * Clean up the terminal.
207 #ifdef SIGTTOU
208 LSIGNAL(SIGTTOU, SIG_IGN);
209 #endif
210 clear_bot();
211 deinit();
212 flush();
213 raw_mode(0);
214 #ifdef SIGTTOU
215 LSIGNAL(SIGTTOU, SIG_DFL);
216 #endif
217 LSIGNAL(SIGTSTP, SIG_DFL);
218 kill(getpid(), SIGTSTP);
220 * ... Bye bye. ...
221 * Hopefully we'll be back later and resume here...
222 * Reset the terminal and arrange to repaint the
223 * screen when we get back to the main command loop.
225 LSIGNAL(SIGTSTP, stop);
226 raw_mode(1);
227 init();
228 screen_trashed = 1;
229 tsignals |= S_WINCH;
231 #endif
232 #ifdef S_WINCH
233 if (tsignals & S_WINCH)
235 int old_width, old_height;
237 * Re-execute scrsize() to read the new window size.
239 old_width = sc_width;
240 old_height = sc_height;
241 get_term();
242 if (sc_width != old_width || sc_height != old_height)
244 wscroll = (sc_height + 1) / 2;
245 calc_jump_sline();
246 calc_shift_count();
247 screen_trashed = 1;
250 #endif
251 if (tsignals & S_INTERRUPT)
253 if (quit_on_intr)
254 quit(QUIT_INTERRUPT);