diff options
Diffstat (limited to 'packages/sysvinit/sysvinit-2.86/sysvinit-2.86.patch')
-rw-r--r-- | packages/sysvinit/sysvinit-2.86/sysvinit-2.86.patch | 4880 |
1 files changed, 0 insertions, 4880 deletions
diff --git a/packages/sysvinit/sysvinit-2.86/sysvinit-2.86.patch b/packages/sysvinit/sysvinit-2.86/sysvinit-2.86.patch deleted file mode 100644 index b83f525c3e..0000000000 --- a/packages/sysvinit/sysvinit-2.86/sysvinit-2.86.patch +++ /dev/null @@ -1,4880 +0,0 @@ -diff -urNd -urNd sysvinit-2.85/COPYRIGHT sysvinit-2.86/COPYRIGHT ---- sysvinit-2.85/COPYRIGHT 2003-04-15 03:45:44.000000000 -0500 -+++ sysvinit-2.86/COPYRIGHT 2004-07-30 07:12:12.000000000 -0500 -@@ -1,4 +1,4 @@ --Sysvinit is Copyright (C) 1991-2003 Miquel van Smoorenburg -+Sysvinit is Copyright (C) 1991-2004 Miquel van Smoorenburg - - This program is free software; you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by -diff -urNd -urNd sysvinit-2.85/doc/Changelog sysvinit-2.86/doc/Changelog ---- sysvinit-2.85/doc/Changelog 2003-04-15 09:37:58.000000000 -0500 -+++ sysvinit-2.86/doc/Changelog 2004-07-30 07:15:06.000000000 -0500 -@@ -1,3 +1,29 @@ -+sysvinit (2.86) cistron; urgency=low -+ -+ * Fixed up bootlogd to read /proc/cmdline. Also keep an internal -+ linebuffer to process \r, \t and ^H. It is becoming useable. -+ * Applied trivial OWL patches -+ * Block signals in syslog(), since syslog() is not re-entrant -+ (James Olin Oden <joden@malachi.lee.k12.nc.us>, redhat bug #97534) -+ * Minor adjustements so that sysvinit compiles on the Hurd -+ * killall5 now skips kernel threads -+ * Inittab entries with both 'S' and other runlevels were broken. -+ Fix by Bryan Kadzban <bryan@kadzban.is-a-geek.net> -+ * Changed initreq.h to be more flexible and forwards-compatible. -+ * You can now through /dev/initctl set environment variables in -+ init that will be inherited by its children. For now, only -+ variables prefixed with INIT_ can be set and the maximum is -+ 16 variables. There's also a length limit due to the size -+ of struct init_request, so it should be safe from abuse. -+ * Option -P and -H to shutdown set INIT_HALT=POWERDOWN and -+ INIT_HALT=HALT as environment variables as described above -+ * Add "mountpoint" utility. -+ * Slightly better algorithm in killall5.c:pidof() -+ * Added some patches from fedora-core (halt-usage, last -t, -+ sulogin-message, user-console) -+ -+ -- Miquel van Smoorenburg <miquels@cistron.nl> Fri, 30 Jul 2004 14:14:58 +0200 -+ - sysvinit (2.85) cistron; urgency=low - - * Add IPv6 support in last(1) -diff -urNd -urNd sysvinit-2.85/doc/Install sysvinit-2.86/doc/Install ---- sysvinit-2.85/doc/Install 2003-04-15 03:46:49.000000000 -0500 -+++ sysvinit-2.86/doc/Install 2004-07-30 07:15:40.000000000 -0500 -@@ -1,5 +1,5 @@ - -- README for the System V style init, version 2.85 -+ README for the System V style init, version 2.86 - - init, shutdown, halt, reboot, wall, last, mesg, runlevel, - killall5, pidof, sulogin. -diff -urNd -urNd sysvinit-2.85/doc/bootlogd.README sysvinit-2.86/doc/bootlogd.README ---- sysvinit-2.85/doc/bootlogd.README 2000-09-12 16:54:31.000000000 -0500 -+++ sysvinit-2.86/doc/bootlogd.README 2004-06-09 07:47:45.000000000 -0500 -@@ -1,10 +1,12 @@ - - bootlogd: a way to capture all console output during bootup -- in a logfile. ** PROOF OF CONCEPT IMPLEMENTATION ** -+ in a logfile. - --- bootlogd opens /dev/console --- finds out what the real console is with an ioctl() --- then opens the real console -+- bootlogd opens /dev/console and finds out what the real console is -+ with an ioctl() if TIOCCONS is available -+- otherwise bootlogd tries to parse /proc/cmdline for console= -+ kernel command line arguments -+- then opens the (most probable) real console - - allocates a pty pair - - redirects console I/O to the pty pair - - then goes in a loop reading from the pty, writing to the real -diff -urNd -urNd sysvinit-2.85/doc/sysvinit-2.85.lsm sysvinit-2.86/doc/sysvinit-2.85.lsm ---- sysvinit-2.85/doc/sysvinit-2.85.lsm 2003-04-18 16:04:12.000000000 -0500 -+++ sysvinit-2.86/doc/sysvinit-2.85.lsm 2004-06-09 07:47:45.000000000 -0500 -@@ -1,7 +1,7 @@ - Begin3 - Title: sysvinit and utilities - Version: 2.85 --Entered-Date: 18APR2003 -+Entered-Date: 15APR2003 - Description: This is the Linux System V init. - This version can be compiled with glibc 2.0.6 and up. - Author: miquels@cistron.nl (Miquel van Smoorenburg) -diff -urNd -urNd sysvinit-2.85/doc/sysvinit-2.86.lsm sysvinit-2.86/doc/sysvinit-2.86.lsm ---- sysvinit-2.85/doc/sysvinit-2.86.lsm 1969-12-31 18:00:00.000000000 -0600 -+++ sysvinit-2.86/doc/sysvinit-2.86.lsm 2004-07-31 08:35:28.000000000 -0500 -@@ -0,0 +1,14 @@ -+Begin3 -+Title: sysvinit and utilities -+Version: 2.86 -+Entered-Date: 30JUL2004 -+Description: This is the Linux System V init. -+ This version can be compiled with glibc 2.0.6 and up. -+Author: miquels@cistron.nl (Miquel van Smoorenburg) -+Primary-Site: ftp.cistron.nl /pub/people/miquels/software -+ 92K sysvinit-2.86.tar.gz -+Alternate-Site: sunsite.unc.edu /pub/Linux/system/daemons/init -+ 92K sysvinit-2.86.tar.gz -+Copying-Policy: GPL -+Keywords: init shutdown halt reboot -+End -diff -urNd -urNd sysvinit-2.85/man/bootlogd.8 sysvinit-2.86/man/bootlogd.8 ---- sysvinit-2.85/man/bootlogd.8 1969-12-31 18:00:00.000000000 -0600 -+++ sysvinit-2.86/man/bootlogd.8 2004-06-09 07:47:45.000000000 -0500 -@@ -0,0 +1,39 @@ -+.TH BOOTLOGD 8 "Jul 21, 2003" "" "Linux System Administrator's Manual" -+.SH NAME -+bootlogd \- record boot messages -+.SH SYNOPSIS -+.B /sbin/bootlogd -+.RB [ \-d ] -+.RB [ \-r ] -+.RB [ \-v ] -+.RB [ " -l logfile " ] -+.RB [ " -p pidfile " ] -+.SH DESCRIPTION -+\fBBootlogd\fP runs in the background and copies all strings sent to the -+\fI/dev/console\fP device to a logfile. If the logfile is not accessible, -+the messages will be buffered in-memory until it is. -+.SH OPTIONS -+.IP \fB\-d\fP -+Do not fork and run in the background. -+.IP \fB\-r\fP -+If there is an existing logfile called \fIlogfile\fP rename it to -+\fIlogfile~\fP unless \fIlogfile~\fP already exists. -+.IP \fB\-v\fP -+Show version. -+.IP \fB\-l logfile\fP -+Log to this logfile. The default is \fI/var/log/boot\fP. -+.IP \fB\-p pidfile\fP -+Put process-id in this file. The default is no pidfile. -+.SH BUGS -+Bootlogd works by redirecting the console output from the console -+device. It copies that output to the real console device and a -+logfile. There is no standard way to find out the real console device -+if you have a new-style \fI/dev/console\fP device (major 5, minor 1). -+\fBBootlogd\fP tries to parse the kernel command line, looking for -+console= lines and deducts the real console device from that. If that -+syntax is ever changed by the kernel, or a console-type is used -+bootlogd does not know about, bootlogd will not work. -+.SH AUTHOR -+Miquel van Smoorenburg, miquels@cistron.nl -+.SH "SEE ALSO" -+.BR dmesg (8) -diff -urNd -urNd sysvinit-2.85/man/init.8 sysvinit-2.86/man/init.8 ---- sysvinit-2.85/man/init.8 2003-04-18 16:05:03.000000000 -0500 -+++ sysvinit-2.86/man/init.8 2004-07-29 06:21:31.000000000 -0500 -@@ -1,6 +1,6 @@ - .\"{{{}}} - .\"{{{ Title --.TH INIT 8 "18 April 2003" "" "Linux System Administrator's Manual" -+.TH INIT 8 "29 Jul 2004" "" "Linux System Administrator's Manual" - .\"}}} - .\"{{{ Name - .SH NAME -@@ -160,7 +160,7 @@ - .SH ENVIRONMENT - \fBInit\fP sets the following environment variables for all its children: - .IP \fBPATH\fP --\fI/usr/local/sbin:/sbin:/bin:/usr/sbin:/usr/bin\fP -+\fI/bin:/usr/bin:/sbin:/usr/sbin\fP - .IP \fBINIT_VERSION\fP - As the name says. Useful to determine if a script runs directly from \fBinit\fP. - .IP \fBRUNLEVEL\fP -diff -urNd -urNd sysvinit-2.85/man/initscript.5 sysvinit-2.86/man/initscript.5 ---- sysvinit-2.85/man/initscript.5 1999-12-24 16:31:21.000000000 -0600 -+++ sysvinit-2.86/man/initscript.5 2004-06-09 07:47:45.000000000 -0500 -@@ -1,4 +1,4 @@ --.TH INITSCRIPT 5 "December 24, 1999" "" "Linux System Administrator's Manual" -+.TH INITSCRIPT 5 "July 10, 2003" "" "Linux System Administrator's Manual" - .SH NAME - initscript \- script that executes inittab commands. - .SH SYNOPSIS -@@ -40,6 +40,12 @@ - - .sp - .RE -+.SH NOTES -+This script is not meant as startup script for daemons or somesuch. -+It has nothing to do with a \fIrc.local\fP style script. It's just -+a handler for things executed from \fB/etc/inittab\fP. Experimenting -+with this can make your system un(re)bootable. -+.RE - .SH FILES - /etc/inittab, - /etc/initscript. -diff -urNd -urNd sysvinit-2.85/man/killall5.8 sysvinit-2.86/man/killall5.8 ---- sysvinit-2.85/man/killall5.8 1997-05-27 05:34:21.000000000 -0500 -+++ sysvinit-2.86/man/killall5.8 2004-06-09 07:47:45.000000000 -0500 -@@ -1,4 +1,4 @@ --.TH KILLALL5 8 "27 May 1997" "" "Linux System Administrator's Manual" -+.TH KILLALL5 8 "04 Nov 2003" "" "Linux System Administrator's Manual" - .SH NAME - killall5 -- send a signal to all processes. - .SH SYNOPSIS -@@ -7,9 +7,9 @@ - .SH DESCRIPTION - .B killall5 - is the SystemV killall command. It sends a signal to all processes except --the processes in its own session, so it won't kill the shell that is --running the script it was called from. Its primary (only) use is in the --\fBrc\fP scripts found in the /etc/init.d directory. -+kernel threads and the processes in its own session, so it won't kill -+the shell that is running the script it was called from. Its primary -+(only) use is in the \fBrc\fP scripts found in the /etc/init.d directory. - .SH SEE ALSO - .BR halt (8), - .BR reboot (8) -diff -urNd -urNd sysvinit-2.85/man/last.1 sysvinit-2.86/man/last.1 ---- sysvinit-2.85/man/last.1 1999-07-29 05:50:34.000000000 -0500 -+++ sysvinit-2.86/man/last.1 2004-07-30 06:39:18.000000000 -0500 -@@ -1,6 +1,6 @@ - .\"{{{}}} - .\"{{{ Title --.TH LAST,LASTB 1 "Jul 29, 1999" "" "Linux System Administrator's Manual" -+.TH LAST,LASTB 1 "Jul 31, 2004" "" "Linux System Administrator's Manual" - .\"}}} - .\"{{{ Name - .SH NAME -@@ -14,6 +14,7 @@ - .RB "[ \-\fBn\fP \fInum\fP ]" - .RB [ \-adiox ] - .RB "[ \-\fBf\fP \fIfile\fP ]" -+.RB "[ \-\fBt\fP \fIYYYYMMDDHHMMSS\fP ]" - .RI [ name... ] - .RI [ tty... ] - .br -@@ -54,6 +55,11 @@ - This is a count telling \fBlast\fP how many lines to show. - .IP "\fB\-n\fP \fInum\fP" - The same. -+.IP "\fB\-t\fP \fIYYYYMMDDHHMMSS\fP" -+Display the state of logins as of the specified time. This is -+useful, e.g., to determine easily who was logged in at a particular -+time -- specify that time with \fB\-t\fP and look for "still logged -+in". - .IP \fB\-R\fP - Suppresses the display of the hostname field. - .IP \fB\-a\fP -diff -urNd -urNd sysvinit-2.85/man/mesg.1 sysvinit-2.86/man/mesg.1 ---- sysvinit-2.85/man/mesg.1 2001-02-26 06:01:10.000000000 -0600 -+++ sysvinit-2.86/man/mesg.1 2004-06-09 07:47:45.000000000 -0500 -@@ -27,7 +27,7 @@ - If no option is given, \fBmesg\fP prints out the current access state of your - terminal. - .PP NOTES --\fBMesg\fP assumes that it's standard input is connected to your -+\fBMesg\fP assumes that its standard input is connected to your - terminal. That also means that if you are logged in multiple times, - you can get/set the mesg status of other sessions by using redirection. - For example "mesg n < /dev/pts/46". -diff -urNd -urNd sysvinit-2.85/man/mountpoint.1 sysvinit-2.86/man/mountpoint.1 ---- sysvinit-2.85/man/mountpoint.1 1969-12-31 18:00:00.000000000 -0600 -+++ sysvinit-2.86/man/mountpoint.1 2004-06-09 07:47:45.000000000 -0500 -@@ -0,0 +1,37 @@ -+.TH MOUNTPOINT 8 "Mar 15, 2004" "" "Linux System Administrator's Manual" -+.SH NAME -+mountpoint \- see if a directory is a mountpoint -+.SH SYNOPSIS -+.B /bin/mountpoint -+.RB [ \-q ] -+.RB [ \-d ] -+.I /path/to/directory -+.br -+.B /bin/mountpoint -+.RB \-x -+.I /dev/device -+.SH DESCRIPTION -+\fBMountpoint\fP checks if the directory is a mountpoint. -+ -+.SH OPTIONS -+.IP \fB\-q\fP -+Be quiet - don't print anything. -+.IP \fB\-d\fP -+Print major/minor device number of the filesystem on stdout. -+.IP \fB\-x\fP -+Print major/minor device number of the blockdevice on stdout. -+.SH EXIT STATUS -+Zero if the directory is a mountpoint, non-zero if not. -+.SH NOTES -+Symbolic links are not followed, except when the \fB-x\fP option is -+used. To force following symlinks, add a trailing slash to the -+path of the directory. -+.PP -+The name of the command is misleading when the -x option is used, -+but the option is useful for comparing if a directory and a device -+match up, and there is no other command that can print the info easily. -+.PP -+.SH AUTHOR -+Miquel van Smoorenburg, miquels@cistron.nl -+.SH "SEE ALSO" -+.BR stat (1) -diff -urNd -urNd sysvinit-2.85/man/shutdown.8 sysvinit-2.86/man/shutdown.8 ---- sysvinit-2.85/man/shutdown.8 2001-10-02 16:27:50.000000000 -0500 -+++ sysvinit-2.86/man/shutdown.8 2004-06-09 07:47:45.000000000 -0500 -@@ -1,6 +1,6 @@ - .\"{{{}}} - .\"{{{ Title --.TH SHUTDOWN 8 "Juli 31, 2001" "" "Linux System Administrator's Manual" -+.TH SHUTDOWN 8 "November 12, 2003" "" "Linux System Administrator's Manual" - .\"}}} - .\"{{{ Name - .SH NAME -@@ -11,7 +11,7 @@ - .B /sbin/shutdown - .RB [ \-t - .IR sec ] --.RB [ \-arkhncfF ] -+.RB [ \-arkhncfFHP ] - .I time - .RI [ warning-message ] - .\"}}} -@@ -54,7 +54,16 @@ - .\"}}} - .\"{{{ -h - .IP \fB\-h\fP --Halt after shutdown. -+Halt or poweroff after shutdown. -+.\"}}} -+.\"{{{ -H -+.IP \fB\-H\fP -+Halt action is to halt or drop into boot monitor on systems that -+support it. -+.\"}}} -+.\"{{{ -P -+.IP \fB\-P\fP -+Halt action is to turn off the power. - .\"}}} - .\"{{{ -n - .IP \fB\-n\fP -@@ -141,6 +150,14 @@ - .sp 1 - Note that if \fI/etc/shutdown.allow\fP is not present, the \fB-a\fP - argument is ignored. -+.SH HALT OR POWEROFF -+The \fB-H\fP option just sets the \fIinit\fP environment variable -+\fIINIT_HALT\fP to \fIHALT\fP, and the \fB-P\fP option just sets -+that variable to \fIPOWEROFF\fP. The shutdown script that calls -+\fBhalt\fP(8) as the last thing in the shutdown sequence should -+check these environment variables and call \fBhalt\fP(8) with -+the right options for these options to actually have any effect. -+Debian 3.1 (sarge) supports this. - .SH FILES - .nf - /fastboot -diff -urNd -urNd sysvinit-2.85/man/sulogin.8 sysvinit-2.86/man/sulogin.8 ---- sysvinit-2.85/man/sulogin.8 2000-09-11 07:19:25.000000000 -0500 -+++ sysvinit-2.86/man/sulogin.8 2004-06-09 07:47:45.000000000 -0500 -@@ -1,4 +1,4 @@ --.TH SULOGIN 8 "11 Sep 2000" "" "Linux System Administrator's Manual" -+.TH SULOGIN 8 "04 Nov 2003" "" "Linux System Administrator's Manual" - .SH NAME - sulogin -- Single-user login - .SH SYNOPSIS -@@ -20,7 +20,7 @@ - .br - (or type Control-D for normal startup): - .PP --\fIsulogin\fP will connected to the current terminal, or to the -+\fIsulogin\fP will be connected to the current terminal, or to the - optional device that can be specified on the command line - (typically \fB/dev/console\fP). - .PP -@@ -45,16 +45,18 @@ - .PP - boot: linux -b rw sushell=/sbin/sash - .SH FALLBACK METHODS --\fIsulogin\fP checks the root password using the standard methods first. --If the \fB-e\fP option was specified, --\fIsulogin\fP examines the next files to find the root password. If --they are damaged, or non-existant, it will use fallback methods that --even go so far as to provide you with a shell prompt without asking --for the root password if they are irrepairably damaged. -+\fIsulogin\fP checks the root password using the standard method (getpwnam) -+first. -+Then, if the \fB-e\fP option was specified, -+\fIsulogin\fP examines these files directly to find the root password: - .PP - /etc/passwd, - .br - /etc/shadow (if present) -+.PP -+If they are damaged or non-existant, sulogin will start a root shell -+without asking for a password. Only use the \fB-e\fP option if you -+are sure the console is physically protected against unauthorized access. - .SH AUTHOR - Miquel van Smoorenburg <miquels@cistron.nl> - .SH SEE ALSO -diff -urNd -urNd sysvinit-2.85/man/wall.1 sysvinit-2.86/man/wall.1 ---- sysvinit-2.85/man/wall.1 2003-04-16 04:17:38.000000000 -0500 -+++ sysvinit-2.86/man/wall.1 2004-06-09 07:47:45.000000000 -0500 -@@ -47,7 +47,7 @@ - .I Wall - ignores the - .B TZ --variable - the time printed in the banner is based on the systems -+variable - the time printed in the banner is based on the system's - local time. - - .SH SEE ALSO -diff -urNd -urNd sysvinit-2.85/src/Makefile sysvinit-2.86/src/Makefile ---- sysvinit-2.85/src/Makefile 2001-11-06 05:58:16.000000000 -0600 -+++ sysvinit-2.86/src/Makefile 2004-06-09 07:47:45.000000000 -0500 -@@ -5,34 +5,56 @@ - # clean cleans up object files - # clobber really cleans up - # --# Version: @(#)Makefile 2.83-3 06-Nov-2001 miquels@cistron.nl -+# Version: @(#)Makefile 2.85-13 23-Mar-2004 miquels@cistron.nl - # - --CC = cc --CFLAGS = -Wall -O2 -D_GNU_SOURCE -+CC = gcc -+CFLAGS = -Wall -O2 -fomit-frame-pointer -D_GNU_SOURCE - LDFLAGS = -s - STATIC = - --# For Debian we do not build all programs, otherwise we do. --ifeq ($(DEBIAN),) --PROGS = init halt shutdown killall5 runlevel sulogin utmpdump \ -- last mesg wall --else --PROGS = init halt shutdown killall5 runlevel sulogin last mesg -+# For some known distributions we do not build all programs, otherwise we do. -+BIN = -+SBIN = init halt shutdown runlevel killall5 -+USRBIN = last mesg -+ -+MAN1 = last.1 lastb.1 mesg.1 -+MAN5 = initscript.5 inittab.5 -+MAN8 = halt.8 init.8 killall5.8 pidof.8 poweroff.8 reboot.8 runlevel.8 -+MAN8 += shutdown.8 telinit.8 -+ -+ifeq ($(DISTRO),) -+BIN += mountpoint -+SBIN += sulogin bootlogd -+USRBIN += utmpdump wall -+MAN1 += mountpoint.1 wall.1 -+MAN8 += sulogin.8 bootlogd.8 -+endif -+ -+ifeq ($(DISTRO),Debian) -+BIN += mountpoint -+SBIN += sulogin bootlogd -+MAN1 += mountpoint.1 -+MAN8 += sulogin.8 bootlogd.8 -+endif -+ -+ifeq ($(DISTRO),Owl) -+USRBIN += wall -+MAN1 += wall.1 - endif - - BIN_OWNER = root - BIN_GROUP = root --BIN_COMBO = $(BIN_OWNER).$(BIN_GROUP) -+BIN_COMBO = $(BIN_OWNER):$(BIN_GROUP) - INSTALL = install -o $(BIN_OWNER) -g $(BIN_GROUP) - MANDIR = /usr/share/man - --# Additional libs for Gnu Libc -+# Additional libs for GNU libc. - ifneq ($(wildcard /usr/lib/libcrypt.a),) - LCRYPT = -lcrypt - endif - --all: $(PROGS) -+all: $(BIN) $(SBIN) $(USRBIN) - - init: init.o init_utmp.o - $(CC) $(LDFLAGS) $(STATIC) -o $@ init.o init_utmp.o -@@ -46,6 +68,9 @@ - mesg: mesg.o - $(CC) $(LDFLAGS) -o $@ mesg.o - -+mountpoint: mountpoint.o -+ $(CC) $(LDFLAGS) -o $@ mountpoint.o -+ - utmpdump: utmpdump.o - $(CC) $(LDFLAGS) -o $@ utmpdump.o - -@@ -62,9 +87,9 @@ - $(CC) $(LDFLAGS) -o $@ dowall.o shutdown.o utmp.o - - bootlogd: bootlogd.o -- $(CC) $(LDFLAGS) -o $@ bootlogd.o -+ $(CC) $(LDFLAGS) -o $@ bootlogd.o -lutil - --init.o: init.c init.h set.h reboot.h -+init.o: init.c init.h set.h reboot.h initreq.h - $(CC) -c $(CFLAGS) init.c - - utmp.o: utmp.c init.h -@@ -80,36 +105,44 @@ - @echo Type \"make clobber\" to really clean up. - - clobber: cleanobjs -- rm -f $(PROGS) -+ rm -f $(BIN) $(SBIN) $(USRBIN) - - distclean: clobber - - install: -- $(INSTALL) -m 755 halt init killall5 sulogin \ -- runlevel shutdown $(ROOT)/sbin -- # These are not installed by default --ifeq ($(DEBIAN),) -- $(INSTALL) -m 555 utmpdump wall $(ROOT)/usr/bin --endif -- # $(INSTALL) -m 755 etc/initscript.sample $(ROOT)/etc -- $(INSTALL) -m 755 mesg last $(ROOT)/usr/bin -- cd $(ROOT)/sbin; ln -sf halt reboot; chown $(BIN_COMBO) reboot -- cd $(ROOT)/sbin; ln -sf halt poweroff; chown $(BIN_COMBO) poweroff -- cd $(ROOT)/sbin; ln -sf init telinit; chown $(BIN_COMBO) telinit -- cd $(ROOT)/bin; ln -sf ../sbin/killall5 pidof; chown $(BIN_COMBO) pidof -- cd $(ROOT)/usr/bin; ln -sf last lastb; chown $(BIN_COMBO) lastb -- $(INSTALL) -m 644 initreq.h $(ROOT)/usr/include -- $(INSTALL) -m 644 ../man/*.8 $(ROOT)$(MANDIR)/man8 -- $(INSTALL) -m 644 ../man/*.5 $(ROOT)$(MANDIR)/man5 --ifeq ($(DEBIAN),) -- $(INSTALL) -m 644 ../man/wall.1 $(ROOT)$(MANDIR)/man1 --endif -- $(INSTALL) -m 644 ../man/last.1 ../man/lastb.1 ../man/mesg.1 \ -- $(ROOT)$(MANDIR)/man1 -+ for i in $(BIN); do \ -+ $(INSTALL) -m 755 $$i $(ROOT)/bin/; \ -+ done -+ for i in $(SBIN); do \ -+ $(INSTALL) -m 755 $$i $(ROOT)/sbin/; \ -+ done -+ for i in $(USRBIN); do \ -+ $(INSTALL) -m 755 $$i $(ROOT)/usr/bin/; \ -+ done -+ # $(INSTALL) -m 755 etc/initscript.sample $(ROOT)/etc/ -+ ln -sf halt $(ROOT)/sbin/reboot -+ ln -sf halt $(ROOT)/sbin/poweroff -+ ln -sf init $(ROOT)/sbin/telinit -+ ln -sf ../sbin/killall5 $(ROOT)/bin/pidof -+ if [ ! -f $(ROOT)/usr/bin/lastb ]; then \ -+ ln -sf last $(ROOT)/usr/bin/lastb; \ -+ fi -+ $(INSTALL) -m 644 initreq.h $(ROOT)/usr/include/ -+ for i in $(MAN1); do \ -+ $(INSTALL) -m 644 ../man/$$i $(ROOT)$(MANDIR)/man1/; \ -+ done -+ for i in $(MAN5); do \ -+ $(INSTALL) -m 644 ../man/$$i $(ROOT)$(MANDIR)/man5/; \ -+ done -+ for i in $(MAN8); do \ -+ $(INSTALL) -m 644 ../man/$$i $(ROOT)$(MANDIR)/man8/; \ -+ done -+ifeq ($(ROOT),) - # -- # This part is skipped on debian systems, the -+ # This part is skipped on Debian systems, the - # debian.preinst script takes care of it. - @if [ ! -p /dev/initctl ]; then \ - echo "Creating /dev/initctl"; \ - rm -f /dev/initctl; \ - mknod -m 600 /dev/initctl p; fi -+endif -Binary files sysvinit-2.85/src/bootlogd and sysvinit-2.86/src/bootlogd differ -diff -urNd -urNd sysvinit-2.85/src/bootlogd.c sysvinit-2.86/src/bootlogd.c ---- sysvinit-2.85/src/bootlogd.c 2001-12-09 08:01:28.000000000 -0600 -+++ sysvinit-2.86/src/bootlogd.c 2004-06-09 07:47:45.000000000 -0500 -@@ -3,12 +3,12 @@ - * The file is usually located on the /var partition, and - * gets written (and fsynced) as soon as possible. - * -- * Version: @(#)bootlogd 2.79 11-Sep-2000 miquels@cistron.nl -+ * Version: @(#)bootlogd 2.86pre 12-Jan-2004 miquels@cistron.nl - * - * Bugs: Uses openpty(), only available in glibc. Sorry. - * - * This file is part of the sysvinit suite, -- * Copyright 1991-2000 Miquel van Smoorenburg. -+ * Copyright 1991-2004 Miquel van Smoorenburg. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License -@@ -17,7 +17,9 @@ - * - * *NOTE* *NOTE* *NOTE* - * This is a PROOF OF CONCEPT IMPLEMENTATION -- * I do not recommend using this on production systems. -+ * -+ * I have bigger plans for Debian, but for now -+ * this has to do ;) - * - */ - -@@ -38,18 +40,14 @@ - #include <dirent.h> - #include <fcntl.h> - #include <pty.h> -- --char *Version = "@(#) bootlogd 2.79 11-Sep-2000 MvS"; -- --/* -- * Until the kernel knows about TIOCGDEV, use a really ugly -- * non-portable (not even between architectures) hack. -- */ --#ifndef TIOCGDEV --# define TIOCTTYGSTRUCT_HACK 1 -+#include <ctype.h> -+#ifdef __linux__ -+#include <sys/mount.h> - #endif - --#define LOGFILE "/var/log/boot.log" -+char *Version = "@(#) bootlogd 2.86 03-Jun-2004 miquels@cistron.nl"; -+ -+#define LOGFILE "/var/log/boot" - - char ringbuf[32768]; - char *endptr = ringbuf + sizeof(ringbuf); -@@ -59,29 +57,32 @@ - int got_signal = 0; - int didnl = 1; - -+struct line { -+ char buf[256]; -+ int pos; -+} line; - --#ifdef TIOCTTYGSTRUCT_HACK --struct tty_offsets { -- char *kver; -- int offset; --} tty_offsets[] = { --#if ((~0UL) == 0xffffffff) /* 32 bits */ -- { "2.0.", 236 }, -- { "2.1.", 268 }, -- { "2.2.", 272 }, -- { "2.3.", 272 }, -- { "2.4.", 272 }, -- { "2.5.", 272 }, --#else /* 64 bits */ -- { "2.2.", 480 }, -- { "2.3.", 480 }, -- { "2.4.", 480 }, -- { "2.5.", 480 }, --#endif -- { NULL, 0 }, -+/* -+ * Console devices as listed on the kernel command line and -+ * the mapping to actual devices in /dev -+ */ -+struct consdev { -+ char *cmdline; -+ char *dev1; -+ char *dev2; -+} consdev[] = { -+ { "ttySC", "/dev/ttySC%s", "/dev/ttsc/%s" }, -+ { "ttyS", "/dev/ttyS%s", "/dev/tts/%s" }, -+ { "tty", "/dev/tty%s", "/dev/vc/%s" }, -+ { "hvc", "/dev/hvc%s", "/dev/hvc/%s" }, -+ { NULL, NULL, NULL }, - }; --#endif - -+/* -+ * Devices to try as console if not found on kernel command line. -+ * Tried from left to right (as opposed to kernel cmdline). -+ */ -+char *defcons[] = { "tty0", "hvc0", "ttyS0", "ttySC0", NULL }; - - /* - * Catch signals. -@@ -95,6 +96,8 @@ - /* - * Scan /dev and find the device name. - * Side-effect: directory is changed to /dev -+ * -+ * FIXME: scan subdirectories for devfs support ? - */ - int findtty(char *res, int rlen, dev_t dev) - { -@@ -117,18 +120,88 @@ - } - } - if (ent == NULL) { -- fprintf(stderr, "bootlogd: cannot find console device\n"); -+ fprintf(stderr, "bootlogd: cannot find console device " -+ "%d:%d in /dev\n", major(dev), minor(dev)); - r = -1; -- } else if (strlen(ent->d_name) >= rlen) { -+ } else if (strlen(ent->d_name) + 5 >= rlen) { - fprintf(stderr, "bootlogd: console device name too long\n"); - r = -1; - } else -- strcpy(res, ent->d_name); -+ snprintf(res, rlen, "/dev/%s", ent->d_name); - closedir(dir); - - return r; - } - -+/* -+ * For some reason, openpty() in glibc sometimes doesn't -+ * work at boot-time. It must be a bug with old-style pty -+ * names, as new-style (/dev/pts) is not available at that -+ * point. So, we find a pty/tty pair ourself if openpty() -+ * fails for whatever reason. -+ */ -+int findpty(int *master, int *slave, char *name) -+{ -+ char pty[16]; -+ char tty[16]; -+ int i, j; -+ int found; -+ -+ if (openpty(master, slave, name, NULL, NULL) >= 0) -+ return 0; -+ -+ found = 0; -+ -+ for (i = 'p'; i <= 'z'; i++) { -+ for (j = '0'; j <= 'f'; j++) { -+ if (j == '9' + 1) j = 'a'; -+ sprintf(pty, "/dev/pty%c%c", i, j); -+ sprintf(tty, "/dev/tty%c%c", i, j); -+ if ((*master = open(pty, O_RDWR|O_NOCTTY)) >= 0) { -+ *slave = open(tty, O_RDWR|O_NOCTTY); -+ if (*slave >= 0) { -+ found = 1; -+ break; -+ } -+ } -+ } -+ if (found) break; -+ } -+ if (found < 0) return -1; -+ -+ if (name) strcpy(name, tty); -+ -+ return 0; -+} -+/* -+ * See if a console taken from the kernel command line maps -+ * to a character device we know about, and if we can open it. -+ */ -+int isconsole(char *s, char *res, int rlen) -+{ -+ struct consdev *c; -+ int l, sl, i, fd; -+ char *p, *q; -+ -+ sl = strlen(s); -+ -+ for (c = consdev; c->cmdline; c++) { -+ l = strlen(c->cmdline); -+ if (sl <= l) continue; -+ p = s + l; -+ if (strncmp(s, c->cmdline, l) != 0 || !isdigit(*p)) -+ continue; -+ for (i = 0; i < 2; i++) { -+ snprintf(res, rlen, i ? c->dev1 : c->dev2, p); -+ if ((q = strchr(res, ',')) != NULL) *q = 0; -+ if ((fd = open(res, O_RDONLY|O_NONBLOCK)) >= 0) { -+ close(fd); -+ return 1; -+ } -+ } -+ } -+ return 0; -+} - - /* - * Find out the _real_ console. Assume that stdin is connected to -@@ -136,21 +209,18 @@ - */ - int consolename(char *res, int rlen) - { -- struct stat st; --#if TIOCTTYGSTRUCT_HACK -- struct utsname uts; -- struct tty_offsets *tt; -- dev_t dev; -- unsigned short *kdev; -- char buf[4096]; -- int offset = -1; --#endif - #ifdef TIOCGDEV -- kdev_t kdev; -+ unsigned int kdev; - #endif -+ struct stat st, st2; -+ char buf[256]; -+ char *p; -+ int didmount = 0; -+ int n, r; -+ int fd; - - fstat(0, &st); -- if (st.st_rdev != 0x0501) { -+ if (major(st.st_rdev) != 5 || minor(st.st_rdev) != 1) { - /* - * Old kernel, can find real device easily. - */ -@@ -160,33 +230,78 @@ - #ifdef TIOCGDEV - if (ioctl(0, TIOCGDEV, &kdev) == 0) - return findtty(res, rlen, (dev_t)kdev); -- return -1; -+ if (errno != ENOIOCTLCMD) return -1; - #endif - -+#ifdef __linux__ - /* -- * New kernel and new console device - hard to find -- * out what device the real console is .. -+ * Read /proc/cmdline. - */ --#if TIOCTTYGSTRUCT_HACK -- if (ioctl(0, TIOCTTYGSTRUCT, buf) != 0) { -- perror("bootlogd: TIOCTTYGSTRUCT"); -+ stat("/", &st); -+ if (stat("/proc", &st2) < 0) { -+ perror("bootlogd: /proc"); - return -1; - } -- uname(&uts); -- for (tt = tty_offsets; tt->kver; tt++) -- if (!strncmp(uts.release, tt->kver, strlen(tt->kver))) { -- offset = tt->offset; -+ if (st.st_dev == st2.st_dev) { -+ if (mount("proc", "/proc", "proc", 0, NULL) < 0) { -+ perror("bootlogd: mount /proc"); -+ return -1; -+ } -+ didmount = 1; -+ } -+ -+ n = 0; -+ r = -1; -+ if ((fd = open("/proc/cmdline", O_RDONLY)) < 0) { -+ perror("bootlogd: /proc/cmdline"); -+ } else { -+ buf[0] = 0; -+ if ((n = read(fd, buf, sizeof(buf) - 1)) >= 0) -+ r = 0; -+ else -+ perror("bootlogd: /proc/cmdline"); -+ close(fd); -+ } -+ if (didmount) umount("/proc"); -+ -+ if (r < 0) return r; -+ -+ /* -+ * OK, so find console= in /proc/cmdline. -+ * Parse in reverse, opening as we go. -+ * -+ * Valid console devices: ttySC, ttyS, tty, hvc. -+ */ -+ p = buf + n; -+ *p-- = 0; -+ r = -1; -+ while (p >= buf) { -+ if (*p == ' ' || *p == '\t' || *p == '\r' || *p == '\n') { -+ *p-- = 0; -+ continue; -+ } -+ if (strncmp(p, "console=", 8) == 0 && -+ isconsole(p + 8, res, rlen)) { -+ r = 0; - break; - } -- if (offset < 0) { -- fprintf(stderr, "bootlogd: don't know offsetof" -- "(struct tty_struct, device) for kernel %s\n", uts.release); -- return -1; -+ p--; - } -- kdev = (unsigned short *)(&buf[offset]); -- dev = (dev_t)(*kdev); -- return findtty(res, rlen, dev); -+ -+ if (r == 0) return r; - #endif -+ -+ /* -+ * Okay, no console on the command line - -+ * guess the default console. -+ */ -+ for (n = 0; defcons[n]; n++) -+ if (isconsole(defcons[n], res, rlen)) -+ return 0; -+ -+ fprintf(stderr, "bootlogd: cannot deduce real console device\n"); -+ -+ return -1; - } - - -@@ -197,9 +312,13 @@ - { - time_t t; - char *s; -+ char tmp[8]; - int olen = len; -+ int dosync = 0; -+ int tlen; - - while (len > 0) { -+ tmp[0] = 0; - if (didnl) { - time(&t); - s = ctime(&t); -@@ -207,24 +326,51 @@ - didnl = 0; - } - switch (*ptr) { -+ case 27: /* ESC */ -+ strcpy(tmp, "^["); -+ break; - case '\r': -+ line.pos = 0; -+ break; -+ case 8: /* ^H */ -+ if (line.pos > 0) line.pos--; - break; - case '\n': - didnl = 1; -+ dosync = 1; -+ break; - case '\t': -+ line.pos += (line.pos / 8 + 1) * 8; -+ if (line.pos >= sizeof(line.buf)) -+ line.pos = sizeof(line.buf) - 1; -+ break; - case 32 ... 127: - case 161 ... 255: -- fputc(*ptr, fp); -+ tmp[0] = *ptr; -+ tmp[1] = 0; - break; - default: -- fprintf(fp, "\\%03o", *ptr); -+ sprintf(tmp, "\\%03o", *ptr); - break; - } - ptr++; - len--; -+ -+ tlen = strlen(tmp); -+ if (tlen && (line.pos + tlen < sizeof(line.buf))) { -+ memcpy(line.buf + line.pos, tmp, tlen); -+ line.pos += tlen; -+ } -+ if (didnl) { -+ fprintf(fp, "%s\n", line.buf); -+ memset(&line, 0, sizeof(line)); -+ } -+ } -+ -+ if (dosync) { -+ fflush(fp); -+ fdatasync(fileno(fp)); - } -- fflush(fp); -- fdatasync(fileno(fp)); - - outptr += olen; - if (outptr >= endptr) -@@ -242,6 +388,40 @@ - exit(1); - } - -+int open_nb(char *buf) -+{ -+ int fd, n; -+ -+ if ((fd = open(buf, O_WRONLY|O_NONBLOCK|O_NOCTTY)) < 0) -+ return -1; -+ n = fcntl(fd, F_GETFL); -+ n &= ~(O_NONBLOCK); -+ fcntl(fd, F_SETFL, n); -+ -+ return fd; -+} -+ -+/* -+ * We got a write error on the real console. If its an EIO, -+ * somebody hung up our filedescriptor, so try to re-open it. -+ */ -+int write_err(int pts, int realfd, char *realcons, int e) -+{ -+ int fd; -+ -+ if (e != EIO) { -+werr: -+ close(pts); -+ fprintf(stderr, "bootlogd: writing to console: %s\n", -+ strerror(e)); -+ return -1; -+ } -+ close(realfd); -+ if ((fd = open_nb(realcons)) < 0) -+ goto werr; -+ -+ return fd; -+} - - int main(int argc, char **argv) - { -@@ -249,6 +429,7 @@ - struct timeval tv; - fd_set fds; - char buf[1024]; -+ char realcons[1024]; - char *p; - char *logfile; - char *pidfile; -@@ -298,23 +479,32 @@ - /* - * Open console device directly. - */ -- if (consolename(buf, sizeof(buf)) < 0) -+ if (consolename(realcons, sizeof(realcons)) < 0) - return 1; -- if ((realfd = open(buf, O_WRONLY|O_NONBLOCK)) < 0) { -+ -+ if (strcmp(realcons, "/dev/tty0") == 0) -+ strcpy(realcons, "/dev/tty1"); -+ if (strcmp(realcons, "/dev/vc/0") == 0) -+ strcpy(realcons, "/dev/vc/1"); -+ -+ if ((realfd = open_nb(realcons)) < 0) { - fprintf(stderr, "bootlogd: %s: %s\n", buf, strerror(errno)); - return 1; - } -- n = fcntl(realfd, F_GETFL); -- n &= ~(O_NONBLOCK); -- fcntl(realfd, F_SETFL, n); - - /* - * Grab a pty, and redirect console messages to it. - */ -- if (openpty(&ptm, &pts, buf, NULL, NULL) < 0) { -- fprintf(stderr, "bootlogd: cannot allocate pseudo tty\n"); -+ ptm = -1; -+ pts = -1; -+ buf[0] = 0; -+ if (findpty(&ptm, &pts, buf) < 0) { -+ fprintf(stderr, -+ "bootlogd: cannot allocate pseudo tty: %s\n", -+ strerror(errno)); - return 1; - } -+ - (void)ioctl(0, TIOCCONS, NULL); - #if 1 - /* Work around bug in 2.1/2.2 kernels. Fixed in 2.2.13 and 2.3.18 */ -@@ -357,8 +547,8 @@ - * open the logfile. There might be buffered messages - * we want to write. - */ -- tv.tv_sec = fp ? 86400 : 5; -- tv.tv_usec = 0; -+ tv.tv_sec = 0; -+ tv.tv_usec = 500000; - FD_ZERO(&fds); - FD_SET(ptm, &fds); - if (select(ptm + 1, &fds, NULL, NULL, &tv) == 1) { -@@ -374,10 +564,22 @@ - p = inptr; - while (m > 0) { - i = write(realfd, p, m); -- if (i <= 0) break; -- m -= i; -- p += i; -+ if (i >= 0) { -+ m -= i; -+ p += i; -+ continue; -+ } -+ /* -+ * Handle EIO (somebody hung -+ * up our filedescriptor) -+ */ -+ realfd = write_err(pts, realfd, -+ realcons, errno); -+ if (realfd >= 0) continue; -+ got_signal = 1; /* Not really */ -+ break; - } -+ - /* - * Increment buffer position. Handle - * wraps, and also drag output pointer -@@ -410,8 +612,8 @@ - writelog(fp, outptr, todo); - } - -- if (fp && !didnl) { -- fputc('\n', fp); -+ if (fp) { -+ if (!didnl) fputc('\n', fp); - fclose(fp); - } - -Binary files sysvinit-2.85/src/bootlogd.o and sysvinit-2.86/src/bootlogd.o differ -diff -urNd -urNd sysvinit-2.85/src/dowall.c sysvinit-2.86/src/dowall.c ---- sysvinit-2.85/src/dowall.c 2003-04-17 06:32:01.000000000 -0500 -+++ sysvinit-2.86/src/dowall.c 2004-06-09 07:47:45.000000000 -0500 -@@ -3,7 +3,7 @@ - * - * Author: Miquel van Smoorenburg, miquels@cistron.nl - * -- * Version: @(#)dowall.c 2.85-1 15-Apr-2003 miquels@cistron.nl -+ * Version: @(#)dowall.c 2.85-5 02-Jul-2003 miquels@cistron.nl - * - * This file is part of the sysvinit suite, - * Copyright 1991-2003 Miquel van Smoorenburg. -@@ -135,6 +135,13 @@ - char *user, *tty; - int fd, flags; - -+ /* -+ * Make sure tp and fd aren't in a register. Some versions -+ * of gcc clobber those after longjmp (or so I understand). -+ */ -+ (void) &tp; -+ (void) &fd; -+ - getuidtty(&user, &tty); - - /* Get the time */ -Binary files sysvinit-2.85/src/dowall.o and sysvinit-2.86/src/dowall.o differ -Binary files sysvinit-2.85/src/halt and sysvinit-2.86/src/halt differ -diff -urNd -urNd sysvinit-2.85/src/halt.c sysvinit-2.86/src/halt.c ---- sysvinit-2.85/src/halt.c 2001-11-27 06:12:03.000000000 -0600 -+++ sysvinit-2.86/src/halt.c 2004-07-30 07:16:18.000000000 -0500 -@@ -8,12 +8,14 @@ - * execute an "shutdown -r". This is for compatibility with - * sysvinit 2.4. - * -- * Usage: halt [-n] [-w] [-d] [-f] [-p] -+ * Usage: halt [-n] [-w] [-d] [-f] [-h] [-i] [-p] - * -n: don't sync before halting the system - * -w: only write a wtmp reboot record and exit. - * -d: don't write a wtmp record. - * -f: force halt/reboot, don't call shutdown. -- * -p: power down the system (if possible, otherwise halt) -+ * -h: put harddisks in standby mode -+ * -i: shut down all network interfaces. -+ * -p: power down the system (if possible, otherwise halt). - * - * Reboot and halt are both this program. Reboot - * is just a link to halt. Invoking the program -@@ -21,10 +23,10 @@ - * - * Author: Miquel van Smoorenburg, miquels@cistron.nl - * -- * Version: 2.84, 27-Nov-2001 -+ * Version: 2.86, 30-Jul-2004 - * - * This file is part of the sysvinit suite, -- * Copyright 1991-2001 Miquel van Smoorenburg. -+ * Copyright 1991-2004 Miquel van Smoorenburg. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License -@@ -47,7 +49,7 @@ - #include <getopt.h> - #include "reboot.h" - --char *Version = "@(#)halt 2.84 27-Nov-2001 miquels@cistron.nl"; -+char *Version = "@(#)halt 2.86 31-Jul-2004 miquels@cistron.nl"; - char *progname; - - #define KERNEL_MONITOR 1 /* If halt() puts you into the kernel monitor. */ -@@ -62,7 +64,16 @@ - */ - void usage(void) - { -- fprintf(stderr, "usage: %s [-n] [-w] [-d] [-f] [-i] [-p]\n", progname); -+ fprintf(stderr, "usage: %s [-n] [-w] [-d] [-f] [-h] [-i]%s\n", -+ progname, strcmp(progname, "halt") ? "" : " [-p]"); -+ fprintf(stderr, "\t-n: don't sync before halting the system\n"); -+ fprintf(stderr, "\t-w: only write a wtmp reboot record and exit.\n"); -+ fprintf(stderr, "\t-d: don't write a wtmp record.\n"); -+ fprintf(stderr, "\t-f: force halt/reboot, don't call shutdown.\n"); -+ fprintf(stderr, "\t-h: put harddisks in standby mode.\n"); -+ fprintf(stderr, "\t-i: shut down all network interfaces.\n"); -+ if (!strcmp(progname, "halt")) -+ fprintf(stderr, "\t-p: power down the system (if possible, otherwise halt).\n"); - exit(1); - } - -@@ -172,11 +183,6 @@ - else - progname = argv[0]; - -- if (geteuid() != 0) { -- fprintf(stderr, "%s: must be superuser.\n", progname); -- exit(1); -- } -- - if (!strcmp(progname, "reboot")) do_reboot = 1; - if (!strcmp(progname, "poweroff")) do_poweroff = 1; - -@@ -216,6 +222,11 @@ - } - if (argc != optind) usage(); - -+ if (geteuid() != 0) { -+ fprintf(stderr, "%s: must be superuser.\n", progname); -+ exit(1); -+ } -+ - (void)chdir("/"); - - if (!do_hard && !do_nothing) { -@@ -236,7 +247,7 @@ - /* - * Exit if all we wanted to do was write a wtmp record. - */ -- if (do_nothing) exit(0); -+ if (do_nothing && !do_hddown && !do_ifdown) exit(0); - - if (do_sync) { - sync(); -@@ -249,13 +260,17 @@ - if (do_hddown) - (void)hddown(); - -+ if (do_nothing) exit(0); -+ - if (do_reboot) { - init_reboot(BMAGIC_REBOOT); - } else { - /* - * Turn on hard reboot, CTRL-ALT-DEL will reboot now - */ -+#ifdef BMAGIC_HARD - init_reboot(BMAGIC_HARD); -+#endif - - /* - * Stop init; it is insensitive to the signals sent -@@ -277,7 +292,9 @@ - /* - * If we return, we (c)ontinued from the kernel monitor. - */ -+#ifdef BMAGIC_SOFT - init_reboot(BMAGIC_SOFT); -+#endif - kill(1, SIGCONT); - - exit(0); -Binary files sysvinit-2.85/src/halt.o and sysvinit-2.86/src/halt.o differ -diff -urNd -urNd sysvinit-2.85/src/hddown.c sysvinit-2.86/src/hddown.c ---- sysvinit-2.85/src/hddown.c 2001-11-07 09:11:21.000000000 -0600 -+++ sysvinit-2.86/src/hddown.c 2004-06-09 07:47:45.000000000 -0500 -@@ -3,7 +3,7 @@ - * shut them down. - * - */ --char *v_hddown = "@(#)hddown.c 1.01 07-Nov-2001 miquels@cistron.nl"; -+char *v_hddown = "@(#)hddown.c 1.02 22-Apr-2003 miquels@cistron.nl"; - - #include <stdio.h> - #include <stdlib.h> -@@ -13,8 +13,9 @@ - #include <fcntl.h> - #include <dirent.h> - --#include <sys/ioctl.h> -+#ifdef __linux__ - -+#include <sys/ioctl.h> - #include <linux/hdreg.h> - - #define MAX_DISKS 64 -@@ -104,6 +105,15 @@ - return 0; - } - -+#else /* __linux__ */ -+ -+int hddown(void) -+{ -+ return 0; -+} -+ -+#endif /* __linux__ */ -+ - #ifdef STANDALONE - int main(int argc, char **argv) - { -Binary files sysvinit-2.85/src/hddown.o and sysvinit-2.86/src/hddown.o differ -Binary files sysvinit-2.85/src/ifdown.o and sysvinit-2.86/src/ifdown.o differ -Binary files sysvinit-2.85/src/init and sysvinit-2.86/src/init differ -diff -urNd -urNd sysvinit-2.85/src/init.c sysvinit-2.86/src/init.c ---- sysvinit-2.85/src/init.c 2003-04-15 06:16:41.000000000 -0500 -+++ sysvinit-2.86/src/init.c 2004-07-30 07:16:20.000000000 -0500 -@@ -5,34 +5,28 @@ - * init [0123456SsQqAaBbCc] - * telinit [0123456SsQqAaBbCc] - * -- * Version: @(#)init.c 2.85 15-Apr-2003 miquels@cistron.nl -+ * Version: @(#)init.c 2.86 30-Jul-2004 miquels@cistron.nl - */ --#define VERSION "2.85" --#define DATE "15-Apr-2003" -+#define VERSION "2.86" -+#define DATE "31-Jul-2004" - /* - * This file is part of the sysvinit suite, -- * Copyright 1991-2003 Miquel van Smoorenburg. -+ * Copyright 1991-2004 Miquel van Smoorenburg. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version - * 2 of the License, or (at your option) any later version. - * -- * Modified: 21 Feb 1998, Al Viro: -- * 'U' flag added to telinit. It forces init to re-exec itself -- * (passing its state through exec, certainly). -- * May be useful for smoother (heh) upgrades. -- * 24 Feb 1998, AV: -- * did_boot made global and added to state - thanks, Miquel. -- * Yet another file descriptors leak - close state pipe if -- * re_exec fails. - */ - - #include <sys/types.h> - #include <sys/stat.h> - #include <sys/ioctl.h> - #include <sys/wait.h> -+#ifdef __linux__ - #include <sys/kd.h> -+#endif - #include <sys/resource.h> - #include <stdlib.h> - #include <unistd.h> -@@ -70,6 +64,13 @@ - # define SIGPWR SIGUSR2 - #endif - -+#ifndef CBAUD -+# define CBAUD 0 -+#endif -+#ifndef CBAUDEX -+# define CBAUDEX 0 -+#endif -+ - /* Set a signal handler. */ - #define SETSIG(sa, sig, fun, flags) \ - do { \ -@@ -88,13 +89,13 @@ - CHILD *newFamily = NULL; /* The list after inittab re-read */ - - CHILD ch_emerg = { /* Emergency shell */ -- 0, 0, 0, 0, 0, -- "~~", -- "S", -- 3, -- "/sbin/sulogin", -- NULL, -- NULL -+ 0, 0, 0, 0, 0, -+ "~~", -+ "S", -+ 3, -+ "/sbin/sulogin", -+ NULL, -+ NULL - }; - - char runlevel = 'S'; /* The current run level */ -@@ -108,8 +109,9 @@ - int wrote_utmp_reboot = 1; /* Set when we wrote the reboot record */ - int sltime = 5; /* Sleep time between TERM and KILL */ - char *argv0; /* First arguments; show up in ps listing */ --int maxproclen; /* Maximal length of argv[0] without \0 */ -+int maxproclen; /* Maximal length of argv[0] with \0 */ - struct utmp utproto; /* Only used for sizeof(utproto.ut_id) */ -+char *user_console = NULL; /* User console device */ - char *console_dev; /* Console device. */ - int pipe_fd = -1; /* /dev/initctl */ - int did_boot = 0; /* Did we already do BOOT* stuff? */ -@@ -186,6 +188,10 @@ - {NULL,0} - }; - -+#define NR_EXTRA_ENV 16 -+char *extra_env[NR_EXTRA_ENV]; -+ -+ - /* - * Sleep a number of seconds. - * -@@ -203,6 +209,35 @@ - ; - } - -+ -+/* -+ * Non-failing allocation routines (init cannot fail). -+ */ -+void *imalloc(size_t size) -+{ -+ void *m; -+ -+ while ((m = malloc(size)) == NULL) { -+ initlog(L_VB, "out of memory"); -+ do_sleep(5); -+ } -+ memset(m, 0, size); -+ return m; -+} -+ -+ -+char *istrdup(char *s) -+{ -+ char *m; -+ int l; -+ -+ l = strlen(s) + 1; -+ m = imalloc(l); -+ memcpy(m, s, l); -+ return m; -+} -+ -+ - /* - * Send the state info of the previous running init to - * the new one, in a version-independant way. -@@ -344,12 +379,9 @@ - } - } while (cmd != C_REC); - -- while ((p = (CHILD *)malloc(sizeof(CHILD))) == NULL ) { -- log(L_VB, "out of memory"); -- do_sleep(5); -- } -- memset(p, 0, sizeof(CHILD)); -+ p = imalloc(sizeof(CHILD)); - get_string(p->id, sizeof(p->id), f); -+ - do switch(cmd = get_cmd(f)) { - case 0: - case C_EOR: -@@ -420,7 +452,7 @@ - #ifdef __GNUC__ - __attribute__ ((format (printf, 1, 2))) - #endif --int setproctitle(char *fmt, ...) -+static int setproctitle(char *fmt, ...) - { - va_list ap; - int len; -@@ -432,8 +464,10 @@ - len = vsnprintf(buf, sizeof(buf), fmt, ap); - va_end(ap); - -- memset(argv0, 0, maxproclen + 1); -- strncpy(argv0, buf, maxproclen - 1); -+ if (maxproclen > 2) { -+ memset(argv0, 0, maxproclen); -+ strncpy(argv0, buf, maxproclen - 2); -+ } - - return len; - } -@@ -448,7 +482,9 @@ - int tried_vtmaster = 0; - char *s; - -- if ((s = getenv("CONSOLE")) != NULL) -+ if (user_console) { -+ console_dev = user_console; -+ } else if ((s = getenv("CONSOLE")) != NULL) - console_dev = s; - else { - console_dev = CONSOLE; -@@ -528,10 +564,9 @@ - if (errno == ECHILD) break; - for( ch = family; ch; ch = ch->next ) - if ( ch->pid == pid && (ch->flags & RUNNING) ) { --#if DEBUG -- log(L_VB, "chld_handler: marked %d as zombie", -+ INITDBG(L_VB, -+ "chld_handler: marked %d as zombie", - ch->pid); --#endif - ADDSET(got_signals, SIGCHLD); - ch->exstat = st; - ch->flags |= ZOMBIE; -@@ -541,11 +576,9 @@ - } - break; - } --#if DEBUG - if (ch == NULL) -- log(L_VB, "chld_handler: unknown child %d exited.", -+ INITDBG(L_VB, "chld_handler: unknown child %d exited.", - pid); --#endif - } - errno = saved_errno; - } -@@ -563,28 +596,34 @@ - } - - /* -- * Dump core. Returns 0 if we are the child, so that the caller -- * can return if it is a signal handler - SIGSEGV is blocked in -- * the handler, so it will be raised when the handler returns. -+ * Fork and dump core in /. - */ --int coredump(void) -+void coredump(void) - { -- static int dumped = 0; -- struct rlimit rlim; -+ static int dumped = 0; -+ struct rlimit rlim; -+ sigset_t mask; - -- if (dumped) return 1; -+ if (dumped) return; - dumped = 1; - -- if (fork() != 0) return 1; -+ if (fork() != 0) return; -+ -+ sigfillset(&mask); -+ sigprocmask(SIG_SETMASK, &mask, NULL); - - rlim.rlim_cur = RLIM_INFINITY; - rlim.rlim_max = RLIM_INFINITY; - setrlimit(RLIMIT_CORE, &rlim); -- - chdir("/"); -+ - signal(SIGSEGV, SIG_DFL); - raise(SIGSEGV); -- return 0; -+ sigdelset(&mask, SIGSEGV); -+ sigprocmask(SIG_SETMASK, &mask, NULL); -+ -+ do_sleep(5); -+ exit(0); - } - - /* -@@ -592,7 +631,7 @@ - * If we have the info, print where it occured. - * Then sleep 30 seconds and try to continue. - */ --#ifdef STACK_DEBUG -+#if defined(STACK_DEBUG) && defined(__linux__) - void segv_handler(int sig, struct sigcontext ctx) - { - char *p = ""; -@@ -601,10 +640,10 @@ - if ((void *)ctx.eip >= (void *)do_sleep && - (void *)ctx.eip < (void *)main) - p = " (code)"; -- log(L_VB, "PANIC: segmentation violation at %p%s! " -+ initlog(L_VB, "PANIC: segmentation violation at %p%s! " - "sleeping for 30 seconds.", (void *)ctx.eip, p); -- if (coredump() != 0) -- do_sleep(30); -+ coredump(); -+ do_sleep(30); - errno = saved_errno; - } - #else -@@ -612,9 +651,10 @@ - { - int saved_errno = errno; - -- log(L_VB, "PANIC: segmentation violation! sleeping for 30 seconds."); -- if (coredump() != 0) -- do_sleep(30); -+ initlog(L_VB, -+ "PANIC: segmentation violation! sleeping for 30 seconds."); -+ coredump(); -+ do_sleep(30); - errno = saved_errno; - } - #endif -@@ -641,7 +681,7 @@ - int fd; - - if ((fd = console_open(O_RDWR|O_NOCTTY)) < 0) { -- log(L_VB, "can't open %s", console_dev); -+ initlog(L_VB, "can't open %s", console_dev); - return; - } - -@@ -697,10 +737,11 @@ - #ifdef __GNUC__ - __attribute__ ((format (printf, 2, 3))) - #endif --void log(int loglevel, char *s, ...) -+void initlog(int loglevel, char *s, ...) - { - va_list va_alist; - char buf[256]; -+ sigset_t nmask, omask; - - va_start(va_alist, s); - vsnprintf(buf, sizeof(buf), s, va_alist); -@@ -708,11 +749,15 @@ - - if (loglevel & L_SY) { - /* -- * Re-etablish connection with syslogd every time. -+ * Re-establish connection with syslogd every time. -+ * Block signals while talking to syslog. - */ -+ sigfillset(&nmask); -+ sigprocmask(SIG_BLOCK, &nmask, &omask); - openlog("init", 0, LOG_DAEMON); - syslog(LOG_INFO, "%s", buf); - closelog(); -+ sigprocmask(SIG_SETMASK, &omask, NULL); - } - - /* -@@ -727,14 +772,51 @@ - - - /* -- * See if one character of s2 is in s1 -+ * Build a new environment for execve(). - */ --int any(char *s1, char *s2) -+char **init_buildenv(int child) - { -- while(*s2) -- if (strchr(s1, *s2++) != NULL) -- return(1); -- return(0); -+ char i_lvl[] = "RUNLEVEL=x"; -+ char i_prev[] = "PREVLEVEL=x"; -+ char i_cons[32]; -+ char **e; -+ int n, i; -+ -+ for (n = 0; environ[n]; n++) -+ ; -+ n += NR_EXTRA_ENV + 8; -+ e = calloc(n, sizeof(char *)); -+ -+ for (n = 0; environ[n]; n++) -+ e[n] = istrdup(environ[n]); -+ -+ for (i = 0; i < NR_EXTRA_ENV; i++) -+ if (extra_env[i]) -+ e[n++] = istrdup(extra_env[i]); -+ -+ if (child) { -+ snprintf(i_cons, sizeof(i_cons), "CONSOLE=%s", console_dev); -+ i_lvl[9] = thislevel; -+ i_prev[10] = prevlevel; -+ e[n++] = istrdup(i_lvl); -+ e[n++] = istrdup(i_prev); -+ e[n++] = istrdup(i_cons); -+ e[n++] = istrdup(E_VERSION); -+ } -+ -+ e[n++] = NULL; -+ -+ return e; -+} -+ -+ -+void init_freeenv(char **e) -+{ -+ int n; -+ -+ for (n = 0; e[n]; n++) -+ free(e[n]); -+ free(e); - } - - -@@ -753,9 +835,6 @@ - time_t t; /* System time */ - int oldAlarm; /* Previous alarm value */ - char *proc = ch->process; /* Command line */ -- char i_lvl[] = "RUNLEVEL=x"; /* Runlevel in environment. */ -- char i_prev[] = "PREVLEVEL=x";/* Previous runlevel. */ -- char i_cons[32]; /* console device. */ - pid_t pid, pgrp; /* child, console process group. */ - sigset_t nmask, omask; /* For blocking SIGCHLD */ - struct sigaction sa; -@@ -781,8 +860,9 @@ - /* Do we try to respawn too fast? */ - if (ch->count >= MAXSPAWN) { - -- log(L_VB, "Id \"%s\" respawning too fast: disabled for %d minutes", -- ch->id, SLEEPTIME / 60); -+ initlog(L_VB, -+ "Id \"%s\" respawning too fast: disabled for %d minutes", -+ ch->id, SLEEPTIME / 60); - ch->flags &= ~RUNNING; - ch->flags |= FAILING; - -@@ -813,7 +893,7 @@ - } - args[6] = proc; - args[7] = NULL; -- } else if (any(proc, "~`!$^&*()=|\\{}[];\"'<>?")) { -+ } else if (strpbrk(proc, "~`!$^&*()=|\\{}[];\"'<>?")) { - /* See if we need to fire off a shell for this command */ - /* Give command line to shell */ - args[1] = SHELL; -@@ -868,15 +948,6 @@ - - sigprocmask(SIG_SETMASK, &omask, NULL); - -- /* Now set RUNLEVEL and PREVLEVEL */ -- snprintf(i_cons, sizeof(i_cons), "CONSOLE=%s", console_dev); -- i_lvl[9] = thislevel; -- i_prev[10] = prevlevel; -- putenv(i_lvl); -- putenv(i_prev); -- putenv(i_cons); -- putenv(E_VERSION); -- - /* - * In sysinit, boot, bootwait or single user mode: - * for any wait-type subprocess we _force_ the console -@@ -896,7 +967,7 @@ - dup(f); - } - if ((pid = fork()) < 0) { -- log(L_VB, "cannot fork"); -+ initlog(L_VB, "cannot fork"); - exit(1); - } - if (pid > 0) { -@@ -926,7 +997,7 @@ - * this with a temporary process. - */ - if ((pid = fork()) < 0) { -- log(L_VB, "cannot fork"); -+ initlog(L_VB, "cannot fork"); - exit(1); - } - if (pid == 0) { -@@ -946,7 +1017,7 @@ - } else { - setsid(); - if ((f = console_open(O_RDWR|O_NOCTTY)) < 0) { -- log(L_VB, "open(%s): %s", console_dev, -+ initlog(L_VB, "open(%s): %s", console_dev, - strerror(errno)); - f = open("/dev/null", O_RDWR); - } -@@ -954,15 +1025,15 @@ - dup(f); - } - -- /* Reset all the signals */ -+ /* Reset all the signals, set up environment */ - for(f = 1; f < NSIG; f++) SETSIG(sa, f, SIG_DFL, SA_RESTART); -- execvp(args[1], args + 1); -+ environ = init_buildenv(1); - - /* -- * Is this a bug in execvp? It does _not_ execute shell -- * scripts (/etc/rc !), so we try again with -- * 'sh -c exec ...' -+ * Execute prog. In case of ENOEXEC try again -+ * as a shell script. - */ -+ execvp(args[1], args + 1); - if (errno == ENOEXEC) { - args[1] = SHELL; - args[2] = "-c"; -@@ -972,18 +1043,16 @@ - args[4] = NULL; - execvp(args[1], args + 1); - } -- log(L_VB, "cannot execute \"%s\"", args[1]); -+ initlog(L_VB, "cannot execute \"%s\"", args[1]); - exit(1); - } - *res = pid; - sigprocmask(SIG_SETMASK, &omask, NULL); - --#if DEBUG -- log(L_VB, "Started id %s (pid %d)", ch->id, pid); --#endif -+ INITDBG(L_VB, "Started id %s (pid %d)", ch->id, pid); - - if (pid == -1) { -- log(L_VB, "cannot fork, retry.."); -+ initlog(L_VB, "cannot fork, retry.."); - do_sleep(5); - continue; - } -@@ -1032,66 +1101,45 @@ - } - } - --/* -- * My version of strtok(3). -- */ --char *get_part(char *str, int tok) --{ -- static char *s; -- char *p, *q; -- -- if (str != NULL) -- s = str; -- if (s == NULL || *s == 0) -- return(NULL); -- q = p = s; -- while(*p != tok && *p) -- p++; -- if (*p == tok) -- *p++ = 0; -- s = p; -- -- return q; --} - - /* - * Read the inittab file. - */ - void read_inittab(void) - { -- FILE *fp; /* The INITTAB file */ -- char buf[256]; /* Line buffer */ -- char err[64]; /* Error message. */ -- char *id, *rlevel, -- *action, *process; /* Fields of a line */ -- char *p; -- CHILD *ch, *old, *i; /* Pointers to CHILD structure */ -- CHILD *head = NULL; /* Head of linked list */ -- int lineNo = 0; /* Line number in INITTAB file */ -- int actionNo; /* Decoded action field */ -- int f; /* Counter */ -- int round; /* round 0 for SIGTERM, round 1 for SIGKILL */ -- int foundOne = 0; /* No killing no sleep */ -- int talk; /* Talk to the user */ -- int done = 0; /* Ready yet? */ -- sigset_t nmask, omask; /* For blocking SIGCHLD. */ -+ FILE *fp; /* The INITTAB file */ -+ CHILD *ch, *old, *i; /* Pointers to CHILD structure */ -+ CHILD *head = NULL; /* Head of linked list */ - #ifdef INITLVL -- struct stat st; /* To stat INITLVL */ -+ struct stat st; /* To stat INITLVL */ - #endif -+ sigset_t nmask, omask; /* For blocking SIGCHLD. */ -+ char buf[256]; /* Line buffer */ -+ char err[64]; /* Error message. */ -+ char *id, *rlevel, -+ *action, *process; /* Fields of a line */ -+ char *p; -+ int lineNo = 0; /* Line number in INITTAB file */ -+ int actionNo; /* Decoded action field */ -+ int f; /* Counter */ -+ int round; /* round 0 for SIGTERM, 1 for SIGKILL */ -+ int foundOne = 0; /* No killing no sleep */ -+ int talk; /* Talk to the user */ -+ int done = 0; /* Ready yet? */ - - #if DEBUG - if (newFamily != NULL) { -- log(L_VB, "PANIC newFamily != NULL"); -+ INITDBG(L_VB, "PANIC newFamily != NULL"); - exit(1); - } -- log(L_VB, "Reading inittab"); -+ INITDBG(L_VB, "Reading inittab"); - #endif - - /* - * Open INITTAB and real line by line. - */ - if ((fp = fopen(INITTAB, "r")) == NULL) -- log(L_VB, "No inittab file found"); -+ initlog(L_VB, "No inittab file found"); - - while(!done) { - /* -@@ -1103,9 +1151,9 @@ - * See if we have a single user entry. - */ - for(old = newFamily; old; old = old->next) -- if (strcmp(old->rlevel, "S") == 0) break; -+ if (strpbrk(old->rlevel, "S")) break; - if (old == NULL) -- snprintf(buf, sizeof(buf), "~~:S:wait:%s\n", SHELL); -+ snprintf(buf, sizeof(buf), "~~:S:wait:%s\n", SULOGIN); - else - continue; - } -@@ -1120,10 +1168,10 @@ - /* - * Decode the fields - */ -- id = get_part(p, ':'); -- rlevel = get_part(NULL, ':'); -- action = get_part(NULL, ':'); -- process = get_part(NULL, '\n'); -+ id = strsep(&p, ":"); -+ rlevel = strsep(&p, ":"); -+ action = strsep(&p, ":"); -+ process = strsep(&p, "\n"); - - /* - * Check if syntax is OK. Be very verbose here, to -@@ -1145,10 +1193,8 @@ - if (action && strlen(action) > 32) - strcpy(err, "action field too long"); - if (err[0] != 0) { -- log(L_VB, "%s[%d]: %s", INITTAB, lineNo, err); --#if DEBUG -- log(L_VB, "%s:%s:%s:%s", id, rlevel, action, process); --#endif -+ initlog(L_VB, "%s[%d]: %s", INITTAB, lineNo, err); -+ INITDBG(L_VB, "%s:%s:%s:%s", id, rlevel, action, process); - continue; - } - -@@ -1162,7 +1208,7 @@ - break; - } - if (actionNo == -1) { -- log(L_VB, "%s[%d]: %s: unknown action field", -+ initlog(L_VB, "%s[%d]: %s: unknown action field", - INITTAB, lineNo, action); - continue; - } -@@ -1172,7 +1218,7 @@ - */ - for(old = newFamily; old; old = old->next) { - if(strcmp(old->id, id) == 0 && strcmp(id, "~~")) { -- log(L_VB, "%s[%d]: duplicate ID field \"%s\"", -+ initlog(L_VB, "%s[%d]: duplicate ID field \"%s\"", - INITTAB, lineNo, id); - break; - } -@@ -1182,11 +1228,7 @@ - /* - * Allocate a CHILD structure - */ -- while ((ch = malloc(sizeof(CHILD))) == NULL) { -- log(L_VB, "out of memory"); -- do_sleep(5); -- } -- memset(ch, 0, sizeof(CHILD)); -+ ch = imalloc(sizeof(CHILD)); - - /* - * And fill it in. -@@ -1275,9 +1317,7 @@ - * be killed. - */ - --#if DEBUG -- log(L_VB, "Checking for children to kill"); --#endif -+ INITDBG(L_VB, "Checking for children to kill"); - for(round = 0; round < 2; round++) { - talk = 1; - for(ch = family; ch; ch = ch->next) { -@@ -1328,19 +1368,19 @@ - ch->flags &= ~KILLME; - continue; - } --#if DEBUG -- log(L_VB, "Killing \"%s\"", ch->process); --#endif -+ INITDBG(L_VB, "Killing \"%s\"", ch->process); - switch(round) { - case 0: /* Send TERM signal */ - if (talk) -- log(L_CO, "Sending processes the TERM signal"); -+ initlog(L_CO, -+ "Sending processes the TERM signal"); - kill(-(ch->pid), SIGTERM); - foundOne = 1; - break; - case 1: /* Send KILL signal and collect status */ - if (talk) -- log(L_CO, "Sending processes the KILL signal"); -+ initlog(L_CO, -+ "Sending processes the KILL signal"); - kill(-(ch->pid), SIGKILL); - break; - } -@@ -1380,12 +1420,11 @@ - for(ch = family; ch; ch = ch->next) - if (ch->flags & KILLME) { - if (!(ch->flags & ZOMBIE)) -- log(L_CO, "Pid %d [id %s] seems to hang", ch->pid, -+ initlog(L_CO, "Pid %d [id %s] seems to hang", ch->pid, - ch->id); - else { --#if DEBUG -- log(L_VB, "Updating utmp for pid %d [id %s]", ch->pid, ch->id); --#endif -+ INITDBG(L_VB, "Updating utmp for pid %d [id %s]", -+ ch->pid, ch->id); - ch->flags &= ~RUNNING; - if (ch->process[0] != '+') - write_utmp_wtmp("", ch->id, ch->pid, DEAD_PROCESS, NULL); -@@ -1451,15 +1490,13 @@ - CHILD *ch; /* Pointer to child */ - int delete; /* Delete this entry from list? */ - --#if DEBUG -- log(L_VB, "Checking for children to start"); --#endif -+ INITDBG(L_VB, "Checking for children to start"); - - for(ch = family; ch; ch = ch->next) { - - #if DEBUG - if (ch->rlevel[0] == 'C') { -- log(L_VB, "%s: flags %d", ch->process, ch->flags); -+ INITDBG(L_VB, "%s: flags %d", ch->process, ch->flags); - } - #endif - -@@ -1545,7 +1582,8 @@ - if (lvl > 0) { - if (islower(lvl)) lvl = toupper(lvl); - if (strchr("0123456789S", lvl) == NULL) { -- log(L_VB, "Initdefault level '%c' is invalid", lvl); -+ initlog(L_VB, -+ "Initdefault level '%c' is invalid", lvl); - lvl = 0; - } - } -@@ -1570,98 +1608,99 @@ - */ - int read_level(int arg) - { -- unsigned char foo = 'X'; /* Contents of INITLVL */ -- CHILD *ch; /* Walk through list */ -- int ok = 1; -+ CHILD *ch; /* Walk through list */ -+ unsigned char foo = 'X'; /* Contents of INITLVL */ -+ int ok = 1; - #ifdef INITLVL -- FILE *fp; -- int st; -- struct stat stt; -+ FILE *fp; -+ struct stat stt; -+ int st; - #endif - -- if (arg) foo = arg; -+ if (arg) foo = arg; - - #ifdef INITLVL -- ok = 0; -+ ok = 0; - -- if (arg == 0) { -- fp = NULL; -- if (stat(INITLVL, &stt) != 0 || stt.st_size != 0L) -- fp = fopen(INITLVL, "r"); -+ if (arg == 0) { -+ fp = NULL; -+ if (stat(INITLVL, &stt) != 0 || stt.st_size != 0L) -+ fp = fopen(INITLVL, "r"); - #ifdef INITLVL2 -- if (fp == NULL && (stat(INITLVL2, &stt) != 0 || stt.st_size != 0L)) -- fp = fopen(INITLVL2, "r"); -+ if (fp == NULL && -+ (stat(INITLVL2, &stt) != 0 || stt.st_size != 0L)) -+ fp = fopen(INITLVL2, "r"); - #endif -- if (fp == NULL) { -- /* INITLVL file is empty or not there - act as 'init q' */ -- log(L_SY, "Re-reading inittab"); -- return(runlevel); -+ if (fp == NULL) { -+ /* INITLVL file empty or not there - act as 'init q' */ -+ initlog(L_SY, "Re-reading inittab"); -+ return(runlevel); -+ } -+ ok = fscanf(fp, "%c %d", &foo, &st); -+ fclose(fp); -+ } else { -+ /* We go to the new runlevel passed as an argument. */ -+ foo = arg; -+ ok = 1; - } -- ok = fscanf(fp, "%c %d", &foo, &st); -- fclose(fp); -- } else { -- /* We go to the new runlevel passed as an argument. */ -- foo = arg; -- ok = 1; -- } -- if (ok == 2) sltime = st; -+ if (ok == 2) sltime = st; - - #endif /* INITLVL */ - -- if (islower(foo)) foo = toupper(foo); -- if (ok < 1 || ok > 2 || strchr("QS0123456789ABCU", foo) == NULL) { -- log(L_VB, "bad runlevel: %c", foo); -- return(runlevel); -- } -+ if (islower(foo)) foo = toupper(foo); -+ if (ok < 1 || ok > 2 || strchr("QS0123456789ABCU", foo) == NULL) { -+ initlog(L_VB, "bad runlevel: %c", foo); -+ return runlevel; -+ } - -- /* Log this action */ -- switch(foo) { -- case 'S': -- log(L_VB, "Going single user"); -- break; -- case 'Q': -- log(L_SY, "Re-reading inittab"); -- break; -- case 'A': -- case 'B': -- case 'C': -- log(L_SY, "Activating demand-procedures for '%c'", foo); -- break; -- case 'U': -- log(L_SY, "Trying to re-exec init"); -- return 'U'; -- default: -- log(L_VB, "Switching to runlevel: %c", foo); -- } -+ /* Log this action */ -+ switch(foo) { -+ case 'S': -+ initlog(L_VB, "Going single user"); -+ break; -+ case 'Q': -+ initlog(L_SY, "Re-reading inittab"); -+ break; -+ case 'A': -+ case 'B': -+ case 'C': -+ initlog(L_SY, -+ "Activating demand-procedures for '%c'", foo); -+ break; -+ case 'U': -+ initlog(L_SY, "Trying to re-exec init"); -+ return 'U'; -+ default: -+ initlog(L_VB, "Switching to runlevel: %c", foo); -+ } - -- if (foo == 'Q') return(runlevel); -+ if (foo == 'Q') return runlevel; - -- /* Check if this is a runlevel a, b or c */ -- if (strchr("ABC", foo)) { -- if (runlevel == 'S') return(runlevel); -+ /* Check if this is a runlevel a, b or c */ -+ if (strchr("ABC", foo)) { -+ if (runlevel == 'S') return(runlevel); - -- /* Read inittab again first! */ -- read_inittab(); -+ /* Read inittab again first! */ -+ read_inittab(); - -- /* Mark those special tasks */ -- for(ch = family; ch; ch = ch->next) -- if (strchr(ch->rlevel, foo) != NULL || -- strchr(ch->rlevel, tolower(foo)) != NULL) { -- ch->flags |= DEMAND; -- ch->flags &= ~XECUTED; --#if DEBUG -- log(L_VB, "Marking (%s) as ondemand, flags %d", -- ch->id, ch->flags); --#endif -- } -- return(runlevel); -- } -+ /* Mark those special tasks */ -+ for(ch = family; ch; ch = ch->next) -+ if (strchr(ch->rlevel, foo) != NULL || -+ strchr(ch->rlevel, tolower(foo)) != NULL) { -+ ch->flags |= DEMAND; -+ ch->flags &= ~XECUTED; -+ INITDBG(L_VB, -+ "Marking (%s) as ondemand, flags %d", -+ ch->id, ch->flags); -+ } -+ return runlevel; -+ } - -- /* Store both the old and the new runlevel. */ -- write_utmp_wtmp("runlevel", "~~", foo + 256*runlevel, RUN_LVL, "~"); -- thislevel = foo; -- prevlevel = runlevel; -- return(foo); -+ /* Store both the old and the new runlevel. */ -+ write_utmp_wtmp("runlevel", "~~", foo + 256*runlevel, RUN_LVL, "~"); -+ thislevel = foo; -+ prevlevel = runlevel; -+ return foo; - } - - -@@ -1674,32 +1713,33 @@ - */ - void fail_check(void) - { -- time_t t; /* System time */ -- CHILD *ch; /* Pointer to child structure */ -- time_t next_alarm = 0; /* When to set next alarm */ -+ CHILD *ch; /* Pointer to child structure */ -+ time_t t; /* System time */ -+ time_t next_alarm = 0; /* When to set next alarm */ - -- time(&t); -+ time(&t); - -- for(ch = family; ch; ch = ch->next) { -+ for(ch = family; ch; ch = ch->next) { - -- if (ch->flags & FAILING) { -- /* Can we free this sucker? */ -- if (ch->tm + SLEEPTIME < t) { -- ch->flags &= ~FAILING; -- ch->count = 0; -- ch->tm = 0; -- } else { -- /* No, we'll look again later */ -- if (next_alarm == 0 || ch->tm + SLEEPTIME > next_alarm) -- next_alarm = ch->tm + SLEEPTIME; -+ if (ch->flags & FAILING) { -+ /* Can we free this sucker? */ -+ if (ch->tm + SLEEPTIME < t) { -+ ch->flags &= ~FAILING; -+ ch->count = 0; -+ ch->tm = 0; -+ } else { -+ /* No, we'll look again later */ -+ if (next_alarm == 0 || -+ ch->tm + SLEEPTIME > next_alarm) -+ next_alarm = ch->tm + SLEEPTIME; -+ } - } - } -- } -- if (next_alarm) { -- next_alarm -= t; -- if (next_alarm < 1) next_alarm = 1; -- alarm(next_alarm); -- } -+ if (next_alarm) { -+ next_alarm -= t; -+ if (next_alarm < 1) next_alarm = 1; -+ alarm(next_alarm); -+ } - } - - /* Set all 'Fail' timers to 0 */ -@@ -1752,9 +1792,9 @@ - */ - int check_pipe(int fd) - { -- struct timeval t; -- fd_set s; -- char signature[8]; -+ struct timeval t; -+ fd_set s; -+ char signature[8]; - - FD_ZERO(&s); - FD_SET(fd, &s); -@@ -1789,10 +1829,11 @@ - */ - void re_exec(void) - { -- sigset_t mask, oldset; -- pid_t pid; -- int fd; -- CHILD *ch; -+ CHILD *ch; -+ sigset_t mask, oldset; -+ pid_t pid; -+ char **env; -+ int fd; - - if (strchr("S12345",runlevel) == NULL) - return; -@@ -1825,27 +1866,26 @@ - */ - for(ch = family; ch; ch = ch->next) - if (ch->flags & ZOMBIE) { --#if DEBUG -- log(L_VB, "Child died, PID= %d", ch->pid); --#endif -+ INITDBG(L_VB, "Child died, PID= %d", ch->pid); - ch->flags &= ~(RUNNING|ZOMBIE|WAITING); - if (ch->process[0] != '+') - write_utmp_wtmp("", ch->id, ch->pid, DEAD_PROCESS, NULL); - } - -- if ((pid = fork()) > 0) { -- /* -- * Yup. _Parent_ exec's ... -- */ -- execl(myname, myname, "--init", NULL); -- } else if (pid == 0) { -+ if ((pid = fork()) == 0) { - /* -- * ... while child sends her the -- * state information and dies -+ * Child sends state information to the parent. - */ - send_state(fd); - exit(0); - } -+ -+ /* -+ * The existing init process execs a new init binary. -+ */ -+ env = init_buildenv(0); -+ execl(myname, myname, "--init", NULL, env); -+ - /* - * We shouldn't be here, something failed. - * Bitch, close the state pipe, unblock signals and return. -@@ -1853,7 +1893,8 @@ - close(fd); - close(STATE_PIPE); - sigprocmask(SIG_SETMASK, &oldset, NULL); -- log(L_CO, "Attempt to re-exec failed"); -+ init_freeenv(env); -+ initlog(L_CO, "Attempt to re-exec failed"); - } - - -@@ -1863,10 +1904,10 @@ - */ - void fifo_new_level(int level) - { -- int oldlevel; - #if CHANGE_WAIT -- CHILD *ch; -+ CHILD *ch; - #endif -+ int oldlevel; - - if (level == runlevel) return; - -@@ -1894,6 +1935,59 @@ - } - } - -+ -+/* -+ * Set/unset environment variables. The variables are -+ * encoded as KEY=VAL\0KEY=VAL\0\0. With "=VAL" it means -+ * setenv, without it means unsetenv. -+ */ -+void initcmd_setenv(char *data, int size) -+{ -+ char *env, *p, *e, *eq; -+ int i, sz; -+ -+ e = data + size; -+ -+ while (*data && data < e) { -+ eq = NULL; -+ for (p = data; *p && p < e; p++) -+ if (*p == '=') eq = p; -+ if (*p) break; -+ env = data; -+ data = ++p; -+ -+ sz = eq ? (eq - env) : (p - env); -+ -+ /*initlog(L_SY, "init_setenv: %s, %s, %d", env, eq, sz);*/ -+ -+ /* -+ * We only allow INIT_* to be set. -+ */ -+ if (strncmp(env, "INIT_", 5) != 0) -+ continue; -+ -+ /* Free existing vars. */ -+ for (i = 0; i < NR_EXTRA_ENV; i++) { -+ if (extra_env[i] == NULL) continue; -+ if (!strncmp(extra_env[i], env, sz) && -+ extra_env[i][sz] == '=') { -+ free(extra_env[i]); -+ extra_env[i] = NULL; -+ } -+ } -+ -+ /* Set new vars if needed. */ -+ if (eq == NULL) continue; -+ for (i = 0; i < NR_EXTRA_ENV; i++) { -+ if (extra_env[i] == NULL) { -+ extra_env[i] = istrdup(env); -+ break; -+ } -+ } -+ } -+} -+ -+ - /* - * Read from the init FIFO. Processes like telnetd and rlogind can - * ask us to create login processes on their behalf. -@@ -1906,12 +2000,12 @@ - */ - void check_init_fifo(void) - { -- struct init_request request; -- int n; -- fd_set fds; -- int quit = 0; -- struct stat st, st2; -- struct timeval tv; -+ struct init_request request; -+ struct timeval tv; -+ struct stat st, st2; -+ fd_set fds; -+ int n; -+ int quit = 0; - - /* - * First, try to create /dev/initctl if not present. -@@ -1940,7 +2034,7 @@ - if ((pipe_fd = open(INIT_FIFO, O_RDWR|O_NONBLOCK)) >= 0) { - fstat(pipe_fd, &st); - if (!S_ISFIFO(st.st_mode)) { -- log(L_VB, "%s is not a fifo", INIT_FIFO); -+ initlog(L_VB, "%s is not a fifo", INIT_FIFO); - close(pipe_fd); - pipe_fd = -1; - } -@@ -1987,7 +2081,7 @@ - } - if (n <= 0) { - if (errno == EINTR) return; -- log(L_VB, "error reading initrequest"); -+ initlog(L_VB, "error reading initrequest"); - continue; - } - -@@ -2001,7 +2095,7 @@ - * Process request. - */ - if (request.magic != INIT_MAGIC || n != sizeof(request)) { -- log(L_VB, "got bogus initrequest"); -+ initlog(L_VB, "got bogus initrequest"); - continue; - } - switch(request.cmd) { -@@ -2025,8 +2119,23 @@ - do_power_fail('O'); - quit = 1; - break; -+ case INIT_CMD_SETENV: -+ initcmd_setenv(request.i.data, sizeof(request.i.data)); -+ break; -+ case INIT_CMD_CHANGECONS: -+ if (user_console) { -+ free(user_console); -+ user_console = NULL; -+ } -+ if (!request.i.bsd.reserved[0]) -+ user_console = NULL; -+ else -+ user_console = strdup(request.i.bsd.reserved); -+ console_init(); -+ quit = 1; -+ break; - default: -- log(L_VB, "got unimplemented initrequest."); -+ initlog(L_VB, "got unimplemented initrequest."); - break; - } - } -@@ -2045,11 +2154,11 @@ - */ - void boot_transitions() - { -- CHILD *ch; -- static int newlevel = 0; -- int loglevel; -- int oldlevel; -- static int warn = 1; -+ CHILD *ch; -+ static int newlevel = 0; -+ static int warn = 1; -+ int loglevel; -+ int oldlevel; - - /* Check if there is something to wait for! */ - for( ch = family; ch; ch = ch->next ) -@@ -2061,9 +2170,7 @@ - oldlevel = 'N'; - switch(runlevel) { - case '#': /* SYSINIT -> BOOT */ --#if DEBUG -- log(L_VB, "SYSINIT -> BOOT"); --#endif -+ INITDBG(L_VB, "SYSINIT -> BOOT"); - - /* Write a boot record. */ - wrote_utmp_reboot = 0; -@@ -2080,9 +2187,7 @@ - runlevel = '*'; - break; - case '*': /* BOOT -> NORMAL */ --#if DEBUG -- log(L_VB, "BOOT -> NORMAL"); --#endif -+ INITDBG(L_VB, "BOOT -> NORMAL"); - if (runlevel != newlevel) - loglevel = newlevel; - runlevel = newlevel; -@@ -2091,9 +2196,7 @@ - break; - case 'S': /* Ended SU mode */ - case 's': --#if DEBUG -- log(L_VB, "END SU MODE"); --#endif -+ INITDBG(L_VB, "END SU MODE"); - newlevel = get_init_default(); - if (!did_boot && newlevel != 'S') - runlevel = '*'; -@@ -2110,7 +2213,8 @@ - break; - default: - if (warn) -- log(L_VB, "no more processes left in this runlevel"); -+ initlog(L_VB, -+ "no more processes left in this runlevel"); - warn = 0; - loglevel = -1; - if (got_signals == 0) -@@ -2118,7 +2222,7 @@ - break; - } - if (loglevel > 0) { -- log(L_VB, "Entering runlevel: %c", runlevel); -+ initlog(L_VB, "Entering runlevel: %c", runlevel); - write_utmp_wtmp("runlevel", "~~", runlevel + 256 * oldlevel, RUN_LVL, "~"); - thislevel = runlevel; - prevlevel = oldlevel; -@@ -2133,16 +2237,14 @@ - */ - void process_signals() - { -- int pwrstat; -- int oldlevel; -- int fd; -- CHILD *ch; -- char c; -+ CHILD *ch; -+ int pwrstat; -+ int oldlevel; -+ int fd; -+ char c; - - if (ISMEMBER(got_signals, SIGPWR)) { --#if DEBUG -- log(L_VB, "got SIGPWR"); --#endif -+ INITDBG(L_VB, "got SIGPWR"); - /* See _what_ kind of SIGPWR this is. */ - pwrstat = 0; - if ((fd = open(PWRSTAT, O_RDONLY)) >= 0) { -@@ -2157,9 +2259,7 @@ - } - - if (ISMEMBER(got_signals, SIGINT)) { --#if DEBUG -- log(L_VB, "got SIGINT"); --#endif -+ INITDBG(L_VB, "got SIGINT"); - /* Tell ctrlaltdel entry to start up */ - for(ch = family; ch; ch = ch->next) - if (ch->action == CTRLALTDEL) -@@ -2168,9 +2268,7 @@ - } - - if (ISMEMBER(got_signals, SIGWINCH)) { --#if DEBUG -- log(L_VB, "got SIGWINCH"); --#endif -+ INITDBG(L_VB, "got SIGWINCH"); - /* Tell kbrequest entry to start up */ - for(ch = family; ch; ch = ch->next) - if (ch->action == KBREQUEST) -@@ -2179,26 +2277,20 @@ - } - - if (ISMEMBER(got_signals, SIGALRM)) { --#if DEBUG -- log(L_VB, "got SIGALRM"); --#endif -+ INITDBG(L_VB, "got SIGALRM"); - /* The timer went off: check it out */ - DELSET(got_signals, SIGALRM); - } - - if (ISMEMBER(got_signals, SIGCHLD)) { --#if DEBUG -- log(L_VB, "got SIGCHLD"); --#endif -+ INITDBG(L_VB, "got SIGCHLD"); - /* First set flag to 0 */ - DELSET(got_signals, SIGCHLD); - - /* See which child this was */ - for(ch = family; ch; ch = ch->next) - if (ch->flags & ZOMBIE) { --#if DEBUG -- log(L_VB, "Child died, PID= %d", ch->pid); --#endif -+ INITDBG(L_VB, "Child died, PID= %d", ch->pid); - ch->flags &= ~(RUNNING|ZOMBIE|WAITING); - if (ch->process[0] != '+') - write_utmp_wtmp("", ch->id, ch->pid, DEAD_PROCESS, NULL); -@@ -2207,9 +2299,7 @@ - } - - if (ISMEMBER(got_signals, SIGHUP)) { --#if DEBUG -- log(L_VB, "got SIGHUP"); --#endif -+ INITDBG(L_VB, "got SIGHUP"); - #if CHANGE_WAIT - /* Are we waiting for a child? */ - for(ch = family; ch; ch = ch->next) -@@ -2240,9 +2330,7 @@ - /* - * SIGUSR1 means close and reopen /dev/initctl - */ --#if DEBUG -- log(L_VB, "got SIGUSR1"); --#endif -+ INITDBG(L_VB, "got SIGUSR1"); - close(pipe_fd); - pipe_fd = -1; - DELSET(got_signals, SIGUSR1); -@@ -2254,11 +2342,11 @@ - */ - int init_main() - { -- int f, st; -- pid_t rc; -- CHILD *ch; -- sigset_t sgt; -- struct sigaction sa; -+ CHILD *ch; -+ struct sigaction sa; -+ sigset_t sgt; -+ pid_t rc; -+ int f, st; - - if (!reload) { - -@@ -2278,6 +2366,7 @@ - } - #endif - -+#ifdef __linux__ - /* - * Tell the kernel to send us SIGINT when CTRL-ALT-DEL - * is pressed, and that we want to handle keyboard signals. -@@ -2288,6 +2377,7 @@ - close(f); - } else - (void) ioctl(0, KDSIGACCEPT, SIGWINCH); -+#endif - - /* - * Ignore all signals. -@@ -2320,9 +2410,9 @@ - setsid(); - - /* -- * Set default PATH variable (for ksh) -+ * Set default PATH variable. - */ -- if (getenv("PATH") == NULL) putenv(PATH_DFL); -+ putenv(PATH_DFL); - - /* - * Initialize /var/run/utmp (only works if /var is on -@@ -2333,7 +2423,7 @@ - /* - * Say hello to the world - */ -- log(L_CO, bootmsg, "booting"); -+ initlog(L_CO, bootmsg, "booting"); - - /* - * See if we have to start an emergency shell. -@@ -2358,7 +2448,7 @@ - /* - * Restart: unblock signals and let the show go on - */ -- log(L_CO, bootmsg, "reloading"); -+ initlog(L_CO, bootmsg, "reloading"); - sigfillset(&sgt); - sigprocmask(SIG_UNBLOCK, &sgt, NULL); - } -@@ -2368,9 +2458,7 @@ - - /* See if we need to make the boot transitions. */ - boot_transitions(); --#if DEBUG -- log(L_VB, "init_main: waiting.."); --#endif -+ INITDBG(L_VB, "init_main: waiting.."); - - /* Check if there are processes to be waited on. */ - for(ch = family; ch; ch = ch->next) -@@ -2405,10 +2493,10 @@ - /* - * Tell the user about the syntax we expect. - */ --void Usage(char *s) -+void usage(char *s) - { -- fprintf(stderr, "Usage: %s 0123456SsQqAaBbCcUu\n", s); -- exit(1); -+ fprintf(stderr, "Usage: %s 0123456SsQqAaBbCcUu\n", s); -+ exit(1); - } - - int telinit(char *progname, int argc, char **argv) -@@ -2418,28 +2506,51 @@ - #endif - struct init_request request; - struct sigaction sa; -- int f, fd; -+ int f, fd, l; -+ char *env = NULL; - -- while((f = getopt(argc, argv, "t:")) != EOF) switch(f) { -+ memset(&request, 0, sizeof(request)); -+ request.magic = INIT_MAGIC; -+ -+ while ((f = getopt(argc, argv, "t:e:")) != EOF) switch(f) { - case 't': - sltime = atoi(optarg); - break; -+ case 'e': -+ if (env == NULL) -+ env = request.i.data; -+ l = strlen(optarg); -+ if (env + l + 2 > request.i.data + sizeof(request.i.data)) { -+ fprintf(stderr, "%s: -e option data " -+ "too large\n", progname); -+ exit(1); -+ } -+ memcpy(env, optarg, l); -+ env += l; -+ *env++ = 0; -+ break; - default: -- Usage(progname); -+ usage(progname); - break; - } - -- /* Check syntax. */ -- if (argc - optind != 1 || strlen(argv[optind]) != 1) Usage(progname); -- if (!strchr("0123456789SsQqAaBbCcUu", argv[optind][0])) Usage(progname); -+ if (env) *env++ = 0; - -- /* Open the fifo and write a command. */ -- memset(&request, 0, sizeof(request)); -- request.magic = INIT_MAGIC; -- request.cmd = INIT_CMD_RUNLVL; -- request.runlevel = argv[optind][0]; -- request.sleeptime = sltime; -+ if (env) { -+ if (argc != optind) -+ usage(progname); -+ request.cmd = INIT_CMD_SETENV; -+ } else { -+ if (argc - optind != 1 || strlen(argv[optind]) != 1) -+ usage(progname); -+ if (!strchr("0123456789SsQqAaBbCcUu", argv[optind][0])) -+ usage(progname); -+ request.cmd = INIT_CMD_RUNLVL; -+ request.runlevel = env ? 0 : argv[optind][0]; -+ request.sleeptime = sltime; -+ } - -+ /* Open the fifo and write a command. */ - /* Make sure we don't hang on opening /dev/initctl */ - SETSIG(sa, SIGALRM, signal_handler, 0); - alarm(3); -@@ -2449,7 +2560,27 @@ - alarm(0); - return 0; - } --#ifndef TELINIT_USES_INITLVL -+ -+#ifdef TELINIT_USES_INITLVL -+ if (request.cmd == INIT_CMD_RUNLVL) { -+ /* Fallthrough to the old method. */ -+ -+ /* Now write the new runlevel. */ -+ if ((fp = fopen(INITLVL, "w")) == NULL) { -+ fprintf(stderr, "%s: cannot create %s\n", -+ progname, INITLVL); -+ exit(1); -+ } -+ fprintf(fp, "%s %d", argv[optind], sltime); -+ fclose(fp); -+ -+ /* And tell init about the pending runlevel change. */ -+ if (kill(INITPID, SIGHUP) < 0) perror(progname); -+ -+ return 0; -+ } -+#endif -+ - fprintf(stderr, "%s: ", progname); - if (ISMEMBER(got_signals, SIGALRM)) { - fprintf(stderr, "timeout opening/writing control channel %s\n", -@@ -2458,24 +2589,6 @@ - perror(INIT_FIFO); - } - return 1; --#endif -- -- /* Fallthrough to the old method. */ -- --#ifdef TELINIT_USES_INITLVL -- /* Now write the new runlevel. */ -- if ((fp = fopen(INITLVL, "w")) == NULL) { -- fprintf(stderr, "%s: cannot create %s\n", progname, INITLVL); -- exit(1); -- } -- fprintf(fp, "%s %d", argv[optind], sltime); -- fclose(fp); -- -- /* And tell init about the pending runlevel change. */ -- if (kill(INITPID, SIGHUP) < 0) perror(progname); -- -- return 0; --#endif - } - - /* -@@ -2518,7 +2631,7 @@ - - receive_state(STATE_PIPE); - -- myname = strdup(argv[0]); -+ myname = istrdup(argv[0]); - argv0 = argv[0]; - maxproclen = 0; - for (f = 0; f < argc; f++) -diff -urNd -urNd sysvinit-2.85/src/init.h sysvinit-2.86/src/init.h ---- sysvinit-2.85/src/init.h 1999-06-03 14:22:59.000000000 -0500 -+++ sysvinit-2.86/src/init.h 2004-07-29 06:21:01.000000000 -0500 -@@ -2,9 +2,8 @@ - * init.h Several defines and declarations to be - * included by all modules of the init program. - * -- * Version: @(#)init.h 2.74 09-Mar-1998 miquels@cistron.nl -+ * Version: @(#)init.h 2.85-5 02-Jul-2003 miquels@cistron.nl - * -- * Modified: Re-exec patch; 24 Feb 1998, Al Viro. - */ - - /* Standard configuration */ -@@ -24,17 +23,26 @@ - #define TESTTIME 120 /* this much seconds */ - #define SLEEPTIME 300 /* Disable time */ - --/* Default path inherited by every child if it's not set. */ --#define PATH_DFL "PATH=/usr/local/sbin:/sbin:/bin:/usr/sbin:/usr/bin" -+/* Default path inherited by every child. */ -+#define PATH_DFL "PATH=/bin:/usr/bin:/sbin:/usr/sbin" - - - /* Prototypes. */ - void write_utmp_wtmp(char *user, char *id, int pid, int type, char *line); - void write_wtmp(char *user, char *id, int pid, int type, char *line); --void log(int loglevel, char *fmt, ...); -+#ifdef __GNUC__ -+__attribute__ ((format (printf, 2, 3))) -+#endif -+void initlog(int loglevel, char *fmt, ...); - void set_term(int how); - void print(char *fmt); - -+#if DEBUG -+# define INITDBG(level, fmt, args...) initlog(level, fmt, ##args) -+#else -+# define INITDBG(level, fmt, args...) -+#endif -+ - /* Actions to be taken by init */ - #define RESPAWN 1 - #define WAIT 2 -Binary files sysvinit-2.85/src/init.o and sysvinit-2.86/src/init.o differ -Binary files sysvinit-2.85/src/init_utmp.o and sysvinit-2.86/src/init_utmp.o differ -diff -urNd -urNd sysvinit-2.85/src/initreq.h sysvinit-2.86/src/initreq.h ---- sysvinit-2.85/src/initreq.h 1996-01-02 12:22:06.000000000 -0600 -+++ sysvinit-2.86/src/initreq.h 2004-07-30 06:56:51.000000000 -0500 -@@ -1,41 +1,77 @@ - /* -- * initreq.h Interface to let init spawn programs on behalf of -- * other programs/daemons. -- * Definitions based on sys_term.c from the BSD 4.4 -- * telnetd source. -+ * initreq.h Interface to talk to init through /dev/initctl. - * -- * Version: @(#)initreq.h 1.25 28-Dec-1995 MvS -+ * Copyright (C) 1995-2004 Miquel van Smoorenburg -+ * -+ * This library is free software; you can redistribute it and/or -+ * modify it under the terms of the GNU Lesser General Public -+ * License as published by the Free Software Foundation; either -+ * version 2 of the License, or (at your option) any later version. -+ * -+ * Version: @(#)initreq.h 1.28 31-Mar-2004 MvS - * -- * Notes: Implemented in sysvinit-2.58 and up, but only -- * for "telinit". Support for rlogind, telnetd -- * and rxvt/xterm will follow shortly. - */ - #ifndef _INITREQ_H - #define _INITREQ_H - - #include <sys/param.h> - -+#if defined(__FreeBSD_kernel__) -+# define INIT_FIFO "/etc/.initctl" -+#else -+# define INIT_FIFO "/dev/initctl" -+#endif -+ - #define INIT_MAGIC 0x03091969 --#define INIT_FIFO "/dev/initctl" --#define INIT_CMD_START 0 --#define INIT_CMD_RUNLVL 1 --#define INIT_CMD_POWERFAIL 2 --#define INIT_CMD_POWERFAILNOW 3 --#define INIT_CMD_POWEROK 4 -+#define INIT_CMD_START 0 -+#define INIT_CMD_RUNLVL 1 -+#define INIT_CMD_POWERFAIL 2 -+#define INIT_CMD_POWERFAILNOW 3 -+#define INIT_CMD_POWEROK 4 -+#define INIT_CMD_BSD 5 -+#define INIT_CMD_SETENV 6 -+#define INIT_CMD_UNSETENV 7 -+ -+#define INIT_CMD_CHANGECONS 12345 -+ -+#ifdef MAXHOSTNAMELEN -+# define INITRQ_HLEN MAXHOSTNAMELEN -+#else -+# define INITRQ_HLEN 64 -+#endif -+ -+/* -+ * This is what BSD 4.4 uses when talking to init. -+ * Linux doesn't use this right now. -+ */ -+struct init_request_bsd { -+ char gen_id[8]; /* Beats me.. telnetd uses "fe" */ -+ char tty_id[16]; /* Tty name minus /dev/tty */ -+ char host[INITRQ_HLEN]; /* Hostname */ -+ char term_type[16]; /* Terminal type */ -+ int signal; /* Signal to send */ -+ int pid; /* Process to send to */ -+ char exec_name[128]; /* Program to execute */ -+ char reserved[128]; /* For future expansion. */ -+}; -+ - -+/* -+ * Because of legacy interfaces, "runlevel" and "sleeptime" -+ * aren't in a seperate struct in the union. -+ * -+ * The weird sizes are because init expects the whole -+ * struct to be 384 bytes. -+ */ - struct 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 gen_id[8]; /* Beats me.. telnetd uses "fe" */ -- char tty_id[16]; /* Tty name minus /dev/tty */ -- char host[MAXHOSTNAMELEN]; /* Hostname */ -- char term_type[16]; /* Terminal type */ -- int signal; /* Signal to send */ -- int pid; /* Process to send to */ -- char exec_name[128]; /* Program to execute */ -- char reserved[128]; /* For future expansion. */ -+ int magic; /* Magic number */ -+ int cmd; /* What kind of request */ -+ int runlevel; /* Runlevel to change to */ -+ int sleeptime; /* Time between TERM and KILL */ -+ union { -+ struct init_request_bsd bsd; -+ char data[368]; -+ } i; - }; - - #endif -Binary files sysvinit-2.85/src/killall5 and sysvinit-2.86/src/killall5 differ -diff -urNd -urNd sysvinit-2.85/src/killall5.c sysvinit-2.86/src/killall5.c ---- sysvinit-2.85/src/killall5.c 2003-04-14 04:59:11.000000000 -0500 -+++ sysvinit-2.86/src/killall5.c 2004-07-30 07:16:23.000000000 -0500 -@@ -5,7 +5,7 @@ - * - * pidof.c Tries to get the pid of the process[es] named. - * -- * Version: 2.85 14-Apr-2003 MvS -+ * Version: 2.86 30-Jul-2004 MvS - * - * Usage: killall5 [-][signal] - * pidof [-s] [-o omitpid [-o omitpid]] program [program..] -@@ -20,7 +20,7 @@ - * - swapped out programs pids are caught now - * - * This file is part of the sysvinit suite, -- * Copyright 1991-2003 Miquel van Smoorenburg. -+ * Copyright 1991-2004 Miquel van Smoorenburg. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License -@@ -41,34 +41,43 @@ - #include <getopt.h> - #include <stdarg.h> - --char *Version = "@(#)killall5 2.85 14-Apr-2003 miquels@cistron.nl"; -+char *Version = "@(#)killall5 2.86 31-Jul-2004 miquels@cistron.nl"; -+ -+#define STATNAMELEN 15 - - /* Info about a process. */ --typedef struct _proc_ --{ -- char *fullname; /* Name as found out from argv[0] */ -- char *basename; /* Only the part after the last / */ -- char *statname; /* the statname without braces */ -- ino_t ino; /* Inode number */ -- dev_t dev; /* Device it is on */ -- pid_t pid; /* Process ID. */ -- int sid; /* Session ID. */ -- struct _proc_ *next; /* Pointer to next struct. */ -+typedef struct proc { -+ char *argv0; /* Name as found out from argv[0] */ -+ char *argv0base; /* `basename argv[1]` */ -+ char *argv1; /* Name as found out from argv[1] */ -+ char *argv1base; /* `basename argv[1]` */ -+ char *statname; /* the statname without braces */ -+ ino_t ino; /* Inode number */ -+ dev_t dev; /* Device it is on */ -+ pid_t pid; /* Process ID. */ -+ int sid; /* Session ID. */ -+ int kernel; /* Kernel thread or zombie. */ -+ struct proc *next; /* Pointer to next struct. */ - } PROC; - - /* pid queue */ --typedef struct _pidq_ { -- struct _pidq_ *front; -- struct _pidq_ *next; -- struct _pidq_ *rear; -- PROC *proc; -+ -+typedef struct pidq { -+ PROC *proc; -+ struct pidq *next; - } PIDQ; - -+typedef struct { -+ PIDQ *head; -+ PIDQ *tail; -+ PIDQ *next; -+} PIDQ_HEAD; -+ - /* List of processes. */ - PROC *plist; - --/* Did we stop a number of processes? */ --int stopped; -+/* Did we stop all processes ? */ -+int sent_sigstop; - - int scripts_too = 0; - -@@ -86,7 +95,7 @@ - void *p; - - if ((p = malloc(bytes)) == NULL) { -- if (stopped) kill(-1, SIGCONT); -+ if (sent_sigstop) kill(-1, SIGCONT); - nsyslog(LOG_ERR, "out of memory"); - exit(1); - } -@@ -98,14 +107,14 @@ - */ - int mount_proc(void) - { -- struct stat st; -- pid_t pid, rc; -- int wst; -- char *args[] = { "mount", "-t", "proc", "none", "/proc", NULL }; -- int did_mount = 0; -+ struct stat st; -+ char *args[] = { "mount", "-t", "proc", "proc", "/proc", 0 }; -+ pid_t pid, rc; -+ int wst; -+ int did_mount = 0; - - /* Stat /proc/version to see if /proc is mounted. */ -- if (stat("/proc/version", &st) < 0) { -+ if (stat("/proc/version", &st) < 0 && errno == ENOENT) { - - /* It's not there, so mount it. */ - if ((pid = fork()) < 0) { -@@ -115,7 +124,6 @@ - if (pid == 0) { - /* Try a few mount binaries. */ - execv("/sbin/mount", args); -- execv("/etc/mount", args); - execv("/bin/mount", args); - - /* Okay, I give up. */ -@@ -134,28 +142,42 @@ - - /* See if mount succeeded. */ - if (stat("/proc/version", &st) < 0) { -- nsyslog(LOG_ERR, "/proc not mounted, failed to mount."); -+ if (errno == ENOENT) -+ nsyslog(LOG_ERR, "/proc not mounted, failed to mount."); -+ else -+ nsyslog(LOG_ERR, "/proc unavailable."); - exit(1); - } - - return did_mount; - } - -+int readarg(FILE *fp, char *buf, int sz) -+{ -+ int c = 0, f = 0; -+ -+ while (f < (sz-1) && (c = fgetc(fp)) != EOF && c) -+ buf[f++] = c; -+ buf[f] = 0; -+ -+ return (c == EOF && f == 0) ? c : f; -+} -+ - /* - * Read the proc filesystem. - */ - int readproc() - { -- DIR *dir; -- struct dirent *d; -- char path[256]; -- char buf[256]; -- char *s, *q; -- FILE *fp; -- int pid, f; -- PROC *p, *n; -- struct stat st; -- int c; -+ DIR *dir; -+ FILE *fp; -+ PROC *p, *n; -+ struct dirent *d; -+ struct stat st; -+ char path[256]; -+ char buf[256]; -+ char *s, *q; -+ unsigned long startcode, endcode; -+ int pid, f; - - /* Open the /proc directory. */ - if ((dir = opendir("/proc")) == NULL) { -@@ -167,7 +189,8 @@ - n = plist; - for (p = plist; n; p = n) { - n = p->next; -- if (p->fullname) free(p->fullname); -+ if (p->argv0) free(p->argv0); -+ if (p->argv1) free(p->argv1); - free(p); - } - plist = NULL; -@@ -188,7 +211,7 @@ - /* Read SID & statname from it. */ - if ((fp = fopen(path, "r")) != NULL) { - buf[0] = 0; -- fgets(buf, 256, fp); -+ fgets(buf, sizeof(buf), fp); - - /* See if name starts with '(' */ - s = buf; -@@ -215,14 +238,21 @@ - p->statname = (char *)xmalloc(strlen(s)+1); - strcpy(p->statname, s); - -- /* This could be replaced by getsid(pid) */ -- if (sscanf(q, "%*c %*d %*d %d", &p->sid) != 1) { -+ /* Get session, startcode, endcode. */ -+ startcode = endcode = 0; -+ if (sscanf(q, "%*c %*d %*d %d %*d %*d %*u %*u " -+ "%*u %*u %*u %*u %*u %*d %*d " -+ "%*d %*d %*d %*d %*u %*u %*d " -+ "%*u %lu %lu", -+ &p->sid, &startcode, &endcode) != 3) { - p->sid = 0; - nsyslog(LOG_ERR, "can't read sid from %s\n", - path); - free(p); - continue; - } -+ if (startcode == 0 && endcode == 0) -+ p->kernel = 1; - fclose(fp); - } else { - /* Process disappeared.. */ -@@ -230,24 +260,44 @@ - continue; - } - -- /* Now read argv[0] */ - snprintf(path, sizeof(path), "/proc/%s/cmdline", d->d_name); - if ((fp = fopen(path, "r")) != NULL) { -- f = 0; -- while(f < 127 && (c = fgetc(fp)) != EOF && c) -- buf[f++] = c; -- buf[f++] = 0; -- fclose(fp); - -- /* Store the name into malloced memory. */ -- p->fullname = (char *)xmalloc(f); -- strcpy(p->fullname, buf); -+ /* Now read argv[0] */ -+ f = readarg(fp, buf, sizeof(buf)); -+ -+ if (buf[0]) { -+ /* Store the name into malloced memory. */ -+ p->argv0 = (char *)xmalloc(f + 1); -+ strcpy(p->argv0, buf); -+ -+ /* Get a pointer to the basename. */ -+ p->argv0base = strrchr(p->argv0, '/'); -+ if (p->argv0base != NULL) -+ p->argv0base++; -+ else -+ p->argv0base = p->argv0; -+ } -+ -+ /* And read argv[1] */ -+ while ((f = readarg(fp, buf, sizeof(buf))) != EOF) -+ if (buf[0] != '-') break; -+ -+ if (buf[0]) { -+ /* Store the name into malloced memory. */ -+ p->argv1 = (char *)xmalloc(f + 1); -+ strcpy(p->argv1, buf); -+ -+ /* Get a pointer to the basename. */ -+ p->argv1base = strrchr(p->argv1, '/'); -+ if (p->argv1base != NULL) -+ p->argv1base++; -+ else -+ p->argv1base = p->argv1; -+ } -+ -+ fclose(fp); - -- /* Get a pointer to the basename. */ -- if ((p->basename = strrchr(p->fullname, '/')) != NULL) -- p->basename++; -- else -- p->basename = p->fullname; - } else { - /* Process disappeared.. */ - free(p); -@@ -272,19 +322,18 @@ - return 0; - } - --PIDQ *init_pid_q(PIDQ *q) -+PIDQ_HEAD *init_pid_q(PIDQ_HEAD *q) - { -- q->front = q->next = q->rear = NULL; -- q->proc = NULL; -+ q->head = q->next = q->tail = NULL; - return q; - } - --int empty_q(PIDQ *q) -+int empty_q(PIDQ_HEAD *q) - { -- return (q->front == NULL); -+ return (q->head == NULL); - } - --int add_pid_to_q(PIDQ *q, PROC *p) -+int add_pid_to_q(PIDQ_HEAD *q, PROC *p) - { - PIDQ *tmp; - -@@ -294,23 +343,23 @@ - tmp->next = NULL; - - if (empty_q(q)) { -- q->front = tmp; -- q->rear = tmp; -+ q->head = tmp; -+ q->tail = tmp; - } else { -- q->rear->next = tmp; -- q->rear = tmp; -+ q->tail->next = tmp; -+ q->tail = tmp; - } - return 0; - } - --PROC *get_next_from_pid_q(PIDQ *q) -+PROC *get_next_from_pid_q(PIDQ_HEAD *q) - { -- PROC *p; -- PIDQ *tmp = q->front; -+ PROC *p; -+ PIDQ *tmp = q->head; - - if (!empty_q(q)) { -- p = q->front->proc; -- q->front = tmp->next; -+ p = q->head->proc; -+ q->head = tmp->next; - free(tmp); - return p; - } -@@ -319,15 +368,15 @@ - } - - /* Try to get the process ID of a given process. */ --PIDQ *pidof(char *prog) -+PIDQ_HEAD *pidof(char *prog) - { -- struct stat st; -- int dostat = 0; -- PROC *p; -- PIDQ *q; -- char *s; -- int foundone = 0; -- int ok = 0; -+ PROC *p; -+ PIDQ_HEAD *q; -+ struct stat st; -+ char *s; -+ int dostat = 0; -+ int foundone = 0; -+ int ok = 0; - - /* Try to stat the executable. */ - if (prog[0] == '/' && stat(prog, &st) == 0) dostat++; -@@ -338,7 +387,7 @@ - else - s++; - -- q = (PIDQ *)xmalloc(sizeof(PIDQ)); -+ q = (PIDQ_HEAD *)xmalloc(sizeof(PIDQ_HEAD)); - q = init_pid_q(q); - - /* First try to find a match based on dev/ino pair. */ -@@ -352,20 +401,31 @@ - } - - /* If we didn't find a match based on dev/ino, try the name. */ -- if (!foundone) { -- for (p = plist; p; p = p->next) { -- ok = 0; -+ if (!foundone) for (p = plist; p; p = p->next) { -+ ok = 0; - -- ok += (strcmp(p->fullname, prog) == 0); -- ok += (strcmp(p->basename, s) == 0); -+ /* Compare name (both basename and full path) */ -+ ok += (p->argv0 && strcmp(p->argv0, prog) == 0); -+ ok += (p->argv0 && strcmp(p->argv0base, s) == 0); - -- if (p->fullname[0] == 0 || -- strchr(p->fullname, ' ') || -- scripts_too) -- ok += (strcmp(p->statname, s) == 0); -+ /* For scripts, compare argv[1] as well. */ -+ if (scripts_too && p->argv1 && -+ !strncmp(p->statname, p->argv1base, STATNAMELEN)) { -+ ok += (strcmp(p->argv1, prog) == 0); -+ ok += (strcmp(p->argv1base, s) == 0); -+ } - -- if (ok) add_pid_to_q(q, p); -+ /* -+ * if we have a space in argv0, process probably -+ * used setproctitle so try statname. -+ */ -+ if (strlen(s) <= STATNAMELEN && -+ (p->argv0 == NULL || -+ p->argv0[0] == 0 || -+ strchr(p->argv0, ' '))) { -+ ok += (strcmp(p->statname, s) == 0); - } -+ if (ok) add_pid_to_q(q, p); - } - - return q; -@@ -410,12 +470,12 @@ - */ - int main_pidof(int argc, char **argv) - { -- PROC *p; -- PIDQ *q; -- int f; -- int first = 1; -- int i,oind, opt, flags = 0; -- pid_t opid[PIDOF_OMITSZ], spid; -+ PIDQ_HEAD *q; -+ PROC *p; -+ pid_t opid[PIDOF_OMITSZ], spid; -+ int f; -+ int first = 1; -+ int i, oind, opt, flags = 0; - - for (oind = PIDOF_OMITSZ-1; oind > 0; oind--) - opid[oind] = 0; -@@ -498,9 +558,9 @@ - /* Main for either killall or pidof. */ - int main(int argc, char **argv) - { -- PROC *p; -- int pid, sid = -1; -- int sig = SIGKILL; -+ PROC *p; -+ int pid, sid = -1; -+ int sig = SIGKILL; - - /* Get program name. */ - if ((progname = strrchr(argv[0], '/')) == NULL) -@@ -511,9 +571,6 @@ - /* Now connect to syslog. */ - openlog(progname, LOG_CONS|LOG_PID, LOG_DAEMON); - -- /* First get the /proc filesystem online. */ -- mount_proc(); -- - /* Were we called as 'pidof' ? */ - if (strcmp(progname, "pidof") == 0) - return main_pidof(argc, argv); -@@ -525,6 +582,9 @@ - if ((sig = atoi(argv[1])) <= 0 || sig > 31) usage(); - } - -+ /* First get the /proc filesystem online. */ -+ mount_proc(); -+ - /* - * Ignoring SIGKILL and SIGSTOP do not make sense, but - * someday kill(-1, sig) might kill ourself if we don't -@@ -537,24 +597,19 @@ - - /* Now stop all processes. */ - kill(-1, SIGSTOP); -- stopped = 1; -+ sent_sigstop = 1; - -- /* Find out our own 'sid'. */ -+ /* Read /proc filesystem */ - if (readproc() < 0) { - kill(-1, SIGCONT); - exit(1); - } - -- pid = getpid(); -- for (p = plist; p; p = p->next) -- if (p->pid == pid) { -- sid = p->sid; -- break; -- } -- - /* Now kill all processes except our session. */ -+ sid = (int)getsid(0); -+ pid = (int)getpid(); - for (p = plist; p; p = p->next) -- if (p->pid != pid && p->sid != sid) -+ if (p->pid != pid && p->sid != sid && !p->kernel) - kill(p->pid, sig); - - /* And let them continue. */ -Binary files sysvinit-2.85/src/last and sysvinit-2.86/src/last differ -diff -urNd -urNd sysvinit-2.85/src/last.c sysvinit-2.86/src/last.c ---- sysvinit-2.85/src/last.c 2003-04-17 06:38:56.000000000 -0500 -+++ sysvinit-2.86/src/last.c 2004-07-30 07:16:26.000000000 -0500 -@@ -6,10 +6,10 @@ - * - * Author: Miquel van Smoorenburg, miquels@cistron.nl - * -- * Version: @(#)last 2.85 16-Apr-2003 miquels@cistron.nl -+ * Version: @(#)last 2.85 30-Jul-2004 miquels@cistron.nl - * - * This file is part of the sysvinit suite, -- * Copyright 1991-2003 Miquel van Smoorenburg. -+ * Copyright 1991-2004 Miquel van Smoorenburg. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License -@@ -40,7 +40,7 @@ - # define SHUTDOWN_TIME 254 - #endif - --char *Version = "@(#) last 2.85 16-Apr-2003 miquels"; -+char *Version = "@(#) last 2.85 31-Apr-2004 miquels"; - - #define CHOP_DOMAIN 0 /* Define to chop off local domainname. */ - #define NEW_UTMP 1 /* Fancy & fast utmp read code. */ -@@ -491,10 +491,48 @@ - void usage(char *s) - { - fprintf(stderr, "Usage: %s [-num | -n num] [-f file] " -+ "[-t YYYYMMDDHHMMSS] " - "[-R] [-x] [-o] [username..] [tty..]\n", s); - exit(1); - } - -+time_t parsetm(char *ts) -+{ -+ struct tm u = {0}, origu; -+ time_t tm; -+ -+ if (sscanf(ts, "%4d%2d%2d%2d%2d%2d", &u.tm_year, -+ &u.tm_mon, &u.tm_mday, &u.tm_hour, &u.tm_min, -+ &u.tm_sec) != 6) -+ return (time_t)-1; -+ -+ u.tm_year -= 1900; -+ u.tm_mon -= 1; -+ u.tm_isdst = -1; -+ -+ origu = u; -+ -+ if ((tm = mktime(&u)) == (time_t)-1) -+ return tm; -+ -+ /* -+ * Unfortunately mktime() is much more forgiving than -+ * it should be. For example, it'll gladly accept -+ * "30" as a valid month number. This behavior is by -+ * design, but we don't like it, so we want to detect -+ * it and complain. -+ */ -+ if (u.tm_year != origu.tm_year || -+ u.tm_mon != origu.tm_mon || -+ u.tm_mday != origu.tm_mday || -+ u.tm_hour != origu.tm_hour || -+ u.tm_min != origu.tm_min || -+ u.tm_sec != origu.tm_sec) -+ return (time_t)-1; -+ -+ return tm; -+} -+ - int main(int argc, char **argv) - { - FILE *fp; /* Filepointer of wtmp file */ -@@ -518,10 +556,12 @@ - int extended = 0; /* Lots of info. */ - char *altufile = NULL;/* Alternate wtmp */ - -+ time_t until = 0; /* at what time to stop parsing the file */ -+ - progname = mybasename(argv[0]); - - /* Process the arguments. */ -- while((c = getopt(argc, argv, "f:n:Rxadio0123456789")) != EOF) -+ while((c = getopt(argc, argv, "f:n:Rxadiot:0123456789")) != EOF) - switch(c) { - case 'R': - showhost = 0; -@@ -552,6 +592,13 @@ - case 'a': - altlist++; - break; -+ case 't': -+ if ((until = parsetm(optarg)) == (time_t)-1) { -+ fprintf(stderr, "%s: Invalid time value \"%s\"\n", -+ progname, optarg); -+ usage(progname); -+ } -+ break; - case '0': case '1': case '2': case '3': case '4': - case '5': case '6': case '7': case '8': case '9': - maxrecs = 10*maxrecs + c - '0'; -@@ -650,6 +697,9 @@ - if (uread(fp, &ut, &quit) != 1) - break; - -+ if (until && until < ut.ut_time) -+ continue; -+ - if (memcmp(&ut, &oldut, sizeof(struct utmp)) == 0) continue; - memcpy(&oldut, &ut, sizeof(struct utmp)); - lastdate = ut.ut_time; -Binary files sysvinit-2.85/src/last.o and sysvinit-2.86/src/last.o differ -Binary files sysvinit-2.85/src/mesg and sysvinit-2.86/src/mesg differ -Binary files sysvinit-2.85/src/mesg.o and sysvinit-2.86/src/mesg.o differ -Binary files sysvinit-2.85/src/mountpoint and sysvinit-2.86/src/mountpoint differ -diff -urNd -urNd sysvinit-2.85/src/mountpoint.c sysvinit-2.86/src/mountpoint.c ---- sysvinit-2.85/src/mountpoint.c 1969-12-31 18:00:00.000000000 -0600 -+++ sysvinit-2.86/src/mountpoint.c 2004-06-09 07:47:45.000000000 -0500 -@@ -0,0 +1,119 @@ -+/* -+ * mountpoint See if a directory is a mountpoint. -+ * -+ * Author: Miquel van Smoorenburg. -+ * -+ * Version: @(#)mountpoint 2.85-12 17-Mar-2004 miquels@cistron.nl -+ * -+ * This file is part of the sysvinit suite, -+ * Copyright 1991-2004 Miquel van Smoorenburg. -+ * -+ * This program is free software; you can redistribute it and/or -+ * modify it under the terms of the GNU General Public License -+ * as published by the Free Software Foundation; either version -+ * 2 of the License, or (at your option) any later version. -+ */ -+ -+#include <sys/stat.h> -+#include <unistd.h> -+#include <stdlib.h> -+#include <string.h> -+#include <errno.h> -+#include <stdarg.h> -+#include <getopt.h> -+#include <stdio.h> -+ -+int dostat(char *path, struct stat *st, int do_lstat, int quiet) -+{ -+ int n; -+ -+ if (do_lstat) -+ n = lstat(path, st); -+ else -+ n = stat(path, st); -+ -+ if (n != 0) { -+ if (!quiet) -+ fprintf(stderr, "mountpoint: %s: %s\n", path, -+ strerror(errno)); -+ return -1; -+ } -+ return 0; -+} -+ -+void usage(void) { -+ fprintf(stderr, "Usage: mountpoint [-q] [-d] [-x] path\n"); -+ exit(1); -+} -+ -+int main(int argc, char **argv) -+{ -+ struct stat st, st2; -+ char buf[256]; -+ char *path; -+ int quiet = 0; -+ int showdev = 0; -+ int xdev = 0; -+ int c, r; -+ -+ while ((c = getopt(argc, argv, "dqx")) != EOF) switch(c) { -+ case 'd': -+ showdev = 1; -+ break; -+ case 'q': -+ quiet = 1; -+ break; -+ case 'x': -+ xdev = 1; -+ break; -+ default: -+ usage(); -+ break; -+ } -+ if (optind != argc - 1) usage(); -+ path = argv[optind]; -+ -+ if (dostat(path, &st, !xdev, quiet) < 0) -+ return 1; -+ -+ if (xdev) { -+#ifdef __linux__ -+ if (!S_ISBLK(st.st_mode)) -+#else -+ if (!S_ISBLK(st.st_mode) && !S_ISCHR(st.st_mode)) -+#endif -+ { -+ if (quiet) -+ printf("\n"); -+ else -+ fprintf(stderr, "mountpoint: %s: not a block device\n", -+ path); -+ return 1; -+ } -+ printf("%u:%u\n", major(st.st_rdev), minor(st.st_rdev)); -+ return 0; -+ } -+ -+ if (!S_ISDIR(st.st_mode)) { -+ if (!quiet) -+ fprintf(stderr, "mountpoint: %s: not a directory\n", -+ path); -+ return 1; -+ } -+ -+ memset(buf, 0, sizeof(buf)); -+ strncpy(buf, path, sizeof(buf) - 4); -+ strcat(buf, "/.."); -+ if (dostat(buf, &st2, 0, quiet) < 0) -+ return 1; -+ -+ r = (st.st_dev != st2.st_dev) || -+ (st.st_dev == st2.st_dev && st.st_ino == st2.st_ino); -+ -+ if (!quiet && !showdev) -+ printf("%s is %sa mountpoint\n", path, r ? "" : "not "); -+ if (showdev) -+ printf("%u:%u\n", major(st.st_dev), minor(st.st_dev)); -+ -+ return r ? 0 : 1; -+} -Binary files sysvinit-2.85/src/mountpoint.o and sysvinit-2.86/src/mountpoint.o differ -diff -urNd -urNd sysvinit-2.85/src/paths.h sysvinit-2.86/src/paths.h ---- sysvinit-2.85/src/paths.h 2003-04-14 06:37:01.000000000 -0500 -+++ sysvinit-2.86/src/paths.h 2004-06-09 07:47:45.000000000 -0500 -@@ -1,7 +1,7 @@ - /* - * paths.h Paths of files that init and related utilities need. - * -- * Version: @(#) paths.h 2.84 27-Nov-2001 -+ * Version: @(#) paths.h 2.85-8 05-Nov-2003 - * - * Author: Miquel van Smoorenburg, <miquels@cistron.nl> - * -@@ -24,6 +24,7 @@ - #define FORCEFSCK "/forcefsck" /* Force fsck on boot */ - #define SDPID "/var/run/shutdown.pid" /* PID of shutdown program */ - #define SHELL "/bin/sh" /* Default shell */ -+#define SULOGIN "/sbin/sulogin" /* Sulogin */ - #define INITSCRIPT "/etc/initscript" /* Initscript. */ - #define PWRSTAT "/etc/powerstatus" /* COMPAT: SIGPWR reason (OK/BAD) */ - -diff -urNd -urNd sysvinit-2.85/src/reboot.h sysvinit-2.86/src/reboot.h ---- sysvinit-2.85/src/reboot.h 1997-09-24 03:55:52.000000000 -0500 -+++ sysvinit-2.86/src/reboot.h 2004-06-09 07:47:45.000000000 -0500 -@@ -2,22 +2,35 @@ - * reboot.h Headerfile that defines how to handle - * the reboot() system call. - * -- * Version: @(#)reboot.h 1.00 23-Jul-1996 miquels@cistron.nl -+ * Version: @(#)reboot.h 2.85-17 04-Jun-2004 miquels@cistron.nl - * - */ - --#if defined(__GLIBC__) --# include <sys/reboot.h> -+#include <sys/reboot.h> -+ -+#ifdef RB_ENABLE_CAD -+# define BMAGIC_HARD RB_ENABLE_CAD - #endif - --#define BMAGIC_HARD 0x89ABCDEF --#define BMAGIC_SOFT 0 --#define BMAGIC_REBOOT 0x01234567 --#define BMAGIC_HALT 0xCDEF0123 --#define BMAGIC_POWEROFF 0x4321FEDC -+#ifdef RB_DISABLE_CAD -+# define BMAGIC_SOFT RB_DISABLE_CAD -+#endif - --#if defined(__GLIBC__) -- #define init_reboot(magic) reboot(magic) -+#ifdef RB_HALT_SYSTEM -+# define BMAGIC_HALT RB_HALT_SYSTEM - #else -- #define init_reboot(magic) reboot(0xfee1dead, 672274793, magic) -+# define BMAGIC_HALT RB_HALT - #endif -+ -+#define BMAGIC_REBOOT RB_AUTOBOOT -+ -+#ifdef RB_POWER_OFF -+# define BMAGIC_POWEROFF RB_POWER_OFF -+#elif defined(RB_POWEROFF) -+# define BMAGIC_POWEROFF RB_POWEROFF -+#else -+# define BMAGIC_POWEROFF BMAGIC_HALT -+#endif -+ -+#define init_reboot(magic) reboot(magic) -+ -Binary files sysvinit-2.85/src/runlevel and sysvinit-2.86/src/runlevel differ -Binary files sysvinit-2.85/src/runlevel.o and sysvinit-2.86/src/runlevel.o differ -Binary files sysvinit-2.85/src/shutdown and sysvinit-2.86/src/shutdown differ -diff -urNd -urNd sysvinit-2.85/src/shutdown.c sysvinit-2.86/src/shutdown.c ---- sysvinit-2.85/src/shutdown.c 2003-04-14 06:35:51.000000000 -0500 -+++ sysvinit-2.86/src/shutdown.c 2004-07-30 06:59:04.000000000 -0500 -@@ -13,10 +13,10 @@ - * - * Author: Miquel van Smoorenburg, miquels@cistron.nl - * -- * Version: @(#)shutdown 2.85 14-Apr-2003 miquels@cistron.nl -+ * Version: @(#)shutdown 2.86-1 31-Jul-2004 miquels@cistron.nl - * - * This file is part of the sysvinit suite, -- * Copyright 1991-2003 Miquel van Smoorenburg. -+ * Copyright 1991-2004 Miquel van Smoorenburg. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License -@@ -36,10 +36,12 @@ - #include <fcntl.h> - #include <stdarg.h> - #include <utmp.h> -+#include <syslog.h> - #include "paths.h" - #include "reboot.h" -+#include "initreq.h" - --char *Version = "@(#) shutdown 2.85 14-Apr-2003 miquels@cistron.nl"; -+char *Version = "@(#) shutdown 2.86-1 31-Jul-2004 miquels@cistron.nl"; - - #define MESSAGELEN 256 - -@@ -52,6 +54,7 @@ - char *sltime = 0; /* Sleep time */ - char newstate[64]; /* What are we gonna do */ - int doself = 0; /* Don't use init */ -+int got_alrm = 0; - - char *clean_env[] = { - "HOME=/", -@@ -67,93 +70,152 @@ - extern void write_wtmp(char *user, char *id, int pid, int type, char *line); - - /* -- * Sleep without being interrupted. -+ * Sleep without being interrupted. - */ - void hardsleep(int secs) - { -- struct timespec ts, rem; -+ struct timespec ts, rem; - -- ts.tv_sec = secs; -- ts.tv_nsec = 0; -+ ts.tv_sec = secs; -+ ts.tv_nsec = 0; - -- while(nanosleep(&ts, &rem) < 0 && errno == EINTR) -+ while(nanosleep(&ts, &rem) < 0 && errno == EINTR) - ts = rem; - } - - /* -- * Break off an already running shutdown. -+ * Break off an already running shutdown. - */ --void stopit() -+void stopit(int sig) - { -- unlink(NOLOGIN); -- unlink(FASTBOOT); -- unlink(FORCEFSCK); -- unlink(SDPID); -- printf("\r\nShutdown cancelled.\r\n"); -- exit(0); -+ unlink(NOLOGIN); -+ unlink(FASTBOOT); -+ unlink(FORCEFSCK); -+ unlink(SDPID); -+ printf("\r\nShutdown cancelled.\r\n"); -+ exit(0); - } - - /* -- * Show usage message. -+ * Show usage message. - */ --void usage() -+void usage(void) - { -- fprintf(stderr, -- "Usage:\t shutdown [-akrhfnc] [-t secs] time [warning message]\n" -+ fprintf(stderr, -+ "Usage:\t shutdown [-akrhHPfnc] [-t secs] time [warning message]\n" - "\t\t -a: use /etc/shutdown.allow\n" - "\t\t -k: don't really shutdown, only warn.\n" - "\t\t -r: reboot after shutdown.\n" - "\t\t -h: halt after shutdown.\n" -+ "\t\t -P: halt action is to turn off power.\n" -+ "\t\t -H: halt action is to just halt.\n" - "\t\t -f: do a 'fast' reboot (skip fsck).\n" - "\t\t -F: Force fsck on reboot.\n" - "\t\t -n: do not go through \"init\" but go down real fast.\n" - "\t\t -c: cancel a running shutdown.\n" - "\t\t -t secs: delay between warning and kill signal.\n" - "\t\t ** the \"time\" argument is mandatory! (try \"now\") **\n"); -- exit(1); -+ exit(1); - } - -+ -+void alrm_handler(int sig) -+{ -+ got_alrm = sig; -+} -+ -+ - /* -- * Tell everyone the system is going down in 'mins' minutes. -+ * Set environment variables in the init process. - */ --void warn(mins) --int mins; -+int init_setenv(char *name, char *value) - { -- char buf[MESSAGELEN + sizeof(newstate)]; -- int len; -+ struct init_request request; -+ struct sigaction sa; -+ int fd; -+ int nl, vl; - -- buf[0] = 0; -- strncat(buf, message, sizeof(buf) - 1); -- len = strlen(buf); -+ memset(&request, 0, sizeof(request)); -+ request.magic = INIT_MAGIC; -+ request.cmd = INIT_CMD_SETENV; -+ nl = strlen(name); -+ vl = value ? strlen(value) : 0; - -- if (mins == 0) -- snprintf(buf + len, sizeof(buf) - len, -- "\rThe system is going down %s NOW!\r\n", -- newstate); -- else -- snprintf(buf + len, sizeof(buf) - len, -- "\rThe system is going DOWN %s in %d minute%s!\r\n", -- newstate, mins, mins == 1 ? "" : "s"); -- wall(buf, 1, 0); -+ if (nl + vl + 3 >= sizeof(request.i.data)) -+ return -1; -+ -+ memcpy(request.i.data, name, nl); -+ if (value) { -+ request.i.data[nl] = '='; -+ memcpy(request.i.data + nl + 1, value, vl); -+ } -+ -+ /* -+ * Open the fifo and write the command. -+ * Make sure we don't hang on opening /dev/initctl -+ */ -+ memset(&sa, 0, sizeof(sa)); -+ sa.sa_handler = alrm_handler; -+ sigaction(SIGALRM, &sa, NULL); -+ got_alrm = 0; -+ alarm(3); -+ if ((fd = open(INIT_FIFO, O_WRONLY)) >= 0 && -+ write(fd, &request, sizeof(request)) == sizeof(request)) { -+ close(fd); -+ alarm(0); -+ return 0; -+ } -+ -+ fprintf(stderr, "shutdown: "); -+ if (got_alrm) { -+ fprintf(stderr, "timeout opening/writing control channel %s\n", -+ INIT_FIFO); -+ } else { -+ perror(INIT_FIFO); -+ } -+ return -1; - } - -+ - /* -- * Create the /etc/nologin file. -+ * Tell everyone the system is going down in 'mins' minutes. -+ */ -+void warn(int mins) -+{ -+ char buf[MESSAGELEN + sizeof(newstate)]; -+ int len; -+ -+ buf[0] = 0; -+ strncat(buf, message, sizeof(buf) - 1); -+ len = strlen(buf); -+ -+ if (mins == 0) -+ snprintf(buf + len, sizeof(buf) - len, -+ "\rThe system is going down %s NOW!\r\n", -+ newstate); -+ else -+ snprintf(buf + len, sizeof(buf) - len, -+ "\rThe system is going DOWN %s in %d minute%s!\r\n", -+ newstate, mins, mins == 1 ? "" : "s"); -+ wall(buf, 1, 0); -+} -+ -+/* -+ * Create the /etc/nologin file. - */ - void donologin(int min) - { -- FILE *fp; -- time_t t; -+ FILE *fp; -+ time_t t; - -- time(&t); -- t += 60 * min; -+ time(&t); -+ t += 60 * min; - -- unlink(NOLOGIN); -- if ((fp = fopen(NOLOGIN, "w")) != NULL) { -- fprintf(fp, "\rThe system is going down on %s\r\n", ctime(&t)); -- if (message[0]) fputs(message, fp); -- fclose(fp); -- } -+ if ((fp = fopen(NOLOGIN, "w")) != NULL) { -+ fprintf(fp, "\rThe system is going down on %s\r\n", ctime(&t)); -+ if (message[0]) fputs(message, fp); -+ fclose(fp); -+ } - } - - /* -@@ -202,131 +264,146 @@ - return 0; - } - --/* Kill all processes, call /etc/init.d/halt (if present) */ -+/* -+ * Kill all processes, call /etc/init.d/halt (if present) -+ */ - void fastdown() - { -- int do_halt = (down_level[0] == '0'); -- int i; -+ int do_halt = (down_level[0] == '0'); -+ int i; - #if 0 -- char cmd[128]; -- char *script; -+ char cmd[128]; -+ char *script; - -- /* Currently, the halt script is either init.d/halt OR rc.d/rc.0, -- * likewise for the reboot script. Test for the presence -- * of either. -- */ -- if (do_halt) { -- if (access(HALTSCRIPT1, X_OK) == 0) -- script = HALTSCRIPT1; -- else -- script = HALTSCRIPT2; -- } else { -- if (access(REBOOTSCRIPT1, X_OK) == 0) -- script = REBOOTSCRIPT1; -- else -- script = REBOOTSCRIPT2; -- } -+ /* -+ * Currently, the halt script is either init.d/halt OR rc.d/rc.0, -+ * likewise for the reboot script. Test for the presence -+ * of either. -+ */ -+ if (do_halt) { -+ if (access(HALTSCRIPT1, X_OK) == 0) -+ script = HALTSCRIPT1; -+ else -+ script = HALTSCRIPT2; -+ } else { -+ if (access(REBOOTSCRIPT1, X_OK) == 0) -+ script = REBOOTSCRIPT1; -+ else -+ script = REBOOTSCRIPT2; -+ } - #endif - -- /* First close all files. */ -- for(i = 0; i < 3; i++) -- if (!isatty(i)) { -- close(i); -- open("/dev/null", O_RDWR); -- } -- for(i = 3; i < 20; i++) close(i); -- close(255); -+ /* First close all files. */ -+ for(i = 0; i < 3; i++) -+ if (!isatty(i)) { -+ close(i); -+ open("/dev/null", O_RDWR); -+ } -+ for(i = 3; i < 20; i++) close(i); -+ close(255); - -- /* First idle init. */ -- if (kill(1, SIGTSTP) < 0) { -- fprintf(stderr, "shutdown: can't idle init.\r\n"); -- exit(1); -- } -+ /* First idle init. */ -+ if (kill(1, SIGTSTP) < 0) { -+ fprintf(stderr, "shutdown: can't idle init.\r\n"); -+ exit(1); -+ } - -- /* Kill all processes. */ -- fprintf(stderr, "shutdown: sending all processes the TERM signal...\r\n"); -- (void) kill(-1, SIGTERM); -- if (sltime) -- sleep(atoi(sltime)); -- else -- sleep(3); -- fprintf(stderr, "shutdown: sending all processes the KILL signal.\r\n"); -- (void) kill(-1, SIGKILL); -+ /* Kill all processes. */ -+ fprintf(stderr, "shutdown: sending all processes the TERM signal...\r\n"); -+ kill(-1, SIGTERM); -+ sleep(sltime ? atoi(sltime) : 3); -+ fprintf(stderr, "shutdown: sending all processes the KILL signal.\r\n"); -+ (void) kill(-1, SIGKILL); - - #if 0 -- /* See if we can run /etc/init.d/halt */ -- if (access(script, X_OK) == 0) { -- spawn(1, cmd, "fast", NULL); -- fprintf(stderr, "shutdown: %s returned - falling back on default routines\r\n", script); -- } -+ /* See if we can run /etc/init.d/halt */ -+ if (access(script, X_OK) == 0) { -+ spawn(1, cmd, "fast", NULL); -+ fprintf(stderr, "shutdown: %s returned - falling back " -+ "on default routines\r\n", script); -+ } - #endif - -- /* script failed or not present: do it ourself. */ -- sleep(1); /* Give init the chance to collect zombies. */ -+ /* script failed or not present: do it ourself. */ -+ sleep(1); /* Give init the chance to collect zombies. */ - -- /* Record the fact that we're going down */ -- write_wtmp("shutdown", "~~", 0, RUN_LVL, "~~"); -+ /* Record the fact that we're going down */ -+ write_wtmp("shutdown", "~~", 0, RUN_LVL, "~~"); - -- /* This is for those who have quota installed. */ -- spawn(1, "accton", NULL); -- spawn(1, "quotaoff", "-a", NULL); -+ /* This is for those who have quota installed. */ -+ spawn(1, "accton", NULL); -+ spawn(1, "quotaoff", "-a", NULL); - -- sync(); -- fprintf(stderr, "shutdown: turning off swap\r\n"); -- spawn(0, "swapoff", "-a", NULL); -- fprintf(stderr, "shutdown: unmounting all file systems\r\n"); -- spawn(0, "umount", "-a", NULL); -+ sync(); -+ fprintf(stderr, "shutdown: turning off swap\r\n"); -+ spawn(0, "swapoff", "-a", NULL); -+ fprintf(stderr, "shutdown: unmounting all file systems\r\n"); -+ spawn(0, "umount", "-a", NULL); - -- /* We're done, halt or reboot now. */ -- if (do_halt) { -- fprintf(stderr, "The system is halted. Press CTRL-ALT-DEL or turn off power\r\n"); -- init_reboot(BMAGIC_HALT); -+ /* We're done, halt or reboot now. */ -+ if (do_halt) { -+ fprintf(stderr, "The system is halted. Press CTRL-ALT-DEL " -+ "or turn off power\r\n"); -+ init_reboot(BMAGIC_HALT); -+ exit(0); -+ } -+ -+ fprintf(stderr, "Please stand by while rebooting the system.\r\n"); -+ init_reboot(BMAGIC_REBOOT); - exit(0); -- } -- fprintf(stderr, "Please stand by while rebooting the system.\r\n"); -- init_reboot(BMAGIC_REBOOT); -- exit(0); - } - - /* -- * Go to runlevel 0, 1 or 6. -+ * Go to runlevel 0, 1 or 6. - */ --void shutdown() -+void shutdown(char *halttype) - { -- char *args[8]; -- int argp = 0; -+ char *args[8]; -+ int argp = 0; -+ int do_halt = (down_level[0] == '0'); - -- /* Warn for the last time (hehe) */ -- warn(0); -- if (dontshut) { -- hardsleep(1); -- stopit(); -- } -+ /* Warn for the last time */ -+ warn(0); -+ if (dontshut) { -+ hardsleep(1); -+ stopit(0); -+ } -+ openlog("shutdown", LOG_PID, LOG_USER); -+ if (do_halt) -+ syslog(LOG_NOTICE, "shutting down for system halt"); -+ else -+ syslog(LOG_NOTICE, "shutting down for system reboot"); -+ closelog(); - -- /* See if we have to do it ourself. */ -- if (doself) fastdown(); -+ /* See if we have to do it ourself. */ -+ if (doself) fastdown(); - -- /* Create the arguments for init. */ -- args[argp++] = INIT; -- if (sltime) { -- args[argp++] = "-t"; -- args[argp++] = sltime; -- } -- args[argp++] = down_level; -- args[argp] = (char *)NULL; -+ /* Create the arguments for init. */ -+ args[argp++] = INIT; -+ if (sltime) { -+ args[argp++] = "-t"; -+ args[argp++] = sltime; -+ } -+ args[argp++] = down_level; -+ args[argp] = (char *)NULL; - -- unlink(SDPID); -- unlink(NOLOGIN); -+ unlink(SDPID); -+ unlink(NOLOGIN); - -- /* Now execute init to change runlevel. */ -- sync(); -- execv(INIT, args); -+ /* Now execute init to change runlevel. */ -+ sync(); -+ init_setenv("INIT_HALT", halttype); -+ execv(INIT, args); - -- /* Oops - failed. */ -- fprintf(stderr, "\rshutdown: cannot execute %s\r\n", INIT); -- unlink(FASTBOOT); -- unlink(FORCEFSCK); -- exit(1); -+ /* Oops - failed. */ -+ fprintf(stderr, "\rshutdown: cannot execute %s\r\n", INIT); -+ unlink(FASTBOOT); -+ unlink(FORCEFSCK); -+ init_setenv("INIT_HALT", NULL); -+ openlog("shutdown", LOG_PID, LOG_USER); -+ syslog(LOG_NOTICE, "shutdown failed"); -+ closelog(); -+ exit(1); - } - - /* -@@ -349,274 +426,287 @@ - } - - /* -- * Main program. -- * Process the options and do the final countdown. -+ * Main program. -+ * Process the options and do the final countdown. - */ --int main(argc, argv) --int argc; --char **argv; -+int main(int argc, char **argv) - { -- extern int getopt(); -- extern int optind; -- int c, i, wt, hours, mins; -- struct tm *lt; -- time_t t; -- char *sp; -- char *when = NULL; -- int didnolog = 0; -- int cancel = 0; -- int useacl = 0; -- int pid = 0; -- uid_t realuid; -- FILE *fp; -- char *downusers[32]; -- char buf[128]; -- char term[UT_LINESIZE + 6]; -- struct stat st; -- struct utmp *ut; -- int user_ok = 0; -- struct sigaction sa; -+ FILE *fp; -+ extern int getopt(); -+ extern int optind; -+ struct sigaction sa; -+ struct tm *lt; -+ struct stat st; -+ struct utmp *ut; -+ time_t t; -+ uid_t realuid; -+ char *halttype; -+ char *downusers[32]; -+ char buf[128]; -+ char term[UT_LINESIZE + 6]; -+ char *sp; -+ char *when = NULL; -+ int c, i, wt; -+ int hours, mins; -+ int didnolog = 0; -+ int cancel = 0; -+ int useacl = 0; -+ int pid = 0; -+ int user_ok = 0; - -- /* We can be installed setuid root (executable for a special group) */ -- realuid = getuid(); -- setuid(geteuid()); -+ /* We can be installed setuid root (executable for a special group) */ -+ realuid = getuid(); -+ setuid(geteuid()); - -- if (getuid() != 0) { -- fprintf(stderr, "shutdown: you must be root to do that!\n"); -- exit(1); -- } -- strcpy(down_level, "1"); -+ if (getuid() != 0) { -+ fprintf(stderr, "shutdown: you must be root to do that!\n"); -+ exit(1); -+ } -+ strcpy(down_level, "1"); -+ halttype = NULL; - -- /* Process the options. */ -- while((c = getopt(argc, argv, "acqkrhnfFyt:g:i:")) != EOF) { -- switch(c) { -- case 'a': /* Access control. */ -- useacl = 1; -- break; -- case 'c': /* Cancel an already running shutdown. */ -- cancel = 1; -- break; -- case 'k': /* Don't really shutdown, only warn.*/ -- dontshut = 1; -- break; -- case 'r': /* Automatic reboot */ -- down_level[0] = '6'; -- break; -- case 'h': /* Halt after shutdown */ -- down_level[0] = '0'; -- break; -- case 'f': /* Don't perform fsck after next boot */ -- fastboot = 1; -- break; -- case 'F': /* Force fsck after next boot */ -- forcefsck = 1; -- break; -- case 'n': /* Don't switch runlevels. */ -- doself = 1; -- break; -- case 't': /* Delay between TERM and KILL */ -- sltime = optarg; -- break; -- case 'y': /* Ignored for sysV compatibility */ -- break; -- case 'g': /* sysv style to specify time. */ -- when = optarg; -- down_level[0] = '0'; -- break; -- case 'i': /* Level to go to. */ -- if (!strchr("0156aAbBcCsS", optarg[0])) { -- fprintf(stderr, "shutdown: `%s': bad runlevel\n", -+ /* Process the options. */ -+ while((c = getopt(argc, argv, "HPacqkrhnfFyt:g:i:")) != EOF) { -+ switch(c) { -+ case 'H': -+ halttype = "HALT"; -+ break; -+ case 'P': -+ halttype = "POWERDOWN"; -+ break; -+ case 'a': /* Access control. */ -+ useacl = 1; -+ break; -+ case 'c': /* Cancel an already running shutdown. */ -+ cancel = 1; -+ break; -+ case 'k': /* Don't really shutdown, only warn.*/ -+ dontshut = 1; -+ break; -+ case 'r': /* Automatic reboot */ -+ down_level[0] = '6'; -+ break; -+ case 'h': /* Halt after shutdown */ -+ down_level[0] = '0'; -+ break; -+ case 'f': /* Don't perform fsck after next boot */ -+ fastboot = 1; -+ break; -+ case 'F': /* Force fsck after next boot */ -+ forcefsck = 1; -+ break; -+ case 'n': /* Don't switch runlevels. */ -+ doself = 1; -+ break; -+ case 't': /* Delay between TERM and KILL */ -+ sltime = optarg; -+ break; -+ case 'y': /* Ignored for sysV compatibility */ -+ break; -+ case 'g': /* sysv style to specify time. */ -+ when = optarg; -+ break; -+ case 'i': /* Level to go to. */ -+ if (!strchr("0156aAbBcCsS", optarg[0])) { -+ fprintf(stderr, -+ "shutdown: `%s': bad runlevel\n", - optarg); -- exit(1); -- } -- down_level[0] = optarg[0]; -- break; -- default: -- usage(); -- break; -- } -- } -+ exit(1); -+ } -+ down_level[0] = optarg[0]; -+ break; -+ default: -+ usage(); -+ break; -+ } -+ } - -- /* Do we need to use the shutdown.allow file ? */ -- if (useacl && (fp = fopen(SDALLOW, "r")) != NULL) { -+ /* Do we need to use the shutdown.allow file ? */ -+ if (useacl && (fp = fopen(SDALLOW, "r")) != NULL) { - -- /* Read /etc/shutdown.allow. */ -- i = 0; -- while(fgets(buf, 128, fp)) { -- if (buf[0] == '#' || buf[0] == '\n') continue; -- if (i > 31) continue; -- for(sp = buf; *sp; sp++) if (*sp == '\n') *sp = 0; -- downusers[i++] = strdup(buf); -- } -- if (i < 32) downusers[i] = 0; -- fclose(fp); -+ /* Read /etc/shutdown.allow. */ -+ i = 0; -+ while(fgets(buf, 128, fp)) { -+ if (buf[0] == '#' || buf[0] == '\n') continue; -+ if (i > 31) continue; -+ for(sp = buf; *sp; sp++) if (*sp == '\n') *sp = 0; -+ downusers[i++] = strdup(buf); -+ } -+ if (i < 32) downusers[i] = 0; -+ fclose(fp); - -- /* Now walk through /var/run/utmp to find logged in users. */ -- while(!user_ok && (ut = getutent()) != NULL) { -+ /* Now walk through /var/run/utmp to find logged in users. */ -+ while(!user_ok && (ut = getutent()) != NULL) { - -- /* See if this is a user process on a VC. */ -- if (ut->ut_type != USER_PROCESS) continue; -- sprintf(term, "/dev/%.*s", UT_LINESIZE, ut->ut_line); -- if (stat(term, &st) < 0) continue; -+ /* See if this is a user process on a VC. */ -+ if (ut->ut_type != USER_PROCESS) continue; -+ sprintf(term, "/dev/%.*s", UT_LINESIZE, ut->ut_line); -+ if (stat(term, &st) < 0) continue; - #ifdef major /* glibc */ -- if (major(st.st_rdev) != 4 || -- minor(st.st_rdev) > 63) continue; -+ if (major(st.st_rdev) != 4 || -+ minor(st.st_rdev) > 63) continue; - #else -- if ((st.st_rdev & 0xFFC0) != 0x0400) continue; -+ if ((st.st_rdev & 0xFFC0) != 0x0400) continue; - #endif -- /* Root is always OK. */ -- if (strcmp(ut->ut_user, "root") == 0) { -- user_ok++; -- break; -- } -- -- /* See if this is an allowed user. */ -- for(i = 0; i < 32 && downusers[i]; i++) -- if (!strncmp(downusers[i], ut->ut_user, UT_NAMESIZE)) { -+ /* Root is always OK. */ -+ if (strcmp(ut->ut_user, "root") == 0) { - user_ok++; - break; - } -- } -- endutent(); - -- /* See if user was allowed. */ -- if (!user_ok) { -- if ((fp = fopen(CONSOLE, "w")) != NULL) { -- fprintf(fp, "\rshutdown: no authorized users logged in.\r\n"); -- fclose(fp); -+ /* See if this is an allowed user. */ -+ for(i = 0; i < 32 && downusers[i]; i++) -+ if (!strncmp(downusers[i], ut->ut_user, -+ UT_NAMESIZE)) { -+ user_ok++; -+ break; -+ } - } -- exit(1); -- } -- } -+ endutent(); - -- /* Read pid of running shutdown from a file */ -- if ((fp = fopen(SDPID, "r")) != NULL) { -- fscanf(fp, "%d", &pid); -- fclose(fp); -- } -+ /* See if user was allowed. */ -+ if (!user_ok) { -+ if ((fp = fopen(CONSOLE, "w")) != NULL) { -+ fprintf(fp, "\rshutdown: no authorized users " -+ "logged in.\r\n"); -+ fclose(fp); -+ } -+ exit(1); -+ } -+ } - -- /* Read remaining words, skip time if needed. */ -- message[0] = 0; -- for(c = optind + (!cancel && !when); c < argc; c++) { -- if (strlen(message) + strlen(argv[c]) + 4 > MESSAGELEN) -- break; -- strcat(message, argv[c]); -- strcat(message, " "); -- } -- if (message[0]) strcat(message, "\r\n"); -+ /* Read pid of running shutdown from a file */ -+ if ((fp = fopen(SDPID, "r")) != NULL) { -+ fscanf(fp, "%d", &pid); -+ fclose(fp); -+ } - -- /* See if we want to run or cancel. */ -- if (cancel) { -- if (pid <= 0) { -- fprintf(stderr, "shutdown: cannot find pid of running shutdown.\n"); -- exit(1); -+ /* Read remaining words, skip time if needed. */ -+ message[0] = 0; -+ for(c = optind + (!cancel && !when); c < argc; c++) { -+ if (strlen(message) + strlen(argv[c]) + 4 > MESSAGELEN) -+ break; -+ strcat(message, argv[c]); -+ strcat(message, " "); - } -- if (kill(pid, SIGINT) < 0) { -- fprintf(stderr, "shutdown: not running.\n"); -- exit(1); -+ if (message[0]) strcat(message, "\r\n"); -+ -+ /* See if we want to run or cancel. */ -+ if (cancel) { -+ if (pid <= 0) { -+ fprintf(stderr, "shutdown: cannot find pid " -+ "of running shutdown.\n"); -+ exit(1); -+ } -+ init_setenv("INIT_HALT", NULL); -+ if (kill(pid, SIGINT) < 0) { -+ fprintf(stderr, "shutdown: not running.\n"); -+ exit(1); -+ } -+ if (message[0]) wall(message, 1, 0); -+ exit(0); - } -- if (message[0]) wall(message, 1, 0); -- exit(0); -- } - -- /* Check syntax. */ -- if (when == NULL) { -- if (optind == argc) usage(); -- when = argv[optind++]; -- } -+ /* Check syntax. */ -+ if (when == NULL) { -+ if (optind == argc) usage(); -+ when = argv[optind++]; -+ } - -- /* See if we are already running. */ -- if (pid > 0 && kill(pid, 0) == 0) { -- fprintf(stderr, "\rshutdown: already running.\r\n"); -- exit(1); -- } -+ /* See if we are already running. */ -+ if (pid > 0 && kill(pid, 0) == 0) { -+ fprintf(stderr, "\rshutdown: already running.\r\n"); -+ exit(1); -+ } - -- /* Extra check. */ -- if (doself && down_level[0] != '0' && down_level[0] != '6') { -- fprintf(stderr, "shutdown: can use \"-n\" for halt or reboot only.\r\n"); -- exit(1); -- } -+ /* Extra check. */ -+ if (doself && down_level[0] != '0' && down_level[0] != '6') { -+ fprintf(stderr, -+ "shutdown: can use \"-n\" for halt or reboot only.\r\n"); -+ exit(1); -+ } - -- /* Tell users what we're gonna do. */ -- switch(down_level[0]) { -- case '0': -- strcpy(newstate, "for system halt"); -- break; -- case '6': -- strcpy(newstate, "for reboot"); -- break; -- case '1': -- strcpy(newstate, "to maintenance mode"); -- break; -- default: -- sprintf(newstate, "to runlevel %s", down_level); -- break; -- } -+ /* Tell users what we're gonna do. */ -+ switch(down_level[0]) { -+ case '0': -+ strcpy(newstate, "for system halt"); -+ break; -+ case '6': -+ strcpy(newstate, "for reboot"); -+ break; -+ case '1': -+ strcpy(newstate, "to maintenance mode"); -+ break; -+ default: -+ sprintf(newstate, "to runlevel %s", down_level); -+ break; -+ } - -- /* Create a new PID file. */ -- unlink(SDPID); -- umask(022); -- if ((fp = fopen(SDPID, "w")) != NULL) { -- fprintf(fp, "%d\n", getpid()); -- fclose(fp); -- } else if (errno != EROFS) -- fprintf(stderr, "shutdown: warning: cannot open %s\n", SDPID); -+ /* Create a new PID file. */ -+ unlink(SDPID); -+ umask(022); -+ if ((fp = fopen(SDPID, "w")) != NULL) { -+ fprintf(fp, "%d\n", getpid()); -+ fclose(fp); -+ } else if (errno != EROFS) -+ fprintf(stderr, "shutdown: warning: cannot open %s\n", SDPID); - -- /* -- * Catch some common signals. -- */ -- signal(SIGQUIT, SIG_IGN); -- signal(SIGCHLD, SIG_IGN); -- signal(SIGHUP, SIG_IGN); -- signal(SIGTSTP, SIG_IGN); -- signal(SIGTTIN, SIG_IGN); -- signal(SIGTTOU, SIG_IGN); -+ /* -+ * Catch some common signals. -+ */ -+ signal(SIGQUIT, SIG_IGN); -+ signal(SIGCHLD, SIG_IGN); -+ signal(SIGHUP, SIG_IGN); -+ signal(SIGTSTP, SIG_IGN); -+ signal(SIGTTIN, SIG_IGN); -+ signal(SIGTTOU, SIG_IGN); - -- sa.sa_handler = stopit; -- sa.sa_flags = SA_RESTART; -- sigaction(SIGINT, &sa, NULL); -+ memset(&sa, 0, sizeof(sa)); -+ sa.sa_handler = stopit; -+ sigaction(SIGINT, &sa, NULL); - -- /* Go to the root directory */ -- chdir("/"); -- if (fastboot) close(open(FASTBOOT, O_CREAT | O_RDWR, 0644)); -- if (forcefsck) close(open(FORCEFSCK, O_CREAT | O_RDWR, 0644)); -+ /* Go to the root directory */ -+ chdir("/"); -+ if (fastboot) close(open(FASTBOOT, O_CREAT | O_RDWR, 0644)); -+ if (forcefsck) close(open(FORCEFSCK, O_CREAT | O_RDWR, 0644)); - -- /* Alias now and take care of old '+mins' notation. */ -- if (!strcmp(when, "now")) strcpy(when, "0"); -- if (when[0] == '+') when++; -+ /* Alias now and take care of old '+mins' notation. */ -+ if (!strcmp(when, "now")) strcpy(when, "0"); -+ if (when[0] == '+') when++; - -- /* Decode shutdown time. */ -- for (sp = when; *sp; sp++) { -- if (*sp != ':' && (*sp < '0' || *sp > '9')) -- usage(); -- } -- if (strchr(when, ':') == NULL) { -- /* Time in minutes. */ -- wt = atoi(when); -- if (wt == 0 && when[0] != '0') usage(); -- } else { -- /* Time in hh:mm format. */ -- if (sscanf(when, "%d:%2d", &hours, &mins) != 2) usage(); -- if (hours > 23 || mins > 59) usage(); -- time(&t); -- lt = localtime(&t); -- wt = (60*hours + mins) - (60*lt->tm_hour + lt->tm_min); -- if (wt < 0) wt += 1440; -- } -- /* Shutdown NOW if time == 0 */ -- if (wt == 0) shutdown(); -+ /* Decode shutdown time. */ -+ for (sp = when; *sp; sp++) { -+ if (*sp != ':' && (*sp < '0' || *sp > '9')) -+ usage(); -+ } -+ if (strchr(when, ':') == NULL) { -+ /* Time in minutes. */ -+ wt = atoi(when); -+ if (wt == 0 && when[0] != '0') usage(); -+ } else { -+ /* Time in hh:mm format. */ -+ if (sscanf(when, "%d:%2d", &hours, &mins) != 2) usage(); -+ if (hours > 23 || mins > 59) usage(); -+ time(&t); -+ lt = localtime(&t); -+ wt = (60*hours + mins) - (60*lt->tm_hour + lt->tm_min); -+ if (wt < 0) wt += 1440; -+ } -+ /* Shutdown NOW if time == 0 */ -+ if (wt == 0) shutdown(halttype); - -- /* Give warnings on regular intervals and finally shutdown. */ -- if (wt < 15 && !needwarning(wt)) warn(wt); -- while(wt) { -- if (wt <= 5 && !didnolog) { -- donologin(wt); -- didnolog++; -+ /* Give warnings on regular intervals and finally shutdown. */ -+ if (wt < 15 && !needwarning(wt)) warn(wt); -+ while(wt) { -+ if (wt <= 5 && !didnolog) { -+ donologin(wt); -+ didnolog++; -+ } -+ if (needwarning(wt)) warn(wt); -+ hardsleep(60); -+ wt--; - } -- if (needwarning(wt)) warn(wt); -- hardsleep(60); -- wt--; -- } -- shutdown(); -- return(0); /* Never happens */ -+ shutdown(halttype); -+ -+ return 0; /* Never happens */ - } -Binary files sysvinit-2.85/src/shutdown.o and sysvinit-2.86/src/shutdown.o differ -Binary files sysvinit-2.85/src/sulogin and sysvinit-2.86/src/sulogin differ -diff -urNd -urNd sysvinit-2.85/src/sulogin.c sysvinit-2.86/src/sulogin.c ---- sysvinit-2.85/src/sulogin.c 2003-04-14 04:53:49.000000000 -0500 -+++ sysvinit-2.86/src/sulogin.c 2004-07-30 06:40:28.000000000 -0500 -@@ -8,7 +8,7 @@ - * encrypted root password is "x" the shadow - * password will be used. - * -- * Version: @(#)sulogin 2.85 14-Apr-2003 miquels@cistron.nl -+ * Version: @(#)sulogin 2.85-3 23-Apr-2003 miquels@cistron.nl - * - */ - -@@ -35,11 +35,15 @@ - #define F_SHADOW "/etc/shadow" - #define BINSH "/bin/sh" - --char *Version = "@(#)sulogin 2.85 14-Apr-2003 miquels@cistron.nl"; -+char *Version = "@(#)sulogin 2.85-3 23-Apr-2003 miquels@cistron.nl"; - - int timeout = 0; - int profile = 0; - -+#ifndef IUCLC -+# define IUCLC 0 -+#endif -+ - #if 0 - /* - * Fix the tty modes and set reasonable defaults. -@@ -252,7 +256,7 @@ - printf("Give root password for maintenance\n"); - else - printf("Press enter for maintenance\n"); -- printf("(or type Control-D for normal startup): "); -+ printf("(or type Control-D to continue): "); - fflush(stdout); - - tcgetattr(0, &old); -Binary files sysvinit-2.85/src/sulogin.o and sysvinit-2.86/src/sulogin.o differ -Binary files sysvinit-2.85/src/utmp.o and sysvinit-2.86/src/utmp.o differ |