/* getty.c - A getty program to get controlling terminal. * * Copyright 2012 Sandeep Sharma <sandeep.jack2756@gamil.com> * Copyright 2013 Kyungwan Han <asura321@gmail.com> * * No Standard. USE_GETTY(NEWTOY(getty, "<2t#<0H:I:l:f:iwnmLh",TOYFLAG_SBIN)) config GETTY bool "getty" default n help usage: getty [OPTIONS] BAUD_RATE[,BAUD_RATE]... TTY [TERMTYPE] -h Enable hardware RTS/CTS flow control -L Set CLOCAL (ignore Carrier Detect state) -m Get baud rate from modem's CONNECT status message -n Don't prompt for login name -w Wait for CR or LF before sending /etc/issue -i Don't display /etc/issue -f ISSUE_FILE Display ISSUE_FILE instead of /etc/issue -l LOGIN Invoke LOGIN instead of /bin/login -t SEC Terminate after SEC if no login name is read -I INITSTR Send INITSTR before anything else -H HOST Log HOST into the utmp file as the hostname */ #define FOR_getty #include "toys.h" #include <utmp.h> GLOBALS( char *issue_str; char *login_str; char *init_str; char *host_str; long timeout; char *tty_name; int speeds[20]; int sc; struct termios termios; char buff[128]; ) #define CTL(x) ((x) ^ 0100) #define HOSTNAME_SIZE 32 typedef void (*sighandler_t)(int); struct speed_mapper { long speed; speed_t code; }; struct speed_mapper speedtab[] = { {50, B50}, {75, B75}, {110, B110}, {134, B134}, {150, B150}, {200, B200}, {300, B300}, {600, B600}, {1200, B1200}, {1800, B1800}, {2400, B2400}, {4800, B4800}, {9600, B9600}, #ifdef B19200 {19200, B19200}, #endif #ifdef B38400 {38400, B38400}, #endif #ifdef EXTA {19200, EXTA}, #endif #ifdef EXTB {38400, B38400}, #endif #ifdef B57600 {57600, B57600}, #endif #ifdef B115200 {115200, B115200}, #endif #ifdef B230400 {230400, B230400}, #endif {0, 0}, }; // Find speed from mapper array static speed_t encode(char *s) { struct speed_mapper *sp; long speed = atolx(s); if (!speed) return 0; for (sp = speedtab; sp->speed; sp++) if (sp->speed == speed) return sp->code; return (speed_t) -1; } static void get_speed(char *sp) { char *ptr; TT.sc = 0; while ((ptr = strsep(&sp, ","))) { TT.speeds[TT.sc] = encode(ptr); if (TT.speeds[TT.sc] < 0) perror_exit("bad speed"); if (++TT.sc > 10) perror_exit("too many speeds, max is 10"); } } // Parse args and set TERM env. variable static void parse_arguments(void) { if (isdigit(**toys.optargs)) { get_speed(*toys.optargs); if (*++toys.optargs) TT.tty_name = xmprintf("%s", *toys.optargs); } else { TT.tty_name = xmprintf("%s", *toys.optargs); if (*++toys.optargs) get_speed(*toys.optargs); } if (*++toys.optargs) setenv("TERM", *toys.optargs, 1); } // Get controlling terminal and redirect stdio static void open_tty(void) { if (strcmp(TT.tty_name, "-")) { if (*(TT.tty_name) != '/') TT.tty_name = xmprintf("/dev/%s", TT.tty_name); // Sends SIGHUP to all foreground process if Session leader don't die,Ignore sighandler_t sig = signal(SIGHUP, SIG_IGN); ioctl(0, TIOCNOTTY, 0); // Giveup if there is any controlling terminal signal(SIGHUP, sig); if ((setsid() < 0) && (getpid() != getsid(0))) perror_exit("setsid"); xclose(0); xopen_stdio(TT.tty_name, O_RDWR|O_NDELAY|O_CLOEXEC); fcntl(0, F_SETFL, fcntl(0, F_GETFL) & ~O_NONBLOCK); // Block read dup2(0, 1); dup2(0, 2); if (ioctl(0, TIOCSCTTY, 1) < 0) perror_msg("ioctl(TIOCSCTTY)"); if (!isatty(0)) perror_exit("/dev/%s: not a tty", TT.tty_name); chown(TT.tty_name, 0, 0); // change ownership, Hope login will change this chmod(TT.tty_name, 0620); } else { // We already have opened TTY if (setsid() < 0) perror_msg("setsid failed"); if ((fcntl(0, F_GETFL) & (O_RDWR|O_RDONLY|O_WRONLY)) != O_RDWR) perror_exit("no read/write permission"); } } // Intialise terminal settings static void termios_init(void) { if (tcgetattr(STDIN_FILENO, &TT.termios) < 0) perror_exit("tcgetattr"); // Flush input and output queues, important for modems! tcflush(STDIN_FILENO, TCIOFLUSH); TT.termios.c_cflag &= (0|CSTOPB|PARENB|PARODD); #ifdef CRTSCTS if (toys.optflags & FLAG_h) TT.termios.c_cflag |= CRTSCTS; #endif if (toys.optflags & FLAG_L) TT.termios.c_cflag |= CLOCAL; TT.termios.c_cc[VTIME] = 0; TT.termios.c_cc[VMIN] = 1; TT.termios.c_oflag = OPOST|ONLCR; TT.termios.c_cflag |= CS8|CREAD|HUPCL|CBAUDEX; // login will disable echo for passwd. TT.termios.c_lflag |= ISIG|ICANON|ECHO|ECHOE|ECHOK|ECHOKE; TT.termios.c_cc[VINTR] = CTL('C'); TT.termios.c_cc[VQUIT] = CTL('\\'); TT.termios.c_cc[VEOF] = CTL('D'); TT.termios.c_cc[VEOL] = '\n'; TT.termios.c_cc[VKILL] = CTL('U'); TT.termios.c_cc[VERASE] = 127; // CERASE TT.termios.c_iflag = ICRNL|IXON|IXOFF; // set non-zero baud rate. Zero baud rate left it unchanged. if (TT.speeds[0] != B0) cfsetspeed(&TT.termios, TT.speeds[0]); if (tcsetattr(STDIN_FILENO, TCSANOW, &TT.termios) < 0) perror_exit("tcsetattr"); } // Get the baud rate from modems CONNECT mesage, Its of form <junk><BAUD><Junk> static void sense_baud(void) { int vmin; ssize_t size; char *ptr; speed_t speed; vmin = TT.termios.c_cc[VMIN]; // Store old TT.termios.c_cc[VMIN] = 0; // No block even queue is empty. if (tcsetattr(STDIN_FILENO, TCSANOW, &TT.termios) < 0) perror_exit("tcsetattr"); size = readall(STDIN_FILENO, TT.buff, sizeof(TT.buff)-1); if (size > 0) { for (ptr = TT.buff; ptr < TT.buff+size; ptr++) { if (isdigit(*ptr)) { speed = encode(ptr); if (speed > 0) cfsetspeed(&TT.termios,speed); break; } } } TT.termios.c_cc[VMIN] = vmin; //restore old value if (tcsetattr(STDIN_FILENO, TCSANOW, &TT.termios) < 0) perror_exit("tcsetattr"); } // Just prompt for login name void print_prompt(void) { char *hostname; struct utsname uts; uname(&uts); hostname = xstrdup(uts.nodename); fputs(hostname, stdout); fputs(" login: ", stdout); fflush(NULL); free(hostname); hostname = NULL; } // Print /etc/isuue with taking care of each escape sequence void write_issue(char *file) { char buff[20] = {0,}; struct utsname u; uname(&u); int size, fd = open(TT.issue_str, O_RDONLY); if (fd < 0) return; while ((size = readall(fd, buff, 1)) > 0) { char *ch = buff; if (*ch == '\\' || *ch == '%') { if (readall(fd, buff, 1) <= 0) perror_exit("readall"); if (*ch == 's') fputs(u.sysname, stdout); if (*ch == 'n'|| *ch == 'h') fputs(u.nodename, stdout); if (*ch == 'r') fputs(u.release, stdout); if (*ch == 'm') fputs(u.machine, stdout); if (*ch == 'l') fputs(TT.tty_name, stdout); } else xputc(*ch); } } // Read login name and print prompt and Issue file. static int read_login_name(void) { tcflush(STDIN_FILENO, TCIFLUSH); // Flush pending speed switches int i = 0; while (1) { // Option -i will overide -f if (!(toys.optflags & FLAG_i)) write_issue(TT.issue_str); print_prompt(); TT.buff[0] = getchar(); if (!TT.buff[0] && TT.sc > 1) return 0; // Switch speed if (TT.buff[0] == '\n') continue; if (TT.buff[0] != '\n') if (!fgets(&TT.buff[1], HOSTNAME_SIZE-1, stdin)) _exit(1); while (i < HOSTNAME_SIZE-1 && isgraph(TT.buff[i])) i++; TT.buff[i] = 0; break; } return 1; } // Put hostname entry in utmp file static void utmp_entry(void) { struct utmp entry; struct utmp *utp_ptr; pid_t pid = getpid(); char *utmperr = "can't make utmp entry, host length greater than UT_HOSTSIZE(256)"; utmpname(_PATH_UTMP); setutent(); // Starts from start while ((utp_ptr = getutent())) if (utp_ptr->ut_pid == pid && utp_ptr->ut_type >= INIT_PROCESS) break; if (!utp_ptr) { entry.ut_type = LOGIN_PROCESS; entry.ut_pid = getpid(); xstrncpy(entry.ut_line, ttyname(STDIN_FILENO) + strlen("/dev/"), UT_LINESIZE); time((time_t *)&entry.ut_time); xstrncpy(entry.ut_user, "LOGIN", UT_NAMESIZE); if (strlen(TT.host_str) > UT_HOSTSIZE) perror_msg(utmperr); else xstrncpy(entry.ut_host, TT.host_str, UT_HOSTSIZE); setutent(); pututline(&entry); return; } xstrncpy(entry.ut_line, ttyname(STDIN_FILENO) + strlen("/dev/"), UT_LINESIZE); xstrncpy(entry.ut_user, "LOGIN", UT_NAMESIZE); if (strlen(TT.host_str) > UT_HOSTSIZE) perror_msg(utmperr); else xstrncpy(entry.ut_host, TT.host_str, UT_HOSTSIZE); time((time_t *)&entry.ut_time); setutent(); pututline(&entry); } void getty_main(void) { pid_t pid = getpid(); char *ptr[3] = {"/bin/login", NULL, NULL}; //2 NULLs so we can add username if (!(toys.optflags & FLAG_f)) TT.issue_str = "/etc/issue"; if (toys.optflags & FLAG_l) ptr[0] = TT.login_str; parse_arguments(); open_tty(); termios_init(); tcsetpgrp(STDIN_FILENO, pid); if (toys.optflags & FLAG_H) utmp_entry(); if (toys.optflags & FLAG_I) writeall(STDOUT_FILENO,TT.init_str,strlen(TT.init_str)); if (toys.optflags & FLAG_m) sense_baud(); if (toys.optflags & FLAG_t) alarm(TT.timeout); if (toys.optflags & FLAG_w) { char ch; while (readall(STDIN_FILENO, &ch, 1) != 1) if (ch == '\n' || ch == '\r') break; } if (!(toys.optflags & FLAG_n)) { int index = 1; // 0th we already set. while (1) { int l = read_login_name(); if (l) break; index = index % TT.sc; cfsetspeed(&TT.termios, TT.speeds[index]); // Select from multiple speeds //Necessary after cfsetspeed if (tcsetattr(STDIN_FILENO, TCSANOW, &TT.termios) < 0) perror_exit("tcsetattr"); } ptr[1]=TT.buff; //put the username in the login command line } xexec(ptr); }