[PATCH 7/7] Further mingetty features

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Allow the system adminstrator to provide options to the login
program.   Some changes for several layouts of the agetty
prompt like short host name or full qualified host name.
Four options enables the user to delay agetty after start,
to change the working directory, to change the root directory,
and to modified scheduling priority.

Signed-off-by: Werner Fink <werner@xxxxxxx>
---
 term-utils/agetty.8 |   78 ++++++++++++++++++++++---------------
 term-utils/agetty.c |  106 +++++++++++++++++++++++++++++++++++++++++++++++---
 2 files changed, 146 insertions(+), 38 deletions(-)

diff --git a/term-utils/agetty.8 b/term-utils/agetty.8
index f998699..dc49193 100644
--- a/term-utils/agetty.8
+++ b/term-utils/agetty.8
@@ -4,12 +4,12 @@ agetty \- alternative Linux getty
 
 .SH SYNOPSIS
 .BR "agetty " [\-8chiLmnsUw]
-.RI "[-a " user ]
-.RI "[-f " issue_file ]
-.RI "[-H " login_host ]
-.RI "[-I " init ]
-.RI "[-l " login_program ]
-.RI "[-t " timeout ]
+.RI "[\-a " user ]
+.RI "[\-f " issue_file ]
+.RI "[\-H " login_host ]
+.RI "[\-I " init ]
+.RI "[\-l " login_program ]
+.RI "[\-t " timeout ]
 .I port
 .I baud_rate,...
 .RI [ term ]
@@ -58,13 +58,13 @@ This program does not use the \fI/etc/gettydefs\fP (System V) or
 .ad
 .TP
 port
-A path name relative to the \fI/dev\fP directory. If a "-" is
+A path name relative to the \fI/dev\fP directory. If a "\-" is
 specified, \fBagetty\fP assumes that its standard input is
 already connected to a tty port and that a connection to a
 remote user has already been established.
 .sp
-Under System V, a "-" \fIport\fP argument should be preceded
-by a "--".
+Under System V, a "\-" \fIport\fP argument should be preceded
+by a "\-\-".
 .TP
 baud_rate,...
 A comma-separated list of one or more baud rates. Each time
@@ -72,7 +72,7 @@ A comma-separated list of one or more baud rates. Each time
 the list, which is treated as if it were circular.
 .sp
 Baud rates should be specified in descending order, so that the
-null character (Ctrl-@) can also be used for baud rate switching.
+null character (Ctrl\-@) can also be used for baud rate switching.
 .TP
 term
 The value to be used for the TERM environment variable. This overrides
@@ -88,19 +88,19 @@ Assume that the tty is 8-bit clean, hence disable parity detection.
 .TP
 \-a, \-\-autologin \fIusername\fP
 Log the specified user automatically in without asking for a login
-name and password. Check the -f option from
+name and password. Check the \-f option from
 \fB/bin/login\fP for this.
 .TP
 \-c, \-\-noreset
 Don't reset terminal cflags (control modes). See \fItermios(3)\fP for more
 details.
 .TP
-\-f, \-\-issue-file \fIissue_file\fP
+\-f, \-\-issue\-file \fIissue_file\fP
 Display the contents of \fIissue_file\fP instead of \fI/etc/issue\fP.
 This allows custom messages to be displayed on different terminals.
 The \-i option will override this option.
 .TP
-\-h, \-\-flow-control
+\-h, \-\-flow\-control
 Enable hardware (RTS/CTS) flow control. It is left up to the
 application to disable software (XON/XOFF) flow protocol where
 appropriate.
@@ -117,7 +117,7 @@ login prompt. Terminals or communications hardware may become confused
 when receiving lots of text at the wrong baud rate; dial-up scripts
 may fail if the login prompt is preceded by too much text.
 .TP
-\-I, \-\-init-string \fIinitstring\fP
+\-I, \-\-init\-string \fIinitstring\fP
 Set an initial string to be sent to the tty or modem before sending
 anything else. This may be used to initialize a modem.  Non printable
 characters may be sent by writing their octal code preceded by a
@@ -148,7 +148,7 @@ Since the \fI\-m\fP feature may fail on heavily-loaded systems,
 you still should enable BREAK processing by enumerating all
 expected baud rates on the command line.
 .TP 
-\-n, \-\-skip-login
+\-n, \-\-skip\-login
 Do not prompt the user for a login name. This can be used in
 connection with \-l option to invoke a non-standard login process such
 as a BBS system. Note that with the \-n option, \fBagetty\fR gets no input from
@@ -158,7 +158,7 @@ space parity, 7 bit characters, and ASCII CR (13) end-of-line character.
 Beware that the program that \fBagetty\fR starts (usually /bin/login)
 is run as root.
 .TP
-\-o, \-\-logopts \fI"login_options"\fP
+\-o, \-\-login\-options \fI"login_options"\fP
 Options  that  are passed to the login program.  \\u is replaced
 by the login name. Defaults to "-- \\u", which is suitable for
 \fB/bin/login\fP.  Please read the SECURITY NOTICE below if
@@ -171,7 +171,7 @@ with \fB\-\-autologin\fP to save memory by lazily spawning shells.
 \-R, \-\-hangup
 Do call vhangup() for a virtually hangup of the specified terminal.
 .TP
-\-s, \-\-keep-baud
+\-s, \-\-keep\-baud
 Try to keep the existing baud rate. The baud rates from
 the command line are used when agetty receives a BREAK character.
 .TP
@@ -180,17 +180,33 @@ Terminate if no user name could be read within \fItimeout\fP
 seconds. This option should probably not be used with hard-wired
 lines.
 .TP
-\-U, \-\-detect-case
+\-U, \-\-detect\-case
 Turn on support for detecting an uppercase only terminal.  This setting will
 detect a login name containing only capitals as indicating an uppercase
 only terminal and turn on some upper to lower case conversions.  Note that
 this has no support for any unicode characters.
 .TP
-\-w, \-\-wait-cr
+\-w, \-\-wait\-cr
 Wait for the user or the modem to send a carriage-return or a
 linefeed character before sending the \fI/etc/issue\fP (or other) file
 and the login prompt. Very useful in connection with the \-I option.
 .TP
+\-\-noclear
+Do not clear the screen before prompting for the login name
+(the screen is normally cleared).
+.TP
+\-\-nonewline
+Do not print a newline before writing out /etc/issue.
+.TP
+\-\-no\-hostname
+By default the hostname will be printed.  With this option enabled,
+no hostname at all will be shown.
+.TP
+\-\-long\-hostname
+By default the hostname is only printed until the first dot.  With
+this option enabled, the full qualified hostname by gethostname()
+or if not found by gethostbyname() is shown.
+.TP
 \-\-version
 Output version information and exit.
 .TP
@@ -227,14 +243,14 @@ dis-connection and turn on auto-answer after 1 ring.)
 If you use the \fB\-\-login\fP and \fB\-\-logopts\fP options, be aware
 that a malicious user may try to enter lognames with embedded options,
 which then get passed to the used login  program.  Agetty does check
-for a leading - and makes sure the logname gets passed as one parameter
+for a leading "\-" and makes sure the logname gets passed as one parameter
 (so embedded spaces will not create yet another parameter), but depending
 on how the login binary parses the command line that might not be sufficient.
 Check that the used login program  can  not  be  abused this way.
 .PP
-Some  programs use -- to indicate that the rest of the commandline should
-not be interpreted as options. Use this feature if available by passing -- before
-the username gets passed by \\u.
+Some  programs use "\-\-" to indicate that the rest of the commandline should
+not be interpreted as options. Use this feature if available by passing "\-\-"
+before the username gets passed by \\u.
 
 .SH ISSUE ESCAPES
 The issue-file (\fI/etc/issue\fP or the file set with the \-f option)
@@ -250,25 +266,25 @@ d
 Insert the current date.
 .TP
 s
-Insert the system name, the name of the operating system. Same as `uname -s'.
+Insert the system name, the name of the operating system. Same as `uname \-s'.
 .TP
 l
 Insert the name of the current tty line.
 .TP
 m
-Insert the architecture identifier of the machine. Same as `uname -m'.
+Insert the architecture identifier of the machine. Same as `uname \-m'.
 .TP
 n
-Insert the nodename of the machine, also known as the hostname. Same as `uname -n'.
+Insert the nodename of the machine, also known as the hostname. Same as `uname \-n'.
 .TP
 o
-Insert the NIS domainname of the machine. Same as `hostname -d'.
+Insert the NIS domainname of the machine. Same as `hostname \-d'.
 .TP
 O
 Insert the DNS domainname of the machine.
 .TP
 r
-Insert the release number of the OS. Same as `uname -r'.
+Insert the release number of the OS. Same as `uname \-r'.
 .TP
 t
 Insert the current time.
@@ -307,7 +323,7 @@ This is thingol.orcan.dk (Linux i386 1.1.9) 18:29:30
 .SH BUGS
 .ad
 .fi
-The baud-rate detection feature (the \fI-m\fP option) requires that
+The baud-rate detection feature (the \fI\-m\fP option) requires that
 \fBagetty\fP be scheduled soon enough after completion of a dial-in
 call (within 30 ms with modems that talk at 2400 baud). For robustness,
 always use the \fI\-m\fP option in combination with a multiple baud
@@ -316,7 +332,7 @@ rate command-line argument, so that BREAK processing is enabled.
 The text in the \fI/etc/issue\fP file (or other) and the login prompt
 are always output with 7-bit characters and space parity.
 
-The baud-rate detection feature (the \fI-m\fP option) requires that
+The baud-rate detection feature (the \fI\-m\fP option) requires that
 the modem emits its status message \fIafter\fP raising the DCD line.
 .SH DIAGNOSTICS
 .ad
@@ -342,4 +358,4 @@ Added \-f option to display custom login messages on different terminals.
 
 .SH AVAILABILITY
 The agetty command is part of the util-linux package and is available from
-ftp://ftp.kernel.org/pub/linux/utils/util-linux/.
+ftp://ftp.kernel.org/pub/linux/utils/util\-linux/.
diff --git a/term-utils/agetty.c b/term-utils/agetty.c
index c49ad4a..1e47d55 100644
--- a/term-utils/agetty.c
+++ b/term-utils/agetty.c
@@ -93,7 +93,7 @@
 #endif
 
 /* Login prompt. */
-#define LOGIN		" login: "
+#define LOGIN		"login: "
 #define ARRAY_SIZE_MAX	16		/* Numbers of args for login beside "-- \\u" */
 
 /* Some shorthands for control characters. */
@@ -133,6 +133,8 @@ struct options {
 	int flags;			/* toggle switches, see below */
 	int timeout;			/* time-out period */
 	char *autolog;			/* login the user automatically */
+	char *chdir;			/* Chdir before the login */
+	char *chroot;			/* Chroot before the login */
 	char *login;			/* login program */
 	char *logopt;			/* options for login program */
 	char *tty;			/* name of tty */
@@ -140,6 +142,8 @@ struct options {
 	char *term;			/* terminal type */
 	char *initstring;		/* modem init string */
 	char *issue;			/* alternative issue file */
+	int delay;			/* Sleep seconds before prompt */
+	int nice;			/* Run login with this priority */
 	int numspeed;			/* number of baud rates to try */
 	speed_t speeds[MAX_SPEED];	/* baud rates to be tried */
 };
@@ -160,6 +164,10 @@ struct options {
 #define F_HANGUP	(1<<13)	/* Do call vhangup(2) */
 #define F_UTF8		(1<<14)	/* We can do UTF8 */
 #define F_LOGINPAUSE	(1<<15)	/* Wait for any key before dropping login prompt */
+#define F_NOCLEAR	(1<<16) /* Do not clear the screen before prompting */
+#define F_NONL		(1<<17) /* No newline before issue */
+#define F_NOHOSTNAME	(1<<18) /* Do not show the hostname */
+#define F_LONGHNAME	(1<<19) /* Show Full qualified hostname */
 
 #define serial_tty_option(opt, flag)	\
 	(((opt)->flags & (F_VCONSOLE|(flag))) == (flag))
@@ -303,6 +311,9 @@ int main(int argc, char **argv)
 	update_utmp(&options);
 #endif
 
+	if (options.delay)
+	    sleep(options.delay);
+
 	debug("calling open_tty\n");
 
 	/* Open the tty as standard { input, output, error }. */
@@ -399,6 +410,22 @@ int main(int argc, char **argv)
 	mkarray(logarr, logcmd);
 	replacename(logarr, logname);
 
+	if (options.chroot) {
+		if (chroot(options.chroot) < 0)
+			log_err(_("%s: can't change root directory %s: %m"),
+				options.tty, options.chroot);
+	}
+	if (options.chdir) {
+		if (chdir(options.chdir) < 0)
+			log_err(_("%s: can't change working directory %s: %m"),
+				options.tty, options.chdir);
+	}
+	if (options.nice) {
+		if (nice(options.nice) < 0)
+			log_warn(_("%s: can't change process priority: %m"),
+				options.tty);
+	}
+
 	/* Let the login program take care of password validation. */
 	execv(options.login, logarr);
 	log_err(_("%s: can't exec %s: %m"), options.tty, options.login);
@@ -413,36 +440,49 @@ static void parse_args(int argc, char **argv, struct options *op)
 
 	enum {
 		VERSION_OPTION = CHAR_MAX + 1,
+		NOHOSTNAME_OPTION,
+		LONGHOSTNAME_OPTION,
 		HELP_OPTION
 	};
 	const struct option longopts[] = {
 		{  "8bits",	     no_argument,	 0,  '8'  },
 		{  "autologin",	     required_argument,	 0,  'a'  },
 		{  "noreset",	     no_argument,	 0,  'c'  },
+		{  "chdir",	     required_argument,	 0,  'C'  },
+		{  "delay",	     required_argument,	 0,  'd'  },
 		{  "issue-file",     required_argument,  0,  'f'  },
 		{  "flow-control",   no_argument,	 0,  'h'  },
 		{  "host",	     required_argument,  0,  'H'  },
 		{  "noissue",	     no_argument,	 0,  'i'  },
 		{  "init-string",    required_argument,  0,  'I'  },
+		{  "noclear",	     no_argument,	 0,  'J'  },
 		{  "login-program",  required_argument,  0,  'l'  },
+		{  "login",	     required_argument,  0,  'l'  },  /* compat option */
+		{  "loginprog",	     required_argument,  0,  'l'  },  /* compat option */
 		{  "local-line",     no_argument,	 0,  'L'  },
 		{  "extract-baud",   no_argument,	 0,  'm'  },
 		{  "skip-login",     no_argument,	 0,  'n'  },
+		{  "nonewline",	     no_argument,	 0,  'N'  },
 		{  "login-options",  required_argument,  0,  'o'  },
 		{  "loginopts",	     required_argument,  0,  'o'  },  /* compat option */
 		{  "logopts",	     required_argument,  0,  'o'  },  /* compat option */
 		{  "loginpause",     no_argument,        0,  'p'  },
+		{  "nice",	     required_argument,  0,  'P'  },
+		{  "chroot",	     required_argument,	 0,  'r'  },
 		{  "hangup",	     no_argument,	 0,  'R'  },
 		{  "keep-baud",      no_argument,	 0,  's'  },
 		{  "timeout",	     required_argument,  0,  't'  },
 		{  "detect-case",    no_argument,	 0,  'U'  },
 		{  "wait-cr",	     no_argument,	 0,  'w'  },
+		{  "no-hostname",    no_argument,	 0,  NOHOSTNAME_OPTION },
+		{  "nohostname",     no_argument,	 0,  NOHOSTNAME_OPTION },  /* compat option */
+		{  "long-hostname",  no_argument,	 0,  LONGHOSTNAME_OPTION },
 		{  "version",	     no_argument,	 0,  VERSION_OPTION  },
 		{  "help",	     no_argument,	 0,  HELP_OPTION     },
 		{ NULL, 0, 0, 0 }
 	};
 
-	while ((c = getopt_long(argc, argv, "8a:cf:hH:iI:l:Lmno:pRst:Uw", longopts,
+	while ((c = getopt_long(argc, argv, "8a:cC:d:f:hH:iI:Jl:LmnNo:pP:r:Rst:Uw", longopts,
 			    NULL)) != -1) {
 		switch (c) {
 		case '8':
@@ -454,6 +494,12 @@ static void parse_args(int argc, char **argv, struct options *op)
 		case 'c':
 			op->flags |= F_KEEPCFLAGS;
 			break;
+		case 'C':
+			op->chdir = optarg;
+			break;
+		case 'd':
+			op->delay = atoi(optarg);
+			break;
 		case 'f':
 			op->flags |= F_CUSTISSUE;
 			op->issue = optarg;
@@ -471,6 +517,9 @@ static void parse_args(int argc, char **argv, struct options *op)
 			init_special_char(optarg, op);
 			op->flags |= F_INITSTRING;
 			break;
+		case 'J':
+			op->flags |= F_NOCLEAR;
+			break;
 		case 'l':
 			op->login = optarg;
 			break;
@@ -489,6 +538,12 @@ static void parse_args(int argc, char **argv, struct options *op)
 		case 'p':
 			op->flags |= F_LOGINPAUSE;
 			break;
+		case 'P':
+			op->nice = atoi(optarg);
+			break;
+		case 'r':
+			op->chroot = optarg;
+			break;
 		case 'R':
 			op->flags |= F_HANGUP;
 			break;
@@ -505,6 +560,12 @@ static void parse_args(int argc, char **argv, struct options *op)
 		case 'w':
 			op->flags |= F_WAITCRLF;
 			break;
+		case NOHOSTNAME_OPTION:
+			op->flags |= F_NOHOSTNAME;
+			break;
+		case LONGHOSTNAME_OPTION:
+			op->flags |= F_LONGHNAME;
+			break;
 		case VERSION_OPTION:
 			printf(_("%s from %s\n"), program_invocation_short_name,
 			       PACKAGE_STRING);
@@ -868,6 +929,18 @@ static void termio_init(struct options *op, struct termios *tp)
 		if ((tp->c_cflag & (CS8|PARODD|PARENB)) == CS8)
 			op->flags |= F_EIGHTBITS;
 
+		if ((op->flags & F_NOCLEAR) == 0) {
+			/*
+			 * Do not write a full reset (ESC c) because this destroys
+			 * the unicode mode again if the terminal was in unicode
+			 * mode.  Also it clears the CONSOLE_MAGIC features which
+			 * are required for some languages/console-fonts.
+			 * Just put the cursor to the home position (ESC [ H),
+			 * erase everything below the cursor (ESC [ J), and set the
+			 * scrolling region to the full window (ESC [ r)
+			 */
+			write_all(STDOUT_FILENO, "\033[r\033[H\033[J", 9);
+		}
 		return;
 	}
 
@@ -1074,8 +1147,10 @@ static void do_prompt(struct options *op, struct termios *tp)
 	FILE *fd;
 #endif				/* ISSUE */
 
-	/* Issue not in use, start with a new line. */
-	write_all(STDOUT_FILENO, "\r\n", 2);
+	if ((op->flags & F_NONL) == 0) {
+		/* Issue not in use, start with a new line. */
+		write_all(STDOUT_FILENO, "\r\n", 2);
+	}
 
 #ifdef	ISSUE
 	if ((op->flags & F_ISSUE) && (fd = fopen(op->issue, "r"))) {
@@ -1161,10 +1236,23 @@ static void do_prompt(struct options *op, struct termios *tp)
 		}
 	}
 #endif /* KDGKBLED */
-	{
+	if ((op->flags & F_NOHOSTNAME) == 0) {
 		char hn[MAXHOSTNAMELEN + 1];
-		if (gethostname(hn, sizeof(hn)) == 0)
-			write_all(STDOUT_FILENO, hn, strlen(hn));
+		if (gethostname(hn, sizeof(hn)) == 0) {
+			struct hostent *ht;
+			char *dot = strchr(hn, '.');
+
+			hn[MAXHOSTNAMELEN] = '\0';
+			if ((op->flags & F_LONGHNAME) == 0) {
+				if (dot)
+					*dot = '\0';
+				write_all(STDOUT_FILENO, hn, strlen(hn));
+			} else if (dot == NULL && (ht = gethostbyname(hn)))
+				write_all(STDOUT_FILENO, ht->h_name, strlen(ht->h_name));
+			else
+				write_all(STDOUT_FILENO, hn, strlen(hn));
+			write_all(STDOUT_FILENO, " ", 1);
+		}
 	}
 	if (op->autolog == (char*)0) {
 		/* Always show login prompt. */
@@ -1474,6 +1562,10 @@ static void __attribute__ ((__noreturn__)) usage(FILE * out)
 		       " -t, --timeout NUMBER       login process timeout\n"
 		       " -U, --detect-case          detect uppercase terminal\n"
 		       " -w, --wait-cr              wait carriage-return\n"
+		       "     --noclear              do not clear the screen before prompt\n"
+		       "     --nonewline            do not print a newline before issue\n"
+		       "     --no-hostname          no hostname at all will be shown\n"
+		       "     --long-hostname        show full qualified hostname\n"
 		       "     --version              output version information and exit\n"
 		       "     --help                 display this help and exit\n\n"));
 
-- 
1.6.0.2

--
To unsubscribe from this list: send the line "unsubscribe util-linux" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Netdev]     [Ethernet Bridging]     [Linux Wireless]     [Kernel Newbies]     [Security]     [Linux for Hams]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux RAID]     [Linux Admin]     [Samba]

  Powered by Linux