blob: c2580b9ebe6914e3fe778914f20f32fad3f04ca7 [file] [log] [blame]
Eric Andersen6f9a7782004-05-01 01:27:30 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Ask for a password
Eric Andersen6f9a7782004-05-01 01:27:30 +00004 *
5 * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
6 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Eric Andersen6f9a7782004-05-01 01:27:30 +00008 */
9
Bernhard Reutner-Fischer20f40002006-01-30 17:17:14 +000010#include "libbb.h"
Eric Andersen6f9a7782004-05-01 01:27:30 +000011
12/* do nothing signal handler */
Denis Vlasenkoa60f84e2008-07-05 09:18:54 +000013static void askpass_timeout(int UNUSED_PARAM ignore)
Eric Andersen6f9a7782004-05-01 01:27:30 +000014{
15}
16
Bernhard Reutner-Fischer82b14292008-12-03 18:48:39 +000017char* FAST_FUNC bb_ask_stdin(const char *prompt)
18{
19 return bb_ask(STDIN_FILENO, 0, prompt);
20}
21char* FAST_FUNC bb_ask(const int fd, int timeout, const char *prompt)
Eric Andersen6f9a7782004-05-01 01:27:30 +000022{
Denis Vlasenko91e149a2007-06-18 10:35:06 +000023 /* Was static char[BIGNUM] */
24 enum { sizeof_passwd = 128 };
Denis Vlasenko6429aab2006-09-23 12:22:11 +000025
Denys Vlasenko02859aa2015-10-09 18:16:40 +020026 char *passwd;
Eric Andersen6f9a7782004-05-01 01:27:30 +000027 char *ret;
Denis Vlasenko6429aab2006-09-23 12:22:11 +000028 int i;
Denis Vlasenkoe5387a02007-10-20 19:20:22 +000029 struct sigaction sa, oldsa;
30 struct termios tio, oldtio;
Eric Andersen6f9a7782004-05-01 01:27:30 +000031
Denys Vlasenkod32fc642014-07-01 13:20:22 +020032 tcflush(fd, TCIFLUSH);
33 /* Was buggy: was printing prompt *before* flushing input,
34 * which was upsetting "expect" based scripts of some users.
35 */
Denys Vlasenko7449e182011-10-22 06:27:41 +020036 fputs(prompt, stdout);
37 fflush_all();
Denys Vlasenko7449e182011-10-22 06:27:41 +020038
39 tcgetattr(fd, &oldtio);
Denis Vlasenkoe5387a02007-10-20 19:20:22 +000040 tio = oldtio;
Denys Vlasenko7449e182011-10-22 06:27:41 +020041#if 0
42 /* Switch off UPPERCASE->lowercase conversion (never used since 198x)
43 * and XON/XOFF (why we want to mess with this??)
44 */
45# ifndef IUCLC
46# define IUCLC 0
47# endif
Denis Vlasenkoe5387a02007-10-20 19:20:22 +000048 tio.c_iflag &= ~(IUCLC|IXON|IXOFF|IXANY);
Denys Vlasenko7449e182011-10-22 06:27:41 +020049#endif
50 /* Switch off echo */
51 tio.c_lflag &= ~(ECHO|ECHOE|ECHOK|ECHONL);
Denys Vlasenko557deb12010-02-03 12:17:06 +010052 tcsetattr(fd, TCSANOW, &tio);
Eric Andersen6f9a7782004-05-01 01:27:30 +000053
Denis Vlasenkoe5387a02007-10-20 19:20:22 +000054 memset(&sa, 0, sizeof(sa));
55 /* sa.sa_flags = 0; - no SA_RESTART! */
Denys Vlasenkoe936c6d2010-02-01 04:55:30 +010056 /* SIGINT and SIGALRM will interrupt reads below */
Denis Vlasenkoe5387a02007-10-20 19:20:22 +000057 sa.sa_handler = askpass_timeout;
58 sigaction(SIGINT, &sa, &oldsa);
Eric Andersen6f9a7782004-05-01 01:27:30 +000059 if (timeout) {
Denis Vlasenko8e2cfec2008-03-12 23:19:35 +000060 sigaction_set(SIGALRM, &sa);
Eric Andersen6f9a7782004-05-01 01:27:30 +000061 alarm(timeout);
62 }
63
Denys Vlasenko02859aa2015-10-09 18:16:40 +020064 passwd = auto_string(xmalloc(sizeof_passwd));
Denys Vlasenkoe936c6d2010-02-01 04:55:30 +010065 ret = passwd;
66 i = 0;
67 while (1) {
68 int r = read(fd, &ret[i], 1);
Jonathan Liub6dc13c2013-05-21 17:01:55 +020069 if ((i == 0 && r == 0) /* EOF (^D) with no password */
70 || r < 0
71 ) {
Denys Vlasenkoe936c6d2010-02-01 04:55:30 +010072 /* read is interrupted by timeout or ^C */
73 ret = NULL;
74 break;
75 }
76 if (r == 0 /* EOF */
77 || ret[i] == '\r' || ret[i] == '\n' /* EOL */
78 || ++i == sizeof_passwd-1 /* line limit */
79 ) {
80 ret[i] = '\0';
81 break;
82 }
Eric Andersen6f9a7782004-05-01 01:27:30 +000083 }
84
85 if (timeout) {
86 alarm(0);
87 }
Denis Vlasenko8e2cfec2008-03-12 23:19:35 +000088 sigaction_set(SIGINT, &oldsa);
Denys Vlasenko557deb12010-02-03 12:17:06 +010089 tcsetattr(fd, TCSANOW, &oldtio);
Denis Vlasenko4daad902007-09-27 10:20:47 +000090 bb_putchar('\n');
Denys Vlasenko8131eea2009-11-02 14:19:51 +010091 fflush_all();
Eric Andersen6f9a7782004-05-01 01:27:30 +000092 return ret;
93}