xref: /minix3/external/bsd/bind/dist/lib/dns/opensslgost_link.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
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