summaryrefslogtreecommitdiff
path: root/init/init.c
diff options
context:
space:
mode:
Diffstat (limited to 'init/init.c')
-rw-r--r--init/init.c269
1 files changed, 151 insertions, 118 deletions
diff --git a/init/init.c b/init/init.c
index ade29c7..ca194dd 100644
--- a/init/init.c
+++ b/init/init.c
@@ -13,19 +13,21 @@
#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;
+#include <linux/serial.h> /* for serial_struct */
+#include <sys/vt.h> /* for vt_stat */
+#include <sys/ioctl.h>
+
+static const char init_usage[] = "Used internally by the system.";
+static char console[16] = "";
+static const char* default_console = "/dev/tty2";
+static char* first_terminal = NULL;
+static const char* second_terminal = "/dev/tty2";
+static const char* log = "/dev/tty3";
+static char* term_ptr = NULL;
static void
-message(const char * terminal, const char * pattern, ...)
+message(const char* terminal, const char * pattern, ...)
{
int fd;
FILE * con = 0;
@@ -36,7 +38,8 @@ message(const char * terminal, const char * pattern, ...)
* 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)
+ if ( !terminal
+ || ((fd = open(terminal, O_WRONLY|O_NOCTTY)) < 0)
|| ((con = fdopen(fd, "w")) == NULL) )
return;
@@ -158,44 +161,59 @@ run(const char* program, const char* const* arguments,
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)));
+ 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;
+ return -1;
}
static void
set_free_pages()
{
- char s[80];
+ 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 int
+get_kernel_revision()
+{
FILE *f;
+ int major=0, minor=0, patch=0;
- 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");
- }
+ f = fopen("/proc/sys/kernel/osrelease","r");
+ fscanf(f,"%d.%d.%d",&major,&minor,&patch);
fclose(f);
+ return major*65536 + minor*256 + patch;
}
+
static void
-shutdown_system(int do_reboot)
+shutdown_system(void)
{
static const char * const umount_args[] = {"/bin/umount", "-a", "-n", 0};
+ static const char * const swapoff_args[] = {"/bin/swapoff", "-a", 0};
+ message(console, "The system is going down NOW !!");
sync();
/* Allow Ctrl-Alt-Del to reboot system. */
reboot(RB_ENABLE_CAD);
@@ -208,59 +226,72 @@ shutdown_system(int do_reboot)
message(console, "Sending SIGKILL to all processes.\r\n");
kill(-1, SIGKILL);
sleep(1);
+ waitfor(run("/bin/swapoff", swapoff_args, console, 0));
waitfor(run("/bin/umount", umount_args, console, 0));
sync();
- bdflush(1, 0);
- sync();
- reboot(do_reboot ?RB_AUTOBOOT : RB_HALT_SYSTEM);
- exit(0);
+ if (get_kernel_revision() <= 2*65536+2*256+11) {
+ /* Removed bdflush call, kupdate in kernels >2.2.11 */
+ bdflush(1, 0);
+ sync();
+ }
}
static void
halt_signal(int sig)
{
- shutdown_system(0);
+ shutdown_system();
+ message(console, "The system is halted. Press CTRL-ALT-DEL or turn off power\r\n");
+ reboot( RB_HALT_SYSTEM);
+ exit(0);
}
static void
reboot_signal(int sig)
{
- shutdown_system(1);
+ shutdown_system();
+ message(console, "Please stand by while rebooting the system.\r\n");
+ reboot( RB_AUTOBOOT);
+ exit(0);
}
static void
-exit_signal(int sig)
+configure_terminals( int serial_cons, int single_user_mode )
{
-
- /* 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 )
-{
- //struct stat statbuf;
char *tty;
+ struct serial_struct sr;
+ struct vt_stat vt;
+
switch (serial_cons) {
+ case -1:
+ /* 2.2 kernels:
+ * identify the real console backend and try to make use of it */
+ if (ioctl(0,TIOCGSERIAL,&sr) == 0) {
+ sprintf( console, "/dev/ttyS%d", sr.line );
+ serial_cons = sr.line+1;
+ }
+ else if (ioctl(0, VT_GETSTATE, &vt) == 0) {
+ sprintf( console, "/dev/tty%d", vt.v_active );
+ serial_cons = 0;
+ }
+ else {
+ /* unknown backend: fallback to /dev/console */
+ strcpy( console, "/dev/console" );
+ serial_cons = 0;
+ }
+ break;
+
case 1:
- strcpy( console, "/dev/ttyS0" );
+ strcpy( console, "/dev/cua0" );
break;
case 2:
- strcpy( console, "/dev/ttyS1" );
+ strcpy( console, "/dev/cua1" );
break;
default:
tty = ttyname(0);
if (tty) {
strcpy( console, tty );
- if (!strncmp( tty, "/dev/ttyS", 9 ))
+ if (!strncmp( tty, "/dev/cua", 8 ))
serial_cons=1;
}
else
@@ -269,24 +300,33 @@ configure_terminals( int serial_cons )
}
if (!first_terminal)
first_terminal = console;
- if (serial_cons && !strncmp(term_ptr,"TERM=linux",10))
- term_ptr = "TERM=vt100";
+#if #cpu (sparc)
+ if (serial_cons > 0 && !strncmp(term_ptr,"TERM=linux",10))
+ term_ptr = "TERM=vt100";
+#endif
+ if (serial_cons) {
+ /* disable other but the first terminal:
+ * VT is not initialized anymore on 2.2 kernel when booting from
+ * serial console, therefore modprobe is flooding the display with
+ * "can't locate module char-major-4" messages. */
+ log = 0;
+ second_terminal = 0;
+ }
}
extern int
init_main(int argc, char * * argv)
{
- static const char* const rc = "etc/rc";
+ const char * rc_arguments[100];
const char * arguments[100];
- int run_rc = 1;
+ int run_rc = TRUE;
int j;
int pid1 = 0;
int pid2 = 0;
- int create_swap= -1;
struct stat statbuf;
- const char * tty_commands[2] = { "bin/sh", "bin/sh"};
- char swap[20];
+ const char * tty_commands[3] = { "etc/init.d/rcS", "bin/sh"};
int serial_console = 0;
+ int retval;
/*
* If I am started as /linuxrc instead of /sbin/init, I don't have the
@@ -299,7 +339,7 @@ init_main(int argc, char * * argv)
signal(SIGUSR1, halt_signal);
signal(SIGUSR2, reboot_signal);
signal(SIGINT, reboot_signal);
- signal(SIGTERM, exit_signal);
+ signal(SIGTERM, reboot_signal);
reboot(RB_DISABLE_CAD);
@@ -307,7 +347,7 @@ init_main(int argc, char * * argv)
for ( j = 1; j < argc; j++ ) {
if ( strcmp(argv[j], "single") == 0 ) {
- run_rc = 0;
+ run_rc = FALSE;
tty_commands[0] = "bin/sh";
tty_commands[1] = 0;
}
@@ -341,7 +381,6 @@ init_main(int argc, char * * argv)
term_ptr=__environ[j];
}
}
- configure_terminals( serial_console );
printf("mounting /proc ...\n");
if (mount("/proc","/proc","proc",0,0)) {
@@ -349,81 +388,75 @@ init_main(int argc, char * * argv)
}
printf("\tdone.\n");
+ if (get_kernel_revision() >= 2*65536+1*256+71) {
+ /* if >= 2.1.71 kernel, /dev/console is not a symlink anymore:
+ * use it as primary console */
+ serial_console=-1;
+ }
+
+ /* Make sure /etc/init.d/rc exists */
+ retval= stat(tty_commands[0],&statbuf);
+ if (retval)
+ run_rc = FALSE;
+
+ configure_terminals( serial_console, run_rc);
+
set_free_pages();
/* not enough memory to do anything useful*/
if (mem_total() < 2000) {
- int retval;
- retval= stat("/etc/swappartition",&statbuf);
- if (retval) {
- printf("You do not have enough RAM, sorry.\n");
- while (1) { sleep(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;
+ retval= stat("/etc/fstab",&statbuf);
+ if (retval) {
+ printf("You do not have enough RAM, sorry.\n");
+ while (1) { sleep(1);}
+ } else {
+ /* Try to turn on swap */
+ static const char * const swapon_args[] = {"/bin/swapon", "-a", 0};
+ waitfor(run("/bin/swapon", swapon_args, console, 0));
+ if (mem_total() < 2000) {
+ printf("You do not have enough RAM, sorry.\n");
+ while (1) { sleep(1);}
+ }
}
- }
}
/*
* 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;
+ rc_arguments[0] = tty_commands[0];
for ( j = 1; j < argc; j++ ) {
- arguments[j] = argv[j];
- }
- arguments[j] = 0;
-
- if ( run_rc ) {
- run(rc, arguments, console, 0);
- //waitfor(run(rc, arguments, console, 0));
- }
-
- if ( 0 == create_swap) {
- if (unlink("/etc/swappartition")) {
- perror("unlinking /etc/swappartition");
- }
+ rc_arguments[j] = argv[j];
}
+ rc_arguments[j] = 0;
arguments[0] = "-sh";
arguments[1] = 0;
+
+ /* Ok, now launch the rc script /etc/init.d/rcS and prepare to start up
+ * some VTs on tty1 and tty2 if somebody hits enter
+ */
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, 1);
+ if ( pid1 == 0 && tty_commands[0] ) {
+ if ( run_rc == TRUE ) {
+ pid1 = run(tty_commands[0], rc_arguments, first_terminal, 0);
+ } else {
+ pid2 = run(tty_commands[1], arguments, first_terminal, 1);
+ }
}
- if ( pid2 == 0 && tty_commands[1] ) {
+ if ( pid2 == 0 && second_terminal && tty_commands[1] ) {
pid2 = run(tty_commands[1], arguments, second_terminal, 1);
}
wpid = wait(&status);
- if ( wpid > 0 ) {
- /* DEBUGGING */
+ if ( wpid > 0 && wpid != pid1) {
message(log, "pid %d exited, status=%x.\n", wpid, status);
}
- if ( wpid == pid1 ) {
- pid1 = 0;
- }
- if ( wpid == pid2 )
+ if ( wpid == pid2 ) {
pid2 = 0;
+ }
}
}