busybox/init.c

439 lines
9.0 KiB
C

#include "internal.h"
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include <termios.h>
#include <sys/types.h>
#include <sys/fcntl.h>
#include <sys/wait.h>
#include <string.h>
#include <sys/mount.h>
#include <sys/reboot.h>
#include <sys/kdaemon.h>
#include <sys/swap.h>
#include <sys/sysmacros.h>
const char init_usage[] = "Used internally by the system.";
char console[16] = "";
const char * default_console = "/dev/tty1";
char * first_terminal = NULL;
const char * second_terminal = "/dev/tty2";
const char log[] = "/dev/tty3";
char * term_ptr = NULL;
static void
message(const char * terminal, const char * pattern, ...)
{
int fd;
FILE * con = 0;
va_list arguments;
/*
* Open the console device each time a message is printed. If the user
* has switched consoles, the open will get the new console. If we kept
* the console open, we'd always print to the same one.
*/
if ( ((fd = open(terminal, O_WRONLY|O_NOCTTY)) < 0)
|| ((con = fdopen(fd, "w")) == NULL) )
return;
va_start(arguments, pattern);
vfprintf(con, pattern, arguments);
va_end(arguments);
fclose(con);
}
static int
waitfor(int pid)
{
int status;
int wpid;
message(log, "Waiting for process %d.\n", pid);
while ( (wpid = wait(&status)) != pid ) {
if ( wpid > 0 ) {
message(
log
,"pid %d exited, status=%x.\n"
,wpid
,status);
}
}
return wpid;
}
static int
run(
const char * program
,const char * const * arguments
,const char * terminal
,int get_enter)
{
static const char control_characters[] = {
'\003',
'\034',
'\177',
'\025',
'\004',
'\0',
'\1',
'\0',
'\021',
'\023',
'\032',
'\0',
'\022',
'\017',
'\027',
'\026',
'\0'
};
static char * environment[] = {
"HOME=/",
"PATH=/bin:/sbin:/usr/bin:/usr/sbin",
"SHELL=/bin/sh",
0,
"USER=root",
0
};
static const char press_enter[] =
"Please press Enter to activate this console. ";
int pid;
environment[3]=term_ptr;
pid = fork();
if ( pid == 0 ) {
struct termios t;
const char * const * arg;
close(0);
close(1);
close(2);
setsid();
open(terminal, O_RDWR);
dup(0);
dup(0);
tcsetpgrp(0, getpgrp());
tcgetattr(0, &t);
memcpy(t.c_cc, control_characters, sizeof(control_characters));
t.c_line = 0;
t.c_iflag = ICRNL|IXON|IXOFF;
t.c_oflag = OPOST|ONLCR;
t.c_lflag = ISIG|ICANON|ECHO|ECHOE|ECHOK|ECHOCTL|ECHOKE|IEXTEN;
tcsetattr(0, TCSANOW, &t);
if ( get_enter ) {
/*
* Save memory by not exec-ing anything large (like a shell)
* before the user wants it. This is critical if swap is not
* enabled and the system has low memory. Generally this will
* be run on the second virtual console, and the first will
* be allowed to start a shell or the installation system.
*/
char c;
write(1, press_enter, sizeof(press_enter) - 1);
read(0, &c, 1);
}
message(log, "Executing ");
arg = arguments;
while ( *arg != 0 )
message(log, "%s ", *arg++);
message(log, "\n");
execve(program, (char * *)arguments, (char * *)environment);
message(log, "%s: could not execute: %s.\r\n", program, strerror(errno));
exit(-1);
}
return pid;
}
static int
mem_total()
{
char s[80];
char *p;
FILE *f;
const char pattern[]="MemTotal:";
f=fopen("/proc/meminfo","r");
while (NULL != fgets(s,79,f)) {
p=strstr(s, pattern);
if (NULL != p) {
fclose(f);
return(atoi(p+strlen(pattern)));
}
}
return -1;
}
static void
set_free_pages()
{
char s[80];
FILE *f;
f=fopen("/proc/sys/vm/freepages","r");
fgets(s,79,f);
if (atoi(s) < 32) {
fclose(f);
f=fopen("/proc/sys/vm/freepages","w");
fprintf(f,"30\t40\t50\n");
printf("\nIncreased /proc/sys/vm/freepages values to 30/40/50\n");
}
fclose(f);
}
static void
shutdown_system(int do_reboot)
{
static const char * const umount_args[] = {"/bin/umount", "-a", "-n", 0};
sync();
/* Allow Ctrl-Alt-Del to reboot system. */
reboot(RB_ENABLE_CAD);
/* Send signals to every process _except_ pid 1 */
message(console, "Sending SIGHUP to all processes.\r\n");
kill(-1, SIGHUP);
sleep(2);
sync();
message(console, "Sending SIGKILL to all processes.\r\n");
kill(-1, SIGKILL);
sleep(1);
waitfor(run("/bin/umount", umount_args, console, 0));
sync();
bdflush(1, 0);
sync();
reboot(do_reboot ?RB_AUTOBOOT : RB_HALT_SYSTEM);
exit(0);
}
static void
halt_signal(int sig)
{
shutdown_system(0);
}
static void
reboot_signal(int sig)
{
shutdown_system(1);
}
static void
exit_signal(int sig)
{
/* initrd doesn't work anyway */
shutdown_system(1);
/* This is used on the initial ramdisk */
/* message(log, "Init exiting.");
exit(0);
*/
}
void
configure_terminals( int serial_cons );
extern int
init_main(struct FileInfo * i, int argc, char * * argv)
{
static const char * const rc = "etc/rc";
const char * arguments[100];
int run_rc = 1;
int j;
int pid1 = 0;
int pid2 = 0;
int create_swap= -1;
struct stat statbuf;
#ifndef INCLUDE_DINSTALL
const char * tty_commands[2] = { "bin/sh", "bin/sh"};
#else
const char * tty_commands[2] = { "sbin/dinstall", "bin/sh"};
#endif
char swap[20];
int serial_console = 0;
/*
* If I am started as /linuxrc instead of /sbin/init, I don't have the
* environment that init expects. I can't fix the signal behavior. Try
* to divorce from the controlling terminal with setsid(). This won't work
* if I am the process group leader.
*/
setsid();
signal(SIGUSR1, halt_signal);
signal(SIGUSR2, reboot_signal);
signal(SIGINT, reboot_signal);
signal(SIGTERM, exit_signal);
reboot(RB_DISABLE_CAD);
message(log, "%s: started. ", argv[0]);
for ( j = 1; j < argc; j++ ) {
if ( strcmp(argv[j], "single") == 0 ) {
run_rc = 0;
tty_commands[0] = "bin/sh";
tty_commands[1] = 0;
}
}
for ( j = 0; __environ[j] != 0; j++ ) {
if ( strncmp(__environ[j], "tty", 3) == 0
&& __environ[j][3] >= '1'
&& __environ[j][3] <= '2'
&& __environ[j][4] == '=' ) {
const char * s = &__environ[j][5];
if ( *s == 0 || strcmp(s, "off") == 0 )
s = 0;
tty_commands[__environ[j][3] - '1'] = s;
}
/* Should catch the syntax of Sparc kernel console setting. */
/* The kernel does not recognize a serial console when getting*/
/* console=/dev/ttySX !! */
else if ( strcmp(__environ[j], "console=ttya") == 0 ) {
serial_console=1;
}
else if ( strcmp(__environ[j], "console=ttyb") == 0 ) {
serial_console=2;
}
/* standard console settings */
else if ( strncmp(__environ[j], "console=", 8) == 0 ) {
first_terminal=&(__environ[j][8]);
}
else if ( strncmp(__environ[j], "TERM=", 5) == 0) {
term_ptr=__environ[j];
}
}
configure_terminals( serial_console );
printf("mounting /proc ...\n");
if (mount("/proc","/proc","proc",0,0)) {
perror("mounting /proc failed\n");
}
printf("\tdone.\n");
set_free_pages();
if (mem_total() < 3500) { /* not enough memory for standard install */
int retval;
retval= stat("/etc/swappartition",&statbuf);
if (retval) {
printf("
You do not have enough RAM, hence you must boot using the Boot Disk
for Low Memory systems.
Read the instructions in the install.html file.
");
while (1) {;}
} else { /* everything OK */
FILE *f;
f=fopen("/etc/swappartition","r");
fgets(swap,19,f);
fclose(f);
*(strstr(swap,"\n"))='\0';
if (swapon(swap,0)) {
perror("swapon failed\n");
} else {
f=fopen("/etc/swaps","w");
fprintf(f,"%s none swap rw 0 0",swap);
fclose(f);
create_swap = 0;
}
}
}
/*
* Don't modify **argv directly, it would show up in the "ps" display.
* I don't want "init" to look like "rc".
*/
arguments[0] = rc;
for ( j = 1; j < argc; j++ ) {
arguments[j] = argv[j];
}
arguments[j] = 0;
if ( run_rc )
waitfor(run(rc, arguments, console, 0));
if ( 0 == create_swap) {
if (unlink("/etc/swappartition")) {
perror("unlinking /etc/swappartition");
}
}
arguments[0] = "-sh";
arguments[1] = 0;
for ( ; ; ) {
int wpid;
int status;
if ( pid1 == 0 && tty_commands[0] ) {
/* Ask before starting a shell */
/*
arguments[0] = tty_commands[0];
*/
pid1 = run(tty_commands[0], arguments, first_terminal, 0);
}
if ( pid2 == 0 && tty_commands[1] )
pid2 = run(tty_commands[1], arguments, second_terminal, 1);
wpid = wait(&status);
if ( wpid > 0 ) {
/* DEBUGGING */
message(log, "pid %d exited, status=%x.\n", wpid, status);
}
if ( wpid == pid1 ) {
pid1 = 0;
}
if ( wpid == pid2 )
pid2 = 0;
}
}
void
configure_terminals( int serial_cons )
{
//struct stat statbuf;
char *tty;
switch (serial_cons) {
case 1:
strcpy( console, "/dev/ttyS0" );
break;
case 2:
strcpy( console, "/dev/ttyS1" );
break;
default:
tty = ttyname(0);
if (tty) {
strcpy( console, tty );
if (!strncmp( tty, "/dev/ttyS", 9 ))
serial_cons=1;
}
else
/* falls back to /dev/tty1 if an error occurs */
strcpy( console, default_console );
}
if (!first_terminal)
first_terminal = console;
if (serial_cons && !strncmp(term_ptr,"TERM=linux",10))
term_ptr = "TERM=vt100";
}