Blob


1 #include <u.h>
2 #include <libc.h>
3 #include <bio.h>
4 #include <auth.h>
5 #include <mp.h>
6 #include <libsec.h>
7 #include "rsa2any.h"
9 RSApriv*
10 getkey(int argc, char **argv, int needprivate, Attr **pa)
11 {
12 char *file, *s, *p;
13 int sz;
14 RSApriv *key;
15 Biobuf *b;
16 int regen;
17 Attr *a;
19 if(argc == 0)
20 file = "/dev/stdin";
21 else
22 file = argv[0];
24 key = mallocz(sizeof(RSApriv), 1);
25 if(key == nil)
26 return nil;
28 if((b = Bopen(file, OREAD)) == nil){
29 werrstr("open %s: %r", file);
30 return nil;
31 }
32 s = Brdstr(b, '\n', 1);
33 if(s == nil){
34 werrstr("read %s: %r", file);
35 return nil;
36 }
37 if(strncmp(s, "key ", 4) != 0){
38 werrstr("bad key format");
39 return nil;
40 }
42 regen = 0;
43 a = _parseattr(s+4);
44 if(a == nil){
45 werrstr("empty key");
46 return nil;
47 }
48 if((p = _strfindattr(a, "proto")) == nil){
49 werrstr("no proto");
50 return nil;
51 }
52 if(strcmp(p, "rsa") != 0){
53 werrstr("proto not rsa");
54 return nil;
55 }
56 if((p = _strfindattr(a, "ek")) == nil){
57 werrstr("no ek");
58 return nil;
59 }
60 if((key->pub.ek = strtomp(p, &p, 16, nil)) == nil || *p != 0){
61 werrstr("bad ek");
62 return nil;
63 }
64 if((p = _strfindattr(a, "n")) == nil){
65 werrstr("no n");
66 return nil;
67 }
68 if((key->pub.n = strtomp(p, &p, 16, nil)) == nil || *p != 0){
69 werrstr("bad n");
70 return nil;
71 }
72 if((p = _strfindattr(a, "size")) == nil)
73 fprint(2, "warning: missing size; will add\n");
74 else if((sz = strtol(p, &p, 10)) == 0 || *p != 0)
75 fprint(2, "warning: bad size; will correct\n");
76 else if(sz != mpsignif(key->pub.n))
77 fprint(2, "warning: wrong size (got %d, expected %d); will correct\n",
78 sz, mpsignif(key->pub.n));
79 if(!needprivate)
80 goto call;
81 if((p = _strfindattr(a, "!dk")) == nil){
82 werrstr("no !dk");
83 return nil;
84 }
85 if((key->dk = strtomp(p, &p, 16, nil)) == nil || *p != 0){
86 werrstr("bad !dk");
87 return nil;
88 }
89 if((p = _strfindattr(a, "!p")) == nil){
90 werrstr("no !p");
91 return nil;
92 }
93 if((key->p = strtomp(p, &p, 16, nil)) == nil || *p != 0){
94 werrstr("bad !p");
95 return nil;
96 }
97 if((p = _strfindattr(a, "!q")) == nil){
98 werrstr("no !q");
99 return nil;
101 if((key->q = strtomp(p, &p, 16, nil)) == nil || *p != 0){
102 werrstr("bad !q");
103 return nil;
105 if((p = _strfindattr(a, "!kp")) == nil){
106 fprint(2, "warning: no !kp\n");
107 regen = 1;
108 goto regen;
110 if((key->kp = strtomp(p, &p, 16, nil)) == nil || *p != 0){
111 fprint(2, "warning: bad !kp\n");
112 regen = 1;
113 goto regen;
115 if((p = _strfindattr(a, "!kq")) == nil){
116 fprint(2, "warning: no !kq\n");
117 regen = 1;
118 goto regen;
120 if((key->kq = strtomp(p, &p, 16, nil)) == nil || *p != 0){
121 fprint(2, "warning: bad !kq\n");
122 regen = 1;
123 goto regen;
125 if((p = _strfindattr(a, "!c2")) == nil){
126 fprint(2, "warning: no !c2\n");
127 regen = 1;
128 goto regen;
130 if((key->c2 = strtomp(p, &p, 16, nil)) == nil || *p != 0){
131 fprint(2, "warning: bad !c2\n");
132 regen = 1;
133 goto regen;
135 regen:
136 if(regen){
137 RSApriv *k2;
139 k2 = rsafill(key->pub.n, key->pub.ek, key->dk, key->p, key->q);
140 if(k2 == nil){
141 werrstr("regenerating chinese-remainder parts failed: %r");
142 return nil;
144 key = k2;
146 call:
147 a = _delattr(a, "ek");
148 a = _delattr(a, "n");
149 a = _delattr(a, "size");
150 a = _delattr(a, "!dk");
151 a = _delattr(a, "!p");
152 a = _delattr(a, "!q");
153 a = _delattr(a, "!c2");
154 a = _delattr(a, "!kp");
155 a = _delattr(a, "!kq");
156 if(pa)
157 *pa = a;
158 return key;
161 DSApriv*
162 getdsakey(int argc, char **argv, int needprivate, Attr **pa)
164 char *file, *s, *p;
165 DSApriv *key;
166 Biobuf *b;
167 int regen;
168 Attr *a;
170 if(argc == 0)
171 file = "/dev/stdin";
172 else
173 file = argv[0];
175 key = mallocz(sizeof(RSApriv), 1);
176 if(key == nil)
177 return nil;
179 if((b = Bopen(file, OREAD)) == nil){
180 werrstr("open %s: %r", file);
181 return nil;
183 s = Brdstr(b, '\n', 1);
184 if(s == nil){
185 werrstr("read %s: %r", file);
186 return nil;
188 if(strncmp(s, "key ", 4) != 0){
189 werrstr("bad key format");
190 return nil;
193 regen = 0;
194 a = _parseattr(s+4);
195 if(a == nil){
196 werrstr("empty key");
197 return nil;
199 if((p = _strfindattr(a, "proto")) == nil){
200 werrstr("no proto");
201 return nil;
203 if(strcmp(p, "dsa") != 0){
204 werrstr("proto not dsa");
205 return nil;
207 if((p = _strfindattr(a, "p")) == nil){
208 werrstr("no p");
209 return nil;
211 if((key->pub.p = strtomp(p, &p, 16, nil)) == nil || *p != 0){
212 werrstr("bad p");
213 return nil;
215 if((p = _strfindattr(a, "q")) == nil){
216 werrstr("no q");
217 return nil;
219 if((key->pub.q = strtomp(p, &p, 16, nil)) == nil || *p != 0){
220 werrstr("bad q");
221 return nil;
223 if((p = _strfindattr(a, "alpha")) == nil){
224 werrstr("no alpha");
225 return nil;
227 if((key->pub.alpha = strtomp(p, &p, 16, nil)) == nil || *p != 0){
228 werrstr("bad alpha");
229 return nil;
231 if((p = _strfindattr(a, "key")) == nil){
232 werrstr("no key=");
233 return nil;
235 if((key->pub.key = strtomp(p, &p, 16, nil)) == nil || *p != 0){
236 werrstr("bad key=");
237 return nil;
239 if(!needprivate)
240 goto call;
241 if((p = _strfindattr(a, "!secret")) == nil){
242 werrstr("no !secret");
243 return nil;
245 if((key->secret = strtomp(p, &p, 16, nil)) == nil || *p != 0){
246 werrstr("bad !secret");
247 return nil;
249 call:
250 a = _delattr(a, "p");
251 a = _delattr(a, "q");
252 a = _delattr(a, "alpha");
253 a = _delattr(a, "key");
254 a = _delattr(a, "!secret");
255 if(pa)
256 *pa = a;
257 return key;
260 uchar*
261 put4(uchar *p, uint n)
263 p[0] = (n>>24)&0xFF;
264 p[1] = (n>>16)&0xFF;
265 p[2] = (n>>8)&0xFF;
266 p[3] = n&0xFF;
267 return p+4;
270 uchar*
271 putn(uchar *p, void *v, uint n)
273 memmove(p, v, n);
274 p += n;
275 return p;
278 uchar*
279 putstr(uchar *p, char *s)
281 p = put4(p, strlen(s));
282 p = putn(p, s, strlen(s));
283 return p;
286 uchar*
287 putmp2(uchar *p, mpint *b)
289 int bits, n;
291 if(mpcmp(b, mpzero) == 0)
292 return put4(p, 0);
293 bits = mpsignif(b);
294 n = (bits+7)/8;
295 if(bits%8 == 0){
296 p = put4(p, n+1);
297 *p++ = 0;
298 }else
299 p = put4(p, n);
300 mptobe(b, p, n, nil);
301 p += n;
302 return p;