bdb756659efe56a01299f4cf439cb8f20739d3d0
[oweals/busybox.git] / libbb / bb_askpass.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Ask for a password
4  * I use a static buffer in this function.  Plan accordingly.
5  *
6  * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
7  *
8  * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
9  */
10
11 #include "libbb.h"
12
13 /* do nothing signal handler */
14 static void askpass_timeout(int UNUSED_PARAM ignore)
15 {
16 }
17
18 char* FAST_FUNC bb_ask_stdin(const char *prompt)
19 {
20         return bb_ask(STDIN_FILENO, 0, prompt);
21 }
22 char* FAST_FUNC bb_ask(const int fd, int timeout, const char *prompt)
23 {
24         /* Was static char[BIGNUM] */
25         enum { sizeof_passwd = 128 };
26         static char *passwd;
27
28         char *ret;
29         int i;
30         struct sigaction sa, oldsa;
31         struct termios tio, oldtio;
32
33         tcgetattr(fd, &oldtio);
34         tcflush(fd, TCIFLUSH);
35         tio = oldtio;
36 #ifndef IUCLC
37 # define IUCLC 0
38 #endif
39         tio.c_iflag &= ~(IUCLC|IXON|IXOFF|IXANY);
40         tio.c_lflag &= ~(ECHO|ECHOE|ECHOK|ECHONL|TOSTOP);
41         tcsetattr_stdin_TCSANOW(&tio);
42
43         memset(&sa, 0, sizeof(sa));
44         /* sa.sa_flags = 0; - no SA_RESTART! */
45         /* SIGINT and SIGALRM will interrupt reads below */
46         sa.sa_handler = askpass_timeout;
47         sigaction(SIGINT, &sa, &oldsa);
48         if (timeout) {
49                 sigaction_set(SIGALRM, &sa);
50                 alarm(timeout);
51         }
52
53         fputs(prompt, stdout);
54         fflush_all();
55
56         if (!passwd)
57                 passwd = xmalloc(sizeof_passwd);
58         memset(passwd, 0, sizeof_passwd);
59         ret = passwd;
60         i = 0;
61         while (1) {
62                 int r = read(fd, &ret[i], 1);
63                 if (r < 0) {
64                         /* read is interrupted by timeout or ^C */
65                         ret = NULL;
66                         break;
67                 }
68                 if (r == 0 /* EOF */
69                  || ret[i] == '\r' || ret[i] == '\n' /* EOL */
70                  || ++i == sizeof_passwd-1 /* line limit */
71                 ) {
72                         ret[i] = '\0';
73                         break;
74                 }
75         }
76
77         if (timeout) {
78                 alarm(0);
79         }
80         sigaction_set(SIGINT, &oldsa);
81
82         tcsetattr_stdin_TCSANOW(&oldtio);
83         bb_putchar('\n');
84         fflush_all();
85         return ret;
86 }