2 * Copyright (c) 2006 Kungliga Tekniska Högskolan
3 * (Royal Institute of Technology, Stockholm, Sweden).
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
17 * 3. Neither the name of the Institute nor the names of its contributors
18 * may be used to endorse or promote products derived from this software
19 * without specific prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 #include <pkcs10_asn1.h>
36 __RCSID("$Heimdal: req.c 21344 2007-06-26 14:22:34Z lha $"
39 struct hx509_request_data
{
41 SubjectPublicKeyInfo key
;
51 _hx509_request_init(hx509_context context
, hx509_request
*req
)
53 *req
= calloc(1, sizeof(**req
));
61 _hx509_request_free(hx509_request
*req
)
64 hx509_name_free(&(*req
)->name
);
65 free_SubjectPublicKeyInfo(&(*req
)->key
);
66 free_ExtKeyUsage(&(*req
)->eku
);
67 free_GeneralNames(&(*req
)->san
);
68 memset(*req
, 0, sizeof(**req
));
74 _hx509_request_set_name(hx509_context context
,
79 hx509_name_free(&req
->name
);
81 int ret
= hx509_name_copy(context
, name
, &req
->name
);
89 _hx509_request_get_name(hx509_context context
,
93 if (req
->name
== NULL
) {
94 hx509_set_error_string(context
, 0, EINVAL
, "Request have no name");
97 return hx509_name_copy(context
, req
->name
, name
);
101 _hx509_request_set_SubjectPublicKeyInfo(hx509_context context
,
103 const SubjectPublicKeyInfo
*key
)
105 free_SubjectPublicKeyInfo(&req
->key
);
106 return copy_SubjectPublicKeyInfo(key
, &req
->key
);
110 _hx509_request_get_SubjectPublicKeyInfo(hx509_context context
,
112 SubjectPublicKeyInfo
*key
)
114 return copy_SubjectPublicKeyInfo(&req
->key
, key
);
118 _hx509_request_add_eku(hx509_context context
,
125 val
= realloc(req
->eku
.val
, sizeof(req
->eku
.val
[0]) * (req
->eku
.len
+ 1));
130 ret
= der_copy_oid(oid
, &req
->eku
.val
[req
->eku
.len
]);
140 _hx509_request_add_dns_name(hx509_context context
,
142 const char *hostname
)
146 memset(&name
, 0, sizeof(name
));
147 name
.element
= choice_GeneralName_dNSName
;
148 name
.u
.dNSName
= rk_UNCONST(hostname
);
150 return add_GeneralNames(&req
->san
, &name
);
154 _hx509_request_add_email(hx509_context context
,
160 memset(&name
, 0, sizeof(name
));
161 name
.element
= choice_GeneralName_rfc822Name
;
162 name
.u
.dNSName
= rk_UNCONST(email
);
164 return add_GeneralNames(&req
->san
, &name
);
170 _hx509_request_to_pkcs10(hx509_context context
,
171 const hx509_request req
,
172 const hx509_private_key signer
,
173 heim_octet_string
*request
)
175 CertificationRequest r
;
176 heim_octet_string data
, os
;
180 if (req
->name
== NULL
) {
181 hx509_set_error_string(context
, 0, EINVAL
,
182 "PKCS10 needs to have a subject");
186 memset(&r
, 0, sizeof(r
));
187 memset(request
, 0, sizeof(*request
));
189 r
.certificationRequestInfo
.version
= pkcs10_v1
;
191 ret
= copy_Name(&req
->name
->der_name
,
192 &r
.certificationRequestInfo
.subject
);
195 ret
= copy_SubjectPublicKeyInfo(&req
->key
,
196 &r
.certificationRequestInfo
.subjectPKInfo
);
199 r
.certificationRequestInfo
.attributes
=
200 calloc(1, sizeof(*r
.certificationRequestInfo
.attributes
));
201 if (r
.certificationRequestInfo
.attributes
== NULL
) {
206 ASN1_MALLOC_ENCODE(CertificationRequestInfo
, data
.data
, data
.length
,
207 &r
.certificationRequestInfo
, &size
, ret
);
210 if (data
.length
!= size
)
213 ret
= _hx509_create_signature(context
,
215 _hx509_crypto_default_sig_alg
,
217 &r
.signatureAlgorithm
,
222 r
.signature
.data
= os
.data
;
223 r
.signature
.length
= os
.length
* 8;
225 ASN1_MALLOC_ENCODE(CertificationRequest
, data
.data
, data
.length
,
229 if (data
.length
!= size
)
235 free_CertificationRequest(&r
);
241 _hx509_request_parse(hx509_context context
,
245 CertificationRequest r
;
246 CertificationRequestInfo
*rinfo
;
252 if (strncmp(path
, "PKCS10:", 7) != 0) {
253 hx509_set_error_string(context
, 0, HX509_UNSUPPORTED_OPERATION
,
254 "unsupport type in %s", path
);
255 return HX509_UNSUPPORTED_OPERATION
;
259 /* XXX PEM request */
261 ret
= _hx509_map_file(path
, &p
, &len
, NULL
);
263 hx509_set_error_string(context
, 0, ret
, "Failed to map file %s", path
);
267 ret
= decode_CertificationRequest(p
, len
, &r
, &size
);
268 _hx509_unmap_file(p
, len
);
270 hx509_set_error_string(context
, 0, ret
, "Failed to decode %s", path
);
274 ret
= _hx509_request_init(context
, req
);
276 free_CertificationRequest(&r
);
280 rinfo
= &r
.certificationRequestInfo
;
282 ret
= _hx509_request_set_SubjectPublicKeyInfo(context
, *req
,
283 &rinfo
->subjectPKInfo
);
285 free_CertificationRequest(&r
);
286 _hx509_request_free(req
);
290 ret
= _hx509_name_from_Name(&rinfo
->subject
, &subject
);
292 free_CertificationRequest(&r
);
293 _hx509_request_free(req
);
296 ret
= _hx509_request_set_name(context
, *req
, subject
);
297 hx509_name_free(&subject
);
298 free_CertificationRequest(&r
);
300 _hx509_request_free(req
);
309 _hx509_request_print(hx509_context context
, hx509_request req
, FILE *f
)
315 ret
= hx509_name_to_string(req
->name
, &subject
);
317 hx509_set_error_string(context
, 0, ret
, "Failed to print name");
320 fprintf(f
, "name: %s\n", subject
);