source: roaraudio/libroar/pinentry.c @ 1254:42feb2f88285

Last change on this file since 1254:42feb2f88285 was 1254:42feb2f88285, checked in by phi, 15 years ago

fixed bug in pinentry to not wait for the child

File size: 6.6 KB
Line 
1//pinentry.c:
2
3/*
4 *      Copyright (C) Philipp 'ph3-der-loewe' Schafft - 2008
5 *
6 *  This file is part of libroar a part of RoarAudio,
7 *  a cross-platform sound system for both, home and professional use.
8 *  See README for details.
9 *
10 *  This file is free software; you can redistribute it and/or modify
11 *  it under the terms of the GNU General Public License version 3
12 *  as published by the Free Software Foundation.
13 *
14 *  libroar is distributed in the hope that it will be useful,
15 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
16 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 *  GNU General Public License for more details.
18 *
19 *  You should have received a copy of the GNU General Public License
20 *  along with this software; see the file COPYING.  If not, write to
21 *  the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
22 *
23 *  NOTE for everyone want's to change something and send patches:
24 *  read README and HACKING! There a addition information on
25 *  the license of this document you need to read before you send
26 *  any patches.
27 *
28 *  NOTE for uses of non-GPL (LGPL,...) software using libesd, libartsc
29 *  or libpulse*:
30 *  The libs libroaresd, libroararts and libroarpulse link this lib
31 *  and are therefore GPL. Because of this it may be illigal to use
32 *  them with any software that uses libesd, libartsc or libpulse*.
33 */
34
35#include "libroar.h"
36
37int roar_pinentry_open (struct roar_pinentry * pe, int flags, char * display, char * tty, char * term) {
38 int in[2], out[2];
39
40 if ( pe == NULL )
41  return -1;
42
43 memset(pe, 0, sizeof(struct roar_pinentry));
44 pe->in  = -1;
45 pe->out = -1;
46
47#ifdef ROAR_HAVE_BIN_PINENTRY
48 if ( display == NULL )
49  display = getenv("DISPLAY");
50
51 if ( display == NULL )
52  display = "(NONE)";
53
54 if ( tty == NULL )
55  tty = ttyname(ROAR_STDIN);
56
57 if ( tty == NULL )
58#ifdef ROAR_DEFAULT_TTY
59  tty = ROAR_DEFAULT_TTY;
60#else
61  return -1;
62#endif
63
64 if ( term == NULL )
65  term = getenv("TERM");
66
67 if ( term == NULL )
68  term = "dumb";
69
70 // open some pipes...
71 if ( pipe(in) != 0 )
72  return -1;
73
74 if ( pipe(out) != 0 ) {
75  close(in[0]);
76  close(in[1]);
77  return -1;
78 }
79
80 pe->pid = fork();
81
82 switch (pe->pid) {
83  case -1:
84    close(in[0]);
85    close(in[1]);
86    close(out[0]);
87    close(out[1]);
88    return -1;
89   break;
90  case 0:
91    close(in[0]);
92    close(out[1]);
93    close(ROAR_STDIN);
94    close(ROAR_STDOUT);
95
96    if ( dup2(out[0], ROAR_STDIN) == -1 )
97     _exit(1);
98
99    if ( dup2(in[1], ROAR_STDOUT) == -1 )
100     _exit(1);
101
102    execlp(ROAR_HAVE_BIN_PINENTRY, "RoarAudio", "--display", display, "--ttytype", term, "--ttyname", tty, NULL);
103
104    _exit(1);
105   break;
106 }
107
108 close(in[1]);
109 close(out[0]);
110
111 pe->in  = in[0];
112 pe->out = out[1];
113
114 pe->fin = fdopen(in[0], "r");
115
116 roar_pinentry_recv(pe, NULL, NULL);
117
118 return 0;
119#else
120 return -1;
121#endif
122}
123
124int roar_pinentry_simple_open(struct roar_pinentry * pe) {
125 return roar_pinentry_open(pe, 0, NULL, NULL, NULL);
126}
127
128int roar_pinentry_close(struct roar_pinentry * pe) {
129 int status;
130
131 if ( pe == NULL )
132  return -1;
133
134 if ( pe->opened == 0 )
135  return 0;
136
137 if ( pe->out != -1 )
138  close(pe->out);
139
140 if ( pe->fin != NULL )
141  fclose(pe->fin);
142
143 if ( pe->in  != -1 )
144  close(pe->in);
145
146 waitpid(pe->pid, &status, 0);
147
148 memset(pe, 0, sizeof(struct roar_pinentry));
149
150 return 0;
151}
152
153int roar_pinentry_send (struct roar_pinentry * pe, char * cmd,  char * args) {
154 size_t len;
155
156 if ( pe == NULL )
157  return -1;
158
159 if ( cmd == NULL )
160  return -1;
161
162 len = strlen(cmd);
163
164 if ( write(pe->out, cmd, len) != len )
165  return -1;
166
167 if ( args != NULL ) {
168  len = strlen(args);
169
170  if ( write(pe->out, args, len) != len )
171   return -1;
172 }
173
174 if ( write(pe->out, "\n", 1) != 1 )
175  return -1;
176
177 return 0;
178}
179
180
181#define MAX_LINE_SIZE 2048
182int roar_pinentry_recv (struct roar_pinentry * pe, char ** line, char ** opts) {
183 char realbuf[MAX_LINE_SIZE];
184 char * tp;
185
186 if ( pe == NULL )
187  return -1;
188
189 if ( pe->fin == NULL )
190  return -1;
191
192 if ( fgets(realbuf, MAX_LINE_SIZE, pe->fin) == NULL )
193  return -1;
194
195 tp = realbuf + strlen(realbuf) - 1;
196
197 for (; *tp == '\r' || *tp == '\n'; tp--)
198  *tp = 0;
199
200 if ( (tp = strstr(realbuf, " ")) == NULL ) {
201  if ( line != NULL )
202   *line = strdup(realbuf);
203
204  if ( opts != NULL )
205   *opts = NULL;
206
207  if ( !strcmp(realbuf, "OK") ) {
208   return 0;
209  } else if ( !strcmp(realbuf, "ERR") ) {
210   return 1;
211  } else {
212   return -1;
213  }
214 } else {
215  *tp = 0;
216
217  if ( !strcmp(realbuf, "D") ) {
218   if ( opts != NULL )
219    *opts = strdup(tp+1);
220
221   return roar_pinentry_recv(pe, line, NULL);
222  }
223
224  if ( line != NULL )
225   *line = strdup(realbuf);
226
227  if ( opts != NULL )
228   *opts = NULL;
229
230  if ( !strcmp(realbuf, "OK") ) {
231   return 0;
232  } else if ( !strcmp(realbuf, "ERR") ) {
233   return 1;
234  } else {
235   return -1;
236  }
237 }
238
239 return -1;
240}
241
242int roar_pinentry_req  (struct roar_pinentry * pe, char * cmd,  char * args, char ** line, char ** opts) {
243 if ( pe == NULL )
244  return -1;
245
246 if ( roar_pinentry_send(pe, cmd, args) != 0 )
247  return -1;
248
249 return roar_pinentry_recv(pe, line, opts);
250}
251
252int roar_pinentry_set_desc (struct roar_pinentry * pe, char * desc) {
253 return roar_pinentry_set(pe, "DESC", desc);
254}
255
256int roar_pinentry_set_prompt(struct roar_pinentry * pe, char * prompt) {
257 return roar_pinentry_set(pe, "PROMPT", prompt);
258}
259
260int roar_pinentry_set_yes  (struct roar_pinentry * pe, char * yes) {
261 return roar_pinentry_set(pe, "OK", yes);
262}
263
264int roar_pinentry_set_no   (struct roar_pinentry * pe, char * no) {
265 return roar_pinentry_set(pe, "CANCEL", no);
266}
267
268int roar_pinentry_set      (struct roar_pinentry * pe, char * obj, char * text) {
269 char req[80] = "SET";
270
271 if ( pe == NULL )
272  return -1;
273
274 if ( obj == NULL )
275  return -1;
276
277 if ( strlen(obj) > (80-1 /* \0 */ + 3 /* "SET" */ + 1 /* " " */) )
278  return -1;
279
280 strncat(req, obj, 80-5);
281 strncat(req, " ", 2);
282
283 return roar_pinentry_req(pe, req, text, NULL, NULL);
284}
285
286int roar_pinentry_getpin   (struct roar_pinentry * pe, char ** pw, char * desc, char * prompt) {
287 if ( pe == NULL )
288  return -1;
289
290 if ( pw == NULL )
291  return -1;
292
293 if ( desc != NULL )
294  if ( roar_pinentry_set_desc(pe, desc) != 0 )
295   return -1;
296
297 if ( prompt != NULL )
298  if ( roar_pinentry_set_prompt(pe, prompt) != 0 )
299   return -1;
300
301 if ( roar_pinentry_req(pe, "GETPIN", NULL, NULL, pw) == -1 )
302  return -1;
303
304 return 0;
305}
306
307int roar_pinentry_confirm  (struct roar_pinentry * pe, char * desc, char * yes, char * no) {
308 if ( pe == NULL )
309  return -1;
310
311 if ( desc != NULL )
312  if ( roar_pinentry_set_desc(pe, desc) != 0 )
313   return -1;
314
315 if ( yes != NULL )
316  if ( roar_pinentry_set_yes(pe, yes) != 0 )
317   return -1;
318
319 if ( no != NULL )
320  if ( roar_pinentry_set_no(pe, no) != 0 )
321   return -1;
322
323 return roar_pinentry_req(pe, "CONFIRM", NULL, NULL, NULL);
324}
325
326//ll
Note: See TracBrowser for help on using the repository browser.