os/ossrv/ssl/libcrypto/src/crypto/engine/eng_openssl.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /* crypto/engine/eng_openssl.c */
     2 /* Written by Geoff Thorpe (geoff@geoffthorpe.net) for the OpenSSL
     3  * project 2000.
     4  */
     5 /* ====================================================================
     6  * Copyright (c) 1999-2001 The OpenSSL Project.  All rights reserved.
     7  *
     8  * Redistribution and use in source and binary forms, with or without
     9  * modification, are permitted provided that the following conditions
    10  * are met:
    11  *
    12  * 1. Redistributions of source code must retain the above copyright
    13  *    notice, this list of conditions and the following disclaimer. 
    14  *
    15  * 2. Redistributions in binary form must reproduce the above copyright
    16  *    notice, this list of conditions and the following disclaimer in
    17  *    the documentation and/or other materials provided with the
    18  *    distribution.
    19  *
    20  * 3. All advertising materials mentioning features or use of this
    21  *    software must display the following acknowledgment:
    22  *    "This product includes software developed by the OpenSSL Project
    23  *    for use in the OpenSSL Toolkit. (http://www.OpenSSL.org/)"
    24  *
    25  * 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to
    26  *    endorse or promote products derived from this software without
    27  *    prior written permission. For written permission, please contact
    28  *    licensing@OpenSSL.org.
    29  *
    30  * 5. Products derived from this software may not be called "OpenSSL"
    31  *    nor may "OpenSSL" appear in their names without prior written
    32  *    permission of the OpenSSL Project.
    33  *
    34  * 6. Redistributions of any form whatsoever must retain the following
    35  *    acknowledgment:
    36  *    "This product includes software developed by the OpenSSL Project
    37  *    for use in the OpenSSL Toolkit (http://www.OpenSSL.org/)"
    38  *
    39  * THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY
    40  * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
    41  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
    42  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE OpenSSL PROJECT OR
    43  * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    44  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
    45  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
    46  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
    47  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
    48  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
    49  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
    50  * OF THE POSSIBILITY OF SUCH DAMAGE.
    51  * ====================================================================
    52  *
    53  * This product includes cryptographic software written by Eric Young
    54  * (eay@cryptsoft.com).  This product includes software written by Tim
    55  * Hudson (tjh@cryptsoft.com).
    56  *
    57  */
    58 /* ====================================================================
    59  * Copyright 2002 Sun Microsystems, Inc. ALL RIGHTS RESERVED.
    60  * ECDH support in OpenSSL originally developed by 
    61  * SUN MICROSYSTEMS, INC., and contributed to the OpenSSL project.
    62  */
    63  /*
    64  © Portions copyright (c) 2006 Nokia Corporation.  All rights reserved.
    65  */
    66 
    67 
    68 #include <stdio.h>
    69 #include <openssl/crypto.h>
    70 #include "cryptlib.h"
    71 #include <openssl/engine.h>
    72 #include <openssl/dso.h>
    73 #include <openssl/pem.h>
    74 #include <openssl/evp.h>
    75 #include <openssl/rand.h>
    76 #ifndef OPENSSL_NO_RSA
    77 #include <openssl/rsa.h>
    78 #endif
    79 #ifndef OPENSSL_NO_DSA
    80 #include <openssl/dsa.h>
    81 #endif
    82 #ifndef OPENSSL_NO_DH
    83 #include <openssl/dh.h>
    84 #endif
    85 #if (defined(SYMBIAN) && (defined(__WINSCW__) || defined(__WINS__)))
    86 #include "libcrypto_wsd_macros.h"
    87 #include "libcrypto_wsd.h"
    88 #endif
    89 
    90 
    91 
    92 /* This testing gunk is implemented (and explained) lower down. It also assumes
    93  * the application explicitly calls "ENGINE_load_openssl()" because this is no
    94  * longer automatic in ENGINE_load_builtin_engines(). */
    95 #define TEST_ENG_OPENSSL_RC4
    96 #define TEST_ENG_OPENSSL_PKEY
    97 /* #define TEST_ENG_OPENSSL_RC4_OTHERS */
    98 #define TEST_ENG_OPENSSL_RC4_P_INIT
    99 /* #define TEST_ENG_OPENSSL_RC4_P_CIPHER */
   100 #define TEST_ENG_OPENSSL_SHA
   101 /* #define TEST_ENG_OPENSSL_SHA_OTHERS */
   102 /* #define TEST_ENG_OPENSSL_SHA_P_INIT */
   103 /* #define TEST_ENG_OPENSSL_SHA_P_UPDATE */
   104 /* #define TEST_ENG_OPENSSL_SHA_P_FINAL */
   105 
   106 /* Now check what of those algorithms are actually enabled */
   107 #ifdef OPENSSL_NO_RC4
   108 #undef TEST_ENG_OPENSSL_RC4
   109 #undef TEST_ENG_OPENSSL_RC4_OTHERS
   110 #undef TEST_ENG_OPENSSL_RC4_P_INIT
   111 #undef TEST_ENG_OPENSSL_RC4_P_CIPHER
   112 #endif
   113 #if defined(OPENSSL_NO_SHA) || defined(OPENSSL_NO_SHA0) || defined(OPENSSL_NO_SHA1)
   114 #undef TEST_ENG_OPENSSL_SHA
   115 #undef TEST_ENG_OPENSSL_SHA_OTHERS
   116 #undef TEST_ENG_OPENSSL_SHA_P_INIT
   117 #undef TEST_ENG_OPENSSL_SHA_P_UPDATE
   118 #undef TEST_ENG_OPENSSL_SHA_P_FINAL 
   119 #endif
   120 
   121 #ifdef TEST_ENG_OPENSSL_RC4
   122 static int openssl_ciphers(ENGINE *e, const EVP_CIPHER **cipher,
   123 				const int **nids, int nid);
   124 #endif
   125 #ifdef TEST_ENG_OPENSSL_SHA
   126 static int openssl_digests(ENGINE *e, const EVP_MD **digest,
   127 				const int **nids, int nid);
   128 #endif
   129 
   130 #ifdef TEST_ENG_OPENSSL_PKEY
   131 static EVP_PKEY *openssl_load_privkey(ENGINE *eng, const char *key_id,
   132 	UI_METHOD *ui_method, void *callback_data);
   133 #endif
   134 
   135 /* The constants used when creating the ENGINE */
   136 #ifndef EMULATOR
   137 static const char *engine_openssl_id = "openssl";
   138 static const char *engine_openssl_name = "Software engine support";
   139 #else
   140 static const char *const engine_openssl_id = "openssl";
   141 static const char *const engine_openssl_name = "Software engine support";
   142 #endif
   143 
   144 /* This internal function is used by ENGINE_openssl() and possibly by the
   145  * "dynamic" ENGINE support too */
   146 static int bind_helper(ENGINE *e)
   147 	{
   148 	if(!ENGINE_set_id(e, engine_openssl_id)
   149 			|| !ENGINE_set_name(e, engine_openssl_name)
   150 #ifndef TEST_ENG_OPENSSL_NO_ALGORITHMS
   151 #ifndef OPENSSL_NO_RSA
   152 			|| !ENGINE_set_RSA(e, RSA_get_default_method())
   153 #endif
   154 #ifndef OPENSSL_NO_DSA
   155 			|| !ENGINE_set_DSA(e, DSA_get_default_method())
   156 #endif
   157 #ifndef OPENSSL_NO_ECDH
   158 			|| !ENGINE_set_ECDH(e, ECDH_OpenSSL())
   159 #endif
   160 #ifndef OPENSSL_NO_ECDSA
   161 			|| !ENGINE_set_ECDSA(e, ECDSA_OpenSSL())
   162 #endif
   163 #ifndef OPENSSL_NO_DH
   164 			|| !ENGINE_set_DH(e, DH_get_default_method())
   165 #endif
   166 			|| !ENGINE_set_RAND(e, RAND_SSLeay())
   167 #ifdef TEST_ENG_OPENSSL_RC4
   168 			|| !ENGINE_set_ciphers(e, openssl_ciphers)
   169 #endif
   170 #ifdef TEST_ENG_OPENSSL_SHA
   171 			|| !ENGINE_set_digests(e, openssl_digests)
   172 #endif
   173 #endif
   174 #ifdef TEST_ENG_OPENSSL_PKEY
   175 			|| !ENGINE_set_load_privkey_function(e, openssl_load_privkey)
   176 #endif
   177 			)
   178 		return 0;
   179 	/* If we add errors to this ENGINE, ensure the error handling is setup here */
   180 	/* openssl_load_error_strings(); */
   181 	return 1;
   182 	}
   183 
   184 static ENGINE *engine_openssl(void)
   185 	{
   186 	ENGINE *ret = ENGINE_new();
   187 	if(!ret)
   188 		return NULL;
   189 	if(!bind_helper(ret))
   190 		{
   191 		ENGINE_free(ret);
   192 		return NULL;
   193 		}
   194 	return ret;
   195 	}
   196 
   197 EXPORT_C void ENGINE_load_openssl(void)
   198 	{
   199 	ENGINE *toadd = engine_openssl();
   200 	if(!toadd) return;
   201 	ENGINE_add(toadd);
   202 	/* If the "add" worked, it gets a structural reference. So either way,
   203 	 * we release our just-created reference. */
   204 	ENGINE_free(toadd);
   205 	ERR_clear_error();
   206 	}
   207 
   208 /* This stuff is needed if this ENGINE is being compiled into a self-contained
   209  * shared-library. */
   210 #ifdef ENGINE_DYNAMIC_SUPPORT
   211 static int bind_fn(ENGINE *e, const char *id)
   212 	{
   213 	if(id && (strcmp(id, engine_openssl_id) != 0))
   214 		return 0;
   215 	if(!bind_helper(e))
   216 		return 0;
   217 	return 1;
   218 	}
   219 IMPLEMENT_DYNAMIC_CHECK_FN()
   220 IMPLEMENT_DYNAMIC_BIND_FN(bind_fn)
   221 #endif /* ENGINE_DYNAMIC_SUPPORT */
   222 
   223 #ifdef TEST_ENG_OPENSSL_RC4
   224 /* This section of code compiles an "alternative implementation" of two modes of
   225  * RC4 into this ENGINE. The result is that EVP_CIPHER operation for "rc4"
   226  * should under normal circumstances go via this support rather than the default
   227  * EVP support. There are other symbols to tweak the testing;
   228  *    TEST_ENC_OPENSSL_RC4_OTHERS - print a one line message to stderr each time
   229  *        we're asked for a cipher we don't support (should not happen).
   230  *    TEST_ENG_OPENSSL_RC4_P_INIT - print a one line message to stderr each time
   231  *        the "init_key" handler is called.
   232  *    TEST_ENG_OPENSSL_RC4_P_CIPHER - ditto for the "cipher" handler.
   233  */
   234 #include <openssl/rc4.h>
   235 #define TEST_RC4_KEY_SIZE		16
   236 #ifndef EMULATOR
   237 static int test_cipher_nids[] = {NID_rc4,NID_rc4_40};
   238 static int test_cipher_nids_number = 2;
   239 #else
   240 static const int test_cipher_nids[] = {NID_rc4,NID_rc4_40};
   241 static const int test_cipher_nids_number = 2;
   242 #endif
   243 typedef struct {
   244 	unsigned char key[TEST_RC4_KEY_SIZE];
   245 	RC4_KEY ks;
   246 	} TEST_RC4_KEY;
   247 #define test(ctx) ((TEST_RC4_KEY *)(ctx)->cipher_data)
   248 static int test_rc4_init_key(EVP_CIPHER_CTX *ctx, const unsigned char *key,
   249 			const unsigned char *iv, int enc)
   250 	{
   251 #ifdef TEST_ENG_OPENSSL_RC4_P_INIT
   252 	fprintf(stderr, "(TEST_ENG_OPENSSL_RC4) test_init_key() called\n");
   253 #endif
   254 	memcpy(&test(ctx)->key[0],key,EVP_CIPHER_CTX_key_length(ctx));
   255 	RC4_set_key(&test(ctx)->ks,EVP_CIPHER_CTX_key_length(ctx),
   256 		test(ctx)->key);
   257 	return 1;
   258 	}
   259 static int test_rc4_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
   260 		      const unsigned char *in, unsigned int inl)
   261 	{
   262 #ifdef TEST_ENG_OPENSSL_RC4_P_CIPHER
   263 	fprintf(stderr, "(TEST_ENG_OPENSSL_RC4) test_cipher() called\n");
   264 #endif
   265 	RC4(&test(ctx)->ks,inl,in,out);
   266 	return 1;
   267 	}
   268 static const EVP_CIPHER test_r4_cipher=
   269 	{
   270 	NID_rc4,
   271 	1,TEST_RC4_KEY_SIZE,0,
   272 	EVP_CIPH_VARIABLE_LENGTH,
   273 	test_rc4_init_key,
   274 	test_rc4_cipher,
   275 	NULL,
   276 	sizeof(TEST_RC4_KEY),
   277 	NULL,
   278 	NULL,
   279 	NULL,
   280 	NULL
   281 	};
   282 static const EVP_CIPHER test_r4_40_cipher=
   283 	{
   284 	NID_rc4_40,
   285 	1,5 /* 40 bit */,0,
   286 	EVP_CIPH_VARIABLE_LENGTH,
   287 	test_rc4_init_key,
   288 	test_rc4_cipher,
   289 	NULL,
   290 	sizeof(TEST_RC4_KEY),
   291 	NULL, 
   292 	NULL,
   293 	NULL,
   294 	NULL
   295 	};
   296 static int openssl_ciphers(ENGINE *e, const EVP_CIPHER **cipher,
   297 			const int **nids, int nid)
   298 	{
   299 	if(!cipher)
   300 		{
   301 		/* We are returning a list of supported nids */
   302 		*nids = test_cipher_nids;
   303 		return test_cipher_nids_number;
   304 		}
   305 	/* We are being asked for a specific cipher */
   306 	if(nid == NID_rc4)
   307 		*cipher = &test_r4_cipher;
   308 	else if(nid == NID_rc4_40)
   309 		*cipher = &test_r4_40_cipher;
   310 	else
   311 		{
   312 #ifdef TEST_ENG_OPENSSL_RC4_OTHERS
   313 		fprintf(stderr, "(TEST_ENG_OPENSSL_RC4) returning NULL for "
   314 				"nid %d\n", nid);
   315 #endif
   316 		*cipher = NULL;
   317 		return 0;
   318 		}
   319 	return 1;
   320 	}
   321 #endif
   322 
   323 #ifdef TEST_ENG_OPENSSL_SHA
   324 /* Much the same sort of comment as for TEST_ENG_OPENSSL_RC4 */
   325 #include <openssl/sha.h>
   326 #ifndef EMULATOR
   327 static int test_digest_nids[] = {NID_sha1};
   328 static int test_digest_nids_number = 1;
   329 #else
   330 static const int test_digest_nids[] = {NID_sha1};
   331 static const int test_digest_nids_number = 1;
   332 #endif
   333 static int test_sha1_init(EVP_MD_CTX *ctx)
   334 	{
   335 #ifdef TEST_ENG_OPENSSL_SHA_P_INIT
   336 	fprintf(stderr, "(TEST_ENG_OPENSSL_SHA) test_sha1_init() called\n");
   337 #endif
   338 	return SHA1_Init(ctx->md_data);
   339 	}
   340 static int test_sha1_update(EVP_MD_CTX *ctx,const void *data,size_t count)
   341 	{
   342 #ifdef TEST_ENG_OPENSSL_SHA_P_UPDATE
   343 	fprintf(stderr, "(TEST_ENG_OPENSSL_SHA) test_sha1_update() called\n");
   344 #endif
   345 	return SHA1_Update(ctx->md_data,data,count);
   346 	}
   347 static int test_sha1_final(EVP_MD_CTX *ctx,unsigned char *md)
   348 	{
   349 #ifdef TEST_ENG_OPENSSL_SHA_P_FINAL
   350 	fprintf(stderr, "(TEST_ENG_OPENSSL_SHA) test_sha1_final() called\n");
   351 #endif
   352 	return SHA1_Final(md,ctx->md_data);
   353 	}
   354 static const EVP_MD test_sha_md=
   355 	{
   356 	NID_sha1,
   357 	NID_sha1WithRSAEncryption,
   358 	SHA_DIGEST_LENGTH,
   359 	0,
   360 	test_sha1_init,
   361 	test_sha1_update,
   362 	test_sha1_final,
   363 	NULL,
   364 	NULL,
   365 	EVP_PKEY_RSA_method,
   366 	SHA_CBLOCK,
   367 	sizeof(EVP_MD *)+sizeof(SHA_CTX),
   368 	};
   369 static int openssl_digests(ENGINE *e, const EVP_MD **digest,
   370 			const int **nids, int nid)
   371 	{
   372 	if(!digest)
   373 		{
   374 		/* We are returning a list of supported nids */
   375 		*nids = test_digest_nids;
   376 		return test_digest_nids_number;
   377 		}
   378 	/* We are being asked for a specific digest */
   379 	if(nid == NID_sha1)
   380 		*digest = &test_sha_md;
   381 	else
   382 		{
   383 #ifdef TEST_ENG_OPENSSL_SHA_OTHERS
   384 		fprintf(stderr, "(TEST_ENG_OPENSSL_SHA) returning NULL for "
   385 				"nid %d\n", nid);
   386 #endif
   387 		*digest = NULL;
   388 		return 0;
   389 		}
   390 	return 1;
   391 	}
   392 #endif
   393 
   394 #ifdef TEST_ENG_OPENSSL_PKEY
   395 static EVP_PKEY *openssl_load_privkey(ENGINE *eng, const char *key_id,
   396 	UI_METHOD *ui_method, void *callback_data)
   397 	{
   398 	BIO *in;
   399 	EVP_PKEY *key;
   400 	fprintf(stderr, "(TEST_ENG_OPENSSL_PKEY)Loading Private key %s\n", key_id);
   401 	in = BIO_new_file(key_id, "r");
   402 	if (!in)
   403 		return NULL;
   404 	key = PEM_read_bio_PrivateKey(in, NULL, 0, NULL);
   405 	BIO_free(in);
   406 	return key;
   407 	}
   408 #endif