blob: 0eb0dde88056a26e5de429ffe21d5ac0cb9cd0ad [file] [log] [blame]
Scott James Remnant50748842006-05-16 21:02:31 +01001/* upstart
2 *
Scott James Remnantb6270dd2006-07-13 02:16:38 +01003 * Copyright © 2006 Canonical Ltd.
4 * Author: Scott James Remnant <scott@ubuntu.com>.
Scott James Remnant50748842006-05-16 21:02:31 +01005 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21#ifdef HAVE_CONFIG_H
22# include <config.h>
23#endif /* HAVE_CONFIG_H */
24
25
Scott James Remnantf43bdf32006-08-27 18:20:29 +010026#include <linux/kd.h>
27
28#include <sys/types.h>
29#include <sys/wait.h>
30#include <sys/ioctl.h>
31#include <sys/reboot.h>
32#include <sys/resource.h>
33
34#include <signal.h>
Scott James Remnant94d00982006-08-25 15:38:22 +020035#include <stdlib.h>
Scott James Remnant12a330f2006-08-24 02:19:09 +020036#include <syslog.h>
Scott James Remnant027dd7b2006-08-21 09:01:25 +020037#include <unistd.h>
38
Scott James Remnant77e8db32006-08-21 08:47:50 +020039#include <nih/macros.h>
40#include <nih/alloc.h>
41#include <nih/list.h>
42#include <nih/timer.h>
43#include <nih/signal.h>
44#include <nih/child.h>
45#include <nih/main.h>
46#include <nih/logging.h>
47
48#include "process.h"
49#include "job.h"
50#include "event.h"
51#include "control.h"
52#include "cfgfile.h"
Scott James Remnant50748842006-05-16 21:02:31 +010053
54
Scott James Remnantf43bdf32006-08-27 18:20:29 +010055/* Prototypes for static functions */
56static void segv_handler (int signum);
57static void cad_handler (void *data, NihSignal *signal);
58static void kbd_handler (void *data, NihSignal *signal);
59
60
Scott James Remnant50748842006-05-16 21:02:31 +010061int
62main (int argc,
63 char *argv[])
64{
Scott James Remnant77e8db32006-08-21 08:47:50 +020065 int ret, i;
Scott James Remnant50748842006-05-16 21:02:31 +010066
Scott James Remnant77e8db32006-08-21 08:47:50 +020067 nih_main_init (argv[0]);
68
Scott James Remnant12a330f2006-08-24 02:19:09 +020069 openlog (program_name, LOG_CONS, LOG_DAEMON);
70
Scott James Remnant77e8db32006-08-21 08:47:50 +020071 nih_log_set_priority (NIH_LOG_DEBUG);
Scott James Remnant12a330f2006-08-24 02:19:09 +020072 nih_log_set_logger (nih_logger_syslog);
Scott James Remnant77e8db32006-08-21 08:47:50 +020073
Scott James Remnant77e8db32006-08-21 08:47:50 +020074 /* Close any file descriptors we inherited, and open the console
75 * device instead
76 */
77 for (i = 0; i < 3; i++)
78 close (i);
79
80 process_setup_console (CONSOLE_OUTPUT);
81
Scott James Remnantf43bdf32006-08-27 18:20:29 +010082
83 /* Reset the signal state and install the signal handler for those
84 * signals we actually want to catch; this also sets those that
85 * can be sent to us, because we're special
86 */
87 nih_signal_reset ();
88 nih_signal_set_handler (SIGALRM, nih_signal_handler);
89 nih_signal_set_handler (SIGHUP, nih_signal_handler);
90 nih_signal_set_handler (SIGINT, nih_signal_handler);
91 nih_signal_set_handler (SIGWINCH, nih_signal_handler);
92 nih_signal_set_handler (SIGSEGV, segv_handler);
93
94 /* Ask the kernel to send us SIGINT when control-alt-delete is
95 * pressed; generate an event with the same name.
96 */
97 reboot (RB_DISABLE_CAD);
98 nih_signal_add_callback (NULL, SIGINT, cad_handler, NULL);
99
100 /* Ask the kernel to send us SIGWINCH when alt-uparrow is pressed;
101 * generate a kbdrequest event.
102 */
103 ioctl (0, KDSIGACCEPT, SIGWINCH);
104 nih_signal_add_callback (NULL, SIGWINCH, kbd_handler, NULL);
105
106
107 /* Reap all children that die */
108 nih_child_add_watch (NULL, -1, job_child_reaper, NULL);
109
110 /* Process the event queue every time through the main loop */
111 nih_main_loop_add_func (NULL, event_queue_run, NULL);
112
Scott James Remnant94d00982006-08-25 15:38:22 +0200113
Scott James Remnant77e8db32006-08-21 08:47:50 +0200114 /* Become session and process group leader (should be already,
115 * but you never know what initramfs did
116 */
117 setsid ();
118
119 /* Open control socket */
120 control_open ();
121
Scott James Remnantceaaf4d2006-08-24 01:40:10 +0200122 /* Read configuration */
123 cfg_watch_dir (NULL, CFG_DIR, NULL);
124
Scott James Remnantf43bdf32006-08-27 18:20:29 +0100125 /* Set the PATH environment variable */
126 setenv ("PATH", PATH, TRUE);
127
Scott James Remnant77e8db32006-08-21 08:47:50 +0200128
129 /* Generate and run the startup event */
130 event_queue_edge ("startup");
131 event_queue_run (NULL, NULL);
132
133 /* Go! */
134 ret = nih_main_loop ();
135
136 return ret;
Scott James Remnant50748842006-05-16 21:02:31 +0100137}
Scott James Remnantf43bdf32006-08-27 18:20:29 +0100138
139
140/**
141 * segv_handler:
142 * @signum:
143 *
144 * Handle receiving the SEGV signal, usually caused by one of our own
145 * mistakes. We deal with it by dumping core in a child process and
146 * just carrying on in the parent.
147 **/
148static void
149segv_handler (int signum)
150{
151 pid_t pid;
152
153 pid = fork ();
154 if (pid == 0) {
155 struct sigaction act;
156 struct rlimit limit;
157 sigset_t mask;
158
159 /* Mask out all signals */
160 sigfillset (&mask);
161 sigprocmask (SIG_SETMASK, &mask, NULL);
162
163 /* Set the SEGV handler to the default so core is dumped */
164 act.sa_handler = SIG_DFL;
165 act.sa_flags = 0;
166 sigemptyset (&act.sa_mask);
167 sigaction (SIGSEGV, &act, NULL);
168
169 /* Dump in the root directory */
170 chdir ("/");
171
172 /* Don't limit the core dump size */
173 limit.rlim_cur = RLIM_INFINITY;
174 limit.rlim_max = RLIM_INFINITY;
175 setrlimit (RLIMIT_CORE, &limit);
176
177 /* Raise the signal */
178 raise (SIGSEGV);
179
180 /* Unmask so that we receive it */
181 sigdelset (&mask, SIGSEGV);
182 sigprocmask (SIG_SETMASK, &mask, NULL);
183
184 /* Wait for death */
185 pause ();
186 exit (0);
187 } else if (pid > 0) {
188 /* Wait for the core to be generated */
189 waitpid (pid, NULL, 0);
190
191 nih_error (_("Caught segmentation fault, core dumped"));
192 } else {
193 nih_error (_("Caught segmentation fault, unable to dump core"));
194 }
195}
196
197/**
198 * cad_handler:
199 * @data: unused,
200 * @signal: signal that called this handler.
201 *
202 * Handle having recieved the SIGINT signal, sent to us when somebody
203 * presses Ctrl-Alt-Delete on the console. We just generate a
204 * control-alt-delete event.
205 **/
206static void
207cad_handler (void *data,
208 NihSignal *signal)
209{
210 event_queue_edge ("control-alt-delete");
211}
212
213/**
214 * kbd_handler:
215 * @data: unused,
216 * @signal: signal that called this handler.
217 *
218 * Handle having recieved the SIGWINCH signal, sent to us when somebody
219 * presses Alt-UpArrow on the console. We just generate a
220 * kbdrequest event.
221 **/
222static void
223kbd_handler (void *data,
224 NihSignal *signal)
225{
226 event_queue_edge ("kbdrequest");
227}