blob: a5bb0d438c89525db6d2f19b52f52765ee6c144f [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 Remnantf4970812006-08-27 18:27:19 +010082 /* Check we're root */
83 if (getuid ()) {
84 nih_error (_("Need to be root"));
85 exit (1);
86 }
87
88 /* Check we're process #1 */
89 if (getpid () > 1) {
90 nih_error (_("Not being executed as init"));
91 exit (1);
92 }
93
Scott James Remnantf43bdf32006-08-27 18:20:29 +010094
95 /* Reset the signal state and install the signal handler for those
96 * signals we actually want to catch; this also sets those that
97 * can be sent to us, because we're special
98 */
99 nih_signal_reset ();
100 nih_signal_set_handler (SIGALRM, nih_signal_handler);
101 nih_signal_set_handler (SIGHUP, nih_signal_handler);
102 nih_signal_set_handler (SIGINT, nih_signal_handler);
103 nih_signal_set_handler (SIGWINCH, nih_signal_handler);
104 nih_signal_set_handler (SIGSEGV, segv_handler);
105
106 /* Ask the kernel to send us SIGINT when control-alt-delete is
107 * pressed; generate an event with the same name.
108 */
109 reboot (RB_DISABLE_CAD);
110 nih_signal_add_callback (NULL, SIGINT, cad_handler, NULL);
111
112 /* Ask the kernel to send us SIGWINCH when alt-uparrow is pressed;
113 * generate a kbdrequest event.
114 */
115 ioctl (0, KDSIGACCEPT, SIGWINCH);
116 nih_signal_add_callback (NULL, SIGWINCH, kbd_handler, NULL);
117
118
119 /* Reap all children that die */
120 nih_child_add_watch (NULL, -1, job_child_reaper, NULL);
121
Scott James Remnantdc8877d2006-08-29 16:56:48 +0100122 /* Process the event queue and check the jobs for idleness
123 * every time through the main loop */
Scott James Remnant0c7e72a2006-08-27 22:22:53 +0100124 nih_main_loop_add_func (NULL, (NihMainLoopCb)event_queue_run, NULL);
Scott James Remnantdc8877d2006-08-29 16:56:48 +0100125 nih_main_loop_add_func (NULL, (NihMainLoopCb)job_detect_idle, NULL);
Scott James Remnantf43bdf32006-08-27 18:20:29 +0100126
Scott James Remnant94d00982006-08-25 15:38:22 +0200127
Scott James Remnant77e8db32006-08-21 08:47:50 +0200128 /* Become session and process group leader (should be already,
129 * but you never know what initramfs did
130 */
131 setsid ();
132
133 /* Open control socket */
134 control_open ();
135
Scott James Remnantceaaf4d2006-08-24 01:40:10 +0200136 /* Read configuration */
137 cfg_watch_dir (NULL, CFG_DIR, NULL);
138
Scott James Remnantf43bdf32006-08-27 18:20:29 +0100139 /* Set the PATH environment variable */
140 setenv ("PATH", PATH, TRUE);
141
Scott James Remnant77e8db32006-08-21 08:47:50 +0200142
143 /* Generate and run the startup event */
144 event_queue_edge ("startup");
Scott James Remnant0c7e72a2006-08-27 22:22:53 +0100145 event_queue_run ();
Scott James Remnant77e8db32006-08-21 08:47:50 +0200146
147 /* Go! */
148 ret = nih_main_loop ();
149
150 return ret;
Scott James Remnant50748842006-05-16 21:02:31 +0100151}
Scott James Remnantf43bdf32006-08-27 18:20:29 +0100152
153
154/**
155 * segv_handler:
156 * @signum:
157 *
158 * Handle receiving the SEGV signal, usually caused by one of our own
159 * mistakes. We deal with it by dumping core in a child process and
160 * just carrying on in the parent.
161 **/
162static void
163segv_handler (int signum)
164{
165 pid_t pid;
166
167 pid = fork ();
168 if (pid == 0) {
169 struct sigaction act;
170 struct rlimit limit;
171 sigset_t mask;
172
173 /* Mask out all signals */
174 sigfillset (&mask);
175 sigprocmask (SIG_SETMASK, &mask, NULL);
176
177 /* Set the SEGV handler to the default so core is dumped */
178 act.sa_handler = SIG_DFL;
179 act.sa_flags = 0;
180 sigemptyset (&act.sa_mask);
181 sigaction (SIGSEGV, &act, NULL);
182
183 /* Dump in the root directory */
184 chdir ("/");
185
186 /* Don't limit the core dump size */
187 limit.rlim_cur = RLIM_INFINITY;
188 limit.rlim_max = RLIM_INFINITY;
189 setrlimit (RLIMIT_CORE, &limit);
190
191 /* Raise the signal */
192 raise (SIGSEGV);
193
194 /* Unmask so that we receive it */
195 sigdelset (&mask, SIGSEGV);
196 sigprocmask (SIG_SETMASK, &mask, NULL);
197
198 /* Wait for death */
199 pause ();
200 exit (0);
201 } else if (pid > 0) {
202 /* Wait for the core to be generated */
203 waitpid (pid, NULL, 0);
204
205 nih_error (_("Caught segmentation fault, core dumped"));
206 } else {
207 nih_error (_("Caught segmentation fault, unable to dump core"));
208 }
209}
210
211/**
212 * cad_handler:
213 * @data: unused,
214 * @signal: signal that called this handler.
215 *
216 * Handle having recieved the SIGINT signal, sent to us when somebody
217 * presses Ctrl-Alt-Delete on the console. We just generate a
218 * control-alt-delete event.
219 **/
220static void
221cad_handler (void *data,
222 NihSignal *signal)
223{
224 event_queue_edge ("control-alt-delete");
225}
226
227/**
228 * kbd_handler:
229 * @data: unused,
230 * @signal: signal that called this handler.
231 *
232 * Handle having recieved the SIGWINCH signal, sent to us when somebody
233 * presses Alt-UpArrow on the console. We just generate a
234 * kbdrequest event.
235 **/
236static void
237kbd_handler (void *data,
238 NihSignal *signal)
239{
240 event_queue_edge ("kbdrequest");
241}