1*00b67f09SDavid van Moolenbroek /* $NetBSD: opensslgost_link.c,v 1.10 2015/07/08 17:28:58 christos Exp $ */
2*00b67f09SDavid van Moolenbroek
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek * Copyright (C) 2010-2014 Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek *
6*00b67f09SDavid van Moolenbroek * Permission to use, copy, modify, and/or distribute this software for any
7*00b67f09SDavid van Moolenbroek * purpose with or without fee is hereby granted, provided that the above
8*00b67f09SDavid van Moolenbroek * copyright notice and this permission notice appear in all copies.
9*00b67f09SDavid van Moolenbroek *
10*00b67f09SDavid van Moolenbroek * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
11*00b67f09SDavid van Moolenbroek * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
12*00b67f09SDavid van Moolenbroek * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
13*00b67f09SDavid van Moolenbroek * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
14*00b67f09SDavid van Moolenbroek * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
15*00b67f09SDavid van Moolenbroek * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
16*00b67f09SDavid van Moolenbroek * PERFORMANCE OF THIS SOFTWARE.
17*00b67f09SDavid van Moolenbroek */
18*00b67f09SDavid van Moolenbroek
19*00b67f09SDavid van Moolenbroek /* Id: opensslgost_link.c,v 1.5 2011/01/19 23:47:12 tbox Exp */
20*00b67f09SDavid van Moolenbroek
21*00b67f09SDavid van Moolenbroek #include <config.h>
22*00b67f09SDavid van Moolenbroek
23*00b67f09SDavid van Moolenbroek #if defined(OPENSSL) && defined(HAVE_OPENSSL_GOST)
24*00b67f09SDavid van Moolenbroek
25*00b67f09SDavid van Moolenbroek #include <isc/entropy.h>
26*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
27*00b67f09SDavid van Moolenbroek #include <isc/string.h>
28*00b67f09SDavid van Moolenbroek #include <isc/util.h>
29*00b67f09SDavid van Moolenbroek
30*00b67f09SDavid van Moolenbroek #include <dst/result.h>
31*00b67f09SDavid van Moolenbroek
32*00b67f09SDavid van Moolenbroek #include "dst_internal.h"
33*00b67f09SDavid van Moolenbroek #include "dst_openssl.h"
34*00b67f09SDavid van Moolenbroek #include "dst_parse.h"
35*00b67f09SDavid van Moolenbroek #include "dst_gost.h"
36*00b67f09SDavid van Moolenbroek
37*00b67f09SDavid van Moolenbroek #include <openssl/err.h>
38*00b67f09SDavid van Moolenbroek #include <openssl/objects.h>
39*00b67f09SDavid van Moolenbroek #include <openssl/rsa.h>
40*00b67f09SDavid van Moolenbroek #include <openssl/engine.h>
41*00b67f09SDavid van Moolenbroek
42*00b67f09SDavid van Moolenbroek static ENGINE *e = NULL;
43*00b67f09SDavid van Moolenbroek static const EVP_MD *opensslgost_digest;
44*00b67f09SDavid van Moolenbroek extern const EVP_MD *EVP_gost(void);
45*00b67f09SDavid van Moolenbroek
EVP_gost(void)46*00b67f09SDavid van Moolenbroek const EVP_MD *EVP_gost(void) {
47*00b67f09SDavid van Moolenbroek return (opensslgost_digest);
48*00b67f09SDavid van Moolenbroek }
49*00b67f09SDavid van Moolenbroek
50*00b67f09SDavid van Moolenbroek /* ISC methods */
51*00b67f09SDavid van Moolenbroek
52*00b67f09SDavid van Moolenbroek isc_result_t
isc_gost_init(isc_gost_t * ctx)53*00b67f09SDavid van Moolenbroek isc_gost_init(isc_gost_t *ctx) {
54*00b67f09SDavid van Moolenbroek const EVP_MD *md;
55*00b67f09SDavid van Moolenbroek int ret;
56*00b67f09SDavid van Moolenbroek
57*00b67f09SDavid van Moolenbroek INSIST(ctx != NULL);
58*00b67f09SDavid van Moolenbroek
59*00b67f09SDavid van Moolenbroek md = EVP_gost();
60*00b67f09SDavid van Moolenbroek if (md == NULL)
61*00b67f09SDavid van Moolenbroek return (DST_R_CRYPTOFAILURE);
62*00b67f09SDavid van Moolenbroek EVP_MD_CTX_init(ctx);
63*00b67f09SDavid van Moolenbroek ret = EVP_DigestInit(ctx, md);
64*00b67f09SDavid van Moolenbroek if (ret != 1)
65*00b67f09SDavid van Moolenbroek return (DST_R_CRYPTOFAILURE);
66*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
67*00b67f09SDavid van Moolenbroek }
68*00b67f09SDavid van Moolenbroek
69*00b67f09SDavid van Moolenbroek void
isc_gost_invalidate(isc_gost_t * ctx)70*00b67f09SDavid van Moolenbroek isc_gost_invalidate(isc_gost_t *ctx) {
71*00b67f09SDavid van Moolenbroek EVP_MD_CTX_cleanup(ctx);
72*00b67f09SDavid van Moolenbroek }
73*00b67f09SDavid van Moolenbroek
74*00b67f09SDavid van Moolenbroek isc_result_t
isc_gost_update(isc_gost_t * ctx,const unsigned char * data,unsigned int len)75*00b67f09SDavid van Moolenbroek isc_gost_update(isc_gost_t *ctx, const unsigned char *data,
76*00b67f09SDavid van Moolenbroek unsigned int len)
77*00b67f09SDavid van Moolenbroek {
78*00b67f09SDavid van Moolenbroek int ret;
79*00b67f09SDavid van Moolenbroek
80*00b67f09SDavid van Moolenbroek INSIST(ctx != NULL);
81*00b67f09SDavid van Moolenbroek INSIST(data != NULL);
82*00b67f09SDavid van Moolenbroek
83*00b67f09SDavid van Moolenbroek ret = EVP_DigestUpdate(ctx, (const void *) data, (size_t) len);
84*00b67f09SDavid van Moolenbroek if (ret != 1)
85*00b67f09SDavid van Moolenbroek return (DST_R_CRYPTOFAILURE);
86*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
87*00b67f09SDavid van Moolenbroek }
88*00b67f09SDavid van Moolenbroek
89*00b67f09SDavid van Moolenbroek isc_result_t
isc_gost_final(isc_gost_t * ctx,unsigned char * digest)90*00b67f09SDavid van Moolenbroek isc_gost_final(isc_gost_t *ctx, unsigned char *digest) {
91*00b67f09SDavid van Moolenbroek int ret;
92*00b67f09SDavid van Moolenbroek
93*00b67f09SDavid van Moolenbroek INSIST(ctx != NULL);
94*00b67f09SDavid van Moolenbroek INSIST(digest != NULL);
95*00b67f09SDavid van Moolenbroek
96*00b67f09SDavid van Moolenbroek ret = EVP_DigestFinal(ctx, digest, NULL);
97*00b67f09SDavid van Moolenbroek if (ret != 1)
98*00b67f09SDavid van Moolenbroek return (DST_R_CRYPTOFAILURE);
99*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
100*00b67f09SDavid van Moolenbroek }
101*00b67f09SDavid van Moolenbroek
102*00b67f09SDavid van Moolenbroek /* DST methods */
103*00b67f09SDavid van Moolenbroek
104*00b67f09SDavid van Moolenbroek #define DST_RET(a) {ret = a; goto err;}
105*00b67f09SDavid van Moolenbroek
106*00b67f09SDavid van Moolenbroek static isc_result_t opensslgost_todns(const dst_key_t *key,
107*00b67f09SDavid van Moolenbroek isc_buffer_t *data);
108*00b67f09SDavid van Moolenbroek
109*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_createctx(dst_key_t * key,dst_context_t * dctx)110*00b67f09SDavid van Moolenbroek opensslgost_createctx(dst_key_t *key, dst_context_t *dctx) {
111*00b67f09SDavid van Moolenbroek EVP_MD_CTX *evp_md_ctx;
112*00b67f09SDavid van Moolenbroek const EVP_MD *md = EVP_gost();
113*00b67f09SDavid van Moolenbroek
114*00b67f09SDavid van Moolenbroek UNUSED(key);
115*00b67f09SDavid van Moolenbroek
116*00b67f09SDavid van Moolenbroek if (md == NULL)
117*00b67f09SDavid van Moolenbroek return (DST_R_OPENSSLFAILURE);
118*00b67f09SDavid van Moolenbroek
119*00b67f09SDavid van Moolenbroek evp_md_ctx = EVP_MD_CTX_create();
120*00b67f09SDavid van Moolenbroek if (evp_md_ctx == NULL)
121*00b67f09SDavid van Moolenbroek return (ISC_R_NOMEMORY);
122*00b67f09SDavid van Moolenbroek
123*00b67f09SDavid van Moolenbroek if (!EVP_DigestInit_ex(evp_md_ctx, md, NULL)) {
124*00b67f09SDavid van Moolenbroek EVP_MD_CTX_destroy(evp_md_ctx);
125*00b67f09SDavid van Moolenbroek return (ISC_R_FAILURE);
126*00b67f09SDavid van Moolenbroek }
127*00b67f09SDavid van Moolenbroek dctx->ctxdata.evp_md_ctx = evp_md_ctx;
128*00b67f09SDavid van Moolenbroek
129*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
130*00b67f09SDavid van Moolenbroek }
131*00b67f09SDavid van Moolenbroek
132*00b67f09SDavid van Moolenbroek static void
opensslgost_destroyctx(dst_context_t * dctx)133*00b67f09SDavid van Moolenbroek opensslgost_destroyctx(dst_context_t *dctx) {
134*00b67f09SDavid van Moolenbroek EVP_MD_CTX *evp_md_ctx = dctx->ctxdata.evp_md_ctx;
135*00b67f09SDavid van Moolenbroek
136*00b67f09SDavid van Moolenbroek if (evp_md_ctx != NULL) {
137*00b67f09SDavid van Moolenbroek EVP_MD_CTX_destroy(evp_md_ctx);
138*00b67f09SDavid van Moolenbroek dctx->ctxdata.evp_md_ctx = NULL;
139*00b67f09SDavid van Moolenbroek }
140*00b67f09SDavid van Moolenbroek }
141*00b67f09SDavid van Moolenbroek
142*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_adddata(dst_context_t * dctx,const isc_region_t * data)143*00b67f09SDavid van Moolenbroek opensslgost_adddata(dst_context_t *dctx, const isc_region_t *data) {
144*00b67f09SDavid van Moolenbroek EVP_MD_CTX *evp_md_ctx = dctx->ctxdata.evp_md_ctx;
145*00b67f09SDavid van Moolenbroek
146*00b67f09SDavid van Moolenbroek if (!EVP_DigestUpdate(evp_md_ctx, data->base, data->length))
147*00b67f09SDavid van Moolenbroek return (ISC_R_FAILURE);
148*00b67f09SDavid van Moolenbroek
149*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
150*00b67f09SDavid van Moolenbroek }
151*00b67f09SDavid van Moolenbroek
152*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_sign(dst_context_t * dctx,isc_buffer_t * sig)153*00b67f09SDavid van Moolenbroek opensslgost_sign(dst_context_t *dctx, isc_buffer_t *sig) {
154*00b67f09SDavid van Moolenbroek dst_key_t *key = dctx->key;
155*00b67f09SDavid van Moolenbroek isc_region_t r;
156*00b67f09SDavid van Moolenbroek unsigned int siglen = 0;
157*00b67f09SDavid van Moolenbroek EVP_MD_CTX *evp_md_ctx = dctx->ctxdata.evp_md_ctx;
158*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = key->keydata.pkey;
159*00b67f09SDavid van Moolenbroek
160*00b67f09SDavid van Moolenbroek isc_buffer_availableregion(sig, &r);
161*00b67f09SDavid van Moolenbroek
162*00b67f09SDavid van Moolenbroek if (r.length < (unsigned int) EVP_PKEY_size(pkey))
163*00b67f09SDavid van Moolenbroek return (ISC_R_NOSPACE);
164*00b67f09SDavid van Moolenbroek
165*00b67f09SDavid van Moolenbroek if (!EVP_SignFinal(evp_md_ctx, r.base, &siglen, pkey))
166*00b67f09SDavid van Moolenbroek return (ISC_R_FAILURE);
167*00b67f09SDavid van Moolenbroek
168*00b67f09SDavid van Moolenbroek isc_buffer_add(sig, siglen);
169*00b67f09SDavid van Moolenbroek
170*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
171*00b67f09SDavid van Moolenbroek }
172*00b67f09SDavid van Moolenbroek
173*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_verify(dst_context_t * dctx,const isc_region_t * sig)174*00b67f09SDavid van Moolenbroek opensslgost_verify(dst_context_t *dctx, const isc_region_t *sig) {
175*00b67f09SDavid van Moolenbroek dst_key_t *key = dctx->key;
176*00b67f09SDavid van Moolenbroek int status = 0;
177*00b67f09SDavid van Moolenbroek EVP_MD_CTX *evp_md_ctx = dctx->ctxdata.evp_md_ctx;
178*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = key->keydata.pkey;
179*00b67f09SDavid van Moolenbroek
180*00b67f09SDavid van Moolenbroek status = EVP_VerifyFinal(evp_md_ctx, sig->base, sig->length, pkey);
181*00b67f09SDavid van Moolenbroek switch (status) {
182*00b67f09SDavid van Moolenbroek case 1:
183*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
184*00b67f09SDavid van Moolenbroek case 0:
185*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult(DST_R_VERIFYFAILURE));
186*00b67f09SDavid van Moolenbroek default:
187*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult3(dctx->category,
188*00b67f09SDavid van Moolenbroek "EVP_VerifyFinal",
189*00b67f09SDavid van Moolenbroek DST_R_VERIFYFAILURE));
190*00b67f09SDavid van Moolenbroek }
191*00b67f09SDavid van Moolenbroek }
192*00b67f09SDavid van Moolenbroek
193*00b67f09SDavid van Moolenbroek static isc_boolean_t
opensslgost_compare(const dst_key_t * key1,const dst_key_t * key2)194*00b67f09SDavid van Moolenbroek opensslgost_compare(const dst_key_t *key1, const dst_key_t *key2) {
195*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey1, *pkey2;
196*00b67f09SDavid van Moolenbroek
197*00b67f09SDavid van Moolenbroek pkey1 = key1->keydata.pkey;
198*00b67f09SDavid van Moolenbroek pkey2 = key2->keydata.pkey;
199*00b67f09SDavid van Moolenbroek
200*00b67f09SDavid van Moolenbroek if (pkey1 == NULL && pkey2 == NULL)
201*00b67f09SDavid van Moolenbroek return (ISC_TRUE);
202*00b67f09SDavid van Moolenbroek else if (pkey1 == NULL || pkey2 == NULL)
203*00b67f09SDavid van Moolenbroek return (ISC_FALSE);
204*00b67f09SDavid van Moolenbroek
205*00b67f09SDavid van Moolenbroek if (EVP_PKEY_cmp(pkey1, pkey2) != 1)
206*00b67f09SDavid van Moolenbroek return (ISC_FALSE);
207*00b67f09SDavid van Moolenbroek return (ISC_TRUE);
208*00b67f09SDavid van Moolenbroek }
209*00b67f09SDavid van Moolenbroek
210*00b67f09SDavid van Moolenbroek static int
progress_cb(EVP_PKEY_CTX * ctx)211*00b67f09SDavid van Moolenbroek progress_cb(EVP_PKEY_CTX *ctx)
212*00b67f09SDavid van Moolenbroek {
213*00b67f09SDavid van Moolenbroek union {
214*00b67f09SDavid van Moolenbroek void *dptr;
215*00b67f09SDavid van Moolenbroek void (*fptr)(int);
216*00b67f09SDavid van Moolenbroek } u;
217*00b67f09SDavid van Moolenbroek int p;
218*00b67f09SDavid van Moolenbroek
219*00b67f09SDavid van Moolenbroek u.dptr = EVP_PKEY_CTX_get_app_data(ctx);
220*00b67f09SDavid van Moolenbroek p = EVP_PKEY_CTX_get_keygen_info(ctx, 0);
221*00b67f09SDavid van Moolenbroek if (u.fptr != NULL)
222*00b67f09SDavid van Moolenbroek u.fptr(p);
223*00b67f09SDavid van Moolenbroek return (1);
224*00b67f09SDavid van Moolenbroek }
225*00b67f09SDavid van Moolenbroek
226*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_generate(dst_key_t * key,int unused,void (* callback)(int))227*00b67f09SDavid van Moolenbroek opensslgost_generate(dst_key_t *key, int unused, void (*callback)(int)) {
228*00b67f09SDavid van Moolenbroek EVP_PKEY_CTX *ctx;
229*00b67f09SDavid van Moolenbroek union {
230*00b67f09SDavid van Moolenbroek void *dptr;
231*00b67f09SDavid van Moolenbroek void (*fptr)(int);
232*00b67f09SDavid van Moolenbroek } u;
233*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = NULL;
234*00b67f09SDavid van Moolenbroek isc_result_t ret;
235*00b67f09SDavid van Moolenbroek
236*00b67f09SDavid van Moolenbroek UNUSED(unused);
237*00b67f09SDavid van Moolenbroek ctx = EVP_PKEY_CTX_new_id(NID_id_GostR3410_2001, NULL);
238*00b67f09SDavid van Moolenbroek if (ctx == NULL)
239*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2("EVP_PKEY_CTX_new_id",
240*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
241*00b67f09SDavid van Moolenbroek if (callback != NULL) {
242*00b67f09SDavid van Moolenbroek u.fptr = callback;
243*00b67f09SDavid van Moolenbroek EVP_PKEY_CTX_set_app_data(ctx, u.dptr);
244*00b67f09SDavid van Moolenbroek EVP_PKEY_CTX_set_cb(ctx, &progress_cb);
245*00b67f09SDavid van Moolenbroek }
246*00b67f09SDavid van Moolenbroek if (EVP_PKEY_keygen_init(ctx) <= 0)
247*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2("EVP_PKEY_keygen_init",
248*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
249*00b67f09SDavid van Moolenbroek if (EVP_PKEY_CTX_ctrl_str(ctx, "paramset", "A") <= 0)
250*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2("EVP_PKEY_CTX_ctrl_str",
251*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
252*00b67f09SDavid van Moolenbroek if (EVP_PKEY_keygen(ctx, &pkey) <= 0)
253*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2("EVP_PKEY_keygen",
254*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
255*00b67f09SDavid van Moolenbroek key->keydata.pkey = pkey;
256*00b67f09SDavid van Moolenbroek key->key_size = EVP_PKEY_bits(pkey);
257*00b67f09SDavid van Moolenbroek EVP_PKEY_CTX_free(ctx);
258*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
259*00b67f09SDavid van Moolenbroek
260*00b67f09SDavid van Moolenbroek err:
261*00b67f09SDavid van Moolenbroek if (pkey != NULL)
262*00b67f09SDavid van Moolenbroek EVP_PKEY_free(pkey);
263*00b67f09SDavid van Moolenbroek if (ctx != NULL)
264*00b67f09SDavid van Moolenbroek EVP_PKEY_CTX_free(ctx);
265*00b67f09SDavid van Moolenbroek return (ret);
266*00b67f09SDavid van Moolenbroek }
267*00b67f09SDavid van Moolenbroek
268*00b67f09SDavid van Moolenbroek static isc_boolean_t
opensslgost_isprivate(const dst_key_t * key)269*00b67f09SDavid van Moolenbroek opensslgost_isprivate(const dst_key_t *key) {
270*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = key->keydata.pkey;
271*00b67f09SDavid van Moolenbroek EC_KEY *ec;
272*00b67f09SDavid van Moolenbroek
273*00b67f09SDavid van Moolenbroek INSIST(pkey != NULL);
274*00b67f09SDavid van Moolenbroek
275*00b67f09SDavid van Moolenbroek ec = EVP_PKEY_get0(pkey);
276*00b67f09SDavid van Moolenbroek return (ISC_TF(ec != NULL && EC_KEY_get0_private_key(ec) != NULL));
277*00b67f09SDavid van Moolenbroek }
278*00b67f09SDavid van Moolenbroek
279*00b67f09SDavid van Moolenbroek static void
opensslgost_destroy(dst_key_t * key)280*00b67f09SDavid van Moolenbroek opensslgost_destroy(dst_key_t *key) {
281*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = key->keydata.pkey;
282*00b67f09SDavid van Moolenbroek
283*00b67f09SDavid van Moolenbroek EVP_PKEY_free(pkey);
284*00b67f09SDavid van Moolenbroek key->keydata.pkey = NULL;
285*00b67f09SDavid van Moolenbroek }
286*00b67f09SDavid van Moolenbroek
287*00b67f09SDavid van Moolenbroek unsigned char gost_prefix[37] = {
288*00b67f09SDavid van Moolenbroek 0x30, 0x63, 0x30, 0x1c, 0x06, 0x06, 0x2a, 0x85,
289*00b67f09SDavid van Moolenbroek 0x03, 0x02, 0x02, 0x13, 0x30, 0x12, 0x06, 0x07,
290*00b67f09SDavid van Moolenbroek 0x2a, 0x85, 0x03, 0x02, 0x02, 0x23, 0x01, 0x06,
291*00b67f09SDavid van Moolenbroek 0x07, 0x2a, 0x85, 0x03, 0x02, 0x02, 0x1e, 0x01,
292*00b67f09SDavid van Moolenbroek 0x03, 0x43, 0x00, 0x04, 0x40
293*00b67f09SDavid van Moolenbroek };
294*00b67f09SDavid van Moolenbroek
295*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_todns(const dst_key_t * key,isc_buffer_t * data)296*00b67f09SDavid van Moolenbroek opensslgost_todns(const dst_key_t *key, isc_buffer_t *data) {
297*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey;
298*00b67f09SDavid van Moolenbroek isc_region_t r;
299*00b67f09SDavid van Moolenbroek unsigned char der[37 + 64], *p;
300*00b67f09SDavid van Moolenbroek int len;
301*00b67f09SDavid van Moolenbroek
302*00b67f09SDavid van Moolenbroek REQUIRE(key->keydata.pkey != NULL);
303*00b67f09SDavid van Moolenbroek
304*00b67f09SDavid van Moolenbroek pkey = key->keydata.pkey;
305*00b67f09SDavid van Moolenbroek
306*00b67f09SDavid van Moolenbroek isc_buffer_availableregion(data, &r);
307*00b67f09SDavid van Moolenbroek if (r.length < 64)
308*00b67f09SDavid van Moolenbroek return (ISC_R_NOSPACE);
309*00b67f09SDavid van Moolenbroek
310*00b67f09SDavid van Moolenbroek p = der;
311*00b67f09SDavid van Moolenbroek len = i2d_PUBKEY(pkey, &p);
312*00b67f09SDavid van Moolenbroek INSIST(len == sizeof(der));
313*00b67f09SDavid van Moolenbroek INSIST(memcmp(gost_prefix, der, 37) == 0);
314*00b67f09SDavid van Moolenbroek memmove(r.base, der + 37, 64);
315*00b67f09SDavid van Moolenbroek isc_buffer_add(data, 64);
316*00b67f09SDavid van Moolenbroek
317*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
318*00b67f09SDavid van Moolenbroek }
319*00b67f09SDavid van Moolenbroek
320*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_fromdns(dst_key_t * key,isc_buffer_t * data)321*00b67f09SDavid van Moolenbroek opensslgost_fromdns(dst_key_t *key, isc_buffer_t *data) {
322*00b67f09SDavid van Moolenbroek isc_region_t r;
323*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = NULL;
324*00b67f09SDavid van Moolenbroek unsigned char der[37 + 64];
325*00b67f09SDavid van Moolenbroek const unsigned char *p;
326*00b67f09SDavid van Moolenbroek
327*00b67f09SDavid van Moolenbroek isc_buffer_remainingregion(data, &r);
328*00b67f09SDavid van Moolenbroek if (r.length == 0)
329*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
330*00b67f09SDavid van Moolenbroek
331*00b67f09SDavid van Moolenbroek if (r.length != 64)
332*00b67f09SDavid van Moolenbroek return (DST_R_INVALIDPUBLICKEY);
333*00b67f09SDavid van Moolenbroek memmove(der, gost_prefix, 37);
334*00b67f09SDavid van Moolenbroek memmove(der + 37, r.base, 64);
335*00b67f09SDavid van Moolenbroek isc_buffer_forward(data, 64);
336*00b67f09SDavid van Moolenbroek
337*00b67f09SDavid van Moolenbroek p = der;
338*00b67f09SDavid van Moolenbroek if (d2i_PUBKEY(&pkey, &p, (long) sizeof(der)) == NULL)
339*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult2("d2i_PUBKEY",
340*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
341*00b67f09SDavid van Moolenbroek key->keydata.pkey = pkey;
342*00b67f09SDavid van Moolenbroek key->key_size = EVP_PKEY_bits(pkey);
343*00b67f09SDavid van Moolenbroek
344*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
345*00b67f09SDavid van Moolenbroek }
346*00b67f09SDavid van Moolenbroek
347*00b67f09SDavid van Moolenbroek #ifdef PREFER_GOSTASN1
348*00b67f09SDavid van Moolenbroek
349*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_tofile(const dst_key_t * key,const char * directory)350*00b67f09SDavid van Moolenbroek opensslgost_tofile(const dst_key_t *key, const char *directory) {
351*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey;
352*00b67f09SDavid van Moolenbroek dst_private_t priv;
353*00b67f09SDavid van Moolenbroek isc_result_t result;
354*00b67f09SDavid van Moolenbroek unsigned char *der, *p;
355*00b67f09SDavid van Moolenbroek int len;
356*00b67f09SDavid van Moolenbroek
357*00b67f09SDavid van Moolenbroek if (key->keydata.pkey == NULL)
358*00b67f09SDavid van Moolenbroek return (DST_R_NULLKEY);
359*00b67f09SDavid van Moolenbroek
360*00b67f09SDavid van Moolenbroek if (key->external) {
361*00b67f09SDavid van Moolenbroek priv.nelements = 0;
362*00b67f09SDavid van Moolenbroek return (dst__privstruct_writefile(key, &priv, directory));
363*00b67f09SDavid van Moolenbroek }
364*00b67f09SDavid van Moolenbroek
365*00b67f09SDavid van Moolenbroek pkey = key->keydata.pkey;
366*00b67f09SDavid van Moolenbroek
367*00b67f09SDavid van Moolenbroek len = i2d_PrivateKey(pkey, NULL);
368*00b67f09SDavid van Moolenbroek der = isc_mem_get(key->mctx, (size_t) len);
369*00b67f09SDavid van Moolenbroek if (der == NULL)
370*00b67f09SDavid van Moolenbroek return (ISC_R_NOMEMORY);
371*00b67f09SDavid van Moolenbroek
372*00b67f09SDavid van Moolenbroek p = der;
373*00b67f09SDavid van Moolenbroek if (i2d_PrivateKey(pkey, &p) != len) {
374*00b67f09SDavid van Moolenbroek result = dst__openssl_toresult2("i2d_PrivateKey",
375*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE);
376*00b67f09SDavid van Moolenbroek goto fail;
377*00b67f09SDavid van Moolenbroek }
378*00b67f09SDavid van Moolenbroek
379*00b67f09SDavid van Moolenbroek priv.elements[0].tag = TAG_GOST_PRIVASN1;
380*00b67f09SDavid van Moolenbroek priv.elements[0].length = len;
381*00b67f09SDavid van Moolenbroek priv.elements[0].data = der;
382*00b67f09SDavid van Moolenbroek priv.nelements = 1;
383*00b67f09SDavid van Moolenbroek
384*00b67f09SDavid van Moolenbroek result = dst__privstruct_writefile(key, &priv, directory);
385*00b67f09SDavid van Moolenbroek fail:
386*00b67f09SDavid van Moolenbroek if (der != NULL)
387*00b67f09SDavid van Moolenbroek isc_mem_put(key->mctx, der, (size_t) len);
388*00b67f09SDavid van Moolenbroek return (result);
389*00b67f09SDavid van Moolenbroek }
390*00b67f09SDavid van Moolenbroek
391*00b67f09SDavid van Moolenbroek #else
392*00b67f09SDavid van Moolenbroek
393*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_tofile(const dst_key_t * key,const char * directory)394*00b67f09SDavid van Moolenbroek opensslgost_tofile(const dst_key_t *key, const char *directory) {
395*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey;
396*00b67f09SDavid van Moolenbroek EC_KEY *eckey;
397*00b67f09SDavid van Moolenbroek const BIGNUM *privkey;
398*00b67f09SDavid van Moolenbroek dst_private_t priv;
399*00b67f09SDavid van Moolenbroek isc_result_t ret;
400*00b67f09SDavid van Moolenbroek unsigned char *buf = NULL;
401*00b67f09SDavid van Moolenbroek
402*00b67f09SDavid van Moolenbroek if (key->keydata.pkey == NULL)
403*00b67f09SDavid van Moolenbroek return (DST_R_NULLKEY);
404*00b67f09SDavid van Moolenbroek
405*00b67f09SDavid van Moolenbroek if (key->external) {
406*00b67f09SDavid van Moolenbroek priv.nelements = 0;
407*00b67f09SDavid van Moolenbroek return (dst__privstruct_writefile(key, &priv, directory));
408*00b67f09SDavid van Moolenbroek }
409*00b67f09SDavid van Moolenbroek
410*00b67f09SDavid van Moolenbroek pkey = key->keydata.pkey;
411*00b67f09SDavid van Moolenbroek eckey = EVP_PKEY_get0(pkey);
412*00b67f09SDavid van Moolenbroek if (eckey == NULL)
413*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult(DST_R_OPENSSLFAILURE));
414*00b67f09SDavid van Moolenbroek privkey = EC_KEY_get0_private_key(eckey);
415*00b67f09SDavid van Moolenbroek if (privkey == NULL)
416*00b67f09SDavid van Moolenbroek return (ISC_R_FAILURE);
417*00b67f09SDavid van Moolenbroek
418*00b67f09SDavid van Moolenbroek buf = isc_mem_get(key->mctx, BN_num_bytes(privkey));
419*00b67f09SDavid van Moolenbroek if (buf == NULL)
420*00b67f09SDavid van Moolenbroek return (ISC_R_NOMEMORY);
421*00b67f09SDavid van Moolenbroek
422*00b67f09SDavid van Moolenbroek priv.elements[0].tag = TAG_GOST_PRIVRAW;
423*00b67f09SDavid van Moolenbroek priv.elements[0].length = BN_num_bytes(privkey);
424*00b67f09SDavid van Moolenbroek BN_bn2bin(privkey, buf);
425*00b67f09SDavid van Moolenbroek priv.elements[0].data = buf;
426*00b67f09SDavid van Moolenbroek priv.nelements = 1;
427*00b67f09SDavid van Moolenbroek
428*00b67f09SDavid van Moolenbroek ret = dst__privstruct_writefile(key, &priv, directory);
429*00b67f09SDavid van Moolenbroek
430*00b67f09SDavid van Moolenbroek if (buf != NULL)
431*00b67f09SDavid van Moolenbroek isc_mem_put(key->mctx, buf, BN_num_bytes(privkey));
432*00b67f09SDavid van Moolenbroek return (ret);
433*00b67f09SDavid van Moolenbroek }
434*00b67f09SDavid van Moolenbroek #endif
435*00b67f09SDavid van Moolenbroek
436*00b67f09SDavid van Moolenbroek static unsigned char gost_dummy_key[71] = {
437*00b67f09SDavid van Moolenbroek 0x30, 0x45, 0x02, 0x01, 0x00, 0x30, 0x1c, 0x06,
438*00b67f09SDavid van Moolenbroek 0x06, 0x2a, 0x85, 0x03, 0x02, 0x02, 0x13, 0x30,
439*00b67f09SDavid van Moolenbroek 0x12, 0x06, 0x07, 0x2a, 0x85, 0x03, 0x02, 0x02,
440*00b67f09SDavid van Moolenbroek 0x23, 0x01, 0x06, 0x07, 0x2a, 0x85, 0x03, 0x02,
441*00b67f09SDavid van Moolenbroek 0x02, 0x1e, 0x01, 0x04, 0x22, 0x02, 0x20, 0x1b,
442*00b67f09SDavid van Moolenbroek 0x3f, 0x94, 0xf7, 0x1a, 0x5f, 0x2f, 0xe7, 0xe5,
443*00b67f09SDavid van Moolenbroek 0x74, 0x0b, 0x8c, 0xd4, 0xb7, 0x18, 0xdd, 0x65,
444*00b67f09SDavid van Moolenbroek 0x68, 0x26, 0xd1, 0x54, 0xfb, 0x77, 0xba, 0x63,
445*00b67f09SDavid van Moolenbroek 0x72, 0xd9, 0xf0, 0x63, 0x87, 0xe0, 0xd6
446*00b67f09SDavid van Moolenbroek };
447*00b67f09SDavid van Moolenbroek
448*00b67f09SDavid van Moolenbroek static isc_result_t
opensslgost_parse(dst_key_t * key,isc_lex_t * lexer,dst_key_t * pub)449*00b67f09SDavid van Moolenbroek opensslgost_parse(dst_key_t *key, isc_lex_t *lexer, dst_key_t *pub) {
450*00b67f09SDavid van Moolenbroek dst_private_t priv;
451*00b67f09SDavid van Moolenbroek isc_result_t ret;
452*00b67f09SDavid van Moolenbroek isc_mem_t *mctx = key->mctx;
453*00b67f09SDavid van Moolenbroek EVP_PKEY *pkey = NULL;
454*00b67f09SDavid van Moolenbroek EC_KEY *eckey;
455*00b67f09SDavid van Moolenbroek const EC_POINT *pubkey = NULL;
456*00b67f09SDavid van Moolenbroek BIGNUM *privkey = NULL;
457*00b67f09SDavid van Moolenbroek const unsigned char *p;
458*00b67f09SDavid van Moolenbroek
459*00b67f09SDavid van Moolenbroek /* read private key file */
460*00b67f09SDavid van Moolenbroek ret = dst__privstruct_parse(key, DST_ALG_ECCGOST, lexer, mctx, &priv);
461*00b67f09SDavid van Moolenbroek if (ret != ISC_R_SUCCESS)
462*00b67f09SDavid van Moolenbroek return (ret);
463*00b67f09SDavid van Moolenbroek
464*00b67f09SDavid van Moolenbroek if (key->external) {
465*00b67f09SDavid van Moolenbroek if (priv.nelements != 0)
466*00b67f09SDavid van Moolenbroek DST_RET(DST_R_INVALIDPRIVATEKEY);
467*00b67f09SDavid van Moolenbroek if (pub == NULL)
468*00b67f09SDavid van Moolenbroek DST_RET(DST_R_INVALIDPRIVATEKEY);
469*00b67f09SDavid van Moolenbroek key->keydata.pkey = pub->keydata.pkey;
470*00b67f09SDavid van Moolenbroek pub->keydata.pkey = NULL;
471*00b67f09SDavid van Moolenbroek key->key_size = pub->key_size;
472*00b67f09SDavid van Moolenbroek dst__privstruct_free(&priv, mctx);
473*00b67f09SDavid van Moolenbroek memset(&priv, 0, sizeof(priv));
474*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
475*00b67f09SDavid van Moolenbroek }
476*00b67f09SDavid van Moolenbroek
477*00b67f09SDavid van Moolenbroek INSIST((priv.elements[0].tag == TAG_GOST_PRIVASN1) ||
478*00b67f09SDavid van Moolenbroek (priv.elements[0].tag == TAG_GOST_PRIVRAW));
479*00b67f09SDavid van Moolenbroek
480*00b67f09SDavid van Moolenbroek if (priv.elements[0].tag == TAG_GOST_PRIVASN1) {
481*00b67f09SDavid van Moolenbroek p = priv.elements[0].data;
482*00b67f09SDavid van Moolenbroek if (d2i_PrivateKey(NID_id_GostR3410_2001, &pkey, &p,
483*00b67f09SDavid van Moolenbroek (long) priv.elements[0].length) == NULL)
484*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2(
485*00b67f09SDavid van Moolenbroek "d2i_PrivateKey",
486*00b67f09SDavid van Moolenbroek DST_R_INVALIDPRIVATEKEY));
487*00b67f09SDavid van Moolenbroek } else {
488*00b67f09SDavid van Moolenbroek if ((pub != NULL) && (pub->keydata.pkey != NULL)) {
489*00b67f09SDavid van Moolenbroek eckey = EVP_PKEY_get0(pub->keydata.pkey);
490*00b67f09SDavid van Moolenbroek pubkey = EC_KEY_get0_public_key(eckey);
491*00b67f09SDavid van Moolenbroek }
492*00b67f09SDavid van Moolenbroek
493*00b67f09SDavid van Moolenbroek privkey = BN_bin2bn(priv.elements[0].data,
494*00b67f09SDavid van Moolenbroek priv.elements[0].length, NULL);
495*00b67f09SDavid van Moolenbroek if (privkey == NULL)
496*00b67f09SDavid van Moolenbroek DST_RET(ISC_R_NOMEMORY);
497*00b67f09SDavid van Moolenbroek
498*00b67f09SDavid van Moolenbroek /* can't create directly the whole key */
499*00b67f09SDavid van Moolenbroek p = gost_dummy_key;
500*00b67f09SDavid van Moolenbroek if (d2i_PrivateKey(NID_id_GostR3410_2001, &pkey, &p,
501*00b67f09SDavid van Moolenbroek (long) sizeof(gost_dummy_key)) == NULL)
502*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2(
503*00b67f09SDavid van Moolenbroek "d2i_PrivateKey",
504*00b67f09SDavid van Moolenbroek DST_R_INVALIDPRIVATEKEY));
505*00b67f09SDavid van Moolenbroek
506*00b67f09SDavid van Moolenbroek eckey = EVP_PKEY_get0(pkey);
507*00b67f09SDavid van Moolenbroek if (eckey == NULL)
508*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult(DST_R_OPENSSLFAILURE));
509*00b67f09SDavid van Moolenbroek if (!EC_KEY_set_private_key(eckey, privkey))
510*00b67f09SDavid van Moolenbroek DST_RET(ISC_R_NOMEMORY);
511*00b67f09SDavid van Moolenbroek
512*00b67f09SDavid van Moolenbroek /* have to (re)set the public key */
513*00b67f09SDavid van Moolenbroek #ifdef notyet
514*00b67f09SDavid van Moolenbroek (void) gost2001_compute_public(eckey);
515*00b67f09SDavid van Moolenbroek #else
516*00b67f09SDavid van Moolenbroek if ((pubkey != NULL) && !EC_KEY_set_public_key(eckey, pubkey))
517*00b67f09SDavid van Moolenbroek DST_RET(ISC_R_NOMEMORY);
518*00b67f09SDavid van Moolenbroek #endif
519*00b67f09SDavid van Moolenbroek BN_clear_free(privkey);
520*00b67f09SDavid van Moolenbroek privkey = NULL;
521*00b67f09SDavid van Moolenbroek }
522*00b67f09SDavid van Moolenbroek key->keydata.pkey = pkey;
523*00b67f09SDavid van Moolenbroek key->key_size = EVP_PKEY_bits(pkey);
524*00b67f09SDavid van Moolenbroek dst__privstruct_free(&priv, mctx);
525*00b67f09SDavid van Moolenbroek memset(&priv, 0, sizeof(priv));
526*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
527*00b67f09SDavid van Moolenbroek
528*00b67f09SDavid van Moolenbroek err:
529*00b67f09SDavid van Moolenbroek if (privkey != NULL)
530*00b67f09SDavid van Moolenbroek BN_clear_free(privkey);
531*00b67f09SDavid van Moolenbroek if (pkey != NULL)
532*00b67f09SDavid van Moolenbroek EVP_PKEY_free(pkey);
533*00b67f09SDavid van Moolenbroek opensslgost_destroy(key);
534*00b67f09SDavid van Moolenbroek dst__privstruct_free(&priv, mctx);
535*00b67f09SDavid van Moolenbroek memset(&priv, 0, sizeof(priv));
536*00b67f09SDavid van Moolenbroek return (ret);
537*00b67f09SDavid van Moolenbroek }
538*00b67f09SDavid van Moolenbroek
539*00b67f09SDavid van Moolenbroek static void
opensslgost_cleanup(void)540*00b67f09SDavid van Moolenbroek opensslgost_cleanup(void) {
541*00b67f09SDavid van Moolenbroek if (e != NULL) {
542*00b67f09SDavid van Moolenbroek ENGINE_finish(e);
543*00b67f09SDavid van Moolenbroek ENGINE_free(e);
544*00b67f09SDavid van Moolenbroek e = NULL;
545*00b67f09SDavid van Moolenbroek }
546*00b67f09SDavid van Moolenbroek }
547*00b67f09SDavid van Moolenbroek
548*00b67f09SDavid van Moolenbroek static dst_func_t opensslgost_functions = {
549*00b67f09SDavid van Moolenbroek opensslgost_createctx,
550*00b67f09SDavid van Moolenbroek NULL, /*%< createctx2 */
551*00b67f09SDavid van Moolenbroek opensslgost_destroyctx,
552*00b67f09SDavid van Moolenbroek opensslgost_adddata,
553*00b67f09SDavid van Moolenbroek opensslgost_sign,
554*00b67f09SDavid van Moolenbroek opensslgost_verify,
555*00b67f09SDavid van Moolenbroek NULL, /*%< verify2 */
556*00b67f09SDavid van Moolenbroek NULL, /*%< computesecret */
557*00b67f09SDavid van Moolenbroek opensslgost_compare,
558*00b67f09SDavid van Moolenbroek NULL, /*%< paramcompare */
559*00b67f09SDavid van Moolenbroek opensslgost_generate,
560*00b67f09SDavid van Moolenbroek opensslgost_isprivate,
561*00b67f09SDavid van Moolenbroek opensslgost_destroy,
562*00b67f09SDavid van Moolenbroek opensslgost_todns,
563*00b67f09SDavid van Moolenbroek opensslgost_fromdns,
564*00b67f09SDavid van Moolenbroek opensslgost_tofile,
565*00b67f09SDavid van Moolenbroek opensslgost_parse,
566*00b67f09SDavid van Moolenbroek opensslgost_cleanup,
567*00b67f09SDavid van Moolenbroek NULL, /*%< fromlabel */
568*00b67f09SDavid van Moolenbroek NULL, /*%< dump */
569*00b67f09SDavid van Moolenbroek NULL /*%< restore */
570*00b67f09SDavid van Moolenbroek };
571*00b67f09SDavid van Moolenbroek
572*00b67f09SDavid van Moolenbroek isc_result_t
dst__opensslgost_init(dst_func_t ** funcp)573*00b67f09SDavid van Moolenbroek dst__opensslgost_init(dst_func_t **funcp) {
574*00b67f09SDavid van Moolenbroek isc_result_t ret;
575*00b67f09SDavid van Moolenbroek
576*00b67f09SDavid van Moolenbroek REQUIRE(funcp != NULL);
577*00b67f09SDavid van Moolenbroek
578*00b67f09SDavid van Moolenbroek /* check if the gost engine works properly */
579*00b67f09SDavid van Moolenbroek e = ENGINE_by_id("gost");
580*00b67f09SDavid van Moolenbroek if (e == NULL)
581*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult2("ENGINE_by_id",
582*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
583*00b67f09SDavid van Moolenbroek if (ENGINE_init(e) <= 0) {
584*00b67f09SDavid van Moolenbroek ENGINE_free(e);
585*00b67f09SDavid van Moolenbroek e = NULL;
586*00b67f09SDavid van Moolenbroek return (dst__openssl_toresult2("ENGINE_init",
587*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
588*00b67f09SDavid van Moolenbroek }
589*00b67f09SDavid van Moolenbroek /* better than to rely on digest_gost symbol */
590*00b67f09SDavid van Moolenbroek opensslgost_digest = ENGINE_get_digest(e, NID_id_GostR3411_94);
591*00b67f09SDavid van Moolenbroek if (opensslgost_digest == NULL)
592*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2("ENGINE_get_digest",
593*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
594*00b67f09SDavid van Moolenbroek /* from openssl.cnf */
595*00b67f09SDavid van Moolenbroek if (ENGINE_register_pkey_asn1_meths(e) <= 0)
596*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2(
597*00b67f09SDavid van Moolenbroek "ENGINE_register_pkey_asn1_meths",
598*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
599*00b67f09SDavid van Moolenbroek if (ENGINE_ctrl_cmd_string(e,
600*00b67f09SDavid van Moolenbroek "CRYPT_PARAMS",
601*00b67f09SDavid van Moolenbroek "id-Gost28147-89-CryptoPro-A-ParamSet",
602*00b67f09SDavid van Moolenbroek 0) <= 0)
603*00b67f09SDavid van Moolenbroek DST_RET(dst__openssl_toresult2("ENGINE_ctrl_cmd_string",
604*00b67f09SDavid van Moolenbroek DST_R_OPENSSLFAILURE));
605*00b67f09SDavid van Moolenbroek
606*00b67f09SDavid van Moolenbroek if (*funcp == NULL)
607*00b67f09SDavid van Moolenbroek *funcp = &opensslgost_functions;
608*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
609*00b67f09SDavid van Moolenbroek
610*00b67f09SDavid van Moolenbroek err:
611*00b67f09SDavid van Moolenbroek ENGINE_finish(e);
612*00b67f09SDavid van Moolenbroek ENGINE_free(e);
613*00b67f09SDavid van Moolenbroek e = NULL;
614*00b67f09SDavid van Moolenbroek return (ret);
615*00b67f09SDavid van Moolenbroek }
616*00b67f09SDavid van Moolenbroek
617*00b67f09SDavid van Moolenbroek #else /* HAVE_OPENSSL_GOST */
618*00b67f09SDavid van Moolenbroek
619*00b67f09SDavid van Moolenbroek #include <isc/util.h>
620*00b67f09SDavid van Moolenbroek
621*00b67f09SDavid van Moolenbroek EMPTY_TRANSLATION_UNIT
622*00b67f09SDavid van Moolenbroek
623*00b67f09SDavid van Moolenbroek #endif /* HAVE_OPENSSL_GOST */
624*00b67f09SDavid van Moolenbroek /*! \file */
625