OLD | NEW |
| (Empty) |
1 /* crypto/ui/ui_lib.c -*- mode:C; c-file-style: "eay" -*- */ | |
2 /* Written by Richard Levitte (richard@levitte.org) for the OpenSSL | |
3 * project 2001. | |
4 */ | |
5 /* ==================================================================== | |
6 * Copyright (c) 2001 The OpenSSL Project. All rights reserved. | |
7 * | |
8 * Redistribution and use in source and binary forms, with or without | |
9 * modification, are permitted provided that the following conditions | |
10 * are met: | |
11 * | |
12 * 1. Redistributions of source code must retain the above copyright | |
13 * notice, this list of conditions and the following disclaimer. | |
14 * | |
15 * 2. Redistributions in binary form must reproduce the above copyright | |
16 * notice, this list of conditions and the following disclaimer in | |
17 * the documentation and/or other materials provided with the | |
18 * distribution. | |
19 * | |
20 * 3. All advertising materials mentioning features or use of this | |
21 * software must display the following acknowledgment: | |
22 * "This product includes software developed by the OpenSSL Project | |
23 * for use in the OpenSSL Toolkit. (http://www.openssl.org/)" | |
24 * | |
25 * 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to | |
26 * endorse or promote products derived from this software without | |
27 * prior written permission. For written permission, please contact | |
28 * openssl-core@openssl.org. | |
29 * | |
30 * 5. Products derived from this software may not be called "OpenSSL" | |
31 * nor may "OpenSSL" appear in their names without prior written | |
32 * permission of the OpenSSL Project. | |
33 * | |
34 * 6. Redistributions of any form whatsoever must retain the following | |
35 * acknowledgment: | |
36 * "This product includes software developed by the OpenSSL Project | |
37 * for use in the OpenSSL Toolkit (http://www.openssl.org/)" | |
38 * | |
39 * THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY | |
40 * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
41 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | |
42 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE OpenSSL PROJECT OR | |
43 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |
44 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | |
45 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | |
46 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | |
47 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, | |
48 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | |
49 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED | |
50 * OF THE POSSIBILITY OF SUCH DAMAGE. | |
51 * ==================================================================== | |
52 * | |
53 * This product includes cryptographic software written by Eric Young | |
54 * (eay@cryptsoft.com). This product includes software written by Tim | |
55 * Hudson (tjh@cryptsoft.com). | |
56 * | |
57 */ | |
58 | |
59 #include <string.h> | |
60 #include "cryptlib.h" | |
61 #include <openssl/e_os2.h> | |
62 #include <openssl/buffer.h> | |
63 #include <openssl/ui.h> | |
64 #include <openssl/err.h> | |
65 #include "ui_locl.h" | |
66 | |
67 IMPLEMENT_STACK_OF(UI_STRING_ST) | |
68 | |
69 static const UI_METHOD *default_UI_meth=NULL; | |
70 | |
71 UI *UI_new(void) | |
72 { | |
73 return(UI_new_method(NULL)); | |
74 } | |
75 | |
76 UI *UI_new_method(const UI_METHOD *method) | |
77 { | |
78 UI *ret; | |
79 | |
80 ret=(UI *)OPENSSL_malloc(sizeof(UI)); | |
81 if (ret == NULL) | |
82 { | |
83 UIerr(UI_F_UI_NEW_METHOD,ERR_R_MALLOC_FAILURE); | |
84 return NULL; | |
85 } | |
86 if (method == NULL) | |
87 ret->meth=UI_get_default_method(); | |
88 else | |
89 ret->meth=method; | |
90 | |
91 ret->strings=NULL; | |
92 ret->user_data=NULL; | |
93 ret->flags=0; | |
94 CRYPTO_new_ex_data(CRYPTO_EX_INDEX_UI, ret, &ret->ex_data); | |
95 return ret; | |
96 } | |
97 | |
98 static void free_string(UI_STRING *uis) | |
99 { | |
100 if (uis->flags & OUT_STRING_FREEABLE) | |
101 { | |
102 OPENSSL_free((char *)uis->out_string); | |
103 switch(uis->type) | |
104 { | |
105 case UIT_BOOLEAN: | |
106 OPENSSL_free((char *)uis->_.boolean_data.action_desc); | |
107 OPENSSL_free((char *)uis->_.boolean_data.ok_chars); | |
108 OPENSSL_free((char *)uis->_.boolean_data.cancel_chars); | |
109 break; | |
110 default: | |
111 break; | |
112 } | |
113 } | |
114 OPENSSL_free(uis); | |
115 } | |
116 | |
117 void UI_free(UI *ui) | |
118 { | |
119 if (ui == NULL) | |
120 return; | |
121 sk_UI_STRING_pop_free(ui->strings,free_string); | |
122 CRYPTO_free_ex_data(CRYPTO_EX_INDEX_UI, ui, &ui->ex_data); | |
123 OPENSSL_free(ui); | |
124 } | |
125 | |
126 static int allocate_string_stack(UI *ui) | |
127 { | |
128 if (ui->strings == NULL) | |
129 { | |
130 ui->strings=sk_UI_STRING_new_null(); | |
131 if (ui->strings == NULL) | |
132 { | |
133 return -1; | |
134 } | |
135 } | |
136 return 0; | |
137 } | |
138 | |
139 static UI_STRING *general_allocate_prompt(UI *ui, const char *prompt, | |
140 int prompt_freeable, enum UI_string_types type, int input_flags, | |
141 char *result_buf) | |
142 { | |
143 UI_STRING *ret = NULL; | |
144 | |
145 if (prompt == NULL) | |
146 { | |
147 UIerr(UI_F_GENERAL_ALLOCATE_PROMPT,ERR_R_PASSED_NULL_PARAMETER); | |
148 } | |
149 else if ((type == UIT_PROMPT || type == UIT_VERIFY | |
150 || type == UIT_BOOLEAN) && result_buf == NULL) | |
151 { | |
152 UIerr(UI_F_GENERAL_ALLOCATE_PROMPT,UI_R_NO_RESULT_BUFFER); | |
153 } | |
154 else if ((ret = (UI_STRING *)OPENSSL_malloc(sizeof(UI_STRING)))) | |
155 { | |
156 ret->out_string=prompt; | |
157 ret->flags=prompt_freeable ? OUT_STRING_FREEABLE : 0; | |
158 ret->input_flags=input_flags; | |
159 ret->type=type; | |
160 ret->result_buf=result_buf; | |
161 } | |
162 return ret; | |
163 } | |
164 | |
165 static int general_allocate_string(UI *ui, const char *prompt, | |
166 int prompt_freeable, enum UI_string_types type, int input_flags, | |
167 char *result_buf, int minsize, int maxsize, const char *test_buf) | |
168 { | |
169 int ret = -1; | |
170 UI_STRING *s = general_allocate_prompt(ui, prompt, prompt_freeable, | |
171 type, input_flags, result_buf); | |
172 | |
173 if (s) | |
174 { | |
175 if (allocate_string_stack(ui) >= 0) | |
176 { | |
177 s->_.string_data.result_minsize=minsize; | |
178 s->_.string_data.result_maxsize=maxsize; | |
179 s->_.string_data.test_buf=test_buf; | |
180 ret=sk_UI_STRING_push(ui->strings, s); | |
181 /* sk_push() returns 0 on error. Let's addapt that */ | |
182 if (ret <= 0) ret--; | |
183 } | |
184 else | |
185 free_string(s); | |
186 } | |
187 return ret; | |
188 } | |
189 | |
190 static int general_allocate_boolean(UI *ui, | |
191 const char *prompt, const char *action_desc, | |
192 const char *ok_chars, const char *cancel_chars, | |
193 int prompt_freeable, enum UI_string_types type, int input_flags, | |
194 char *result_buf) | |
195 { | |
196 int ret = -1; | |
197 UI_STRING *s; | |
198 const char *p; | |
199 | |
200 if (ok_chars == NULL) | |
201 { | |
202 UIerr(UI_F_GENERAL_ALLOCATE_BOOLEAN,ERR_R_PASSED_NULL_PARAMETER)
; | |
203 } | |
204 else if (cancel_chars == NULL) | |
205 { | |
206 UIerr(UI_F_GENERAL_ALLOCATE_BOOLEAN,ERR_R_PASSED_NULL_PARAMETER)
; | |
207 } | |
208 else | |
209 { | |
210 for(p = ok_chars; *p; p++) | |
211 { | |
212 if (strchr(cancel_chars, *p)) | |
213 { | |
214 UIerr(UI_F_GENERAL_ALLOCATE_BOOLEAN, | |
215 UI_R_COMMON_OK_AND_CANCEL_CHARACTERS); | |
216 } | |
217 } | |
218 | |
219 s = general_allocate_prompt(ui, prompt, prompt_freeable, | |
220 type, input_flags, result_buf); | |
221 | |
222 if (s) | |
223 { | |
224 if (allocate_string_stack(ui) >= 0) | |
225 { | |
226 s->_.boolean_data.action_desc = action_desc; | |
227 s->_.boolean_data.ok_chars = ok_chars; | |
228 s->_.boolean_data.cancel_chars = cancel_chars; | |
229 ret=sk_UI_STRING_push(ui->strings, s); | |
230 /* sk_push() returns 0 on error. | |
231 Let's addapt that */ | |
232 if (ret <= 0) ret--; | |
233 } | |
234 else | |
235 free_string(s); | |
236 } | |
237 } | |
238 return ret; | |
239 } | |
240 | |
241 /* Returns the index to the place in the stack or -1 for error. Uses a | |
242 direct reference to the prompt. */ | |
243 int UI_add_input_string(UI *ui, const char *prompt, int flags, | |
244 char *result_buf, int minsize, int maxsize) | |
245 { | |
246 return general_allocate_string(ui, prompt, 0, | |
247 UIT_PROMPT, flags, result_buf, minsize, maxsize, NULL); | |
248 } | |
249 | |
250 /* Same as UI_add_input_string(), excepts it takes a copy of the prompt */ | |
251 int UI_dup_input_string(UI *ui, const char *prompt, int flags, | |
252 char *result_buf, int minsize, int maxsize) | |
253 { | |
254 char *prompt_copy=NULL; | |
255 | |
256 if (prompt) | |
257 { | |
258 prompt_copy=BUF_strdup(prompt); | |
259 if (prompt_copy == NULL) | |
260 { | |
261 UIerr(UI_F_UI_DUP_INPUT_STRING,ERR_R_MALLOC_FAILURE); | |
262 return 0; | |
263 } | |
264 } | |
265 | |
266 return general_allocate_string(ui, prompt_copy, 1, | |
267 UIT_PROMPT, flags, result_buf, minsize, maxsize, NULL); | |
268 } | |
269 | |
270 int UI_add_verify_string(UI *ui, const char *prompt, int flags, | |
271 char *result_buf, int minsize, int maxsize, const char *test_buf) | |
272 { | |
273 return general_allocate_string(ui, prompt, 0, | |
274 UIT_VERIFY, flags, result_buf, minsize, maxsize, test_buf); | |
275 } | |
276 | |
277 int UI_dup_verify_string(UI *ui, const char *prompt, int flags, | |
278 char *result_buf, int minsize, int maxsize, const char *test_buf) | |
279 { | |
280 char *prompt_copy=NULL; | |
281 | |
282 if (prompt) | |
283 { | |
284 prompt_copy=BUF_strdup(prompt); | |
285 if (prompt_copy == NULL) | |
286 { | |
287 UIerr(UI_F_UI_DUP_VERIFY_STRING,ERR_R_MALLOC_FAILURE); | |
288 return -1; | |
289 } | |
290 } | |
291 | |
292 return general_allocate_string(ui, prompt_copy, 1, | |
293 UIT_VERIFY, flags, result_buf, minsize, maxsize, test_buf); | |
294 } | |
295 | |
296 int UI_add_input_boolean(UI *ui, const char *prompt, const char *action_desc, | |
297 const char *ok_chars, const char *cancel_chars, | |
298 int flags, char *result_buf) | |
299 { | |
300 return general_allocate_boolean(ui, prompt, action_desc, | |
301 ok_chars, cancel_chars, 0, UIT_BOOLEAN, flags, result_buf); | |
302 } | |
303 | |
304 int UI_dup_input_boolean(UI *ui, const char *prompt, const char *action_desc, | |
305 const char *ok_chars, const char *cancel_chars, | |
306 int flags, char *result_buf) | |
307 { | |
308 char *prompt_copy = NULL; | |
309 char *action_desc_copy = NULL; | |
310 char *ok_chars_copy = NULL; | |
311 char *cancel_chars_copy = NULL; | |
312 | |
313 if (prompt) | |
314 { | |
315 prompt_copy=BUF_strdup(prompt); | |
316 if (prompt_copy == NULL) | |
317 { | |
318 UIerr(UI_F_UI_DUP_INPUT_BOOLEAN,ERR_R_MALLOC_FAILURE); | |
319 goto err; | |
320 } | |
321 } | |
322 | |
323 if (action_desc) | |
324 { | |
325 action_desc_copy=BUF_strdup(action_desc); | |
326 if (action_desc_copy == NULL) | |
327 { | |
328 UIerr(UI_F_UI_DUP_INPUT_BOOLEAN,ERR_R_MALLOC_FAILURE); | |
329 goto err; | |
330 } | |
331 } | |
332 | |
333 if (ok_chars) | |
334 { | |
335 ok_chars_copy=BUF_strdup(ok_chars); | |
336 if (ok_chars_copy == NULL) | |
337 { | |
338 UIerr(UI_F_UI_DUP_INPUT_BOOLEAN,ERR_R_MALLOC_FAILURE); | |
339 goto err; | |
340 } | |
341 } | |
342 | |
343 if (cancel_chars) | |
344 { | |
345 cancel_chars_copy=BUF_strdup(cancel_chars); | |
346 if (cancel_chars_copy == NULL) | |
347 { | |
348 UIerr(UI_F_UI_DUP_INPUT_BOOLEAN,ERR_R_MALLOC_FAILURE); | |
349 goto err; | |
350 } | |
351 } | |
352 | |
353 return general_allocate_boolean(ui, prompt_copy, action_desc_copy, | |
354 ok_chars_copy, cancel_chars_copy, 1, UIT_BOOLEAN, flags, | |
355 result_buf); | |
356 err: | |
357 if (prompt_copy) OPENSSL_free(prompt_copy); | |
358 if (action_desc_copy) OPENSSL_free(action_desc_copy); | |
359 if (ok_chars_copy) OPENSSL_free(ok_chars_copy); | |
360 if (cancel_chars_copy) OPENSSL_free(cancel_chars_copy); | |
361 return -1; | |
362 } | |
363 | |
364 int UI_add_info_string(UI *ui, const char *text) | |
365 { | |
366 return general_allocate_string(ui, text, 0, UIT_INFO, 0, NULL, 0, 0, | |
367 NULL); | |
368 } | |
369 | |
370 int UI_dup_info_string(UI *ui, const char *text) | |
371 { | |
372 char *text_copy=NULL; | |
373 | |
374 if (text) | |
375 { | |
376 text_copy=BUF_strdup(text); | |
377 if (text_copy == NULL) | |
378 { | |
379 UIerr(UI_F_UI_DUP_INFO_STRING,ERR_R_MALLOC_FAILURE); | |
380 return -1; | |
381 } | |
382 } | |
383 | |
384 return general_allocate_string(ui, text_copy, 1, UIT_INFO, 0, NULL, | |
385 0, 0, NULL); | |
386 } | |
387 | |
388 int UI_add_error_string(UI *ui, const char *text) | |
389 { | |
390 return general_allocate_string(ui, text, 0, UIT_ERROR, 0, NULL, 0, 0, | |
391 NULL); | |
392 } | |
393 | |
394 int UI_dup_error_string(UI *ui, const char *text) | |
395 { | |
396 char *text_copy=NULL; | |
397 | |
398 if (text) | |
399 { | |
400 text_copy=BUF_strdup(text); | |
401 if (text_copy == NULL) | |
402 { | |
403 UIerr(UI_F_UI_DUP_ERROR_STRING,ERR_R_MALLOC_FAILURE); | |
404 return -1; | |
405 } | |
406 } | |
407 return general_allocate_string(ui, text_copy, 1, UIT_ERROR, 0, NULL, | |
408 0, 0, NULL); | |
409 } | |
410 | |
411 char *UI_construct_prompt(UI *ui, const char *object_desc, | |
412 const char *object_name) | |
413 { | |
414 char *prompt = NULL; | |
415 | |
416 if (ui->meth->ui_construct_prompt) | |
417 prompt = ui->meth->ui_construct_prompt(ui, | |
418 object_desc, object_name); | |
419 else | |
420 { | |
421 char prompt1[] = "Enter "; | |
422 char prompt2[] = " for "; | |
423 char prompt3[] = ":"; | |
424 int len = 0; | |
425 | |
426 if (object_desc == NULL) | |
427 return NULL; | |
428 len = sizeof(prompt1) - 1 + strlen(object_desc); | |
429 if (object_name) | |
430 len += sizeof(prompt2) - 1 + strlen(object_name); | |
431 len += sizeof(prompt3) - 1; | |
432 | |
433 prompt = (char *)OPENSSL_malloc(len + 1); | |
434 BUF_strlcpy(prompt, prompt1, len + 1); | |
435 BUF_strlcat(prompt, object_desc, len + 1); | |
436 if (object_name) | |
437 { | |
438 BUF_strlcat(prompt, prompt2, len + 1); | |
439 BUF_strlcat(prompt, object_name, len + 1); | |
440 } | |
441 BUF_strlcat(prompt, prompt3, len + 1); | |
442 } | |
443 return prompt; | |
444 } | |
445 | |
446 void *UI_add_user_data(UI *ui, void *user_data) | |
447 { | |
448 void *old_data = ui->user_data; | |
449 ui->user_data = user_data; | |
450 return old_data; | |
451 } | |
452 | |
453 void *UI_get0_user_data(UI *ui) | |
454 { | |
455 return ui->user_data; | |
456 } | |
457 | |
458 const char *UI_get0_result(UI *ui, int i) | |
459 { | |
460 if (i < 0) | |
461 { | |
462 UIerr(UI_F_UI_GET0_RESULT,UI_R_INDEX_TOO_SMALL); | |
463 return NULL; | |
464 } | |
465 if (i >= sk_UI_STRING_num(ui->strings)) | |
466 { | |
467 UIerr(UI_F_UI_GET0_RESULT,UI_R_INDEX_TOO_LARGE); | |
468 return NULL; | |
469 } | |
470 return UI_get0_result_string(sk_UI_STRING_value(ui->strings, i)); | |
471 } | |
472 | |
473 static int print_error(const char *str, size_t len, UI *ui) | |
474 { | |
475 UI_STRING uis; | |
476 | |
477 memset(&uis, 0, sizeof(uis)); | |
478 uis.type = UIT_ERROR; | |
479 uis.out_string = str; | |
480 | |
481 if (ui->meth->ui_write_string | |
482 && !ui->meth->ui_write_string(ui, &uis)) | |
483 return -1; | |
484 return 0; | |
485 } | |
486 | |
487 int UI_process(UI *ui) | |
488 { | |
489 int i, ok=0; | |
490 | |
491 if (ui->meth->ui_open_session && !ui->meth->ui_open_session(ui)) | |
492 return -1; | |
493 | |
494 if (ui->flags & UI_FLAG_PRINT_ERRORS) | |
495 ERR_print_errors_cb( | |
496 (int (*)(const char *, size_t, void *))print_error, | |
497 (void *)ui); | |
498 | |
499 for(i=0; i<sk_UI_STRING_num(ui->strings); i++) | |
500 { | |
501 if (ui->meth->ui_write_string | |
502 && !ui->meth->ui_write_string(ui, | |
503 sk_UI_STRING_value(ui->strings, i))) | |
504 { | |
505 ok=-1; | |
506 goto err; | |
507 } | |
508 } | |
509 | |
510 if (ui->meth->ui_flush) | |
511 switch(ui->meth->ui_flush(ui)) | |
512 { | |
513 case -1: /* Interrupt/Cancel/something... */ | |
514 ok = -2; | |
515 goto err; | |
516 case 0: /* Errors */ | |
517 ok = -1; | |
518 goto err; | |
519 default: /* Success */ | |
520 ok = 0; | |
521 break; | |
522 } | |
523 | |
524 for(i=0; i<sk_UI_STRING_num(ui->strings); i++) | |
525 { | |
526 if (ui->meth->ui_read_string) | |
527 { | |
528 switch(ui->meth->ui_read_string(ui, | |
529 sk_UI_STRING_value(ui->strings, i))) | |
530 { | |
531 case -1: /* Interrupt/Cancel/something... */ | |
532 ok = -2; | |
533 goto err; | |
534 case 0: /* Errors */ | |
535 ok = -1; | |
536 goto err; | |
537 default: /* Success */ | |
538 ok = 0; | |
539 break; | |
540 } | |
541 } | |
542 } | |
543 err: | |
544 if (ui->meth->ui_close_session && !ui->meth->ui_close_session(ui)) | |
545 return -1; | |
546 return ok; | |
547 } | |
548 | |
549 int UI_ctrl(UI *ui, int cmd, long i, void *p, void (*f)(void)) | |
550 { | |
551 if (ui == NULL) | |
552 { | |
553 UIerr(UI_F_UI_CTRL,ERR_R_PASSED_NULL_PARAMETER); | |
554 return -1; | |
555 } | |
556 switch(cmd) | |
557 { | |
558 case UI_CTRL_PRINT_ERRORS: | |
559 { | |
560 int save_flag = !!(ui->flags & UI_FLAG_PRINT_ERRORS); | |
561 if (i) | |
562 ui->flags |= UI_FLAG_PRINT_ERRORS; | |
563 else | |
564 ui->flags &= ~UI_FLAG_PRINT_ERRORS; | |
565 return save_flag; | |
566 } | |
567 case UI_CTRL_IS_REDOABLE: | |
568 return !!(ui->flags & UI_FLAG_REDOABLE); | |
569 default: | |
570 break; | |
571 } | |
572 UIerr(UI_F_UI_CTRL,UI_R_UNKNOWN_CONTROL_COMMAND); | |
573 return -1; | |
574 } | |
575 | |
576 int UI_get_ex_new_index(long argl, void *argp, CRYPTO_EX_new *new_func, | |
577 CRYPTO_EX_dup *dup_func, CRYPTO_EX_free *free_func) | |
578 { | |
579 return CRYPTO_get_ex_new_index(CRYPTO_EX_INDEX_UI, argl, argp, | |
580 new_func, dup_func, free_func); | |
581 } | |
582 | |
583 int UI_set_ex_data(UI *r, int idx, void *arg) | |
584 { | |
585 return(CRYPTO_set_ex_data(&r->ex_data,idx,arg)); | |
586 } | |
587 | |
588 void *UI_get_ex_data(UI *r, int idx) | |
589 { | |
590 return(CRYPTO_get_ex_data(&r->ex_data,idx)); | |
591 } | |
592 | |
593 void UI_set_default_method(const UI_METHOD *meth) | |
594 { | |
595 default_UI_meth=meth; | |
596 } | |
597 | |
598 const UI_METHOD *UI_get_default_method(void) | |
599 { | |
600 if (default_UI_meth == NULL) | |
601 { | |
602 default_UI_meth=UI_OpenSSL(); | |
603 } | |
604 return default_UI_meth; | |
605 } | |
606 | |
607 const UI_METHOD *UI_get_method(UI *ui) | |
608 { | |
609 return ui->meth; | |
610 } | |
611 | |
612 const UI_METHOD *UI_set_method(UI *ui, const UI_METHOD *meth) | |
613 { | |
614 ui->meth=meth; | |
615 return ui->meth; | |
616 } | |
617 | |
618 | |
619 UI_METHOD *UI_create_method(char *name) | |
620 { | |
621 UI_METHOD *ui_method = (UI_METHOD *)OPENSSL_malloc(sizeof(UI_METHOD)); | |
622 | |
623 if (ui_method) | |
624 { | |
625 memset(ui_method, 0, sizeof(*ui_method)); | |
626 ui_method->name = BUF_strdup(name); | |
627 } | |
628 return ui_method; | |
629 } | |
630 | |
631 /* BIG FSCKING WARNING!!!! If you use this on a statically allocated method | |
632 (that is, it hasn't been allocated using UI_create_method(), you deserve | |
633 anything Murphy can throw at you and more! You have been warned. */ | |
634 void UI_destroy_method(UI_METHOD *ui_method) | |
635 { | |
636 OPENSSL_free(ui_method->name); | |
637 ui_method->name = NULL; | |
638 OPENSSL_free(ui_method); | |
639 } | |
640 | |
641 int UI_method_set_opener(UI_METHOD *method, int (*opener)(UI *ui)) | |
642 { | |
643 if (method) | |
644 { | |
645 method->ui_open_session = opener; | |
646 return 0; | |
647 } | |
648 else | |
649 return -1; | |
650 } | |
651 | |
652 int UI_method_set_writer(UI_METHOD *method, int (*writer)(UI *ui, UI_STRING *uis
)) | |
653 { | |
654 if (method) | |
655 { | |
656 method->ui_write_string = writer; | |
657 return 0; | |
658 } | |
659 else | |
660 return -1; | |
661 } | |
662 | |
663 int UI_method_set_flusher(UI_METHOD *method, int (*flusher)(UI *ui)) | |
664 { | |
665 if (method) | |
666 { | |
667 method->ui_flush = flusher; | |
668 return 0; | |
669 } | |
670 else | |
671 return -1; | |
672 } | |
673 | |
674 int UI_method_set_reader(UI_METHOD *method, int (*reader)(UI *ui, UI_STRING *uis
)) | |
675 { | |
676 if (method) | |
677 { | |
678 method->ui_read_string = reader; | |
679 return 0; | |
680 } | |
681 else | |
682 return -1; | |
683 } | |
684 | |
685 int UI_method_set_closer(UI_METHOD *method, int (*closer)(UI *ui)) | |
686 { | |
687 if (method) | |
688 { | |
689 method->ui_close_session = closer; | |
690 return 0; | |
691 } | |
692 else | |
693 return -1; | |
694 } | |
695 | |
696 int UI_method_set_prompt_constructor(UI_METHOD *method, char *(*prompt_construct
or)(UI* ui, const char* object_desc, const char* object_name)) | |
697 { | |
698 if (method) | |
699 { | |
700 method->ui_construct_prompt = prompt_constructor; | |
701 return 0; | |
702 } | |
703 else | |
704 return -1; | |
705 } | |
706 | |
707 int (*UI_method_get_opener(UI_METHOD *method))(UI*) | |
708 { | |
709 if (method) | |
710 return method->ui_open_session; | |
711 else | |
712 return NULL; | |
713 } | |
714 | |
715 int (*UI_method_get_writer(UI_METHOD *method))(UI*,UI_STRING*) | |
716 { | |
717 if (method) | |
718 return method->ui_write_string; | |
719 else | |
720 return NULL; | |
721 } | |
722 | |
723 int (*UI_method_get_flusher(UI_METHOD *method))(UI*) | |
724 { | |
725 if (method) | |
726 return method->ui_flush; | |
727 else | |
728 return NULL; | |
729 } | |
730 | |
731 int (*UI_method_get_reader(UI_METHOD *method))(UI*,UI_STRING*) | |
732 { | |
733 if (method) | |
734 return method->ui_read_string; | |
735 else | |
736 return NULL; | |
737 } | |
738 | |
739 int (*UI_method_get_closer(UI_METHOD *method))(UI*) | |
740 { | |
741 if (method) | |
742 return method->ui_close_session; | |
743 else | |
744 return NULL; | |
745 } | |
746 | |
747 char* (*UI_method_get_prompt_constructor(UI_METHOD *method))(UI*, const char*, c
onst char*) | |
748 { | |
749 if (method) | |
750 return method->ui_construct_prompt; | |
751 else | |
752 return NULL; | |
753 } | |
754 | |
755 enum UI_string_types UI_get_string_type(UI_STRING *uis) | |
756 { | |
757 if (!uis) | |
758 return UIT_NONE; | |
759 return uis->type; | |
760 } | |
761 | |
762 int UI_get_input_flags(UI_STRING *uis) | |
763 { | |
764 if (!uis) | |
765 return 0; | |
766 return uis->input_flags; | |
767 } | |
768 | |
769 const char *UI_get0_output_string(UI_STRING *uis) | |
770 { | |
771 if (!uis) | |
772 return NULL; | |
773 return uis->out_string; | |
774 } | |
775 | |
776 const char *UI_get0_action_string(UI_STRING *uis) | |
777 { | |
778 if (!uis) | |
779 return NULL; | |
780 switch(uis->type) | |
781 { | |
782 case UIT_PROMPT: | |
783 case UIT_BOOLEAN: | |
784 return uis->_.boolean_data.action_desc; | |
785 default: | |
786 return NULL; | |
787 } | |
788 } | |
789 | |
790 const char *UI_get0_result_string(UI_STRING *uis) | |
791 { | |
792 if (!uis) | |
793 return NULL; | |
794 switch(uis->type) | |
795 { | |
796 case UIT_PROMPT: | |
797 case UIT_VERIFY: | |
798 return uis->result_buf; | |
799 default: | |
800 return NULL; | |
801 } | |
802 } | |
803 | |
804 const char *UI_get0_test_string(UI_STRING *uis) | |
805 { | |
806 if (!uis) | |
807 return NULL; | |
808 switch(uis->type) | |
809 { | |
810 case UIT_VERIFY: | |
811 return uis->_.string_data.test_buf; | |
812 default: | |
813 return NULL; | |
814 } | |
815 } | |
816 | |
817 int UI_get_result_minsize(UI_STRING *uis) | |
818 { | |
819 if (!uis) | |
820 return -1; | |
821 switch(uis->type) | |
822 { | |
823 case UIT_PROMPT: | |
824 case UIT_VERIFY: | |
825 return uis->_.string_data.result_minsize; | |
826 default: | |
827 return -1; | |
828 } | |
829 } | |
830 | |
831 int UI_get_result_maxsize(UI_STRING *uis) | |
832 { | |
833 if (!uis) | |
834 return -1; | |
835 switch(uis->type) | |
836 { | |
837 case UIT_PROMPT: | |
838 case UIT_VERIFY: | |
839 return uis->_.string_data.result_maxsize; | |
840 default: | |
841 return -1; | |
842 } | |
843 } | |
844 | |
845 int UI_set_result(UI *ui, UI_STRING *uis, const char *result) | |
846 { | |
847 int l = strlen(result); | |
848 | |
849 ui->flags &= ~UI_FLAG_REDOABLE; | |
850 | |
851 if (!uis) | |
852 return -1; | |
853 switch (uis->type) | |
854 { | |
855 case UIT_PROMPT: | |
856 case UIT_VERIFY: | |
857 { | |
858 char number1[DECIMAL_SIZE(uis->_.string_data.result_minsize)+1]; | |
859 char number2[DECIMAL_SIZE(uis->_.string_data.result_maxsize)+1]; | |
860 | |
861 BIO_snprintf(number1, sizeof(number1), "%d", | |
862 uis->_.string_data.result_minsize); | |
863 BIO_snprintf(number2, sizeof(number2), "%d", | |
864 uis->_.string_data.result_maxsize); | |
865 | |
866 if (l < uis->_.string_data.result_minsize) | |
867 { | |
868 ui->flags |= UI_FLAG_REDOABLE; | |
869 UIerr(UI_F_UI_SET_RESULT,UI_R_RESULT_TOO_SMALL); | |
870 ERR_add_error_data(5,"You must type in ", | |
871 number1," to ",number2," characters"); | |
872 return -1; | |
873 } | |
874 if (l > uis->_.string_data.result_maxsize) | |
875 { | |
876 ui->flags |= UI_FLAG_REDOABLE; | |
877 UIerr(UI_F_UI_SET_RESULT,UI_R_RESULT_TOO_LARGE); | |
878 ERR_add_error_data(5,"You must type in ", | |
879 number1," to ",number2," characters"); | |
880 return -1; | |
881 } | |
882 } | |
883 | |
884 if (!uis->result_buf) | |
885 { | |
886 UIerr(UI_F_UI_SET_RESULT,UI_R_NO_RESULT_BUFFER); | |
887 return -1; | |
888 } | |
889 | |
890 BUF_strlcpy(uis->result_buf, result, | |
891 uis->_.string_data.result_maxsize + 1); | |
892 break; | |
893 case UIT_BOOLEAN: | |
894 { | |
895 const char *p; | |
896 | |
897 if (!uis->result_buf) | |
898 { | |
899 UIerr(UI_F_UI_SET_RESULT,UI_R_NO_RESULT_BUFFER); | |
900 return -1; | |
901 } | |
902 | |
903 uis->result_buf[0] = '\0'; | |
904 for(p = result; *p; p++) | |
905 { | |
906 if (strchr(uis->_.boolean_data.ok_chars, *p)) | |
907 { | |
908 uis->result_buf[0] = | |
909 uis->_.boolean_data.ok_chars[0]; | |
910 break; | |
911 } | |
912 if (strchr(uis->_.boolean_data.cancel_chars, *p)) | |
913 { | |
914 uis->result_buf[0] = | |
915 uis->_.boolean_data.cancel_chars[0]; | |
916 break; | |
917 } | |
918 } | |
919 default: | |
920 break; | |
921 } | |
922 } | |
923 return 0; | |
924 } | |
OLD | NEW |