⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 finit-alt.c

📁 Linux系统快速启动 一个华硕eeepc快速启动的clone。 能装在大多数Linux系统上。 http://helllabs.org/finit/
💻 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 + -