Version 7.6.3.2-android, tag libreoffice-7.6.3.2-android
[LibreOffice.git] / onlineupdate / source / libmar / sign / nss_secutil.c
blob875c14309ec0fe35a4763f7617b503e0d743ee99
1 /* This Source Code Form is subject to the terms of the Mozilla Public
2 * License, v. 2.0. If a copy of the MPL was not distributed with this
3 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
5 /* With the exception of GetPasswordString, this file was
6 copied from NSS's cmd/lib/secutil.c hg revision 8f011395145e */
8 #include "nss_secutil.h"
10 #include "prprf.h"
11 #ifdef _WIN32
12 #include <io.h>
13 #else
14 #include <unistd.h>
15 #endif
17 static char consoleName[] = {
18 #ifdef UNIX
19 "/dev/tty"
20 #else
21 "CON:"
22 #endif
25 #if defined(_WINDOWS)
26 static char * quiet_fgets (char *buf, int length, FILE *input)
28 char *end = buf;
30 /* fflush (input); */
31 memset (buf, 0, length);
33 if (!isatty(fileno(input))) {
34 return fgets(buf,length,input);
37 while (1)
39 int c;
40 #if defined (_WIN32_WCE)
41 c = getchar(); /* gets a character from stdin */
42 #else
43 c = getch(); /* getch gets a character from the console */
44 #endif
45 if (c == '\b')
47 if (end > buf)
48 end--;
51 else if (--length > 0)
52 *end++ = c;
54 if (!c || c == '\n' || c == '\r')
55 break;
58 return buf;
60 #endif
62 char *
63 GetPasswordString(void *arg, char *prompt)
65 FILE *input = stdin;
66 char phrase[200] = {'\0'};
67 int isInputTerminal = isatty(fileno(stdin));
69 (void) arg; (void) prompt; // avoid warnings
71 #ifndef _WINDOWS
72 if (isInputTerminal) {
73 input = fopen(consoleName, "r");
74 if (input == NULL) {
75 fprintf(stderr, "Error opening input terminal for read\n");
76 return NULL;
79 #endif
81 if (isInputTerminal) {
82 fprintf(stdout, "Please enter your password:\n");
83 fflush(stdout);
86 QUIET_FGETS (phrase, sizeof(phrase), input);
88 if (isInputTerminal) {
89 fprintf(stdout, "\n");
92 #ifndef _WINDOWS
93 if (isInputTerminal) {
94 fclose(input);
96 #endif
98 /* Strip off the newlines if present */
99 if (phrase[PORT_Strlen(phrase)-1] == '\n' ||
100 phrase[PORT_Strlen(phrase)-1] == '\r') {
101 phrase[PORT_Strlen(phrase)-1] = 0;
103 return (char*) PORT_Strdup(phrase);
106 char *
107 SECU_FilePasswd(PK11SlotInfo *slot, PRBool retry, void *arg)
109 char* phrases, *phrase;
110 PRFileDesc *fd;
111 int32_t nb;
112 char *pwFile = arg;
113 int i;
114 const long maxPwdFileSize = 4096;
115 char* tokenName = NULL;
116 int tokenLen = 0;
118 if (!pwFile)
119 return 0;
121 if (retry) {
122 return 0; /* no good retrying - the files contents will be the same */
125 phrases = PORT_ZAlloc(maxPwdFileSize + 1);
127 if (!phrases) {
128 return 0; /* out of memory */
131 fd = PR_Open(pwFile, PR_RDONLY, 0);
132 if (!fd) {
133 fprintf(stderr, "No password file \"%s\" exists.\n", pwFile);
134 PORT_Free(phrases);
135 return NULL;
138 nb = PR_Read(fd, phrases, maxPwdFileSize);
140 PR_Close(fd);
142 if (nb == 0) {
143 fprintf(stderr,"password file contains no data\n");
144 PORT_Free(phrases);
145 return NULL;
148 if (slot) {
149 tokenName = PK11_GetTokenName(slot);
150 if (tokenName) {
151 tokenLen = PORT_Strlen(tokenName);
154 i = 0;
157 int startphrase = i;
158 int phraseLen;
160 /* handle the Windows EOL case */
161 while (i < nb && phrases[i] != '\r' && phrases[i] != '\n') i++;
162 /* terminate passphrase */
163 phrases[i++] = '\0';
164 /* clean up any EOL before the start of the next passphrase */
165 while ( (i<nb) && (phrases[i] == '\r' || phrases[i] == '\n')) {
166 phrases[i++] = '\0';
168 /* now analyze the current passphrase */
169 phrase = &phrases[startphrase];
170 if (!tokenName)
171 break;
172 if (PORT_Strncmp(phrase, tokenName, tokenLen)) continue;
173 phraseLen = PORT_Strlen(phrase);
174 if (phraseLen < (tokenLen+1)) continue;
175 if (phrase[tokenLen] != ':') continue;
176 phrase = &phrase[tokenLen+1];
177 break;
179 } while (i<nb);
181 phrase = PORT_Strdup((char*)phrase);
182 PORT_Free(phrases);
183 return phrase;
186 char *
187 SECU_GetModulePassword(PK11SlotInfo *slot, PRBool retry, void *arg)
189 char prompt[255];
190 secuPWData *pwdata = (secuPWData *)arg;
191 secuPWData pwnull = { PW_NONE, 0 };
192 secuPWData pwxtrn = { PW_EXTERNAL, "external" };
193 char *pw;
195 if (pwdata == NULL)
196 pwdata = &pwnull;
198 if (PK11_ProtectedAuthenticationPath(slot)) {
199 pwdata = &pwxtrn;
201 if (retry && pwdata->source != PW_NONE) {
202 PR_fprintf(PR_STDERR, "Incorrect password/PIN entered.\n");
203 return NULL;
206 switch (pwdata->source) {
207 case PW_NONE:
208 sprintf(prompt, "Enter Password or Pin for \"%s\":",
209 PK11_GetTokenName(slot));
210 return GetPasswordString(NULL, prompt);
211 case PW_FROMFILE:
212 /* Instead of opening and closing the file every time, get the pw
213 * once, then keep it in memory (duh).
215 pw = SECU_FilePasswd(slot, retry, pwdata->data);
216 pwdata->source = PW_PLAINTEXT;
217 pwdata->data = strdup(pw);
218 /* it's already been dup'ed */
219 return pw;
220 case PW_EXTERNAL:
221 sprintf(prompt,
222 "Press Enter, then enter PIN for \"%s\" on external device.\n",
223 PK11_GetTokenName(slot));
224 pw = GetPasswordString(NULL, prompt);
225 if (pw) {
226 memset(pw, 0, PORT_Strlen(pw));
227 PORT_Free(pw);
229 /* Fall Through */
230 case PW_PLAINTEXT:
231 return strdup(pwdata->data);
232 default:
233 break;
236 PR_fprintf(PR_STDERR, "Password check failed: No password found.\n");
237 return NULL;