📄 finit-alt.c
字号:
/*Improved fast initCopyright (c) 2008 Claudio MatsuokaPermission is hereby granted, free of charge, to any person obtaining a copyof this software and associated documentation files (the "Software"), to dealin the Software without restriction, including without limitation the rightsto use, copy, modify, merge, publish, distribute, sublicense, and/or sellcopies of the Software, and to permit persons to whom the Software isfurnished to do so, subject to the following conditions:The above copyright notice and this permission notice shall be included inall copies or substantial portions of the Software.THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS ORIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THEAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHERLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS INTHE SOFTWARE.*/#include <stdio.h>#include <string.h>#include <stdlib.h>#include <ctype.h>#include <sys/mount.h>#include <sys/types.h>#include <sys/stat.h>#include <fcntl.h>#include <dirent.h>#include <errno.h>#include <signal.h>#include <unistd.h>#include <sys/reboot.h>#include <sys/wait.h>#include <linux/fs.h>#include <utmp.h>#include "helpers.h"#ifndef DEFUSER#define DEFUSER "user"#endif/* Distribution configuration */#if defined DIST_MDV /* Mandriva */#define RANDOMSEED "/var/lib/random-seed"#define SYSROOT "/sysroot"#define GETTY "/usr/bin/openvt /sbin/mingetty tty2"#define RUNPARTS "/usr/bin/run-parts"#define REMOUNT_ROOTFS_RW#define MAKE_DEVICES#define PAM_CONSOLE#define LISTEN_INITCTL#define RUNLEVEL 5#define USE_VAR_RUN_RESOLVCONF#define USE_MESSAGE_BUS#define USE_CONSOLEKIT#elif defined DIST_EEEXUBUNTU /* eeeXubuntu */#define RANDOMSEED "/var/lib/urandom/random-seed"#define SYSROOT "/sysroot"#define GETTY "/usr/bin/openvt /sbin/getty 38400 tty2"#define RUNPARTS "/bin/run-parts"#define REMOUNT_ROOTFS_RW#define MAKE_DEVICES#define TOUCH_ETC_NETWORK_RUN_IFSTATE#define LISTEN_INITCTL#define USE_USPLASH#else /* original Eeepc distribution */#define RANDOMSEED "/var/lib/urandom/random-seed"#define SYSROOT "/mnt"#define GETTY "/usr/bin/openvt /sbin/getty 38400 tty2"#define RUNPARTS "/bin/run-parts"#define TOUCH_ETC_NETWORK_RUN_IFSTATE#define USE_ETC_RESOLVCONF_RUN#endif#ifdef DIRECTISA#define HWCLOCK_DIRECTISA " --directisa"#else#define HWCLOCK_DIRECTISA#endif/* From sysvinit *//* Set a signal handler. */#define SETSIG(sa, sig, fun, flags) \ do { \ sa.sa_handler = fun; \ sa.sa_flags = flags; \ sigemptyset(&sa.sa_mask); \ sigaction(sig, &sa, NULL); \ } while (0)#ifdef LISTEN_INITCTL#define INIT_MAGIC 0x03091969#define INIT_CMD_RUNLVL 1struct init_request { int magic; /* Magic number */ int cmd; /* What kind of request */ int runlevel; /* Runlevel to change to */ int sleeptime; /* Time between TERM and KILL */ char data[368];};#endif#define touch(x) mknod((x), S_IFREG|0644, 0)#define chardev(x,m,maj,min) mknod((x), S_IFCHR|(m), makedev((maj),(min)))#define blkdev(x,m,maj,min) mknod((x), S_IFBLK|(m), makedev((maj),(min)))#define MATCH_CMD(l,c,x) \ ((!strncmp((l), (c), strlen((c)))) && ((x) = (l) + strlen((c))))#define LINE_SIZE 1024#define CMD_SIZE 256#define USERNAME_SIZE 16#define HOSTNAME_SIZE 32void shutdown(int);void signal_handler(int);void chld_handler(int);#ifdef DEBUG_TIMESTAMP /* For profiling */#include <time.h>#include <stdarg.h>static int debug = 1;static struct timeval t0;void _d(char *fmt, ...){ va_list ap; struct timeval t; int s, ms; if (!debug) return; va_start(ap, fmt); gettimeofday(&t, NULL); s = t.tv_sec - t0.tv_sec; ms = (t.tv_usec - t0.tv_usec) / 1000; if (ms < 0) { ms += 1000; s -= 1; } printf("[%d.%03d] ", s, ms); vprintf(fmt, ap); printf("\n"); va_end(ap);}#else#define _d(x...) do { if (debug) { printf(x); printf("\n"); } } while (0)static int debug = 0;#endifstatic void build_cmd(char *cmd, char *x, int len){ int l; char *c; c = cmd + strlen(cmd); /* skip spaces */ for (; *x && (*x == ' ' || *x == '\t'); x++); /* copy next arg */ for (l = 0; *x && *x != '#' && *x != '\t' && l < len; l++) *c++ = *x++; *c = 0; _d("cmd = %s", cmd);}#ifdef LISTEN_INITCTL/* Standard reboot/shutdown utilities talk to init using /dev/initctl. * We should check if the fifo was recreated and reopen it. */static void listen_initctl(){ if (!fork()) { int ctl; fd_set fds; struct init_request request; mkfifo("/dev/initctl", 0600); ctl = open("/dev/initctl", O_RDONLY); while (1) { FD_ZERO(&fds); FD_SET(ctl, &fds); if (select(ctl + 1, &fds, NULL, NULL, NULL) <= 0) continue; read(ctl, &request, sizeof(request)); if (request.magic != INIT_MAGIC) continue; if (request.cmd == INIT_CMD_RUNLVL) { switch (request.runlevel) { case '0': shutdown(SIGUSR2); break; case '6': shutdown(SIGUSR1); } } } }}#endifint main(){ int i; FILE *f; char line[LINE_SIZE]; int fd; struct sigaction sa, act; sigset_t nmask, nmask2; char username[USERNAME_SIZE] = DEFUSER; char hostname[HOSTNAME_SIZE] = "eviltwin"; char cmd[CMD_SIZE];#ifdef USE_ETC_RESOLVCONF_RUN DIR *dir; struct dirent *d;#endif#ifdef RUNLEVEL struct utmp entry;#endif#ifdef DEBUG_TIMESTAMP gettimeofday(&t0, NULL);#endif puts("finit-alt " VERSION " (built " __DATE__ " " __TIME__ " by " WHOAMI ")");#ifdef USE_USPLASH system("usplash_write \"TEXT-URGENT finit " VERSION " (build "__DATE__ " " __TIME__ ") \""); system("usplash_write \"PROGRESS 80\"");#endif chdir("/"); umask(022); /* * Signal management */ for (i = 1; i < NSIG; i++) SETSIG(sa, i, SIG_IGN, SA_RESTART); SETSIG(sa, SIGINT, shutdown, 0); SETSIG(sa, SIGPWR, SIG_IGN, 0); SETSIG(sa, SIGUSR1, shutdown, 0); SETSIG(sa, SIGUSR2, shutdown, 0); SETSIG(sa, SIGTERM, SIG_IGN, 0); SETSIG(sa, SIGALRM, SIG_IGN, 0); SETSIG(sa, SIGHUP, SIG_IGN, 0); SETSIG(sa, SIGCONT, SIG_IGN, SA_RESTART); SETSIG(sa, SIGCHLD, chld_handler, SA_RESTART); /* Block sigchild while forking */ sigemptyset(&nmask); sigaddset(&nmask, SIGCHLD); sigprocmask(SIG_BLOCK, &nmask, NULL); reboot(RB_DISABLE_CAD); mount("none", "/proc", "proc", 0, NULL); /* * Parse kernel parameters */ if ((f = fopen("/proc/cmdline", "r")) != NULL) { fgets(line, LINE_SIZE, f); if (strstr(line, "quiet")) { close(0); close(1); close(2); } if ((strstr(line, "finit_debug"))) { debug = 1; } fclose(f); } setsid(); /* * Parse configuration file */ if ((f = fopen("/etc/finit.conf", "r")) != NULL) { char *x; while (!feof(f)) { fgets(line, LINE_SIZE, f); chomp(line); _d("conf: %s", line); /* Do this before mounting / read-write */ if (MATCH_CMD(line, "check ", x)) { strcpy(cmd, "/sbin/fsck -C -a "); build_cmd(cmd, x, CMD_SIZE); system(cmd); continue; } if (MATCH_CMD(line, "user ", x)) { *username = 0; build_cmd(username, x, USERNAME_SIZE); continue; } if (MATCH_CMD(line, "host ", x)) { *hostname = 0; build_cmd(hostname, x, HOSTNAME_SIZE); continue; } if (MATCH_CMD(line, "module ", x)) { strcpy(cmd, "/sbin/modprobe "); build_cmd(cmd, x, CMD_SIZE); system(cmd); continue; }#ifdef MAKE_DEVICES /* This is used if the system uses udev and doesn't * mount /dev as tmpfs in the initrd, or if it doesn't * use an initrd. Mount /dev as tmpfs and create basic * device nodes. */ if (MATCH_CMD(line, "mountdev", x)) { mount("none", "/dev", "tmpfs", 0, "mode=0755"); blkdev("/dev/sda", 0660, 8, 0); blkdev("/dev/sda1", 0660, 8, 1); blkdev("/dev/sda2", 0660, 8, 2); blkdev("/dev/sda3", 0660, 8, 3); blkdev("/dev/sda4", 0660, 8, 4); blkdev("/dev/sda5", 0660, 8, 5); blkdev("/dev/sda6", 0660, 8, 6); continue; } /* This works only if /dev is tmpfs! If not, create * devices statically on the filesystem */ if (MATCH_CMD(line, "mknod ", x)) { strcpy(cmd, "/bin/mknod "); build_cmd(cmd, x, CMD_SIZE); system(cmd); continue; }#endif } fclose(f); } /* * Mount filesystems */ _d("mount filesystems");#ifdef REMOUNT_ROOTFS_RW system("/bin/mount -n -o remount,rw /");#endif umask(0);#ifdef MAKE_DEVICES mkdir("/dev/shm", 0755); mkdir("/dev/pts", 0755);#endif mount("none", "/sys", "sysfs", 0, NULL); mount("none", "/dev/pts", "devpts", 0, "gid=5,mode=620"); mount("none", "/dev/shm", "tmpfs", 0, NULL); mount("none", "/tmp", "tmpfs", 0, "mode=1777,size=128m"); mount("none", "/var/run", "tmpfs", 0, "mode=0755"); mount("none", "/var/lock", "tmpfs", 0, "mode=1777"); mount("none", "/proc/bus/usb", "usbfs", 0, NULL); mount(SYSROOT, "/", NULL, MS_MOVE, NULL); _d("make devices");#ifdef MAKE_DEVICES mkdir("/dev/input", 0755); chardev("/dev/urandom", 0666, 1, 9); chardev("/dev/ptmx", 0666, 5, 2); chardev("/dev/null", 0666, 1, 3); chmod("/dev/null", 0666); chardev("/dev/mem", 0640, 1, 1); chardev("/dev/tty0", 0660, 4, 0); chardev("/dev/tty1", 0660, 4, 1); chardev("/dev/tty2", 0660, 4, 2); chardev("/dev/tty3", 0660, 4, 3); chardev("/dev/input/mice", 0660, 13, 63); chardev("/dev/input/event0", 0660, 13, 64); chardev("/dev/agpgart", 0660, 10, 175); blkdev("/dev/loop0", 0600, 7, 0);#endif unlink("/etc/mtab"); system("mount -a"); umask(0022); /* * Time adjustments */ if ((fd = open("/etc/adjtime", O_CREAT|O_WRONLY|O_TRUNC, 0644)) >= 0) { write(fd, "0.0 0 0.0\n", 10); close(fd); } _d("adjust clock");#ifndef NO_HCTOSYS#ifndef DEBUG_TIMESTAMP system("/sbin/hwclock --hctosys --localtime" HWCLOCK_DIRECTISA);#endif#endif /* * Network stuff */#ifdef USE_ETC_RESOLVCONF_RUN makepath("/dev/shm/network"); makepath("/dev/shm/resolvconf/interface"); if ((dir = opendir("/etc/resolvconf/run/interface")) != NULL) { while ((d = readdir(dir)) != NULL) { if (isalnum(d->d_name[0])) continue; snprintf(line, LINE_SIZE, "/etc/resolvconf/run/interface/%s", d->d_name); unlink(line); } closedir(dir); }#endif#ifdef USE_VAR_RUN_RESOLVCONF makepath("/var/run/resolvconf/interface"); symlink("../../../etc/resolv.conf", "/var/run/resolvconf/resolv.conf");#endif touch("/var/run/utmp"); chown("/var/run/utmp", 0, getgroup("utmp")); chmod("/var/run/utmp", 0664);#ifdef RUNLEVEL memset(&entry, 0, sizeof(struct utmp)); entry.ut_type = RUN_LVL; entry.ut_pid = '0' + RUNLEVEL; setutent(); pututline(&entry); endutent();#endif touch("/etc/resolvconf/run/enable-updates"); chdir("/etc/resolvconf/run/interface");#ifdef BUILTIN_RUNPARTS run_parts("/etc/resolvconf/update.d", "-i", NULL);#else system(RUNPARTS " --arg=i /etc/resolvconf/update.d");#endif chdir("/"); #ifdef TOUCH_ETC_NETWORK_RUN_IFSTATE touch("/etc/network/run/ifstate");#endif if ((f = fopen("/etc/hostname", "r")) != NULL) { fgets(hostname, HOSTNAME_SIZE, f); chomp(hostname); fclose(f); } sethostname(hostname, strlen(hostname)); ifconfig("lo", "127.0.0.1", "255.0.0.0", 1); /* * Set random seed */ copyfile(RANDOMSEED, "/dev/urandom", 0); unlink(RANDOMSEED); umask(077); copyfile("/dev/urandom", RANDOMSEED, 4096); umask(0); /* * Console setup (for X) */ makepath("/var/run/console"); snprintf(line, LINE_SIZE, "/var/run/console/%s", username); touch(line);#ifdef PAM_CONSOLE if ((fd = open("/var/run/console/console.lock", O_CREAT|O_WRONLY|O_TRUNC, 0644)) >= 0) { write(fd, username, strlen(username)); close(fd); system("/sbin/pam_console_apply"); }#endif /* * Misc setup */ mkdir("/tmp/.X11-unix", 01777); mkdir("/tmp/.ICE-unix", 01777); umask(022);#ifdef USE_USPLASH system("usplash_write \"PROGRESS 90\"");#endif system(GETTY "&"); _d("forking"); if (!fork()) { /* child process */ vhangup();#ifndef DEBUG_TIMESTAMP close(2); close(1); close(0); if (open("/dev/tty1", O_RDWR) != 0) exit(1);#endif sigemptyset(&act.sa_mask); act.sa_handler = SIG_DFL; sigemptyset(&nmask2); sigaddset(&nmask2, SIGCHLD); sigprocmask(SIG_UNBLOCK, &nmask2, NULL); for (i = 1; i < NSIG; i++) sigaction(i, &sa, NULL);#ifndef DEBUG_TIMESTAMP dup2(0, 0); dup2(0, 1); dup2(0, 2);#endif touch("/tmp/nologin");#ifdef USE_MESSAGE_BUS _d("dbus"); mkdir("/var/run/dbus", 0755); mkdir("/var/lock/subsys/messagebus", 0755); system("dbus-uuidgen --ensure;dbus-daemon --system"); /* dbus-daemon needs this delay, even after checking * availability of /var/run/dbus/system_bus_socket */ usleep(500000);#endif#ifdef USE_CONSOLEKIT _d("consolekit"); system("/usr/sbin/console-kit-daemon");#endif#ifdef LISTEN_INITCTL listen_initctl();#endif while (access("/tmp/shutdown", F_OK) < 0) { _d("start X as %s\n", username);#ifdef DEBUG_TIMESTAMP sleep(10);#endif if (debug) { snprintf(line, LINE_SIZE, "su -c startx -l %s\n", username); system(line); system("/bin/sh"); } else { snprintf(line, LINE_SIZE, "su -c startx -l %s\n &> /dev/null", username); system(line); } } exit(0); } /* parent process */ sleep(1); system("/usr/sbin/services.sh &>/dev/null&"); while (1) { sigemptyset(&nmask); pselect(0, NULL, NULL, NULL, NULL, &nmask); }}/* * Shut down on INT USR1 USR2 */void shutdown(int sig){ touch("/tmp/shutdown"); kill(-1, SIGTERM); write(1, "\033[?25l\033[30;40m", 14); copyfile("/boot/shutdown.fb", "/dev/fb/0", 0); sleep(2); system("/usr/sbin/alsactl store > /dev/null 2>&1"); system("/sbin/hwclock --systohc --localtime" HWCLOCK_DIRECTISA); kill(-1, SIGKILL); sync(); sync(); system("/bin/mount -n -o remount,ro /"); system("/sbin/unionctl.static / --remove / > /dev/null 2>&1"); if (sig == SIGINT || sig == SIGUSR1) reboot(RB_AUTOBOOT); reboot(RB_POWER_OFF);}/* * SIGCHLD: one of our children has died */void chld_handler(int sig){ int status; while (waitpid(-1, &status, WNOHANG) != 0) { if (errno == ECHILD) break; }}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -