xref: /dragonfly/crypto/libressl/apps/openssl/ec.c (revision cca6fc52)
1cca6fc52SDaniel Fojt /* $OpenBSD: ec.c,v 1.14 2019/07/14 03:30:45 guenther Exp $ */
2f5b1c8a1SJohn Marino /*
3f5b1c8a1SJohn Marino  * Written by Nils Larsch for the OpenSSL project.
4f5b1c8a1SJohn Marino  */
5f5b1c8a1SJohn Marino /* ====================================================================
6f5b1c8a1SJohn Marino  * Copyright (c) 1998-2005 The OpenSSL Project.  All rights reserved.
7f5b1c8a1SJohn Marino  *
8f5b1c8a1SJohn Marino  * Redistribution and use in source and binary forms, with or without
9f5b1c8a1SJohn Marino  * modification, are permitted provided that the following conditions
10f5b1c8a1SJohn Marino  * are met:
11f5b1c8a1SJohn Marino  *
12f5b1c8a1SJohn Marino  * 1. Redistributions of source code must retain the above copyright
13f5b1c8a1SJohn Marino  *    notice, this list of conditions and the following disclaimer.
14f5b1c8a1SJohn Marino  *
15f5b1c8a1SJohn Marino  * 2. Redistributions in binary form must reproduce the above copyright
16f5b1c8a1SJohn Marino  *    notice, this list of conditions and the following disclaimer in
17f5b1c8a1SJohn Marino  *    the documentation and/or other materials provided with the
18f5b1c8a1SJohn Marino  *    distribution.
19f5b1c8a1SJohn Marino  *
20f5b1c8a1SJohn Marino  * 3. All advertising materials mentioning features or use of this
21f5b1c8a1SJohn Marino  *    software must display the following acknowledgment:
22f5b1c8a1SJohn Marino  *    "This product includes software developed by the OpenSSL Project
23f5b1c8a1SJohn Marino  *    for use in the OpenSSL Toolkit. (http://www.openssl.org/)"
24f5b1c8a1SJohn Marino  *
25f5b1c8a1SJohn Marino  * 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to
26f5b1c8a1SJohn Marino  *    endorse or promote products derived from this software without
27f5b1c8a1SJohn Marino  *    prior written permission. For written permission, please contact
28f5b1c8a1SJohn Marino  *    openssl-core@openssl.org.
29f5b1c8a1SJohn Marino  *
30f5b1c8a1SJohn Marino  * 5. Products derived from this software may not be called "OpenSSL"
31f5b1c8a1SJohn Marino  *    nor may "OpenSSL" appear in their names without prior written
32f5b1c8a1SJohn Marino  *    permission of the OpenSSL Project.
33f5b1c8a1SJohn Marino  *
34f5b1c8a1SJohn Marino  * 6. Redistributions of any form whatsoever must retain the following
35f5b1c8a1SJohn Marino  *    acknowledgment:
36f5b1c8a1SJohn Marino  *    "This product includes software developed by the OpenSSL Project
37f5b1c8a1SJohn Marino  *    for use in the OpenSSL Toolkit (http://www.openssl.org/)"
38f5b1c8a1SJohn Marino  *
39f5b1c8a1SJohn Marino  * THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY
40f5b1c8a1SJohn Marino  * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
41f5b1c8a1SJohn Marino  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
42f5b1c8a1SJohn Marino  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE OpenSSL PROJECT OR
43f5b1c8a1SJohn Marino  * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
44f5b1c8a1SJohn Marino  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
45f5b1c8a1SJohn Marino  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
46f5b1c8a1SJohn Marino  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
47f5b1c8a1SJohn Marino  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
48f5b1c8a1SJohn Marino  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
49f5b1c8a1SJohn Marino  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
50f5b1c8a1SJohn Marino  * OF THE POSSIBILITY OF SUCH DAMAGE.
51f5b1c8a1SJohn Marino  * ====================================================================
52f5b1c8a1SJohn Marino  *
53f5b1c8a1SJohn Marino  * This product includes cryptographic software written by Eric Young
54f5b1c8a1SJohn Marino  * (eay@cryptsoft.com).  This product includes software written by Tim
55f5b1c8a1SJohn Marino  * Hudson (tjh@cryptsoft.com).
56f5b1c8a1SJohn Marino  *
57f5b1c8a1SJohn Marino  */
58f5b1c8a1SJohn Marino 
59f5b1c8a1SJohn Marino #include <openssl/opensslconf.h>
60f5b1c8a1SJohn Marino 
61f5b1c8a1SJohn Marino #ifndef OPENSSL_NO_EC
62f5b1c8a1SJohn Marino 
63f5b1c8a1SJohn Marino #include <stdio.h>
64f5b1c8a1SJohn Marino #include <stdlib.h>
65f5b1c8a1SJohn Marino #include <string.h>
66f5b1c8a1SJohn Marino 
67f5b1c8a1SJohn Marino #include "apps.h"
68f5b1c8a1SJohn Marino 
69f5b1c8a1SJohn Marino #include <openssl/bio.h>
70f5b1c8a1SJohn Marino #include <openssl/err.h>
71f5b1c8a1SJohn Marino #include <openssl/evp.h>
72f5b1c8a1SJohn Marino #include <openssl/pem.h>
73f5b1c8a1SJohn Marino 
74f5b1c8a1SJohn Marino static struct {
75f5b1c8a1SJohn Marino 	int asn1_flag;
76f5b1c8a1SJohn Marino 	const EVP_CIPHER *enc;
77f5b1c8a1SJohn Marino 	point_conversion_form_t form;
78f5b1c8a1SJohn Marino 	char *infile;
79f5b1c8a1SJohn Marino 	int informat;
80f5b1c8a1SJohn Marino 	char *outfile;
81f5b1c8a1SJohn Marino 	int outformat;
82f5b1c8a1SJohn Marino 	int new_asn1_flag;
83f5b1c8a1SJohn Marino 	int new_form;
84f5b1c8a1SJohn Marino 	int noout;
85f5b1c8a1SJohn Marino 	int param_out;
86f5b1c8a1SJohn Marino 	char *passargin;
87f5b1c8a1SJohn Marino 	char *passargout;
88f5b1c8a1SJohn Marino 	int pubin;
89f5b1c8a1SJohn Marino 	int pubout;
90f5b1c8a1SJohn Marino 	int text;
91f5b1c8a1SJohn Marino } ec_config;
92f5b1c8a1SJohn Marino 
93f5b1c8a1SJohn Marino static int
ec_opt_enc(int argc,char ** argv,int * argsused)94f5b1c8a1SJohn Marino ec_opt_enc(int argc, char **argv, int *argsused)
95f5b1c8a1SJohn Marino {
96f5b1c8a1SJohn Marino 	char *name = argv[0];
97f5b1c8a1SJohn Marino 
98f5b1c8a1SJohn Marino 	if (*name++ != '-')
99f5b1c8a1SJohn Marino 		return (1);
100f5b1c8a1SJohn Marino 
101f5b1c8a1SJohn Marino 	if ((ec_config.enc = EVP_get_cipherbyname(name)) != NULL) {
102f5b1c8a1SJohn Marino 		*argsused = 1;
103f5b1c8a1SJohn Marino 		return (0);
104f5b1c8a1SJohn Marino 	}
105f5b1c8a1SJohn Marino 
106f5b1c8a1SJohn Marino 	return (1);
107f5b1c8a1SJohn Marino }
108f5b1c8a1SJohn Marino 
109f5b1c8a1SJohn Marino static int
ec_opt_form(char * arg)110f5b1c8a1SJohn Marino ec_opt_form(char *arg)
111f5b1c8a1SJohn Marino {
112f5b1c8a1SJohn Marino 	if (strcmp(arg, "compressed") == 0)
113f5b1c8a1SJohn Marino 		ec_config.form = POINT_CONVERSION_COMPRESSED;
114f5b1c8a1SJohn Marino 	else if (strcmp(arg, "uncompressed") == 0)
115f5b1c8a1SJohn Marino 		ec_config.form = POINT_CONVERSION_UNCOMPRESSED;
116f5b1c8a1SJohn Marino 	else if (strcmp(arg, "hybrid") == 0)
117f5b1c8a1SJohn Marino 		ec_config.form = POINT_CONVERSION_HYBRID;
118f5b1c8a1SJohn Marino 	else {
119f5b1c8a1SJohn Marino 		fprintf(stderr, "Invalid point conversion: %s\n", arg);
120f5b1c8a1SJohn Marino 		return (1);
121f5b1c8a1SJohn Marino 	}
122f5b1c8a1SJohn Marino 
123f5b1c8a1SJohn Marino 	ec_config.new_form = 1;
124f5b1c8a1SJohn Marino 	return (0);
125f5b1c8a1SJohn Marino }
126f5b1c8a1SJohn Marino 
127f5b1c8a1SJohn Marino static int
ec_opt_named(char * arg)128f5b1c8a1SJohn Marino ec_opt_named(char *arg)
129f5b1c8a1SJohn Marino {
130f5b1c8a1SJohn Marino 	if (strcmp(arg, "named_curve") == 0)
131f5b1c8a1SJohn Marino 		ec_config.asn1_flag = OPENSSL_EC_NAMED_CURVE;
132f5b1c8a1SJohn Marino 	else if (strcmp(arg, "explicit") == 0)
133f5b1c8a1SJohn Marino 		ec_config.asn1_flag = 0;
134f5b1c8a1SJohn Marino 	else {
135f5b1c8a1SJohn Marino 		fprintf(stderr, "Invalid curve type: %s\n", arg);
136f5b1c8a1SJohn Marino 		return (1);
137f5b1c8a1SJohn Marino 	}
138f5b1c8a1SJohn Marino 
139f5b1c8a1SJohn Marino 	ec_config.new_asn1_flag = 1;
140f5b1c8a1SJohn Marino 	return (0);
141f5b1c8a1SJohn Marino }
142f5b1c8a1SJohn Marino 
143cca6fc52SDaniel Fojt static const struct option ec_options[] = {
144f5b1c8a1SJohn Marino 	{
145f5b1c8a1SJohn Marino 		.name = "conv_form",
146f5b1c8a1SJohn Marino 		.argname = "form",
147f5b1c8a1SJohn Marino 		.desc = "Specify the point conversion form (default"
148f5b1c8a1SJohn Marino 		    " \"named_curve\")",
149f5b1c8a1SJohn Marino 		.type = OPTION_ARG_FUNC,
150f5b1c8a1SJohn Marino 		.opt.argfunc = ec_opt_form,
151f5b1c8a1SJohn Marino 	},
152f5b1c8a1SJohn Marino 	{
153f5b1c8a1SJohn Marino 		.name = "in",
154f5b1c8a1SJohn Marino 		.argname = "file",
155f5b1c8a1SJohn Marino 		.desc = "Input file (default stdin)",
156f5b1c8a1SJohn Marino 		.type = OPTION_ARG,
157f5b1c8a1SJohn Marino 		.opt.arg = &ec_config.infile,
158f5b1c8a1SJohn Marino 	},
159f5b1c8a1SJohn Marino 	{
160f5b1c8a1SJohn Marino 		.name = "inform",
161f5b1c8a1SJohn Marino 		.argname = "format",
162f5b1c8a1SJohn Marino 		.desc = "Input format (DER or PEM (default))",
163f5b1c8a1SJohn Marino 		.type = OPTION_ARG_FORMAT,
164f5b1c8a1SJohn Marino 		.opt.value = &ec_config.informat,
165f5b1c8a1SJohn Marino 	},
166f5b1c8a1SJohn Marino 	{
167f5b1c8a1SJohn Marino 		.name = "noout",
168f5b1c8a1SJohn Marino 		.desc = "No output",
169f5b1c8a1SJohn Marino 		.type = OPTION_FLAG,
170f5b1c8a1SJohn Marino 		.opt.flag = &ec_config.noout,
171f5b1c8a1SJohn Marino 	},
172f5b1c8a1SJohn Marino 	{
173f5b1c8a1SJohn Marino 		.name = "out",
174f5b1c8a1SJohn Marino 		.argname = "file",
175f5b1c8a1SJohn Marino 		.desc = "Output file (default stdout)",
176f5b1c8a1SJohn Marino 		.type = OPTION_ARG,
177f5b1c8a1SJohn Marino 		.opt.arg = &ec_config.outfile,
178f5b1c8a1SJohn Marino 	},
179f5b1c8a1SJohn Marino 	{
180f5b1c8a1SJohn Marino 		.name = "outform",
181f5b1c8a1SJohn Marino 		.argname = "format",
182f5b1c8a1SJohn Marino 		.desc = "Output format (DER or PEM (default))",
183f5b1c8a1SJohn Marino 		.type = OPTION_ARG_FORMAT,
184f5b1c8a1SJohn Marino 		.opt.value = &ec_config.outformat,
185f5b1c8a1SJohn Marino 	},
186f5b1c8a1SJohn Marino 	{
187f5b1c8a1SJohn Marino 		.name = "param_enc",
188f5b1c8a1SJohn Marino 		.argname = "type",
189f5b1c8a1SJohn Marino 		.desc = "Specify the way the ec parameters are encoded"
190f5b1c8a1SJohn Marino 		    " (default \"uncompressed\")",
191f5b1c8a1SJohn Marino 		.type = OPTION_ARG_FUNC,
192f5b1c8a1SJohn Marino 		.opt.argfunc = ec_opt_named,
193f5b1c8a1SJohn Marino 	},
194f5b1c8a1SJohn Marino 	{
195f5b1c8a1SJohn Marino 		.name = "param_out",
196f5b1c8a1SJohn Marino 		.desc = "Print the elliptic curve parameters",
197f5b1c8a1SJohn Marino 		.type = OPTION_FLAG,
198f5b1c8a1SJohn Marino 		.opt.flag = &ec_config.param_out,
199f5b1c8a1SJohn Marino 	},
200f5b1c8a1SJohn Marino 	{
201f5b1c8a1SJohn Marino 		.name = "passin",
202f5b1c8a1SJohn Marino 		.argname = "source",
203f5b1c8a1SJohn Marino 		.desc = "Input file passphrase source",
204f5b1c8a1SJohn Marino 		.type = OPTION_ARG,
205f5b1c8a1SJohn Marino 		.opt.arg = &ec_config.passargin,
206f5b1c8a1SJohn Marino 	},
207f5b1c8a1SJohn Marino 	{
208f5b1c8a1SJohn Marino 		.name = "passout",
209f5b1c8a1SJohn Marino 		.argname = "source",
210f5b1c8a1SJohn Marino 		.desc = "Output file passphrase source",
211f5b1c8a1SJohn Marino 		.type = OPTION_ARG,
212f5b1c8a1SJohn Marino 		.opt.arg = &ec_config.passargout,
213f5b1c8a1SJohn Marino 	},
214f5b1c8a1SJohn Marino 	{
215f5b1c8a1SJohn Marino 		.name = "pubin",
216f5b1c8a1SJohn Marino 		.desc = "Read public key instead of private key from input",
217f5b1c8a1SJohn Marino 		.type = OPTION_FLAG,
218f5b1c8a1SJohn Marino 		.opt.flag = &ec_config.pubin,
219f5b1c8a1SJohn Marino 	},
220f5b1c8a1SJohn Marino 	{
221f5b1c8a1SJohn Marino 		.name = "pubout",
222f5b1c8a1SJohn Marino 		.desc = "Output public key instead of private key in output",
223f5b1c8a1SJohn Marino 		.type = OPTION_FLAG,
224f5b1c8a1SJohn Marino 		.opt.flag = &ec_config.pubout,
225f5b1c8a1SJohn Marino 	},
226f5b1c8a1SJohn Marino 	{
227f5b1c8a1SJohn Marino 		.name = "text",
228f5b1c8a1SJohn Marino 		.desc = "Print the public/private key components and parameters",
229f5b1c8a1SJohn Marino 		.type = OPTION_FLAG,
230f5b1c8a1SJohn Marino 		.opt.flag = &ec_config.text,
231f5b1c8a1SJohn Marino 	},
232f5b1c8a1SJohn Marino 	{
233f5b1c8a1SJohn Marino 		.name = NULL,
234f5b1c8a1SJohn Marino 		.desc = "Cipher to encrypt the output if using PEM format",
235f5b1c8a1SJohn Marino 		.type = OPTION_ARGV_FUNC,
236f5b1c8a1SJohn Marino 		.opt.argvfunc = ec_opt_enc,
237f5b1c8a1SJohn Marino 	},
238f5b1c8a1SJohn Marino 	{ NULL },
239f5b1c8a1SJohn Marino };
240f5b1c8a1SJohn Marino 
241f5b1c8a1SJohn Marino static void
ec_usage(void)242f5b1c8a1SJohn Marino ec_usage(void)
243f5b1c8a1SJohn Marino {
24472c33676SMaxim Ag 	int n = 0;
24572c33676SMaxim Ag 
246f5b1c8a1SJohn Marino 	fprintf(stderr,
247f5b1c8a1SJohn Marino 	    "usage: ec [-conv_form form] [-in file]\n"
248f5b1c8a1SJohn Marino 	    "    [-inform format] [-noout] [-out file] [-outform format]\n"
249f5b1c8a1SJohn Marino 	    "    [-param_enc type] [-param_out] [-passin file]\n"
250f5b1c8a1SJohn Marino 	    "    [-passout file] [-pubin] [-pubout] [-text] [-ciphername]\n\n");
251f5b1c8a1SJohn Marino 	options_usage(ec_options);
252f5b1c8a1SJohn Marino 
253f5b1c8a1SJohn Marino 	fprintf(stderr, "\n");
254f5b1c8a1SJohn Marino 
255f5b1c8a1SJohn Marino 	fprintf(stderr, "Valid ciphername values:\n\n");
25672c33676SMaxim Ag 	OBJ_NAME_do_all_sorted(OBJ_NAME_TYPE_CIPHER_METH, show_cipher, &n);
257f5b1c8a1SJohn Marino 	fprintf(stderr, "\n");
258f5b1c8a1SJohn Marino }
259f5b1c8a1SJohn Marino 
260f5b1c8a1SJohn Marino int
ec_main(int argc,char ** argv)261f5b1c8a1SJohn Marino ec_main(int argc, char **argv)
262f5b1c8a1SJohn Marino {
263f5b1c8a1SJohn Marino 	int ret = 1;
264f5b1c8a1SJohn Marino 	EC_KEY *eckey = NULL;
265f5b1c8a1SJohn Marino 	const EC_GROUP *group;
266f5b1c8a1SJohn Marino 	int i;
267f5b1c8a1SJohn Marino 	BIO *in = NULL, *out = NULL;
268f5b1c8a1SJohn Marino 	char *passin = NULL, *passout = NULL;
269f5b1c8a1SJohn Marino 
270f5b1c8a1SJohn Marino 	if (single_execution) {
27172c33676SMaxim Ag 		if (pledge("stdio cpath wpath rpath tty", NULL) == -1) {
272f5b1c8a1SJohn Marino 			perror("pledge");
273f5b1c8a1SJohn Marino 			exit(1);
274f5b1c8a1SJohn Marino 		}
275f5b1c8a1SJohn Marino 	}
276f5b1c8a1SJohn Marino 
277f5b1c8a1SJohn Marino 	memset(&ec_config, 0, sizeof(ec_config));
278f5b1c8a1SJohn Marino 
279f5b1c8a1SJohn Marino 	ec_config.asn1_flag = OPENSSL_EC_NAMED_CURVE;
280f5b1c8a1SJohn Marino 	ec_config.form = POINT_CONVERSION_UNCOMPRESSED;
281f5b1c8a1SJohn Marino 	ec_config.informat = FORMAT_PEM;
282f5b1c8a1SJohn Marino 	ec_config.outformat = FORMAT_PEM;
283f5b1c8a1SJohn Marino 
284f5b1c8a1SJohn Marino 	if (options_parse(argc, argv, ec_options, NULL, NULL) != 0) {
285f5b1c8a1SJohn Marino 		ec_usage();
286f5b1c8a1SJohn Marino 		goto end;
287f5b1c8a1SJohn Marino 	}
288f5b1c8a1SJohn Marino 
289f5b1c8a1SJohn Marino 	if (!app_passwd(bio_err, ec_config.passargin, ec_config.passargout,
290f5b1c8a1SJohn Marino 	    &passin, &passout)) {
291f5b1c8a1SJohn Marino 		BIO_printf(bio_err, "Error getting passwords\n");
292f5b1c8a1SJohn Marino 		goto end;
293f5b1c8a1SJohn Marino 	}
294f5b1c8a1SJohn Marino 	in = BIO_new(BIO_s_file());
295f5b1c8a1SJohn Marino 	out = BIO_new(BIO_s_file());
296f5b1c8a1SJohn Marino 	if (in == NULL || out == NULL) {
297f5b1c8a1SJohn Marino 		ERR_print_errors(bio_err);
298f5b1c8a1SJohn Marino 		goto end;
299f5b1c8a1SJohn Marino 	}
300f5b1c8a1SJohn Marino 	if (ec_config.infile == NULL)
301f5b1c8a1SJohn Marino 		BIO_set_fp(in, stdin, BIO_NOCLOSE);
302f5b1c8a1SJohn Marino 	else {
303f5b1c8a1SJohn Marino 		if (BIO_read_filename(in, ec_config.infile) <= 0) {
304f5b1c8a1SJohn Marino 			perror(ec_config.infile);
305f5b1c8a1SJohn Marino 			goto end;
306f5b1c8a1SJohn Marino 		}
307f5b1c8a1SJohn Marino 	}
308f5b1c8a1SJohn Marino 
309f5b1c8a1SJohn Marino 	BIO_printf(bio_err, "read EC key\n");
310f5b1c8a1SJohn Marino 	if (ec_config.informat == FORMAT_ASN1) {
311f5b1c8a1SJohn Marino 		if (ec_config.pubin)
312f5b1c8a1SJohn Marino 			eckey = d2i_EC_PUBKEY_bio(in, NULL);
313f5b1c8a1SJohn Marino 		else
314f5b1c8a1SJohn Marino 			eckey = d2i_ECPrivateKey_bio(in, NULL);
315f5b1c8a1SJohn Marino 	} else if (ec_config.informat == FORMAT_PEM) {
316f5b1c8a1SJohn Marino 		if (ec_config.pubin)
317f5b1c8a1SJohn Marino 			eckey = PEM_read_bio_EC_PUBKEY(in, NULL, NULL,
318f5b1c8a1SJohn Marino 			    NULL);
319f5b1c8a1SJohn Marino 		else
320f5b1c8a1SJohn Marino 			eckey = PEM_read_bio_ECPrivateKey(in, NULL, NULL,
321f5b1c8a1SJohn Marino 			    passin);
322f5b1c8a1SJohn Marino 	} else {
323f5b1c8a1SJohn Marino 		BIO_printf(bio_err, "bad input format specified for key\n");
324f5b1c8a1SJohn Marino 		goto end;
325f5b1c8a1SJohn Marino 	}
326f5b1c8a1SJohn Marino 	if (eckey == NULL) {
327f5b1c8a1SJohn Marino 		BIO_printf(bio_err, "unable to load Key\n");
328f5b1c8a1SJohn Marino 		ERR_print_errors(bio_err);
329f5b1c8a1SJohn Marino 		goto end;
330f5b1c8a1SJohn Marino 	}
331f5b1c8a1SJohn Marino 	if (ec_config.outfile == NULL) {
332f5b1c8a1SJohn Marino 		BIO_set_fp(out, stdout, BIO_NOCLOSE);
333f5b1c8a1SJohn Marino 	} else {
334f5b1c8a1SJohn Marino 		if (BIO_write_filename(out, ec_config.outfile) <= 0) {
335f5b1c8a1SJohn Marino 			perror(ec_config.outfile);
336f5b1c8a1SJohn Marino 			goto end;
337f5b1c8a1SJohn Marino 		}
338f5b1c8a1SJohn Marino 	}
339f5b1c8a1SJohn Marino 
340f5b1c8a1SJohn Marino 	group = EC_KEY_get0_group(eckey);
341f5b1c8a1SJohn Marino 
342f5b1c8a1SJohn Marino 	if (ec_config.new_form)
343f5b1c8a1SJohn Marino 		EC_KEY_set_conv_form(eckey, ec_config.form);
344f5b1c8a1SJohn Marino 
345f5b1c8a1SJohn Marino 	if (ec_config.new_asn1_flag)
346f5b1c8a1SJohn Marino 		EC_KEY_set_asn1_flag(eckey, ec_config.asn1_flag);
347f5b1c8a1SJohn Marino 
348f5b1c8a1SJohn Marino 	if (ec_config.text)
349f5b1c8a1SJohn Marino 		if (!EC_KEY_print(out, eckey, 0)) {
350f5b1c8a1SJohn Marino 			perror(ec_config.outfile);
351f5b1c8a1SJohn Marino 			ERR_print_errors(bio_err);
352f5b1c8a1SJohn Marino 			goto end;
353f5b1c8a1SJohn Marino 		}
354f5b1c8a1SJohn Marino 	if (ec_config.noout) {
355f5b1c8a1SJohn Marino 		ret = 0;
356f5b1c8a1SJohn Marino 		goto end;
357f5b1c8a1SJohn Marino 	}
358f5b1c8a1SJohn Marino 	BIO_printf(bio_err, "writing EC key\n");
359f5b1c8a1SJohn Marino 	if (ec_config.outformat == FORMAT_ASN1) {
360f5b1c8a1SJohn Marino 		if (ec_config.param_out)
361f5b1c8a1SJohn Marino 			i = i2d_ECPKParameters_bio(out, group);
362f5b1c8a1SJohn Marino 		else if (ec_config.pubin || ec_config.pubout)
363f5b1c8a1SJohn Marino 			i = i2d_EC_PUBKEY_bio(out, eckey);
364f5b1c8a1SJohn Marino 		else
365f5b1c8a1SJohn Marino 			i = i2d_ECPrivateKey_bio(out, eckey);
366f5b1c8a1SJohn Marino 	} else if (ec_config.outformat == FORMAT_PEM) {
367f5b1c8a1SJohn Marino 		if (ec_config.param_out)
368f5b1c8a1SJohn Marino 			i = PEM_write_bio_ECPKParameters(out, group);
369f5b1c8a1SJohn Marino 		else if (ec_config.pubin || ec_config.pubout)
370f5b1c8a1SJohn Marino 			i = PEM_write_bio_EC_PUBKEY(out, eckey);
371f5b1c8a1SJohn Marino 		else
372f5b1c8a1SJohn Marino 			i = PEM_write_bio_ECPrivateKey(out, eckey,
373f5b1c8a1SJohn Marino 			    ec_config.enc, NULL, 0, NULL, passout);
374f5b1c8a1SJohn Marino 	} else {
375f5b1c8a1SJohn Marino 		BIO_printf(bio_err, "bad output format specified for "
376f5b1c8a1SJohn Marino 		    "outfile\n");
377f5b1c8a1SJohn Marino 		goto end;
378f5b1c8a1SJohn Marino 	}
379f5b1c8a1SJohn Marino 
380f5b1c8a1SJohn Marino 	if (!i) {
381f5b1c8a1SJohn Marino 		BIO_printf(bio_err, "unable to write private key\n");
382f5b1c8a1SJohn Marino 		ERR_print_errors(bio_err);
383f5b1c8a1SJohn Marino 	} else
384f5b1c8a1SJohn Marino 		ret = 0;
385f5b1c8a1SJohn Marino  end:
386f5b1c8a1SJohn Marino 	BIO_free(in);
387f5b1c8a1SJohn Marino 	BIO_free_all(out);
388f5b1c8a1SJohn Marino 	EC_KEY_free(eckey);
389f5b1c8a1SJohn Marino 	free(passin);
390f5b1c8a1SJohn Marino 	free(passout);
391f5b1c8a1SJohn Marino 
392f5b1c8a1SJohn Marino 	return (ret);
393f5b1c8a1SJohn Marino }
394f5b1c8a1SJohn Marino #endif
395