| /* vi: set sw=4 ts=4: */ |
| /* |
| * vlock implementation for busybox |
| * |
| * Copyright (C) 2000 by spoon <spoon@ix.netcom.com> |
| * Written by spoon <spon@ix.netcom.com> |
| * |
| * 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. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| * |
| */ |
| |
| /* Shoutz to Michael K. Johnson <johnsonm@redhat.com>, author of the |
| * original vlock. I snagged a bunch of his code to write this |
| * minimalistic vlock. |
| */ |
| /* Fixed by Erik Andersen to do passwords the tinylogin way... |
| * It now works with md5, sha1, etc passwords. */ |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <sys/vt.h> |
| #include <signal.h> |
| #include <string.h> |
| #include <unistd.h> |
| #include <fcntl.h> |
| #include <errno.h> |
| #include <sys/ioctl.h> |
| #include <termios.h> |
| |
| #include "busybox.h" |
| |
| static struct passwd *pw; |
| static struct vt_mode ovtm; |
| static struct termios oterm; |
| static int vfd; |
| static int o_lock_all = 0; |
| |
| #ifdef CONFIG_FEATURE_SHADOWPASSWDS |
| static struct spwd *spw; |
| |
| /* getspuid - get a shadow entry by uid */ |
| struct spwd *getspuid(uid_t uid) |
| { |
| struct spwd *sp; |
| struct passwd *mypw; |
| |
| if ((mypw = getpwuid(getuid())) == NULL) { |
| return (NULL); |
| } |
| setspent(); |
| while ((sp = getspent()) != NULL) { |
| if (strcmp(mypw->pw_name, sp->sp_namp) == 0) |
| break; |
| } |
| endspent(); |
| return (sp); |
| } |
| #endif |
| |
| static void release_vt(int signo) |
| { |
| if (!o_lock_all) |
| ioctl(vfd, VT_RELDISP, 1); |
| else |
| ioctl(vfd, VT_RELDISP, 0); |
| } |
| |
| static void acquire_vt(int signo) |
| { |
| ioctl(vfd, VT_RELDISP, VT_ACKACQ); |
| } |
| |
| static void restore_terminal(void) |
| { |
| ioctl(vfd, VT_SETMODE, &ovtm); |
| tcsetattr(STDIN_FILENO, TCSANOW, &oterm); |
| } |
| |
| extern int vlock_main(int argc, char **argv) |
| { |
| sigset_t sig; |
| struct sigaction sa; |
| struct vt_mode vtm; |
| int times = 0; |
| struct termios term; |
| |
| if (argc > 2) { |
| bb_show_usage(); |
| } |
| |
| if (argc == 2) { |
| if (strncmp(argv[1], "-a", 2)) { |
| bb_show_usage(); |
| } else { |
| o_lock_all = 1; |
| } |
| } |
| |
| if ((pw = getpwuid(getuid())) == NULL) { |
| bb_error_msg_and_die("no password for uid %d\n", getuid()); |
| } |
| #ifdef CONFIG_FEATURE_SHADOWPASSWDS |
| if ((strcmp(pw->pw_passwd, "x") == 0) |
| || (strcmp(pw->pw_passwd, "*") == 0)) { |
| |
| if ((spw = getspuid(getuid())) == NULL) { |
| bb_error_msg_and_die("could not read shadow password for uid %d: %s\n", |
| getuid(), strerror(errno)); |
| } |
| if (spw->sp_pwdp) { |
| pw->pw_passwd = spw->sp_pwdp; |
| } |
| } |
| #endif /* CONFIG_FEATURE_SHADOWPASSWDS */ |
| if (pw->pw_passwd[0] == '!' || pw->pw_passwd[0] == '*') { |
| bb_error_msg_and_die("Account disabled for uid %d\n", getuid()); |
| } |
| |
| /* we no longer need root privs */ |
| setuid(getuid()); |
| setgid(getgid()); |
| |
| if ((vfd = open("/dev/tty", O_RDWR)) < 0) { |
| bb_error_msg_and_die("/dev/tty"); |
| }; |
| |
| if (ioctl(vfd, VT_GETMODE, &vtm) < 0) { |
| bb_error_msg_and_die("/dev/tty"); |
| }; |
| |
| /* mask a bunch of signals */ |
| sigprocmask(SIG_SETMASK, NULL, &sig); |
| sigdelset(&sig, SIGUSR1); |
| sigdelset(&sig, SIGUSR2); |
| sigaddset(&sig, SIGTSTP); |
| sigaddset(&sig, SIGTTIN); |
| sigaddset(&sig, SIGTTOU); |
| sigaddset(&sig, SIGHUP); |
| sigaddset(&sig, SIGCHLD); |
| sigaddset(&sig, SIGQUIT); |
| sigaddset(&sig, SIGINT); |
| |
| sigemptyset(&(sa.sa_mask)); |
| sa.sa_flags = SA_RESTART; |
| sa.sa_handler = release_vt; |
| sigaction(SIGUSR1, &sa, NULL); |
| sa.sa_handler = acquire_vt; |
| sigaction(SIGUSR2, &sa, NULL); |
| |
| /* need to handle some signals so that we don't get killed by them */ |
| sa.sa_handler = SIG_IGN; |
| sigaction(SIGHUP, &sa, NULL); |
| sigaction(SIGQUIT, &sa, NULL); |
| sigaction(SIGINT, &sa, NULL); |
| sigaction(SIGTSTP, &sa, NULL); |
| |
| ovtm = vtm; |
| vtm.mode = VT_PROCESS; |
| vtm.relsig = SIGUSR1; |
| vtm.acqsig = SIGUSR2; |
| ioctl(vfd, VT_SETMODE, &vtm); |
| |
| tcgetattr(STDIN_FILENO, &oterm); |
| term = oterm; |
| term.c_iflag &= ~BRKINT; |
| term.c_iflag |= IGNBRK; |
| term.c_lflag &= ~ISIG; |
| term.c_lflag &= ~(ECHO | ECHOCTL); |
| tcsetattr(STDIN_FILENO, TCSANOW, &term); |
| |
| do { |
| char *pass, *crypt_pass; |
| char prompt[100]; |
| |
| if (o_lock_all) { |
| printf("All Virtual Consoles locked.\n"); |
| } else { |
| printf("This Virtual Console locked.\n"); |
| } |
| fflush(stdout); |
| |
| snprintf(prompt, 100, "%s's password: ", pw->pw_name); |
| |
| if ((pass = getpass(prompt)) == NULL) { |
| perror("getpass"); |
| restore_terminal(); |
| exit(1); |
| } |
| |
| crypt_pass = pw_encrypt(pass, pw->pw_passwd); |
| if (strncmp(crypt_pass, pw->pw_passwd, sizeof(crypt_pass)) == 0) { |
| memset(pass, 0, strlen(pass)); |
| memset(crypt_pass, 0, strlen(crypt_pass)); |
| restore_terminal(); |
| return 0; |
| } |
| memset(pass, 0, strlen(pass)); |
| memset(crypt_pass, 0, strlen(crypt_pass)); |
| |
| if (isatty(STDIN_FILENO) == 0) { |
| perror("isatty"); |
| restore_terminal(); |
| exit(1); |
| } |
| |
| sleep(++times); |
| printf("Password incorrect.\n"); |
| if (times >= 3) { |
| sleep(15); |
| times = 2; |
| } |
| } while (1); |
| } |
| |
| /* |
| Local Variables: |
| c-file-style: "linux" |
| c-basic-offset: 4 |
| tab-width: 4 |
| End: |
| */ |