Annotation of embedaddon/curl/src/tool_getpass.c, revision 1.1.1.1
1.1 misho 1: /***************************************************************************
2: * _ _ ____ _
3: * Project ___| | | | _ \| |
4: * / __| | | | |_) | |
5: * | (__| |_| | _ <| |___
6: * \___|\___/|_| \_\_____|
7: *
8: * Copyright (C) 1998 - 2019, Daniel Stenberg, <daniel@haxx.se>, et al.
9: *
10: * This software is licensed as described in the file COPYING, which
11: * you should have received as part of this distribution. The terms
12: * are also available at https://curl.haxx.se/docs/copyright.html.
13: *
14: * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15: * copies of the Software, and permit persons to whom the Software is
16: * furnished to do so, under the terms of the COPYING file.
17: *
18: * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19: * KIND, either express or implied.
20: *
21: ***************************************************************************/
22: #include "tool_setup.h"
23:
24: #if defined(__AMIGA__) && !defined(__amigaos4__)
25: # undef HAVE_TERMIOS_H
26: #endif
27:
28: #ifndef HAVE_GETPASS_R
29: /* this file is only for systems without getpass_r() */
30:
31: #ifdef HAVE_FCNTL_H
32: # include <fcntl.h>
33: #endif
34:
35: #ifdef HAVE_TERMIOS_H
36: # include <termios.h>
37: #elif defined(HAVE_TERMIO_H)
38: # include <termio.h>
39: #endif
40:
41: #ifdef __VMS
42: # include descrip
43: # include starlet
44: # include iodef
45: #endif
46:
47: #ifdef WIN32
48: # include <conio.h>
49: #endif
50:
51: #ifdef NETWARE
52: # ifdef __NOVELL_LIBC__
53: # include <screen.h>
54: # else
55: # include <nwconio.h>
56: # endif
57: #endif
58:
59: #ifdef HAVE_UNISTD_H
60: #include <unistd.h>
61: #endif
62: #include "tool_getpass.h"
63:
64: #include "memdebug.h" /* keep this as LAST include */
65:
66: #ifdef __VMS
67: /* VMS implementation */
68: char *getpass_r(const char *prompt, char *buffer, size_t buflen)
69: {
70: long sts;
71: short chan;
72:
73: /* MSK, 23-JAN-2004, iosbdef.h wasn't in VAX V7.2 or CC 6.4 */
74: /* distribution so I created this. May revert back later to */
75: /* struct _iosb iosb; */
76: struct _iosb
77: {
78: short int iosb$w_status; /* status */
79: short int iosb$w_bcnt; /* byte count */
80: int unused; /* unused */
81: } iosb;
82:
83: $DESCRIPTOR(ttdesc, "TT");
84:
85: buffer[0] = '\0';
86: sts = sys$assign(&ttdesc, &chan, 0, 0);
87: if(sts & 1) {
88: sts = sys$qiow(0, chan,
89: IO$_READPROMPT | IO$M_NOECHO,
90: &iosb, 0, 0, buffer, buflen, 0, 0,
91: prompt, strlen(prompt));
92:
93: if((sts & 1) && (iosb.iosb$w_status & 1))
94: buffer[iosb.iosb$w_bcnt] = '\0';
95:
96: sys$dassgn(chan);
97: }
98: return buffer; /* we always return success */
99: }
100: #define DONE
101: #endif /* __VMS */
102:
103: #ifdef __SYMBIAN32__
104: # define getch() getchar()
105: #endif
106:
107: #if defined(WIN32) || defined(__SYMBIAN32__)
108:
109: char *getpass_r(const char *prompt, char *buffer, size_t buflen)
110: {
111: size_t i;
112: fputs(prompt, stderr);
113:
114: for(i = 0; i < buflen; i++) {
115: buffer[i] = (char)getch();
116: if(buffer[i] == '\r' || buffer[i] == '\n') {
117: buffer[i] = '\0';
118: break;
119: }
120: else
121: if(buffer[i] == '\b')
122: /* remove this letter and if this is not the first key, remove the
123: previous one as well */
124: i = i - (i >= 1 ? 2 : 1);
125: }
126: #ifndef __SYMBIAN32__
127: /* since echo is disabled, print a newline */
128: fputs("\n", stderr);
129: #endif
130: /* if user didn't hit ENTER, terminate buffer */
131: if(i == buflen)
132: buffer[buflen-1] = '\0';
133:
134: return buffer; /* we always return success */
135: }
136: #define DONE
137: #endif /* WIN32 || __SYMBIAN32__ */
138:
139: #ifdef NETWARE
140: /* NetWare implementation */
141: #ifdef __NOVELL_LIBC__
142: char *getpass_r(const char *prompt, char *buffer, size_t buflen)
143: {
144: return getpassword(prompt, buffer, buflen);
145: }
146: #else
147: char *getpass_r(const char *prompt, char *buffer, size_t buflen)
148: {
149: size_t i = 0;
150:
151: printf("%s", prompt);
152: do {
153: buffer[i++] = getch();
154: if(buffer[i-1] == '\b') {
155: /* remove this letter and if this is not the first key,
156: remove the previous one as well */
157: if(i > 1) {
158: printf("\b \b");
159: i = i - 2;
160: }
161: else {
162: RingTheBell();
163: i = i - 1;
164: }
165: }
166: else if(buffer[i-1] != 13)
167: putchar('*');
168:
169: } while((buffer[i-1] != 13) && (i < buflen));
170: buffer[i-1] = '\0';
171: printf("\r\n");
172: return buffer;
173: }
174: #endif /* __NOVELL_LIBC__ */
175: #define DONE
176: #endif /* NETWARE */
177:
178: #ifndef DONE /* not previously provided */
179:
180: #ifdef HAVE_TERMIOS_H
181: # define struct_term struct termios
182: #elif defined(HAVE_TERMIO_H)
183: # define struct_term struct termio
184: #else
185: # undef struct_term
186: #endif
187:
188: static bool ttyecho(bool enable, int fd)
189: {
190: #ifdef struct_term
191: static struct_term withecho;
192: static struct_term noecho;
193: #endif
194: if(!enable) {
195: /* disable echo by extracting the current 'withecho' mode and remove the
196: ECHO bit and set back the struct */
197: #ifdef HAVE_TERMIOS_H
198: tcgetattr(fd, &withecho);
199: noecho = withecho;
200: noecho.c_lflag &= ~ECHO;
201: tcsetattr(fd, TCSANOW, &noecho);
202: #elif defined(HAVE_TERMIO_H)
203: ioctl(fd, TCGETA, &withecho);
204: noecho = withecho;
205: noecho.c_lflag &= ~ECHO;
206: ioctl(fd, TCSETA, &noecho);
207: #else
208: /* neither HAVE_TERMIO_H nor HAVE_TERMIOS_H, we can't disable echo! */
209: (void)fd;
210: return FALSE; /* not disabled */
211: #endif
212: return TRUE; /* disabled */
213: }
214: /* re-enable echo, assumes we disabled it before (and set the structs we
215: now use to reset the terminal status) */
216: #ifdef HAVE_TERMIOS_H
217: tcsetattr(fd, TCSAFLUSH, &withecho);
218: #elif defined(HAVE_TERMIO_H)
219: ioctl(fd, TCSETA, &withecho);
220: #else
221: return FALSE; /* not enabled */
222: #endif
223: return TRUE; /* enabled */
224: }
225:
226: char *getpass_r(const char *prompt, /* prompt to display */
227: char *password, /* buffer to store password in */
228: size_t buflen) /* size of buffer to store password in */
229: {
230: ssize_t nread;
231: bool disabled;
232: int fd = open("/dev/tty", O_RDONLY);
233: if(-1 == fd)
234: fd = STDIN_FILENO; /* use stdin if the tty couldn't be used */
235:
236: disabled = ttyecho(FALSE, fd); /* disable terminal echo */
237:
238: fputs(prompt, stderr);
239: nread = read(fd, password, buflen);
240: if(nread > 0)
241: password[--nread] = '\0'; /* zero terminate where enter is stored */
242: else
243: password[0] = '\0'; /* got nothing */
244:
245: if(disabled) {
246: /* if echo actually was disabled, add a newline */
247: fputs("\n", stderr);
248: (void)ttyecho(TRUE, fd); /* enable echo */
249: }
250:
251: if(STDIN_FILENO != fd)
252: close(fd);
253:
254: return password; /* return pointer to buffer */
255: }
256:
257: #endif /* DONE */
258: #endif /* HAVE_GETPASS_R */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>