123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188 |
- /* vi: set sw=4 ts=4: */
- /*
- * Copyright (c) 2007 Denys Vlasenko <vda.linux@googlemail.com>
- *
- * Licensed under GPLv2, see file LICENSE in this source tree.
- */
- #include "libbb.h"
- //applet:IF_CTTYHACK(APPLET(cttyhack, BB_DIR_BIN, BB_SUID_DROP))
- //kbuild:lib-$(CONFIG_CTTYHACK) += cttyhack.o
- //config:config CTTYHACK
- //config: bool "cttyhack"
- //config: default y
- //config: help
- //config: One common problem reported on the mailing list is the "can't
- //config: access tty; job control turned off" error message, which typically
- //config: appears when one tries to use a shell with stdin/stdout on
- //config: /dev/console.
- //config: This device is special - it cannot be a controlling tty.
- //config:
- //config: The proper solution is to use the correct device instead of
- //config: /dev/console.
- //config:
- //config: cttyhack provides a "quick and dirty" solution to this problem.
- //config: It analyzes stdin with various ioctls, trying to determine whether
- //config: it is a /dev/ttyN or /dev/ttySN (virtual terminal or serial line).
- //config: On Linux it also checks sysfs for a pointer to the active console.
- //config: If cttyhack is able to find the real console device, it closes
- //config: stdin/out/err and reopens that device.
- //config: Then it executes the given program. Opening the device will make
- //config: that device a controlling tty. This may require cttyhack
- //config: to be a session leader.
- //config:
- //config: Example for /etc/inittab (for busybox init):
- //config:
- //config: ::respawn:/bin/cttyhack /bin/sh
- //config:
- //config: Starting an interactive shell from boot shell script:
- //config:
- //config: setsid cttyhack sh
- //config:
- //config: Giving controlling tty to shell running with PID 1:
- //config:
- //config: # exec cttyhack sh
- //config:
- //config: Without cttyhack, you need to know exact tty name,
- //config: and do something like this:
- //config:
- //config: # exec setsid sh -c 'exec sh </dev/tty1 >/dev/tty1 2>&1'
- //config:
- //config: Starting getty on a controlling tty from a shell script:
- //config:
- //config: # getty 115200 $(cttyhack)
- //usage:#define cttyhack_trivial_usage
- //usage: "[PROG ARGS]"
- //usage:#define cttyhack_full_usage "\n\n"
- //usage: "Give PROG a controlling tty if possible."
- //usage: "\nExample for /etc/inittab (for busybox init):"
- //usage: "\n ::respawn:/bin/cttyhack /bin/sh"
- //usage: "\nGiving controlling tty to shell running with PID 1:"
- //usage: "\n $ exec cttyhack sh"
- //usage: "\nStarting interactive shell from boot shell script:"
- //usage: "\n setsid cttyhack sh"
- #if !defined(__linux__) && !defined(TIOCGSERIAL) && !ENABLE_WERROR
- # warning cttyhack will not be able to detect a controlling tty on this system
- #endif
- /* From <linux/vt.h> */
- struct vt_stat {
- unsigned short v_active; /* active vt */
- unsigned short v_signal; /* signal to send */
- unsigned short v_state; /* vt bitmask */
- };
- enum { VT_GETSTATE = 0x5603 }; /* get global vt state info */
- /* From <linux/serial.h> */
- struct serial_struct {
- int type;
- int line;
- unsigned int port;
- int irq;
- int flags;
- int xmit_fifo_size;
- int custom_divisor;
- int baud_base;
- unsigned short close_delay;
- char io_type;
- char reserved_char[1];
- int hub6;
- unsigned short closing_wait; /* time to wait before closing */
- unsigned short closing_wait2; /* no longer used... */
- unsigned char *iomem_base;
- unsigned short iomem_reg_shift;
- unsigned int port_high;
- unsigned long iomap_base; /* cookie passed into ioremap */
- int reserved[1];
- };
- int cttyhack_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
- int cttyhack_main(int argc UNUSED_PARAM, char **argv)
- {
- int fd;
- char console[sizeof(int)*3 + 16];
- union {
- struct vt_stat vt;
- struct serial_struct sr;
- char paranoia[sizeof(struct serial_struct) * 3];
- } u;
- strcpy(console, "/dev/tty");
- fd = open(console, O_RDWR);
- if (fd < 0) {
- /* We don't have ctty (or don't have "/dev/tty" node...) */
- do {
- #ifdef __linux__
- /* Note that this method does not use _stdin_.
- * Thus, "cttyhack </dev/something" can't be used.
- * However, this method is more reliable than
- * TIOCGSERIAL check, which assumes that all
- * serial lines follow /dev/ttySn convention -
- * which is not always the case.
- * Therefore, we use this method first:
- */
- int s = open_read_close("/sys/class/tty/console/active",
- console + 5, sizeof(console) - 5);
- if (s > 0) {
- char *last;
- /* Found active console via sysfs (Linux 2.6.38+).
- * It looks like "[tty0 ]ttyS0\n" so zap the newline:
- */
- console[4 + s] = '\0';
- /* If there are multiple consoles,
- * take the last one:
- */
- last = strrchr(console + 5, ' ');
- if (last)
- overlapping_strcpy(console + 5, last + 1);
- break;
- }
- if (ioctl(0, VT_GETSTATE, &u.vt) == 0) {
- /* this is linux virtual tty */
- sprintf(console + 8, "S%u" + 1, (int)u.vt.v_active);
- break;
- }
- #endif
- #ifdef TIOCGSERIAL
- if (ioctl(0, TIOCGSERIAL, &u.sr) == 0) {
- /* this is a serial console; assuming it is named /dev/ttySn */
- sprintf(console + 8, "S%u", (int)u.sr.line);
- break;
- }
- #endif
- /* nope, could not find it */
- console[0] = '\0';
- } while (0);
- }
- argv++;
- if (!argv[0]) {
- if (!console[0])
- return EXIT_FAILURE;
- puts(console);
- return EXIT_SUCCESS;
- }
- if (fd < 0) {
- fd = open_or_warn(console, O_RDWR);
- if (fd < 0)
- goto ret;
- }
- //bb_error_msg("switching to '%s'", console);
- dup2(fd, 0);
- dup2(fd, 1);
- dup2(fd, 2);
- while (fd > 2)
- close(fd--);
- /* Some other session may have it as ctty,
- * try to steal it from them:
- */
- ioctl(0, TIOCSCTTY, 1);
- ret:
- BB_EXECVP_or_die(argv);
- }
|