You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			1141 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			1141 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /* Copyright (C) 1995-1998 Eric Young (eay@cryptsoft.com)
 | ||
|  |  * All rights reserved. | ||
|  |  * | ||
|  |  * This package is an SSL implementation written | ||
|  |  * by Eric Young (eay@cryptsoft.com). | ||
|  |  * The implementation was written so as to conform with Netscapes SSL. | ||
|  |  * | ||
|  |  * This library is free for commercial and non-commercial use as long as | ||
|  |  * the following conditions are aheared to.  The following conditions | ||
|  |  * apply to all code found in this distribution, be it the RC4, RSA, | ||
|  |  * lhash, DES, etc., code; not just the SSL code.  The SSL documentation | ||
|  |  * included with this distribution is covered by the same copyright terms | ||
|  |  * except that the holder is Tim Hudson (tjh@cryptsoft.com). | ||
|  |  * | ||
|  |  * Copyright remains Eric Young's, and as such any Copyright notices in | ||
|  |  * the code are not to be removed. | ||
|  |  * If this package is used in a product, Eric Young should be given attribution | ||
|  |  * as the author of the parts of the library used. | ||
|  |  * This can be in the form of a textual message at program startup or | ||
|  |  * in documentation (online or textual) provided with the package. | ||
|  |  * | ||
|  |  * Redistribution and use in source and binary forms, with or without | ||
|  |  * modification, are permitted provided that the following conditions | ||
|  |  * are met: | ||
|  |  * 1. Redistributions of source code must retain the copyright | ||
|  |  *    notice, this list of conditions and the following disclaimer. | ||
|  |  * 2. Redistributions in binary form must reproduce the above copyright | ||
|  |  *    notice, this list of conditions and the following disclaimer in the | ||
|  |  *    documentation and/or other materials provided with the distribution. | ||
|  |  * 3. All advertising materials mentioning features or use of this software | ||
|  |  *    must display the following acknowledgement: | ||
|  |  *    "This product includes cryptographic software written by | ||
|  |  *     Eric Young (eay@cryptsoft.com)" | ||
|  |  *    The word 'cryptographic' can be left out if the rouines from the library | ||
|  |  *    being used are not cryptographic related :-). | ||
|  |  * 4. If you include any Windows specific code (or a derivative thereof) from | ||
|  |  *    the apps directory (application code) you must include an acknowledgement: | ||
|  |  *    "This product includes software written by Tim Hudson (tjh@cryptsoft.com)" | ||
|  |  * | ||
|  |  * THIS SOFTWARE IS PROVIDED BY ERIC YOUNG ``AS IS'' AND | ||
|  |  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||
|  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||
|  |  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE | ||
|  |  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||
|  |  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS | ||
|  |  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | ||
|  |  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | ||
|  |  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY | ||
|  |  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | ||
|  |  * SUCH DAMAGE. | ||
|  |  * | ||
|  |  * The licence and distribution terms for any publically available version or | ||
|  |  * derivative of this code cannot be changed.  i.e. this code cannot simply be | ||
|  |  * copied and put under another distribution licence | ||
|  |  * [including the GNU Public Licence.] */ | ||
|  | 
 | ||
|  | #include <openssl/rsa.h>
 | ||
|  | 
 | ||
|  | #include <assert.h>
 | ||
|  | #include <string.h>
 | ||
|  | 
 | ||
|  | #include <openssl/bn.h>
 | ||
|  | #include <openssl/err.h>
 | ||
|  | #include <openssl/mem.h>
 | ||
|  | #include <openssl/thread.h>
 | ||
|  | 
 | ||
|  | #include "internal.h"
 | ||
|  | #include "../internal.h"
 | ||
|  | 
 | ||
|  | 
 | ||
|  | static int check_modulus_and_exponent_sizes(const RSA *rsa) { | ||
|  |   unsigned rsa_bits = BN_num_bits(rsa->n); | ||
|  | 
 | ||
|  |   if (rsa_bits > 16 * 1024) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_MODULUS_TOO_LARGE); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* Mitigate DoS attacks by limiting the exponent size. 33 bits was chosen as
 | ||
|  |    * the limit based on the recommendations in [1] and [2]. Windows CryptoAPI | ||
|  |    * doesn't support values larger than 32 bits [3], so it is unlikely that | ||
|  |    * exponents larger than 32 bits are being used for anything Windows commonly | ||
|  |    * does. | ||
|  |    * | ||
|  |    * [1] https://www.imperialviolet.org/2012/03/16/rsae.html
 | ||
|  |    * [2] https://www.imperialviolet.org/2012/03/17/rsados.html
 | ||
|  |    * [3] https://msdn.microsoft.com/en-us/library/aa387685(VS.85).aspx */
 | ||
|  |   static const unsigned kMaxExponentBits = 33; | ||
|  | 
 | ||
|  |   if (BN_num_bits(rsa->e) > kMaxExponentBits) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_BAD_E_VALUE); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* Verify |n > e|. Comparing |rsa_bits| to |kMaxExponentBits| is a small
 | ||
|  |    * shortcut to comparing |n| and |e| directly. In reality, |kMaxExponentBits| | ||
|  |    * is much smaller than the minimum RSA key size that any application should | ||
|  |    * accept. */ | ||
|  |   if (rsa_bits <= kMaxExponentBits) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_KEY_SIZE_TOO_SMALL); | ||
|  |     return 0; | ||
|  |   } | ||
|  |   assert(BN_ucmp(rsa->n, rsa->e) > 0); | ||
|  | 
 | ||
|  |   return 1; | ||
|  | } | ||
|  | 
 | ||
|  | size_t rsa_default_size(const RSA *rsa) { | ||
|  |   return BN_num_bytes(rsa->n); | ||
|  | } | ||
|  | 
 | ||
|  | int rsa_default_encrypt(RSA *rsa, size_t *out_len, uint8_t *out, size_t max_out, | ||
|  |                         const uint8_t *in, size_t in_len, int padding) { | ||
|  |   const unsigned rsa_size = RSA_size(rsa); | ||
|  |   BIGNUM *f, *result; | ||
|  |   uint8_t *buf = NULL; | ||
|  |   BN_CTX *ctx = NULL; | ||
|  |   int i, ret = 0; | ||
|  | 
 | ||
|  |   if (max_out < rsa_size) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_OUTPUT_BUFFER_TOO_SMALL); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!check_modulus_and_exponent_sizes(rsa)) { | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   ctx = BN_CTX_new(); | ||
|  |   if (ctx == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   BN_CTX_start(ctx); | ||
|  |   f = BN_CTX_get(ctx); | ||
|  |   result = BN_CTX_get(ctx); | ||
|  |   buf = OPENSSL_malloc(rsa_size); | ||
|  |   if (!f || !result || !buf) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_MALLOC_FAILURE); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   switch (padding) { | ||
|  |     case RSA_PKCS1_PADDING: | ||
|  |       i = RSA_padding_add_PKCS1_type_2(buf, rsa_size, in, in_len); | ||
|  |       break; | ||
|  |     case RSA_PKCS1_OAEP_PADDING: | ||
|  |       /* Use the default parameters: SHA-1 for both hashes and no label. */ | ||
|  |       i = RSA_padding_add_PKCS1_OAEP_mgf1(buf, rsa_size, in, in_len, | ||
|  |                                           NULL, 0, NULL, NULL); | ||
|  |       break; | ||
|  |     case RSA_NO_PADDING: | ||
|  |       i = RSA_padding_add_none(buf, rsa_size, in, in_len); | ||
|  |       break; | ||
|  |     default: | ||
|  |       OPENSSL_PUT_ERROR(RSA, RSA_R_UNKNOWN_PADDING_TYPE); | ||
|  |       goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (i <= 0) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_bin2bn(buf, rsa_size, f) == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_ucmp(f, rsa->n) >= 0) { | ||
|  |     /* usually the padding functions would catch this */ | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_DATA_TOO_LARGE_FOR_MODULUS); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_MONT_CTX_set_locked(&rsa->mont_n, &rsa->lock, rsa->n, ctx) || | ||
|  |       !BN_mod_exp_mont(result, f, rsa->e, rsa->n, ctx, rsa->mont_n)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* put in leading 0 bytes if the number is less than the length of the
 | ||
|  |    * modulus */ | ||
|  |   if (!BN_bn2bin_padded(out, rsa_size, result)) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_INTERNAL_ERROR); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   *out_len = rsa_size; | ||
|  |   ret = 1; | ||
|  | 
 | ||
|  | err: | ||
|  |   if (ctx != NULL) { | ||
|  |     BN_CTX_end(ctx); | ||
|  |     BN_CTX_free(ctx); | ||
|  |   } | ||
|  |   if (buf != NULL) { | ||
|  |     OPENSSL_cleanse(buf, rsa_size); | ||
|  |     OPENSSL_free(buf); | ||
|  |   } | ||
|  | 
 | ||
|  |   return ret; | ||
|  | } | ||
|  | 
 | ||
|  | /* MAX_BLINDINGS_PER_RSA defines the maximum number of cached BN_BLINDINGs per
 | ||
|  |  * RSA*. Then this limit is exceeded, BN_BLINDING objects will be created and | ||
|  |  * destroyed as needed. */ | ||
|  | #define MAX_BLINDINGS_PER_RSA 1024
 | ||
|  | 
 | ||
|  | /* rsa_blinding_get returns a BN_BLINDING to use with |rsa|. It does this by
 | ||
|  |  * allocating one of the cached BN_BLINDING objects in |rsa->blindings|. If | ||
|  |  * none are free, the cache will be extended by a extra element and the new | ||
|  |  * BN_BLINDING is returned. | ||
|  |  * | ||
|  |  * On success, the index of the assigned BN_BLINDING is written to | ||
|  |  * |*index_used| and must be passed to |rsa_blinding_release| when finished. */ | ||
|  | static BN_BLINDING *rsa_blinding_get(RSA *rsa, unsigned *index_used, | ||
|  |                                      BN_CTX *ctx) { | ||
|  |   assert(ctx != NULL); | ||
|  |   assert(rsa->mont_n != NULL); | ||
|  | 
 | ||
|  |   BN_BLINDING *ret = NULL; | ||
|  |   BN_BLINDING **new_blindings; | ||
|  |   uint8_t *new_blindings_inuse; | ||
|  |   char overflow = 0; | ||
|  | 
 | ||
|  |   CRYPTO_MUTEX_lock_write(&rsa->lock); | ||
|  | 
 | ||
|  |   unsigned i; | ||
|  |   for (i = 0; i < rsa->num_blindings; i++) { | ||
|  |     if (rsa->blindings_inuse[i] == 0) { | ||
|  |       rsa->blindings_inuse[i] = 1; | ||
|  |       ret = rsa->blindings[i]; | ||
|  |       *index_used = i; | ||
|  |       break; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (ret != NULL) { | ||
|  |     CRYPTO_MUTEX_unlock_write(&rsa->lock); | ||
|  |     return ret; | ||
|  |   } | ||
|  | 
 | ||
|  |   overflow = rsa->num_blindings >= MAX_BLINDINGS_PER_RSA; | ||
|  | 
 | ||
|  |   /* We didn't find a free BN_BLINDING to use so increase the length of
 | ||
|  |    * the arrays by one and use the newly created element. */ | ||
|  | 
 | ||
|  |   CRYPTO_MUTEX_unlock_write(&rsa->lock); | ||
|  |   ret = BN_BLINDING_new(); | ||
|  |   if (ret == NULL) { | ||
|  |     return NULL; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (overflow) { | ||
|  |     /* We cannot add any more cached BN_BLINDINGs so we use |ret|
 | ||
|  |      * and mark it for destruction in |rsa_blinding_release|. */ | ||
|  |     *index_used = MAX_BLINDINGS_PER_RSA; | ||
|  |     return ret; | ||
|  |   } | ||
|  | 
 | ||
|  |   CRYPTO_MUTEX_lock_write(&rsa->lock); | ||
|  | 
 | ||
|  |   new_blindings = | ||
|  |       OPENSSL_malloc(sizeof(BN_BLINDING *) * (rsa->num_blindings + 1)); | ||
|  |   if (new_blindings == NULL) { | ||
|  |     goto err1; | ||
|  |   } | ||
|  |   memcpy(new_blindings, rsa->blindings, | ||
|  |          sizeof(BN_BLINDING *) * rsa->num_blindings); | ||
|  |   new_blindings[rsa->num_blindings] = ret; | ||
|  | 
 | ||
|  |   new_blindings_inuse = OPENSSL_malloc(rsa->num_blindings + 1); | ||
|  |   if (new_blindings_inuse == NULL) { | ||
|  |     goto err2; | ||
|  |   } | ||
|  |   memcpy(new_blindings_inuse, rsa->blindings_inuse, rsa->num_blindings); | ||
|  |   new_blindings_inuse[rsa->num_blindings] = 1; | ||
|  |   *index_used = rsa->num_blindings; | ||
|  | 
 | ||
|  |   OPENSSL_free(rsa->blindings); | ||
|  |   rsa->blindings = new_blindings; | ||
|  |   OPENSSL_free(rsa->blindings_inuse); | ||
|  |   rsa->blindings_inuse = new_blindings_inuse; | ||
|  |   rsa->num_blindings++; | ||
|  | 
 | ||
|  |   CRYPTO_MUTEX_unlock_write(&rsa->lock); | ||
|  |   return ret; | ||
|  | 
 | ||
|  | err2: | ||
|  |   OPENSSL_free(new_blindings); | ||
|  | 
 | ||
|  | err1: | ||
|  |   CRYPTO_MUTEX_unlock_write(&rsa->lock); | ||
|  |   BN_BLINDING_free(ret); | ||
|  |   return NULL; | ||
|  | } | ||
|  | 
 | ||
|  | /* rsa_blinding_release marks the cached BN_BLINDING at the given index as free
 | ||
|  |  * for other threads to use. */ | ||
|  | static void rsa_blinding_release(RSA *rsa, BN_BLINDING *blinding, | ||
|  |                                  unsigned blinding_index) { | ||
|  |   if (blinding_index == MAX_BLINDINGS_PER_RSA) { | ||
|  |     /* This blinding wasn't cached. */ | ||
|  |     BN_BLINDING_free(blinding); | ||
|  |     return; | ||
|  |   } | ||
|  | 
 | ||
|  |   CRYPTO_MUTEX_lock_write(&rsa->lock); | ||
|  |   rsa->blindings_inuse[blinding_index] = 0; | ||
|  |   CRYPTO_MUTEX_unlock_write(&rsa->lock); | ||
|  | } | ||
|  | 
 | ||
|  | /* signing */ | ||
|  | int rsa_default_sign_raw(RSA *rsa, size_t *out_len, uint8_t *out, | ||
|  |                          size_t max_out, const uint8_t *in, size_t in_len, | ||
|  |                          int padding) { | ||
|  |   const unsigned rsa_size = RSA_size(rsa); | ||
|  |   uint8_t *buf = NULL; | ||
|  |   int i, ret = 0; | ||
|  | 
 | ||
|  |   if (max_out < rsa_size) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_OUTPUT_BUFFER_TOO_SMALL); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   buf = OPENSSL_malloc(rsa_size); | ||
|  |   if (buf == NULL) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_MALLOC_FAILURE); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   switch (padding) { | ||
|  |     case RSA_PKCS1_PADDING: | ||
|  |       i = RSA_padding_add_PKCS1_type_1(buf, rsa_size, in, in_len); | ||
|  |       break; | ||
|  |     case RSA_NO_PADDING: | ||
|  |       i = RSA_padding_add_none(buf, rsa_size, in, in_len); | ||
|  |       break; | ||
|  |     default: | ||
|  |       OPENSSL_PUT_ERROR(RSA, RSA_R_UNKNOWN_PADDING_TYPE); | ||
|  |       goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (i <= 0) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!RSA_private_transform(rsa, out, buf, rsa_size)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   *out_len = rsa_size; | ||
|  |   ret = 1; | ||
|  | 
 | ||
|  | err: | ||
|  |   if (buf != NULL) { | ||
|  |     OPENSSL_cleanse(buf, rsa_size); | ||
|  |     OPENSSL_free(buf); | ||
|  |   } | ||
|  | 
 | ||
|  |   return ret; | ||
|  | } | ||
|  | 
 | ||
|  | int rsa_default_decrypt(RSA *rsa, size_t *out_len, uint8_t *out, size_t max_out, | ||
|  |                         const uint8_t *in, size_t in_len, int padding) { | ||
|  |   const unsigned rsa_size = RSA_size(rsa); | ||
|  |   int r = -1; | ||
|  |   uint8_t *buf = NULL; | ||
|  |   int ret = 0; | ||
|  | 
 | ||
|  |   if (max_out < rsa_size) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_OUTPUT_BUFFER_TOO_SMALL); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (padding == RSA_NO_PADDING) { | ||
|  |     buf = out; | ||
|  |   } else { | ||
|  |     /* Allocate a temporary buffer to hold the padded plaintext. */ | ||
|  |     buf = OPENSSL_malloc(rsa_size); | ||
|  |     if (buf == NULL) { | ||
|  |       OPENSSL_PUT_ERROR(RSA, ERR_R_MALLOC_FAILURE); | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (in_len != rsa_size) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_DATA_LEN_NOT_EQUAL_TO_MOD_LEN); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!RSA_private_transform(rsa, buf, in, rsa_size)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   switch (padding) { | ||
|  |     case RSA_PKCS1_PADDING: | ||
|  |       r = RSA_padding_check_PKCS1_type_2(out, rsa_size, buf, rsa_size); | ||
|  |       break; | ||
|  |     case RSA_PKCS1_OAEP_PADDING: | ||
|  |       /* Use the default parameters: SHA-1 for both hashes and no label. */ | ||
|  |       r = RSA_padding_check_PKCS1_OAEP_mgf1(out, rsa_size, buf, rsa_size, | ||
|  |                                             NULL, 0, NULL, NULL); | ||
|  |       break; | ||
|  |     case RSA_NO_PADDING: | ||
|  |       r = rsa_size; | ||
|  |       break; | ||
|  |     default: | ||
|  |       OPENSSL_PUT_ERROR(RSA, RSA_R_UNKNOWN_PADDING_TYPE); | ||
|  |       goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (r < 0) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_PADDING_CHECK_FAILED); | ||
|  |   } else { | ||
|  |     *out_len = r; | ||
|  |     ret = 1; | ||
|  |   } | ||
|  | 
 | ||
|  | err: | ||
|  |   if (padding != RSA_NO_PADDING && buf != NULL) { | ||
|  |     OPENSSL_cleanse(buf, rsa_size); | ||
|  |     OPENSSL_free(buf); | ||
|  |   } | ||
|  | 
 | ||
|  |   return ret; | ||
|  | } | ||
|  | 
 | ||
|  | static int mod_exp(BIGNUM *r0, const BIGNUM *I, RSA *rsa, BN_CTX *ctx); | ||
|  | 
 | ||
|  | int RSA_verify_raw(RSA *rsa, size_t *out_len, uint8_t *out, size_t max_out, | ||
|  |                    const uint8_t *in, size_t in_len, int padding) { | ||
|  |   if (rsa->n == NULL || rsa->e == NULL) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_VALUE_MISSING); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   const unsigned rsa_size = RSA_size(rsa); | ||
|  |   BIGNUM *f, *result; | ||
|  |   int r = -1; | ||
|  | 
 | ||
|  |   if (max_out < rsa_size) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_OUTPUT_BUFFER_TOO_SMALL); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (in_len != rsa_size) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_DATA_LEN_NOT_EQUAL_TO_MOD_LEN); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!check_modulus_and_exponent_sizes(rsa)) { | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   BN_CTX *ctx = BN_CTX_new(); | ||
|  |   if (ctx == NULL) { | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   int ret = 0; | ||
|  |   uint8_t *buf = NULL; | ||
|  | 
 | ||
|  |   BN_CTX_start(ctx); | ||
|  |   f = BN_CTX_get(ctx); | ||
|  |   result = BN_CTX_get(ctx); | ||
|  |   if (f == NULL || result == NULL) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_MALLOC_FAILURE); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (padding == RSA_NO_PADDING) { | ||
|  |     buf = out; | ||
|  |   } else { | ||
|  |     /* Allocate a temporary buffer to hold the padded plaintext. */ | ||
|  |     buf = OPENSSL_malloc(rsa_size); | ||
|  |     if (buf == NULL) { | ||
|  |       OPENSSL_PUT_ERROR(RSA, ERR_R_MALLOC_FAILURE); | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_bin2bn(in, in_len, f) == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_ucmp(f, rsa->n) >= 0) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_DATA_TOO_LARGE_FOR_MODULUS); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_MONT_CTX_set_locked(&rsa->mont_n, &rsa->lock, rsa->n, ctx) || | ||
|  |       !BN_mod_exp_mont(result, f, rsa->e, rsa->n, ctx, rsa->mont_n)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_bn2bin_padded(buf, rsa_size, result)) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_INTERNAL_ERROR); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   switch (padding) { | ||
|  |     case RSA_PKCS1_PADDING: | ||
|  |       r = RSA_padding_check_PKCS1_type_1(out, rsa_size, buf, rsa_size); | ||
|  |       break; | ||
|  |     case RSA_NO_PADDING: | ||
|  |       r = rsa_size; | ||
|  |       break; | ||
|  |     default: | ||
|  |       OPENSSL_PUT_ERROR(RSA, RSA_R_UNKNOWN_PADDING_TYPE); | ||
|  |       goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (r < 0) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_PADDING_CHECK_FAILED); | ||
|  |   } else { | ||
|  |     *out_len = r; | ||
|  |     ret = 1; | ||
|  |   } | ||
|  | 
 | ||
|  | err: | ||
|  |   BN_CTX_end(ctx); | ||
|  |   BN_CTX_free(ctx); | ||
|  |   if (buf != out) { | ||
|  |     OPENSSL_free(buf); | ||
|  |   } | ||
|  |   return ret; | ||
|  | } | ||
|  | 
 | ||
|  | int rsa_default_private_transform(RSA *rsa, uint8_t *out, const uint8_t *in, | ||
|  |                                   size_t len) { | ||
|  |   BIGNUM *f, *result; | ||
|  |   BN_CTX *ctx = NULL; | ||
|  |   unsigned blinding_index = 0; | ||
|  |   BN_BLINDING *blinding = NULL; | ||
|  |   int ret = 0; | ||
|  | 
 | ||
|  |   ctx = BN_CTX_new(); | ||
|  |   if (ctx == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   BN_CTX_start(ctx); | ||
|  |   f = BN_CTX_get(ctx); | ||
|  |   result = BN_CTX_get(ctx); | ||
|  | 
 | ||
|  |   if (f == NULL || result == NULL) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_MALLOC_FAILURE); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_bin2bn(in, len, f) == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_ucmp(f, rsa->n) >= 0) { | ||
|  |     /* Usually the padding functions would catch this. */ | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_DATA_TOO_LARGE_FOR_MODULUS); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_MONT_CTX_set_locked(&rsa->mont_n, &rsa->lock, rsa->n, ctx)) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_INTERNAL_ERROR); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* We cannot do blinding or verification without |e|, and continuing without
 | ||
|  |    * those countermeasures is dangerous. However, the Java/Android RSA API | ||
|  |    * requires support for keys where only |d| and |n| (and not |e|) are known. | ||
|  |    * The callers that require that bad behavior set |RSA_FLAG_NO_BLINDING|. */ | ||
|  |   int disable_security = (rsa->flags & RSA_FLAG_NO_BLINDING) && rsa->e == NULL; | ||
|  | 
 | ||
|  |   if (!disable_security) { | ||
|  |     /* Keys without public exponents must have blinding explicitly disabled to
 | ||
|  |      * be used. */ | ||
|  |     if (rsa->e == NULL) { | ||
|  |       OPENSSL_PUT_ERROR(RSA, RSA_R_NO_PUBLIC_EXPONENT); | ||
|  |       goto err; | ||
|  |     } | ||
|  | 
 | ||
|  |     blinding = rsa_blinding_get(rsa, &blinding_index, ctx); | ||
|  |     if (blinding == NULL) { | ||
|  |       OPENSSL_PUT_ERROR(RSA, ERR_R_INTERNAL_ERROR); | ||
|  |       goto err; | ||
|  |     } | ||
|  |     if (!BN_BLINDING_convert(f, blinding, rsa->e, rsa->mont_n, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (rsa->p != NULL && rsa->q != NULL && rsa->e != NULL && rsa->dmp1 != NULL && | ||
|  |       rsa->dmq1 != NULL && rsa->iqmp != NULL) { | ||
|  |     if (!mod_exp(result, f, rsa, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } else { | ||
|  |     BIGNUM local_d; | ||
|  |     BIGNUM *d = NULL; | ||
|  | 
 | ||
|  |     BN_init(&local_d); | ||
|  |     d = &local_d; | ||
|  |     BN_with_flags(d, rsa->d, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |     if (!BN_mod_exp_mont_consttime(result, f, d, rsa->n, ctx, rsa->mont_n)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   /* Verify the result to protect against fault attacks as described in the
 | ||
|  |    * 1997 paper "On the Importance of Checking Cryptographic Protocols for | ||
|  |    * Faults" by Dan Boneh, Richard A. DeMillo, and Richard J. Lipton. Some | ||
|  |    * implementations do this only when the CRT is used, but we do it in all | ||
|  |    * cases. Section 6 of the aforementioned paper describes an attack that | ||
|  |    * works when the CRT isn't used. That attack is much less likely to succeed | ||
|  |    * than the CRT attack, but there have likely been improvements since 1997. | ||
|  |    * | ||
|  |    * This check is cheap assuming |e| is small; it almost always is. */ | ||
|  |   if (!disable_security) { | ||
|  |     BIGNUM *vrfy = BN_CTX_get(ctx); | ||
|  |     if (vrfy == NULL || | ||
|  |         !BN_mod_exp_mont(vrfy, result, rsa->e, rsa->n, ctx, rsa->mont_n) || | ||
|  |         !BN_equal_consttime(vrfy, f)) { | ||
|  |       OPENSSL_PUT_ERROR(RSA, ERR_R_INTERNAL_ERROR); | ||
|  |       goto err; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!BN_BLINDING_invert(result, blinding, rsa->mont_n, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_bn2bin_padded(out, len, result)) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_R_INTERNAL_ERROR); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   ret = 1; | ||
|  | 
 | ||
|  | err: | ||
|  |   if (ctx != NULL) { | ||
|  |     BN_CTX_end(ctx); | ||
|  |     BN_CTX_free(ctx); | ||
|  |   } | ||
|  |   if (blinding != NULL) { | ||
|  |     rsa_blinding_release(rsa, blinding, blinding_index); | ||
|  |   } | ||
|  | 
 | ||
|  |   return ret; | ||
|  | } | ||
|  | 
 | ||
|  | static int mod_exp(BIGNUM *r0, const BIGNUM *I, RSA *rsa, BN_CTX *ctx) { | ||
|  |   assert(ctx != NULL); | ||
|  | 
 | ||
|  |   assert(rsa->n != NULL); | ||
|  |   assert(rsa->e != NULL); | ||
|  |   assert(rsa->d != NULL); | ||
|  |   assert(rsa->p != NULL); | ||
|  |   assert(rsa->q != NULL); | ||
|  |   assert(rsa->dmp1 != NULL); | ||
|  |   assert(rsa->dmq1 != NULL); | ||
|  |   assert(rsa->iqmp != NULL); | ||
|  | 
 | ||
|  |   BIGNUM *r1, *m1, *vrfy; | ||
|  |   BIGNUM local_dmp1, local_dmq1, local_c, local_r1; | ||
|  |   BIGNUM *dmp1, *dmq1, *c, *pr1; | ||
|  |   int ret = 0; | ||
|  |   size_t i, num_additional_primes = 0; | ||
|  | 
 | ||
|  |   if (rsa->additional_primes != NULL) { | ||
|  |     num_additional_primes = sk_RSA_additional_prime_num(rsa->additional_primes); | ||
|  |   } | ||
|  | 
 | ||
|  |   BN_CTX_start(ctx); | ||
|  |   r1 = BN_CTX_get(ctx); | ||
|  |   m1 = BN_CTX_get(ctx); | ||
|  |   vrfy = BN_CTX_get(ctx); | ||
|  |   if (r1 == NULL || | ||
|  |       m1 == NULL || | ||
|  |       vrfy == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   { | ||
|  |     BIGNUM local_p, local_q; | ||
|  |     BIGNUM *p = NULL, *q = NULL; | ||
|  | 
 | ||
|  |     /* Make sure BN_mod_inverse in Montgomery intialization uses the
 | ||
|  |      * BN_FLG_CONSTTIME flag. */ | ||
|  |     BN_init(&local_p); | ||
|  |     p = &local_p; | ||
|  |     BN_with_flags(p, rsa->p, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |     BN_init(&local_q); | ||
|  |     q = &local_q; | ||
|  |     BN_with_flags(q, rsa->q, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |     if (!BN_MONT_CTX_set_locked(&rsa->mont_p, &rsa->lock, p, ctx) || | ||
|  |         !BN_MONT_CTX_set_locked(&rsa->mont_q, &rsa->lock, q, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_MONT_CTX_set_locked(&rsa->mont_n, &rsa->lock, rsa->n, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* compute I mod q */ | ||
|  |   c = &local_c; | ||
|  |   BN_with_flags(c, I, BN_FLG_CONSTTIME); | ||
|  |   if (!BN_mod(r1, c, rsa->q, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* compute r1^dmq1 mod q */ | ||
|  |   dmq1 = &local_dmq1; | ||
|  |   BN_with_flags(dmq1, rsa->dmq1, BN_FLG_CONSTTIME); | ||
|  |   if (!BN_mod_exp_mont_consttime(m1, r1, dmq1, rsa->q, ctx, rsa->mont_q)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* compute I mod p */ | ||
|  |   c = &local_c; | ||
|  |   BN_with_flags(c, I, BN_FLG_CONSTTIME); | ||
|  |   if (!BN_mod(r1, c, rsa->p, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* compute r1^dmp1 mod p */ | ||
|  |   dmp1 = &local_dmp1; | ||
|  |   BN_with_flags(dmp1, rsa->dmp1, BN_FLG_CONSTTIME); | ||
|  |   if (!BN_mod_exp_mont_consttime(r0, r1, dmp1, rsa->p, ctx, rsa->mont_p)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_sub(r0, r0, m1)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   /* This will help stop the size of r0 increasing, which does
 | ||
|  |    * affect the multiply if it optimised for a power of 2 size */ | ||
|  |   if (BN_is_negative(r0)) { | ||
|  |     if (!BN_add(r0, r0, rsa->p)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_mul(r1, r0, rsa->iqmp, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* Turn BN_FLG_CONSTTIME flag on before division operation */ | ||
|  |   pr1 = &local_r1; | ||
|  |   BN_with_flags(pr1, r1, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |   if (!BN_mod(r0, pr1, rsa->p, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* If p < q it is occasionally possible for the correction of
 | ||
|  |    * adding 'p' if r0 is negative above to leave the result still | ||
|  |    * negative. This can break the private key operations: the following | ||
|  |    * second correction should *always* correct this rare occurrence. | ||
|  |    * This will *never* happen with OpenSSL generated keys because | ||
|  |    * they ensure p > q [steve] */ | ||
|  |   if (BN_is_negative(r0)) { | ||
|  |     if (!BN_add(r0, r0, rsa->p)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  |   if (!BN_mul(r1, r0, rsa->q, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!BN_add(r0, r1, m1)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   for (i = 0; i < num_additional_primes; i++) { | ||
|  |     /* multi-prime RSA. */ | ||
|  |     BIGNUM local_exp, local_prime; | ||
|  |     BIGNUM *exp = &local_exp, *prime = &local_prime; | ||
|  |     RSA_additional_prime *ap = | ||
|  |         sk_RSA_additional_prime_value(rsa->additional_primes, i); | ||
|  | 
 | ||
|  |     BN_with_flags(exp, ap->exp, BN_FLG_CONSTTIME); | ||
|  |     BN_with_flags(prime, ap->prime, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |     /* c will already point to a BIGNUM with the correct flags. */ | ||
|  |     if (!BN_mod(r1, c, prime, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!BN_MONT_CTX_set_locked(&ap->mont, &rsa->lock, prime, ctx) || | ||
|  |         !BN_mod_exp_mont_consttime(m1, r1, exp, prime, ctx, ap->mont)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  | 
 | ||
|  |     BN_set_flags(m1, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |     if (!BN_sub(m1, m1, r0) || | ||
|  |         !BN_mul(m1, m1, ap->coeff, ctx) || | ||
|  |         !BN_mod(m1, m1, prime, ctx) || | ||
|  |         (BN_is_negative(m1) && !BN_add(m1, m1, prime)) || | ||
|  |         !BN_mul(m1, m1, ap->r, ctx) || | ||
|  |         !BN_add(r0, r0, m1)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   ret = 1; | ||
|  | 
 | ||
|  | err: | ||
|  |   BN_CTX_end(ctx); | ||
|  |   return ret; | ||
|  | } | ||
|  | 
 | ||
|  | int rsa_default_multi_prime_keygen(RSA *rsa, int bits, int num_primes, | ||
|  |                                    BIGNUM *e_value, BN_GENCB *cb) { | ||
|  |   BIGNUM *r0 = NULL, *r1 = NULL, *r2 = NULL, *r3 = NULL, *tmp; | ||
|  |   BIGNUM local_r0, local_d, local_p; | ||
|  |   BIGNUM *pr0, *d, *p; | ||
|  |   int prime_bits, ok = -1, n = 0, i, j; | ||
|  |   BN_CTX *ctx = NULL; | ||
|  |   STACK_OF(RSA_additional_prime) *additional_primes = NULL; | ||
|  | 
 | ||
|  |   if (num_primes < 2) { | ||
|  |     ok = 0; /* we set our own err */ | ||
|  |     OPENSSL_PUT_ERROR(RSA, RSA_R_MUST_HAVE_AT_LEAST_TWO_PRIMES); | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   ctx = BN_CTX_new(); | ||
|  |   if (ctx == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   BN_CTX_start(ctx); | ||
|  |   r0 = BN_CTX_get(ctx); | ||
|  |   r1 = BN_CTX_get(ctx); | ||
|  |   r2 = BN_CTX_get(ctx); | ||
|  |   r3 = BN_CTX_get(ctx); | ||
|  |   if (r0 == NULL || r1 == NULL || r2 == NULL || r3 == NULL) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (num_primes > 2) { | ||
|  |     additional_primes = sk_RSA_additional_prime_new_null(); | ||
|  |     if (additional_primes == NULL) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   for (i = 2; i < num_primes; i++) { | ||
|  |     RSA_additional_prime *ap = OPENSSL_malloc(sizeof(RSA_additional_prime)); | ||
|  |     if (ap == NULL) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |     memset(ap, 0, sizeof(RSA_additional_prime)); | ||
|  |     ap->prime = BN_new(); | ||
|  |     ap->exp = BN_new(); | ||
|  |     ap->coeff = BN_new(); | ||
|  |     ap->r = BN_new(); | ||
|  |     if (ap->prime == NULL || | ||
|  |         ap->exp == NULL || | ||
|  |         ap->coeff == NULL || | ||
|  |         ap->r == NULL || | ||
|  |         !sk_RSA_additional_prime_push(additional_primes, ap)) { | ||
|  |       RSA_additional_prime_free(ap); | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   /* We need the RSA components non-NULL */ | ||
|  |   if (!rsa->n && ((rsa->n = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->d && ((rsa->d = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->e && ((rsa->e = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->p && ((rsa->p = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->q && ((rsa->q = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->dmp1 && ((rsa->dmp1 = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->dmq1 && ((rsa->dmq1 = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   if (!rsa->iqmp && ((rsa->iqmp = BN_new()) == NULL)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_copy(rsa->e, e_value)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* generate p and q */ | ||
|  |   prime_bits = (bits + (num_primes - 1)) / num_primes; | ||
|  |   for (;;) { | ||
|  |     if (!BN_generate_prime_ex(rsa->p, prime_bits, 0, NULL, NULL, cb) || | ||
|  |         !BN_sub(r2, rsa->p, BN_value_one()) || | ||
|  |         !BN_gcd(r1, r2, rsa->e, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |     if (BN_is_one(r1)) { | ||
|  |       break; | ||
|  |     } | ||
|  |     if (!BN_GENCB_call(cb, 2, n++)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  |   if (!BN_GENCB_call(cb, 3, 0)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  |   prime_bits = ((bits - prime_bits) + (num_primes - 2)) / (num_primes - 1); | ||
|  |   for (;;) { | ||
|  |     /* When generating ridiculously small keys, we can get stuck
 | ||
|  |      * continually regenerating the same prime values. Check for | ||
|  |      * this and bail if it happens 3 times. */ | ||
|  |     unsigned int degenerate = 0; | ||
|  |     do { | ||
|  |       if (!BN_generate_prime_ex(rsa->q, prime_bits, 0, NULL, NULL, cb)) { | ||
|  |         goto err; | ||
|  |       } | ||
|  |     } while ((BN_cmp(rsa->p, rsa->q) == 0) && (++degenerate < 3)); | ||
|  |     if (degenerate == 3) { | ||
|  |       ok = 0; /* we set our own err */ | ||
|  |       OPENSSL_PUT_ERROR(RSA, RSA_R_KEY_SIZE_TOO_SMALL); | ||
|  |       goto err; | ||
|  |     } | ||
|  |     if (!BN_sub(r2, rsa->q, BN_value_one()) || | ||
|  |         !BN_gcd(r1, r2, rsa->e, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |     if (BN_is_one(r1)) { | ||
|  |       break; | ||
|  |     } | ||
|  |     if (!BN_GENCB_call(cb, 2, n++)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!BN_GENCB_call(cb, 3, 1) || | ||
|  |       !BN_mul(rsa->n, rsa->p, rsa->q, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   for (i = 2; i < num_primes; i++) { | ||
|  |     RSA_additional_prime *ap = | ||
|  |         sk_RSA_additional_prime_value(additional_primes, i - 2); | ||
|  |     prime_bits = ((bits - BN_num_bits(rsa->n)) + (num_primes - (i + 1))) / | ||
|  |                  (num_primes - i); | ||
|  | 
 | ||
|  |     for (;;) { | ||
|  |       if (!BN_generate_prime_ex(ap->prime, prime_bits, 0, NULL, NULL, cb)) { | ||
|  |         goto err; | ||
|  |       } | ||
|  |       if (BN_cmp(rsa->p, ap->prime) == 0 || | ||
|  |           BN_cmp(rsa->q, ap->prime) == 0) { | ||
|  |         continue; | ||
|  |       } | ||
|  | 
 | ||
|  |       for (j = 0; j < i - 2; j++) { | ||
|  |         if (BN_cmp(sk_RSA_additional_prime_value(additional_primes, j)->prime, | ||
|  |                    ap->prime) == 0) { | ||
|  |           break; | ||
|  |         } | ||
|  |       } | ||
|  |       if (j != i - 2) { | ||
|  |         continue; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (!BN_sub(r2, ap->prime, BN_value_one()) || | ||
|  |           !BN_gcd(r1, r2, rsa->e, ctx)) { | ||
|  |         goto err; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (!BN_is_one(r1)) { | ||
|  |         continue; | ||
|  |       } | ||
|  |       if (i != num_primes - 1) { | ||
|  |         break; | ||
|  |       } | ||
|  | 
 | ||
|  |       /* For the last prime we'll check that it makes n large enough. In the
 | ||
|  |        * two prime case this isn't a problem because we generate primes with | ||
|  |        * the top two bits set and so the product is always of the expected | ||
|  |        * size. In the multi prime case, this doesn't follow. */ | ||
|  |       if (!BN_mul(r1, rsa->n, ap->prime, ctx)) { | ||
|  |         goto err; | ||
|  |       } | ||
|  |       if (BN_num_bits(r1) == (unsigned) bits) { | ||
|  |         break; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (!BN_GENCB_call(cb, 2, n++)) { | ||
|  |         goto err; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     /* ap->r is is the product of all the primes prior to the current one
 | ||
|  |      * (including p and q). */ | ||
|  |     if (!BN_copy(ap->r, rsa->n)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |     if (i == num_primes - 1) { | ||
|  |       /* In the case of the last prime, we calculated n as |r1| in the loop
 | ||
|  |        * above. */ | ||
|  |       if (!BN_copy(rsa->n, r1)) { | ||
|  |         goto err; | ||
|  |       } | ||
|  |     } else if (!BN_mul(rsa->n, rsa->n, ap->prime, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!BN_GENCB_call(cb, 3, 1)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (BN_cmp(rsa->p, rsa->q) < 0) { | ||
|  |     tmp = rsa->p; | ||
|  |     rsa->p = rsa->q; | ||
|  |     rsa->q = tmp; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* calculate d */ | ||
|  |   if (!BN_sub(r1, rsa->p, BN_value_one())) { | ||
|  |     goto err; /* p-1 */ | ||
|  |   } | ||
|  |   if (!BN_sub(r2, rsa->q, BN_value_one())) { | ||
|  |     goto err; /* q-1 */ | ||
|  |   } | ||
|  |   if (!BN_mul(r0, r1, r2, ctx)) { | ||
|  |     goto err; /* (p-1)(q-1) */ | ||
|  |   } | ||
|  |   for (i = 2; i < num_primes; i++) { | ||
|  |     RSA_additional_prime *ap = | ||
|  |         sk_RSA_additional_prime_value(additional_primes, i - 2); | ||
|  |     if (!BN_sub(r3, ap->prime, BN_value_one()) || | ||
|  |         !BN_mul(r0, r0, r3, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  |   pr0 = &local_r0; | ||
|  |   BN_with_flags(pr0, r0, BN_FLG_CONSTTIME); | ||
|  |   if (!BN_mod_inverse(rsa->d, rsa->e, pr0, ctx)) { | ||
|  |     goto err; /* d */ | ||
|  |   } | ||
|  | 
 | ||
|  |   /* set up d for correct BN_FLG_CONSTTIME flag */ | ||
|  |   d = &local_d; | ||
|  |   BN_with_flags(d, rsa->d, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |   /* calculate d mod (p-1) */ | ||
|  |   if (!BN_mod(rsa->dmp1, d, r1, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* calculate d mod (q-1) */ | ||
|  |   if (!BN_mod(rsa->dmq1, d, r2, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   /* calculate inverse of q mod p */ | ||
|  |   p = &local_p; | ||
|  |   BN_with_flags(p, rsa->p, BN_FLG_CONSTTIME); | ||
|  | 
 | ||
|  |   if (!BN_mod_inverse(rsa->iqmp, rsa->q, p, ctx)) { | ||
|  |     goto err; | ||
|  |   } | ||
|  | 
 | ||
|  |   for (i = 2; i < num_primes; i++) { | ||
|  |     RSA_additional_prime *ap = | ||
|  |         sk_RSA_additional_prime_value(additional_primes, i - 2); | ||
|  |     if (!BN_sub(ap->exp, ap->prime, BN_value_one()) || | ||
|  |         !BN_mod(ap->exp, rsa->d, ap->exp, ctx) || | ||
|  |         !BN_mod_inverse(ap->coeff, ap->r, ap->prime, ctx)) { | ||
|  |       goto err; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   ok = 1; | ||
|  |   rsa->additional_primes = additional_primes; | ||
|  |   additional_primes = NULL; | ||
|  | 
 | ||
|  | err: | ||
|  |   if (ok == -1) { | ||
|  |     OPENSSL_PUT_ERROR(RSA, ERR_LIB_BN); | ||
|  |     ok = 0; | ||
|  |   } | ||
|  |   if (ctx != NULL) { | ||
|  |     BN_CTX_end(ctx); | ||
|  |     BN_CTX_free(ctx); | ||
|  |   } | ||
|  |   sk_RSA_additional_prime_pop_free(additional_primes, | ||
|  |                                    RSA_additional_prime_free); | ||
|  |   return ok; | ||
|  | } | ||
|  | 
 | ||
|  | int rsa_default_keygen(RSA *rsa, int bits, BIGNUM *e_value, BN_GENCB *cb) { | ||
|  |   return rsa_default_multi_prime_keygen(rsa, bits, 2 /* num primes */, e_value, | ||
|  |                                         cb); | ||
|  | } | ||
|  | 
 | ||
|  | /* All of the methods are NULL to make it easier for the compiler/linker to drop
 | ||
|  |  * unused functions. The wrapper functions will select the appropriate | ||
|  |  * |rsa_default_*| implementation. */ | ||
|  | const RSA_METHOD RSA_default_method = { | ||
|  |   { | ||
|  |     0 /* references */, | ||
|  |     1 /* is_static */, | ||
|  |   }, | ||
|  |   NULL /* app_data */, | ||
|  | 
 | ||
|  |   NULL /* init */, | ||
|  |   NULL /* finish (defaults to rsa_default_finish) */, | ||
|  | 
 | ||
|  |   NULL /* size (defaults to rsa_default_size) */, | ||
|  | 
 | ||
|  |   NULL /* sign */, | ||
|  |   NULL /* verify */, | ||
|  | 
 | ||
|  |   NULL /* encrypt (defaults to rsa_default_encrypt) */, | ||
|  |   NULL /* sign_raw (defaults to rsa_default_sign_raw) */, | ||
|  |   NULL /* decrypt (defaults to rsa_default_decrypt) */, | ||
|  |   NULL /* verify_raw (defaults to rsa_default_verify_raw) */, | ||
|  | 
 | ||
|  |   NULL /* private_transform (defaults to rsa_default_private_transform) */, | ||
|  | 
 | ||
|  |   NULL /* mod_exp (ignored) */, | ||
|  |   NULL /* bn_mod_exp (ignored) */, | ||
|  | 
 | ||
|  |   RSA_FLAG_CACHE_PUBLIC | RSA_FLAG_CACHE_PRIVATE, | ||
|  | 
 | ||
|  |   NULL /* keygen (defaults to rsa_default_keygen) */, | ||
|  |   NULL /* multi_prime_keygen (defaults to rsa_default_multi_prime_keygen) */, | ||
|  | 
 | ||
|  |   NULL /* supports_digest */, | ||
|  | }; |