7f614fca11ca19b47e19076cbc603d3049fe4dfc
[oweals/busybox.git] / loginutils / vlock.c
1 /* vi: set sw=4 ts=4: */
2
3 /*
4  * vlock implementation for busybox
5  *
6  * Copyright (C) 2000 by spoon <spoon@ix.netcom.com>
7  * Written by spoon <spon@ix.netcom.com>
8  *
9  * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
10  */
11
12 /* Shoutz to Michael K. Johnson <johnsonm@redhat.com>, author of the
13  * original vlock.  I snagged a bunch of his code to write this
14  * minimalistic vlock.
15  */
16 /* Fixed by Erik Andersen to do passwords the tinylogin way...
17  * It now works with md5, sha1, etc passwords. */
18
19 #include "libbb.h"
20 #include <sys/vt.h>
21
22 static struct passwd *pw;
23 static struct vt_mode ovtm;
24 static struct termios oterm;
25 static int vfd;
26 static unsigned long o_lock_all;
27
28 static void release_vt(int signo)
29 {
30         ioctl(vfd, VT_RELDISP, !o_lock_all);
31 }
32
33 static void acquire_vt(int signo)
34 {
35         ioctl(vfd, VT_RELDISP, VT_ACKACQ);
36 }
37
38 static void restore_terminal(void)
39 {
40         ioctl(vfd, VT_SETMODE, &ovtm);
41         tcsetattr(STDIN_FILENO, TCSANOW, &oterm);
42 }
43
44 int vlock_main(int argc, char **argv);
45 int vlock_main(int argc, char **argv)
46 {
47         sigset_t sig;
48         struct sigaction sa;
49         struct vt_mode vtm;
50         struct termios term;
51         uid_t uid = getuid();
52
53         pw = getpwuid(uid);
54         if (pw == NULL)
55                 bb_error_msg_and_die("unknown uid %d", uid);
56
57         if (argc > 2) {
58                 bb_show_usage();
59         }
60
61         o_lock_all = getopt32(argc, argv, "a");
62
63         vfd = xopen(CURRENT_TTY, O_RDWR);
64
65         if (ioctl(vfd, VT_GETMODE, &vtm) < 0) {
66                 bb_perror_msg_and_die("VT_GETMODE");
67         }
68
69         /* mask a bunch of signals */
70         sigprocmask(SIG_SETMASK, NULL, &sig);
71         sigdelset(&sig, SIGUSR1);
72         sigdelset(&sig, SIGUSR2);
73         sigaddset(&sig, SIGTSTP);
74         sigaddset(&sig, SIGTTIN);
75         sigaddset(&sig, SIGTTOU);
76         sigaddset(&sig, SIGHUP);
77         sigaddset(&sig, SIGCHLD);
78         sigaddset(&sig, SIGQUIT);
79         sigaddset(&sig, SIGINT);
80
81         sigemptyset(&(sa.sa_mask));
82         sa.sa_flags = SA_RESTART;
83         sa.sa_handler = release_vt;
84         sigaction(SIGUSR1, &sa, NULL);
85         sa.sa_handler = acquire_vt;
86         sigaction(SIGUSR2, &sa, NULL);
87
88         /* need to handle some signals so that we don't get killed by them */
89         sa.sa_handler = SIG_IGN;
90         sigaction(SIGHUP, &sa, NULL);
91         sigaction(SIGQUIT, &sa, NULL);
92         sigaction(SIGINT, &sa, NULL);
93         sigaction(SIGTSTP, &sa, NULL);
94
95         ovtm = vtm;
96         vtm.mode = VT_PROCESS;
97         vtm.relsig = SIGUSR1;
98         vtm.acqsig = SIGUSR2;
99         ioctl(vfd, VT_SETMODE, &vtm);
100
101         tcgetattr(STDIN_FILENO, &oterm);
102         term = oterm;
103         term.c_iflag &= ~BRKINT;
104         term.c_iflag |= IGNBRK;
105         term.c_lflag &= ~ISIG;
106         term.c_lflag &= ~(ECHO | ECHOCTL);
107         tcsetattr(STDIN_FILENO, TCSANOW, &term);
108
109         do {
110                 printf("Virtual Console%s locked by %s.\n", (o_lock_all) ? "s" : "", pw->pw_name);
111                 if (correct_password(pw)) {
112                         break;
113                 }
114                 bb_do_delay(FAIL_DELAY);
115                 puts("Password incorrect");
116         } while (1);
117         restore_terminal();
118         fflush_stdout_and_exit(0);
119 }