summaryrefslogtreecommitdiff
path: root/init/init.c
diff options
context:
space:
mode:
Diffstat (limited to 'init/init.c')
-rw-r--r--init/init.c525
1 files changed, 20 insertions, 505 deletions
diff --git a/init/init.c b/init/init.c
index 4a6e0b90..bf6ef8b5 100644
--- a/init/init.c
+++ b/init/init.c
@@ -56,12 +56,10 @@
#include "mung_msg_S.h"
/* host_reboot flags for when we crash. */
-int crash_flags = RB_AUTOBOOT;
+static int crash_flags = RB_AUTOBOOT;
#define BOOT(flags) ((flags & RB_HALT) ? "halt" : "reboot")
-#define _PATH_RUNCOM "/libexec/rc"
-
const char *argp_program_version = STANDARD_HURD_VERSION (init);
@@ -78,9 +76,9 @@ options[] =
{0}
};
-char doc[] = "Start and maintain hurd core servers and system run state";
+static char doc[] = "Start and maintain hurd core servers and system run state";
-int booted; /* Set when the core servers are up. */
+static int booted; /* Set when the core servers are up. */
/* This structure keeps track of each notified task. */
struct ntfy_task
@@ -99,46 +97,39 @@ struct ess_task
};
/* These are linked lists of all of the registered items. */
-struct ess_task *ess_tasks;
-struct ntfy_task *ntfy_tasks;
-
-/* Mapped time */
-volatile struct mapped_time_value *mapped_time;
+static struct ess_task *ess_tasks;
+static struct ntfy_task *ntfy_tasks;
/* Our receive right */
-mach_port_t startup;
+static mach_port_t startup;
/* Ports to the kernel */
-mach_port_t host_priv, device_master;
+static mach_port_t host_priv, device_master;
/* Args to bootstrap, expressed as flags */
-int bootstrap_args = 0;
-
-/* Set if something determines we should no longer pass the `autoboot'
- flag to _PATH_RUNCOM. */
-int do_fastboot;
+static int bootstrap_args = 0;
/* Stored information for returning proc and auth startup messages. */
-mach_port_t procreply, authreply;
-mach_msg_type_name_t procreplytype, authreplytype;
+static mach_port_t procreply, authreply;
+static mach_msg_type_name_t procreplytype, authreplytype;
/* Our ports to auth and proc. */
-mach_port_t authserver;
-mach_port_t procserver;
+static mach_port_t authserver;
+static mach_port_t procserver;
/* Our bootstrap port, on which we call fsys_getpriv and fsys_init. */
-mach_port_t bootport;
+static mach_port_t bootport;
/* Set iff we are a `fake' bootstrap. */
-int fakeboot;
+static int fakeboot;
/* The tasks of auth and proc and the bootstrap filesystem. */
-task_t authtask, proctask, fstask;
+static task_t authtask, proctask, fstask;
-mach_port_t default_ports[INIT_PORT_MAX];
-mach_port_t default_dtable[3];
-int default_ints[INIT_INT_MAX];
+static mach_port_t default_ports[INIT_PORT_MAX];
+static mach_port_t default_dtable[3];
+static int default_ints[INIT_INT_MAX];
static char **global_argv;
static char *startup_envz;
@@ -745,168 +736,6 @@ init_stdarrays ()
mach_port_deallocate (mach_task_self (), std_port_array[i]);
}
-#ifndef SPLIT_INIT
-/* Open /dev/console. If it isn't there, or it isn't a terminal, then
- create /tmp/console and put the terminal on it. If we get EROFS,
- in trying to create /tmp/console then as a last resort, put the
- console on /tmp itself.
-
- In any case, after the console has been opened, set it appropriately
- in default_dtable. Also return a port right for the terminal. */
-file_t
-open_console ()
-{
-#define TERMINAL_FIRST_TRY "/hurd/term\0/tmp/console\0device\0console"
-#define TERMINAL_SECOND_TRY "/hurd/term\0/tmp\0device\0console"
- mach_port_t term;
- static char *termname;
- struct stat st;
- error_t err = 0;
-
- if (booted)
- {
- term = file_name_lookup (termname, O_RDWR, 0);
- return term;
- }
-
- termname = _PATH_CONSOLE;
- term = file_name_lookup (termname, O_RDWR, 0);
- if (term != MACH_PORT_NULL)
- err = io_stat (term, &st);
- else
- err = errno;
- if (err)
- error (0, err, "%s", termname);
- else if (st.st_fstype != FSTYPE_TERM)
- error (0, 0, "%s: Not a terminal", termname);
-
- if (term == MACH_PORT_NULL || err || st.st_fstype != FSTYPE_TERM)
- /* Start the terminal server ourselves. */
- {
- size_t argz_len; /* Length of args passed to translator. */
- char *terminal; /* Name of term translator. */
- mach_port_t control; /* Control port for term translator. */
- int try = 1;
-
- error_t open_node (int flags,
- mach_port_t *underlying,
- mach_msg_type_name_t *underlying_type,
- task_t task, void *cookie)
- {
- term = file_name_lookup (termname, flags | O_CREAT|O_NOTRANS, 0666);
- if (term == MACH_PORT_NULL)
- {
- error (0, errno, "%s", termname);
- return errno;
- }
-
- *underlying = term;
- *underlying_type = MACH_MSG_TYPE_COPY_SEND;
-
- return 0;
- }
-
- retry:
- bootstrap_args |= RB_SINGLE;
-
- if (try == 1)
- {
- terminal = TERMINAL_FIRST_TRY;
- argz_len = sizeof TERMINAL_FIRST_TRY;
- try = 2;
- }
- else if (try == 2)
- {
- terminal = TERMINAL_SECOND_TRY;
- argz_len = sizeof TERMINAL_SECOND_TRY;
- try = 3;
- }
- else
- goto fail;
-
- termname = terminal + strlen (terminal) + 1; /* first arg is name */
-
- /* The callback to start_translator opens TERM as a side effect. */
- errno =
- fshelp_start_translator (open_node, NULL, terminal, terminal,
- argz_len, 3000, &control);
- if (errno)
- {
- error (0, errno, "%s", terminal);
- goto retry;
- }
-
- errno = file_set_translator (term, 0, FS_TRANS_SET, 0, 0, 0,
- control, MACH_MSG_TYPE_COPY_SEND);
- mach_port_deallocate (mach_task_self (), control);
- if (errno)
- {
- error (0, errno, "%s", termname);
- goto retry;
- }
- mach_port_deallocate (mach_task_self (), term);
-
- /* Now repeat the open. */
- term = file_name_lookup (termname, O_RDWR, 0);
- if (term == MACH_PORT_NULL)
- {
- error (0, errno, "%s", termname);
- goto retry;
- }
- errno = io_stat (term, &st);
- if (errno)
- {
- error (0, errno, "%s", termname);
- term = MACH_PORT_NULL;
- goto retry;
- }
- if (st.st_fstype != FSTYPE_TERM)
- {
- error (0, 0, "%s: Not a terminal", termname);
- term = MACH_PORT_NULL;
- goto retry;
- }
-
- if (term)
- error (0, 0, "Using temporary console %s", termname);
- }
-
- fail:
-
- /* At this point either TERM is the console or it's null. If it's
- null, then don't do anything, and our fd's will be copied.
- Otherwise, open fd's 0, 1, and 2. */
- if (term != MACH_PORT_NULL)
- {
- int fd = openport (term, O_RDWR);
- if (fd < 0)
- assert_perror (errno);
-
- dup2 (fd, 0);
- close (fd);
- dup2 (0, 1);
- dup2 (0, 2);
-
- fclose (stdin);
- stdin = fdopen (0, "r");
-
- /* Don't reopen our output channel for reliability's sake. */
-
- /* Set ports in init_dtable for programs we start. */
- mach_port_deallocate (mach_task_self (), default_dtable[0]);
- mach_port_deallocate (mach_task_self (), default_dtable[1]);
- mach_port_deallocate (mach_task_self (), default_dtable[2]);
- default_dtable[0] = getdport (0);
- default_dtable[1] = getdport (1);
- default_dtable[2] = getdport (2);
- }
- else
- error (0, 0, "Cannot open console");
-
- return term;
-}
-#endif
-
/* Frobnicate the kernel task and the proc server's idea of it (PID 2),
so the kernel command line can be read as for a normal Hurd process. */
@@ -1037,8 +866,6 @@ frob_kernel_process (void)
error (0, err, "proc_set_arg_locations for kernel task");
}
-#ifdef SPLIT_INIT
-
/** Running userland. **/
/* In the "split-init" setup, we just run a single program (usually
@@ -1047,8 +874,8 @@ frob_kernel_process (void)
shell as a fallback. */
-pid_t child_pid; /* PID of the child we run */
-task_t child_task; /* and its (original) task port */
+static pid_t child_pid; /* PID of the child we run */
+static task_t child_task; /* and its (original) task port */
error_t send_signal (mach_port_t msgport, int signal, mach_port_t refport,
mach_msg_timeout_t);
@@ -1258,318 +1085,6 @@ launch_system (void)
{
launch_something (0);
}
-
-
-#else
-
-/** Single and multi user transitions **/
-
-/* Current state of the system. */
-enum
-{
- INITIAL,
- SINGLE,
- RUNCOM,
- MULTI,
-} system_state;
-
-pid_t shell_pid; /* PID of single-user shell. */
-pid_t rc_pid; /* PID of rc script */
-
-
-#include "ttys.h"
-
-
-/* Start the single-user environment. This can only be done
- when the core servers have fully started. We know that
- startup_essential_task is the last thing they do before being
- ready to handle requests, so we start this once all the necessary
- servers have identified themselves that way. */
-void
-launch_single_user ()
-{
- char shell[1024];
- mach_port_t term;
-
- printf ("Single-user environment:");
- fflush (stdout);
-
- term = open_console ();
-
- system_state = SINGLE;
-
-#if 0
- printf ("Shell program [%s]: ", _PATH_BSHELL);
- if (! getstring (shell, sizeof shell))
-#endif
- strcpy (shell, _PATH_BSHELL);
-
- /* The shell needs a real controlling terminal, so set that up here. */
- shell_pid = run_for_real (shell, shell, strlen (shell) + 1, term, 1);
- mach_port_deallocate (mach_task_self (), term);
- if (shell_pid == 0)
- crash_system ();
- printf (" shell.\n");
- fflush (stdout);
-}
-
-/* Run /etc/rc as a shell script. Return non-zero if we fail. */
-int
-process_rc_script ()
-{
- char *rcargs;
- size_t rcargslen;
- mach_port_t term;
-
- if (do_fastboot)
- {
- rcargs = malloc (rcargslen = sizeof _PATH_RUNCOM);
- if (!rcargs)
- return ENOMEM;
- strcpy (rcargs, _PATH_RUNCOM);
- }
- else
- {
- rcargslen = asprintf (&rcargs, "%s%c%s", _PATH_RUNCOM, '\0', "autoboot");
- rcargslen++; /* final null */
- }
-
- term = open_console ();
-
- system_state = RUNCOM;
-
- rc_pid = run_for_real (rcargs, rcargs, rcargslen, term, 1);
- free (rcargs);
- mach_port_deallocate (mach_task_self (), term);
- return ! rc_pid;
-}
-
-/* Start up multi-user state. */
-void
-launch_multi_user ()
-{
- int fail;
-
- if (!mapped_time)
- maptime_map (1, 0, &mapped_time);
-
- fail = init_ttys ();
- if (fail)
- launch_single_user ();
- else
- {
- system_state = MULTI;
- fail = startup_ttys ();
- if (fail)
- launch_single_user ();
- }
-}
-
-void
-launch_system ()
-{
- if (bootstrap_args & RB_SINGLE)
- launch_single_user ();
- else
- {
- if (process_rc_script ())
- launch_single_user ();
- }
-}
-
-
-/* Kill all the outstanding processes with SIGNO. Return 1 if
- there were no tasks left to kill. */
-int
-kill_everyone (int signo)
-{
- pid_t pidbuf[100], *pids = pidbuf;
- mach_msg_type_number_t i, npids = 100;
- task_t tk;
- struct ess_task *es;
- mach_port_t msg;
- int didany;
- int nloops;
- error_t err;
-
- for (nloops = 10; nloops; nloops--)
- {
- if (nloops < 9)
- /* Give it a rest for folks to have a chance to die */
- sleep (1);
-
- didany = 0;
- err = proc_getallpids (procserver, &pids, &npids);
- if (!err)
- {
- for (i = 0; i < npids; i++)
- {
- if (pids[i] == 1 /* us */
- || pids[i] == 3 /* default pager for now XXX */)
- continue;
-
- /* See if the task is essential */
- err = proc_pid2task (procserver, pids[i], &tk);
- if (err)
- continue;
-
- for (es = ess_tasks; es; es = es->next)
- if (tk == es->task_port)
- {
- /* Skip this one */
- mach_port_deallocate (mach_task_self (), tk);
- break;
- }
- if (es)
- continue;
-
- /* Kill it */
- if (signo == SIGKILL)
- {
- task_terminate (tk);
- didany = 1;
- }
- else
- {
- err = proc_getmsgport (procserver, pids[i], &msg);
- if (err)
- {
- mach_port_deallocate (mach_task_self (), tk);
- continue;
- }
-
- didany = 1;
- msg_sig_post (msg, signo, 0, tk);
- mach_port_deallocate (mach_task_self (), tk);
- }
- }
- }
- if (pids != pidbuf)
- munmap (pids, npids * sizeof (pid_t));
- if (!didany)
- return 1;
- }
- return 0;
-}
-
-/* Kill outstanding multi-user sessions */
-void
-kill_multi_user ()
-{
- int sigs[3] = {SIGHUP, SIGTERM, SIGKILL};
- int stage;
-
- free_ttys ();
-
- for (stage = 0; stage < 3; stage++)
- if (kill_everyone (sigs[stage]))
- break;
-
- /* Notify tasks that they are about to die. */
- notify_shutdown ("transition to single-user");
-
- if (stage == 3)
- error (0, 0, "warning: some processes wouldn't die; `ps -AlM' advised");
-}
-
-/* SIGNO has arrived and has been validated. Do whatever work it
- implies. */
-void
-process_signal (int signo)
-{
- int fail;
-
- switch (signo)
- {
- case SIGTERM:
- if (system_state == MULTI)
- {
- /* Drop back to single user. */
- kill_multi_user ();
- launch_single_user ();
- }
- break;
-
- case SIGHUP:
- if (system_state == MULTI)
- reread_ttys ();
- break;
-
- case SIGCHLD:
- {
- /* A child died. Find its status. */
- int status;
- pid_t pid;
-
- for (;;)
- {
- pid = waitpid (WAIT_ANY, &status, WNOHANG | WUNTRACED);
- if (pid <= 0)
- return;
-
- if (pid == shell_pid && system_state == SINGLE)
- {
- if (WIFSIGNALED (status))
- {
- error (0, 0,
- "Single-user terminated abnormally (%s), restarting",
- strsignal (WTERMSIG (status)));
- launch_single_user ();
- }
- else if (WIFSTOPPED (status))
- {
- error (0, 0,
- "Single-user stopped (%s), killing and restarting",
- strsignal (WSTOPSIG (status)));
- kill (shell_pid, SIGKILL);
- launch_single_user ();
- }
- else
- {
- do_fastboot = 1;
- fail = process_rc_script ();
- if (fail)
- {
- do_fastboot = 0;
- launch_single_user ();
- }
- }
- }
- else if (pid == rc_pid && system_state == RUNCOM)
- {
- if (WIFSIGNALED (status))
- {
- error (0, 0,
- "%s terminated abnormally (%s), \
-going to single user mode",
- _PATH_RUNCOM, strsignal (WTERMSIG (status)));
- launch_single_user ();
- }
- else if (WIFSTOPPED (status))
- {
- error (0, 0,
- "%s stopped (%s), \
-killing it and going to single user mode",
- _PATH_RUNCOM, strsignal (WSTOPSIG (status)));
- kill (rc_pid, SIGKILL);
- launch_single_user ();
- }
- else if (WEXITSTATUS (status))
- launch_single_user ();
- else
- launch_multi_user ();
- }
- else if (system_state == MULTI)
- restart_terminal (pid);
- }
- break;
- }
- default:
- break;
- }
-}
-
-#endif /* SPLIT_INIT */
/** RPC servers **/