Patch from Jason Schoon to add optional SIGUSR1 support to dd.
[oweals/busybox.git] / init / init.c
index 058a47a83c4b5fc0b4fb86ef60d7bbfe70c4e2ec..22fb33dfd8d3a5b8d07215d072a69f97dfa3079e 100644 (file)
@@ -6,20 +6,7 @@
  * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
  * Adjusted by so many folks, it's impossible to keep track.
  *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- *
+ * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
  */
 
 /* Turn this on to disable all the dangerous
@@ -37,7 +24,7 @@
 #include <termios.h>
 #include <unistd.h>
 #include <limits.h>
-#include <sys/fcntl.h>
+#include <fcntl.h>
 #include <sys/ioctl.h>
 #include <sys/types.h>
 #include <sys/wait.h>
 #endif
 
 
+#ifdef CONFIG_SELINUX
+# include <selinux/selinux.h>
+#endif /* CONFIG_SELINUX */
+
+
 #define INIT_BUFFS_SIZE 256
 
 /* From <linux/vt.h> */
@@ -60,7 +52,7 @@ struct vt_stat {
        unsigned short v_signal;        /* signal to send */
        unsigned short v_state; /* vt bitmask */
 };
-static const int VT_GETSTATE = 0x5603; /* get global vt state info */
+enum { VT_GETSTATE = 0x5603 }; /* get global vt state info */
 
 /* From <linux/serial.h> */
 struct serial_struct {
@@ -158,22 +150,25 @@ static char console[CONSOLE_BUFF_SIZE] = _PATH_CONSOLE;
 static char *log_console = VC_5;
 #endif
 static sig_atomic_t got_cont = 0;
-static const int LOG = 0x1;
-static const int CONSOLE = 0x2;
+
+enum {
+       LOG = 0x1,
+       CONSOLE = 0x2,
 
 #if defined CONFIG_FEATURE_EXTRA_QUIET
-static const int MAYBE_CONSOLE = 0x0;
+       MAYBE_CONSOLE = 0x0,
 #else
-#define MAYBE_CONSOLE  CONSOLE
+       MAYBE_CONSOLE = CONSOLE,
 #endif
-#ifndef RB_HALT_SYSTEM
-static const int RB_HALT_SYSTEM = 0xcdef0123;
-static const int RB_ENABLE_CAD = 0x89abcdef;
-static const int RB_DISABLE_CAD = 0;
 
-#define RB_POWER_OFF    0x4321fedc
-static const int RB_AUTOBOOT = 0x01234567;
+#ifndef RB_HALT_SYSTEM
+       RB_HALT_SYSTEM = 0xcdef0123,
+       RB_ENABLE_CAD = 0x89abcdef,
+       RB_DISABLE_CAD = 0,
+       RB_POWER_OFF = 0x4321fedc,
+       RB_AUTOBOOT = 0x01234567,
 #endif
+};
 
 static const char * const environment[] = {
        "HOME=/",
@@ -198,7 +193,8 @@ static void loop_forever(void)
 /* Print a message to the specified device.
  * Device may be bitwise-or'd from LOG | CONSOLE */
 #ifndef DEBUG_INIT
-static inline void messageD(int device, const char *fmt, ...)
+static inline void messageD(int ATTRIBUTE_UNUSED device, 
+                               const char ATTRIBUTE_UNUSED *fmt, ...)
 {
 }
 #else
@@ -225,7 +221,7 @@ static void message(int device, const char *fmt, ...)
        if (device & LOG) {
                /* don`t out "\r\n" */
                openlog(bb_applet_name, 0, LOG_DAEMON);
-               syslog(LOG_INFO, "%s", msg);
+               syslog(LOG_INFO, "%s", msg + 1);
                closelog();
        }
 
@@ -353,7 +349,7 @@ static void console_init(void)
 #ifndef CONFIG_SYSLOGD
                log_console =
 #endif
-               safe_strncpy(console, "/dev/null", sizeof(console));
+               safe_strncpy(console, bb_dev_null, sizeof(console));
        } else {
                s = getenv("TERM");
                /* check for serial console */
@@ -475,7 +471,7 @@ static pid_t run(const struct init_action *a)
                                                break;
                                        }
                                        /* FIXME handle other errors */
-                }
+               }
 
                                /* See if stealing the controlling tty back is necessary */
                                pgrp = tcgetpgrp(0);
@@ -629,7 +625,9 @@ static void run_actions(int action)
        for (a = init_action_list; a; a = tmp) {
                tmp = a->next;
                if (a->action == action) {
-                       if (a->action & (SYSINIT | WAIT | CTRLALTDEL | SHUTDOWN | RESTART)) {
+                       if (access(a->terminal, R_OK | W_OK)) {
+                               delete_init_action(a);
+                       } else if (a->action & (SYSINIT | WAIT | CTRLALTDEL | SHUTDOWN | RESTART)) {
                                waitfor(a);
                                delete_init_action(a);
                        } else if (a->action & ONCE) {
@@ -702,7 +700,7 @@ static void shutdown_system(void)
        sync();
 }
 
-static void exec_signal(int sig)
+static void exec_signal(int sig ATTRIBUTE_UNUSED)
 {
        struct init_action *a, *tmp;
        sigset_t unblock_signals;
@@ -761,17 +759,10 @@ static void exec_signal(int sig)
        }
 }
 
-static void halt_signal(int sig)
+static void halt_signal(int sig ATTRIBUTE_UNUSED)
 {
        shutdown_system();
-       message(CONSOLE | LOG,
-#if #cpu(s390)
-                       /* Seems the s390 console is Wierd(tm). */
-                       "The system is halted. You may reboot now."
-#else
-                       "The system is halted. Press Reset or turn off power"
-#endif
-               );
+       message(CONSOLE | LOG, "The system is halted.");
        sync();
 
        /* allow time for last message to reach serial console */
@@ -785,7 +776,7 @@ static void halt_signal(int sig)
        loop_forever();
 }
 
-static void reboot_signal(int sig)
+static void reboot_signal(int sig ATTRIBUTE_UNUSED)
 {
        shutdown_system();
        message(CONSOLE | LOG, "Please stand by while rebooting the system.");
@@ -799,13 +790,13 @@ static void reboot_signal(int sig)
        loop_forever();
 }
 
-static void ctrlaltdel_signal(int sig)
+static void ctrlaltdel_signal(int sig ATTRIBUTE_UNUSED)
 {
        run_actions(CTRLALTDEL);
 }
 
 /* The SIGSTOP & SIGTSTP handler */
-static void stop_handler(int sig)
+static void stop_handler(int sig ATTRIBUTE_UNUSED)
 {
        int saved_errno = errno;
 
@@ -817,7 +808,7 @@ static void stop_handler(int sig)
 }
 
 /* The SIGCONT handler */
-static void cont_handler(int sig)
+static void cont_handler(int sig ATTRIBUTE_UNUSED)
 {
        got_cont = 1;
 }
@@ -831,10 +822,7 @@ static void new_init_action(int action, const char *command, const char *cons)
        if (*cons == '\0')
                cons = console;
 
-       /* do not run entries if console device is not available */
-       if (access(cons, R_OK | W_OK))
-               return;
-       if (strcmp(cons, "/dev/null") == 0 && (action & ASKFIRST))
+       if (strcmp(cons, bb_dev_null) == 0 && (action & ASKFIRST))
                return;
 
        new_action = calloc((size_t) (1), sizeof(struct init_action));
@@ -1000,7 +988,7 @@ static void parse_inittab(void)
 }
 
 #ifdef CONFIG_FEATURE_USE_INITTAB
-static void reload_signal(int sig)
+static void reload_signal(int sig ATTRIBUTE_UNUSED)
 {
        struct init_action *a, *tmp;
 
@@ -1026,22 +1014,20 @@ static void reload_signal(int sig)
 }
 #endif                                                 /* CONFIG_FEATURE_USE_INITTAB */
 
-extern int init_main(int argc, char **argv)
+int init_main(int argc, char **argv)
 {
        struct init_action *a;
        pid_t wpid;
        int status;
 
        if (argc > 1 && !strcmp(argv[1], "-q")) {
-               return kill_init(SIGHUP);
+               return kill(1,SIGHUP);
        }
 #ifndef DEBUG_INIT
        /* Expect to be invoked as init with PID=1 or be invoked as linuxrc */
-       if (getpid() != 1
-#ifdef CONFIG_FEATURE_INITRD
-               && strstr(bb_applet_name, "linuxrc") == NULL
-#endif
-               ) {
+       if (getpid() != 1 &&
+               (!ENABLE_FEATURE_INITRD || !strstr(bb_applet_name, "linuxrc")))
+       {
                bb_show_usage();
        }
        /* Set up sig handlers  -- be sure to
@@ -1116,6 +1102,22 @@ extern int init_main(int argc, char **argv)
                parse_inittab();
        }
 
+#ifdef CONFIG_SELINUX
+       if (getenv("SELINUX_INIT") == NULL) {
+               int enforce = 0;
+
+               putenv("SELINUX_INIT=YES");
+               if (selinux_init_load_policy(&enforce) == 0) {
+                       execv(argv[0], argv);
+               } else if (enforce > 0) {
+                       /* SELinux in enforcing mode but load_policy failed */
+                       /* At this point, we probably can't open /dev/console, so log() won't work */
+                       message(CONSOLE,"Unable to load SELinux Policy. Machine is in enforcing mode. Halting now.");
+                       exit(1);
+               }
+       }
+#endif /* CONFIG_SELINUX */
+
        /* Make the command line just say "init"  -- thats all, nothing else */
        fixup_argv(argc, argv, "init");