The core of the fix to allow opens to go async inside a compound request.
[Samba.git] / lib / replace / getpass.c
blobf95109f2b43ddd0fc7b751ffde93aae3a8b4640c
1 /* Copyright (C) 1992-1998 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public License as
6 published by the Free Software Foundation; either version 3 of the
7 License, or (at your option) any later version.
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Library General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; see the file COPYING.LIB. If
16 not, see <http://www.gnu.org/licenses/>. */
18 /* Modified to use with samba by Jeremy Allison, 8th July 1995. */
20 #include "replace.h"
21 #include "system/filesys.h"
22 #include "system/wait.h"
23 #include "system/terminal.h"
24 #include "system/passwd.h"
27 * Define additional missing types
29 #ifndef HAVE_SIG_ATOMIC_T_TYPE
30 typedef int sig_atomic_t;
31 #endif
33 #ifndef SIGCLD
34 #define SIGCLD SIGCHLD
35 #endif
37 #ifdef SYSV_TERMIO
39 /* SYSTEM V TERMIO HANDLING */
41 static struct termio t;
43 #define ECHO_IS_ON(t) ((t).c_lflag & ECHO)
44 #define TURN_ECHO_OFF(t) ((t).c_lflag &= ~ECHO)
45 #define TURN_ECHO_ON(t) ((t).c_lflag |= ECHO)
47 #ifndef TCSAFLUSH
48 #define TCSAFLUSH 1
49 #endif
51 #ifndef TCSANOW
52 #define TCSANOW 0
53 #endif
55 static int tcgetattr(int fd, struct termio *_t)
57 return ioctl(fd, TCGETA, _t);
60 static int tcsetattr(int fd, int flags, struct termio *_t)
62 if(flags & TCSAFLUSH)
63 ioctl(fd, TCFLSH, TCIOFLUSH);
64 return ioctl(fd, TCSETS, _t);
67 #elif !defined(TCSAFLUSH)
69 /* BSD TERMIO HANDLING */
71 static struct sgttyb t;
73 #define ECHO_IS_ON(t) ((t).sg_flags & ECHO)
74 #define TURN_ECHO_OFF(t) ((t).sg_flags &= ~ECHO)
75 #define TURN_ECHO_ON(t) ((t).sg_flags |= ECHO)
77 #define TCSAFLUSH 1
78 #define TCSANOW 0
80 static int tcgetattr(int fd, struct sgttyb *_t)
82 return ioctl(fd, TIOCGETP, (char *)_t);
85 static int tcsetattr(int fd, int flags, struct sgttyb *_t)
87 return ioctl(fd, TIOCSETP, (char *)_t);
90 #else /* POSIX TERMIO HANDLING */
91 #define ECHO_IS_ON(t) ((t).c_lflag & ECHO)
92 #define TURN_ECHO_OFF(t) ((t).c_lflag &= ~ECHO)
93 #define TURN_ECHO_ON(t) ((t).c_lflag |= ECHO)
95 static struct termios t;
96 #endif /* SYSV_TERMIO */
98 static void catch_signal(int signum, void (*handler)(int ))
100 #ifdef HAVE_SIGACTION
101 struct sigaction act;
102 struct sigaction oldact;
104 memset(&act, 0, sizeof(act));
106 act.sa_handler = handler;
107 #ifdef SA_RESTART
109 * We *want* SIGALRM to interrupt a system call.
111 if(signum != SIGALRM)
112 act.sa_flags = SA_RESTART;
113 #endif
114 sigemptyset(&act.sa_mask);
115 sigaddset(&act.sa_mask,signum);
116 sigaction(signum,&act,&oldact);
117 #else /* !HAVE_SIGACTION */
118 /* FIXME: need to handle sigvec and systems with broken signal() */
119 signal(signum, handler);
120 #endif
123 static sig_atomic_t gotintr;
124 static int in_fd = -1;
126 /***************************************************************
127 Signal function to tell us were ^C'ed.
128 ****************************************************************/
130 static void gotintr_sig(int signum)
132 gotintr = 1;
133 if (in_fd != -1)
134 close(in_fd); /* Safe way to force a return. */
135 in_fd = -1;
138 char *rep_getpass(const char *prompt)
140 FILE *in, *out;
141 int echo_off;
142 static char buf[256];
143 static size_t bufsize = sizeof(buf);
144 size_t nread;
146 /* Catch problematic signals */
147 catch_signal(SIGINT, gotintr_sig);
149 /* Try to write to and read from the terminal if we can.
150 If we can't open the terminal, use stderr and stdin. */
152 in = fopen ("/dev/tty", "w+");
153 if (in == NULL) {
154 in = stdin;
155 out = stderr;
156 } else {
157 out = in;
160 setvbuf(in, NULL, _IONBF, 0);
162 /* Turn echoing off if it is on now. */
164 if (tcgetattr (fileno (in), &t) == 0) {
165 if (ECHO_IS_ON(t)) {
166 TURN_ECHO_OFF(t);
167 echo_off = tcsetattr (fileno (in), TCSAFLUSH, &t) == 0;
168 TURN_ECHO_ON(t);
169 } else {
170 echo_off = 0;
172 } else {
173 echo_off = 0;
176 /* Write the prompt. */
177 fputs(prompt, out);
178 fflush(out);
180 /* Read the password. */
181 buf[0] = 0;
182 if (!gotintr) {
183 in_fd = fileno(in);
184 if (fgets(buf, bufsize, in) == NULL) {
185 buf[0] = 0;
188 nread = strlen(buf);
189 if (nread) {
190 if (buf[nread - 1] == '\n')
191 buf[nread - 1] = '\0';
194 /* Restore echoing. */
195 if (echo_off) {
196 if (gotintr && in_fd == -1) {
197 in = fopen ("/dev/tty", "w+");
199 if (in != NULL)
200 tcsetattr (fileno (in), TCSANOW, &t);
203 fprintf(out, "\n");
204 fflush(out);
206 if (in && in != stdin) /* We opened the terminal; now close it. */
207 fclose(in);
209 /* Catch problematic signals */
210 catch_signal(SIGINT, SIG_DFL);
212 if (gotintr) {
213 printf("Interrupted by signal.\n");
214 fflush(stdout);
215 exit(1);
217 return buf;