diff options
| author | Rob Landley <rob@landley.net> | 2006-11-25 13:48:02 -0500 | 
|---|---|---|
| committer | Rob Landley <rob@landley.net> | 2006-11-25 13:48:02 -0500 | 
| commit | b23d230ad55e028a4516e0cd60b7ecdc25504e4b (patch) | |
| tree | 4d18c658bc429a40263e20cf6634c51824d88bcb | |
| parent | 1674e2ace4dcce67c3920dcdbf2f441903194b68 (diff) | |
| download | toybox-b23d230ad55e028a4516e0cd60b7ecdc25504e4b.tar.gz | |
Add oneit.
| -rw-r--r-- | toys/Config.in | 16 | ||||
| -rw-r--r-- | toys/oneit.c | 45 | ||||
| -rw-r--r-- | toys/toylist.h | 1 | 
3 files changed, 62 insertions, 0 deletions
diff --git a/toys/Config.in b/toys/Config.in index 77e59e4d..59271ddb 100644 --- a/toys/Config.in +++ b/toys/Config.in @@ -34,6 +34,22 @@ config HELLO  	help  	  A hello world program.  You don't need this. +config ONEIT +	bool "oneit" +	default n +	help +	  usage: oneit [-p] command [...] + +	  A simple init program that runs a single supplied command line with a +	  controlling tty (so CTRL-C can kill it). + +	  -p	Power off instead of rebooting when command exits. + +	  The oneit command runs the supplied command line as a child process +	  (because PID 1 has signals blocked), attached to /dev/tty0, in its +	  own session.  Then oneit reaps zombies until the child exits, at +	  which point it reboots (or with -p, powers off) the system. +  config PWD  	bool "pwd"  	default n diff --git a/toys/oneit.c b/toys/oneit.c new file mode 100644 index 00000000..1f87e5f2 --- /dev/null +++ b/toys/oneit.c @@ -0,0 +1,45 @@ +/* oneit.c, tiny one-process init replacement. + * + * Copyright 2005 by Rob Landley <rob@landley.net>. + */ + +#include "toys.h" +#include <sys/reboot.h> + +// The minimum amount of work necessary to get ctrl-c and such to work is: +// +// - Fork a child (PID 1 is special: can't exit, has various signals blocked). +// - Do a setsid() (so we have our own session). +// - In the child, attach stdio to /dev/tty0 (/dev/console is special) +// - Exec the rest of the command line. +// +// PID 1 then reaps zombies until the child process it spawned exits, at which +// point it calls sync() and reboot().  I could stick a kill -1 in there. + +int oneit_main(void) +{ +  int i; +  pid_t pid; + +  // Create a new child process. +  pid = vfork(); +  if (pid) { + +    // pid 1 just reaps zombies until it gets its child, then halts the system. +    while (pid!=wait(&i)); +    sync(); +    reboot(toys.optflags ? RB_POWER_OFF : RB_AUTOBOOT); +  } +     +  // Redirect stdio to /dev/tty0, with new session ID, so ctrl-c works. +  setsid(); +  for (i=0; i<3; i++) { +    close(i); +    open("/dev/tty0",O_RDWR); +  } + +  // Can't xexec() here because we vforked so we don't want to error_exit(). +  toy_exec(toys.optargs); +  execvp(*toys.optargs, toys.optargs); +  _exit(127); +} diff --git a/toys/toylist.h b/toys/toylist.h index a61aff93..ef427491 100644 --- a/toys/toylist.h +++ b/toys/toylist.h @@ -57,6 +57,7 @@ USE_TOYSH(NEWTOY(cd, NULL, TOYFLAG_NOFORK))  USE_DF(NEWTOY(df, "Pkt*a", TOYFLAG_USR|TOYFLAG_SBIN))  USE_TOYSH(NEWTOY(exit, NULL, TOYFLAG_NOFORK))  USE_HELLO(NEWTOY(hello, NULL, TOYFLAG_NOFORK|TOYFLAG_USR)) +USE_ONEIT(NEWTOY(oneit, "+p<1", TOYFLAG_SBIN))  USE_PWD(NEWTOY(pwd, NULL, TOYFLAG_BIN))  USE_TOYSH(OLDTOY(sh, toysh, "c:i", TOYFLAG_BIN))  USE_TOYSH(NEWTOY(toysh, "c:i", TOYFLAG_BIN))  | 
