TA的每日心情 | 开心 2024-12-9 18:45 |
---|
签到天数: 124 天 [LV.7]常住居民III
|
欢迎您注册加入!这里有您将更精采!
您需要 登录 才可以下载或查看,没有账号?注册
x
-
- /* c code
- Header source file: rsa.h
- */
- /**
- * \file rsa.h
- */
- #ifndef XYSSL_RSA_H
- #define XYSSL_RSA_H
- #include "xyssl/bignum.h"
- #define XYSSL_ERR_RSA_BAD_INPUT_DATA -0x0400
- #define XYSSL_ERR_RSA_INVALID_PADDING -0x0410
- #define XYSSL_ERR_RSA_KEY_GEN_FAILED -0x0420
- #define XYSSL_ERR_RSA_KEY_CHECK_FAILED -0x0430
- #define XYSSL_ERR_RSA_PUBLIC_FAILED -0x0440
- #define XYSSL_ERR_RSA_PRIVATE_FAILED -0x0450
- #define XYSSL_ERR_RSA_VERIFY_FAILED -0x0460
- /*
- * PKCS#1 constants
- */
- #define RSA_RAW 0
- #define RSA_MD2 2
- #define RSA_MD4 3
- #define RSA_MD5 4
- #define RSA_SHA1 5
- #define RSA_SHA256 6
- #define RSA_PUBLIC 0
- #define RSA_PRIVATE 1
- #define RSA_PKCS_V15 0
- #define RSA_PKCS_V21 1
- #define RSA_SIGN 1
- #define RSA_CRYPT 2
- /*
- * DigestInfo ::= SEQUENCE {
- * digestAlgorithm DigestAlgorithmIdentifier,
- * digest Digest }
- *
- * DigestAlgorithmIdentifier ::= AlgorithmIdentifier
- *
- * Digest ::= OCTET STRING
- */
- #define ASN1_HASH_MDX \
- "\x30\x20\x30\x0C\x06\x08\x2A\x86\x48" \
- "\x86\xF7\x0D\x02\x00\x05\x00\x04\x10"
- #define ASN1_HASH_SHA1 \
- "\x30\x21\x30\x09\x06\x05\x2B\x0E\x03" \
- "\x02\x1A\x05\x00\x04\x14"
- /**
- * \brief RSA context structure
- */
- typedef struct
- {
- int ver; /*!< always 0 */
- int len; /*!< size(N) in chars */
- mpi N; /*!< public modulus */
- mpi E; /*!< public exponent */
- mpi D; /*!< private exponent */
- mpi P; /*!< 1st prime factor */
- mpi Q; /*!< 2nd prime factor */
- mpi DP; /*!< D % (P - 1) */
- mpi DQ; /*!< D % (Q - 1) */
- mpi QP; /*!< 1 / (Q % P) */
- mpi RN; /*!< cached R^2 mod N */
- mpi RP; /*!< cached R^2 mod P */
- mpi RQ; /*!< cached R^2 mod Q */
- int padding; /*!< 1.5 or OAEP/PSS */
- int hash_id; /*!< hash identifier */
- int (*f_rng)(void *); /*!< RNG function */
- void *p_rng; /*!< RNG parameter */
- }
- rsa_context;
- #ifdef __cplusplus
- extern "C" {
- #endif
- /**
- * \brief Initialize an RSA context
- *
- * \param ctx RSA context to be initialized
- * \param padding RSA_PKCS_V15 or RSA_PKCS_V21
- * \param hash_id RSA_PKCS_V21 hash identifier
- * \param f_rng RNG function
- * \param p_rng RNG parameter
- *
- * \note The hash_id parameter is actually ignored
- * when using RSA_PKCS_V15 padding.
- *
- * \note Currently (xyssl-0.8), RSA_PKCS_V21 padding
- * is not supported.
- */
- void rsa_init( rsa_context *ctx,
- int padding,
- int hash_id,
- int (*f_rng)(void *),
- void *p_rng );
- /**
- * \brief Generate an RSA keypair
- *
- * \param ctx RSA context that will hold the key
- * \param nbits size of the public key in bits
- * \param exponent public exponent (e.g., 65537)
- *
- * \note rsa_init() must be called beforehand to setup
- * the RSA context (especially f_rng and p_rng).
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- */
- int rsa_gen_key( rsa_context *ctx, int nbits, int exponent );
- /**
- * \brief Check a public RSA key
- *
- * \param ctx RSA context to be checked
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- */
- int rsa_check_pubkey( rsa_context *ctx );
- /**
- * \brief Check a private RSA key
- *
- * \param ctx RSA context to be checked
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- */
- int rsa_check_privkey( rsa_context *ctx );
- /**
- * \brief Do an RSA public key operation
- *
- * \param ctx RSA context
- * \param input input buffer
- * \param output output buffer
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- *
- * \note This function does NOT take care of message
- * padding. Also, be sure to set input[0] = 0.
- *
- * \note The input and output buffers must be large
- * enough (eg. 128 bytes if RSA-1024 is used).
- */
- int rsa_public( rsa_context *ctx,
- unsigned char *input,
- unsigned char *output );
- /**
- * \brief Do an RSA private key operation
- *
- * \param ctx RSA context
- * \param input input buffer
- * \param output output buffer
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- *
- * \note The input and output buffers must be large
- * enough (eg. 128 bytes if RSA-1024 is used).
- */
- int rsa_private( rsa_context *ctx,
- unsigned char *input,
- unsigned char *output );
- /**
- * \brief Add the message padding, then do an RSA operation
- *
- * \param ctx RSA context
- * \param mode RSA_PUBLIC or RSA_PRIVATE
- * \param ilen contains the the plaintext length
- * \param input buffer holding the data to be encrypted
- * \param output buffer that will hold the ciphertext
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- *
- * \note The output buffer must be as large as the size
- * of ctx->N (eg. 128 bytes if RSA-1024 is used).
- */
- int rsa_pkcs1_encrypt( rsa_context *ctx,
- int mode, int ilen,
- unsigned char *input,
- unsigned char *output );
- /**
- * \brief Do an RSA operation, then remove the message padding
- *
- * \param ctx RSA context
- * \param mode RSA_PUBLIC or RSA_PRIVATE
- * \param input buffer holding the encrypted data
- * \param output buffer that will hold the plaintext
- * \param olen will contain the plaintext length
- *
- * \return 0 if successful, or an XYSSL_ERR_RSA_XXX error code
- *
- * \note The output buffer must be as large as the size
- * of ctx->N (eg. 128 bytes if RSA-1024 is used).
- */
- int rsa_pkcs1_decrypt( rsa_context *ctx,
- int mode, int *olen,
- unsigned char *input,
- unsigned char *output );
- /**
- * \brief Do a private RSA to sign a message digest
- *
- * \param ctx RSA context
- * \param mode RSA_PUBLIC or RSA_PRIVATE
- * \param hash_id RSA_RAW, RSA_MD{2,4,5} or RSA_SHA{1,256}
- * \param hashlen message digest length (for RSA_RAW only)
- * \param hash buffer holding the message digest
- * \param sig buffer that will hold the ciphertext
- *
- * \return 0 if the signing operation was successful,
- * or an XYSSL_ERR_RSA_XXX error code
- *
- * \note The "sig" buffer must be as large as the size
- * of ctx->N (eg. 128 bytes if RSA-1024 is used).
- */
- int rsa_pkcs1_sign( rsa_context *ctx,
- int mode,
- int hash_id,
- int hashlen,
- unsigned char *hash,
- unsigned char *sig );
- /**
- * \brief Do a public RSA and check the message digest
- *
- * \param ctx points to an RSA public key
- * \param mode RSA_PUBLIC or RSA_PRIVATE
- * \param hash_id RSA_RAW, RSA_MD{2,4,5} or RSA_SHA{1,256}
- * \param hashlen message digest length (for RSA_RAW only)
- * \param hash buffer holding the message digest
- * \param sig buffer holding the ciphertext
- *
- * \return 0 if the verify operation was successful,
- * or an XYSSL_ERR_RSA_XXX error code
- *
- * \note The "sig" buffer must be as large as the size
- * of ctx->N (eg. 128 bytes if RSA-1024 is used).
- */
- int rsa_pkcs1_verify( rsa_context *ctx,
- int mode,
- int hash_id,
- int hashlen,
- unsigned char *hash,
- unsigned char *sig );
- /**
- * \brief Free the components of an RSA key
- */
- void rsa_free( rsa_context *ctx );
- /**
- * \brief Checkup routine
- *
- * \return 0 if successful, or 1 if the test failed
- */
- int rsa_self_test( int verbose );
- #ifdef __cplusplus
- }
- #endif
- #endif /* rsa.h */C source file: rsa.c
- /*
- * The RSA public-key cryptosystem
- *
- * Copyright (C) 2006-2007 Christophe Devine
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License, version 2.1 as published by the Free Software Foundation.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
- * MA 02110-1301 USA
- */
- /*
- * RSA was designed by Ron Rivest, Adi Shamir and Len Adleman.
- *
- * http://theory.lcs.mit.edu/~rivest/rsapaper.pdf
- * http://www.cacr.math.uwaterloo.ca/hac/about/chap8.pdf
- */
- #include "xyssl/config.h"
- #if defined(XYSSL_RSA_C)
- #include "xyssl/rsa.h"
- #include <stdlib.h>
- #include <string.h>
- #include <stdio.h>
- /*
- * Initialize an RSA context
- */
- void rsa_init( rsa_context *ctx,
- int padding,
- int hash_id,
- int (*f_rng)(void *),
- void *p_rng )
- {
- memset( ctx, 0, sizeof( rsa_context ) );
- ctx->padding = padding;
- ctx->hash_id = hash_id;
- ctx->f_rng = f_rng;
- ctx->p_rng = p_rng;
- }
- #if defined(XYSSL_GENPRIME)
- /*
- * Generate an RSA keypair
- */
- int rsa_gen_key( rsa_context *ctx, int nbits, int exponent )
- {
- int ret;
- mpi P1, Q1, H, G;
- if( ctx->f_rng == NULL || nbits < 128 || exponent < 3 )
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- mpi_init( &P1, &Q1, &H, &G, NULL );
- /*
- * find primes P and Q with Q < P so that:
- * GCD( E, (P-1)*(Q-1) ) == 1
- */
- MPI_CHK( mpi_lset( &ctx->E, exponent ) );
- nbits >>= 1;
- do
- {
- MPI_CHK( mpi_gen_prime( &ctx->P, nbits, 0,
- ctx->f_rng, ctx->p_rng ) );
- MPI_CHK( mpi_gen_prime( &ctx->Q, nbits, 0,
- ctx->f_rng, ctx->p_rng ) );
- if( mpi_cmp_mpi( &ctx->P, &ctx->Q ) < 0 )
- mpi_swap( &ctx->P, &ctx->Q );
- if( mpi_cmp_mpi( &ctx->P, &ctx->Q ) == 0 )
- continue;
- MPI_CHK( mpi_mul_mpi( &ctx->N, &ctx->P, &ctx->Q ) );
- MPI_CHK( mpi_sub_int( &P1, &ctx->P, 1 ) );
- MPI_CHK( mpi_sub_int( &Q1, &ctx->Q, 1 ) );
- MPI_CHK( mpi_mul_mpi( &H, &P1, &Q1 ) );
- MPI_CHK( mpi_gcd( &G, &ctx->E, &H ) );
- }
- while( mpi_cmp_int( &G, 1 ) != 0 );
- /*
- * D = E^-1 mod ((P-1)*(Q-1))
- * DP = D mod (P - 1)
- * DQ = D mod (Q - 1)
- * QP = Q^-1 mod P
- */
- MPI_CHK( mpi_inv_mod( &ctx->D , &ctx->E, &H ) );
- MPI_CHK( mpi_mod_mpi( &ctx->DP, &ctx->D, &P1 ) );
- MPI_CHK( mpi_mod_mpi( &ctx->DQ, &ctx->D, &Q1 ) );
- MPI_CHK( mpi_inv_mod( &ctx->QP, &ctx->Q, &ctx->P ) );
- ctx->len = ( mpi_msb( &ctx->N ) + 7 ) >> 3;
- cleanup:
- mpi_free( &G, &H, &Q1, &P1, NULL );
- if( ret != 0 )
- {
- rsa_free( ctx );
- return( XYSSL_ERR_RSA_KEY_GEN_FAILED | ret );
- }
- return( 0 );
- }
- #endif
- /*
- * Check a public RSA key
- */
- int rsa_check_pubkey( rsa_context *ctx )
- {
- if( ( ctx->N.p[0] & 1 ) == 0 ||
- ( ctx->E.p[0] & 1 ) == 0 )
- return( XYSSL_ERR_RSA_KEY_CHECK_FAILED );
- if( mpi_msb( &ctx->N ) < 128 ||
- mpi_msb( &ctx->N ) > 4096 )
- return( XYSSL_ERR_RSA_KEY_CHECK_FAILED );
- if( mpi_msb( &ctx->E ) < 2 ||
- mpi_msb( &ctx->E ) > 64 )
- return( XYSSL_ERR_RSA_KEY_CHECK_FAILED );
- return( 0 );
- }
- /*
- * Check a private RSA key
- */
- int rsa_check_privkey( rsa_context *ctx )
- {
- int ret;
- mpi TN, P1, Q1, H, G;
- if( ( ret = rsa_check_pubkey( ctx ) ) != 0 )
- return( ret );
- mpi_init( &TN, &P1, &Q1, &H, &G, NULL );
- MPI_CHK( mpi_mul_mpi( &TN, &ctx->P, &ctx->Q ) );
- MPI_CHK( mpi_sub_int( &P1, &ctx->P, 1 ) );
- MPI_CHK( mpi_sub_int( &Q1, &ctx->Q, 1 ) );
- MPI_CHK( mpi_mul_mpi( &H, &P1, &Q1 ) );
- MPI_CHK( mpi_gcd( &G, &ctx->E, &H ) );
- if( mpi_cmp_mpi( &TN, &ctx->N ) == 0 &&
- mpi_cmp_int( &G, 1 ) == 0 )
- {
- mpi_free( &TN, &P1, &Q1, &H, &G, NULL );
- return( 0 );
- }
- cleanup:
- mpi_free( &TN, &P1, &Q1, &H, &G, NULL );
- return( XYSSL_ERR_RSA_KEY_CHECK_FAILED | ret );
- }
- /*
- * Do an RSA public key operation
- */
- int rsa_public( rsa_context *ctx,
- unsigned char *input,
- unsigned char *output )
- {
- int ret, olen;
- mpi T;
- mpi_init( &T, NULL );
- MPI_CHK( mpi_read_binary( &T, input, ctx->len ) );
- if( mpi_cmp_mpi( &T, &ctx->N ) >= 0 )
- {
- mpi_free( &T, NULL );
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- }
- olen = ctx->len;
- MPI_CHK( mpi_exp_mod( &T, &T, &ctx->E, &ctx->N, &ctx->RN ) );
- MPI_CHK( mpi_write_binary( &T, output, olen ) );
- cleanup:
- mpi_free( &T, NULL );
- if( ret != 0 )
- return( XYSSL_ERR_RSA_PUBLIC_FAILED | ret );
- return( 0 );
- }
- /*
- * Do an RSA private key operation
- */
- int rsa_private( rsa_context *ctx,
- unsigned char *input,
- unsigned char *output )
- {
- int ret, olen;
- mpi T, T1, T2;
- mpi_init( &T, &T1, &T2, NULL );
- MPI_CHK( mpi_read_binary( &T, input, ctx->len ) );
- if( mpi_cmp_mpi( &T, &ctx->N ) >= 0 )
- {
- mpi_free( &T, NULL );
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- }
- #if 0
- MPI_CHK( mpi_exp_mod( &T, &T, &ctx->D, &ctx->N, &ctx->RN ) );
- #else
- /*
- * faster decryption using the CRT
- *
- * T1 = input ^ dP mod P
- * T2 = input ^ dQ mod Q
- */
- MPI_CHK( mpi_exp_mod( &T1, &T, &ctx->DP, &ctx->P, &ctx->RP ) );
- MPI_CHK( mpi_exp_mod( &T2, &T, &ctx->DQ, &ctx->Q, &ctx->RQ ) );
- /*
- * T = (T1 - T2) * (Q^-1 mod P) mod P
- */
- MPI_CHK( mpi_sub_mpi( &T, &T1, &T2 ) );
- MPI_CHK( mpi_mul_mpi( &T1, &T, &ctx->QP ) );
- MPI_CHK( mpi_mod_mpi( &T, &T1, &ctx->P ) );
- /*
- * output = T2 + T * Q
- */
- MPI_CHK( mpi_mul_mpi( &T1, &T, &ctx->Q ) );
- MPI_CHK( mpi_add_mpi( &T, &T2, &T1 ) );
- #endif
- olen = ctx->len;
- MPI_CHK( mpi_write_binary( &T, output, olen ) );
- cleanup:
- mpi_free( &T, &T1, &T2, NULL );
- if( ret != 0 )
- return( XYSSL_ERR_RSA_PRIVATE_FAILED | ret );
- return( 0 );
- }
- /*
- * Add the message padding, then do an RSA operation
- */
- int rsa_pkcs1_encrypt( rsa_context *ctx,
- int mode, int ilen,
- unsigned char *input,
- unsigned char *output )
- {
- int nb_pad, olen;
- unsigned char *p = output;
- olen = ctx->len;
- switch( ctx->padding )
- {
- case RSA_PKCS_V15:
- if( ilen < 0 || olen < ilen + 11 )
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- nb_pad = olen - 3 - ilen;
- *p++ = 0;
- *p++ = RSA_CRYPT;
- while( nb_pad-- > 0 )
- {
- do {
- *p = (unsigned char) rand();
- } while( *p == 0 );
- p++;
- }
- *p++ = 0;
- memcpy( p, input, ilen );
- break;
- default:
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- }
- return( ( mode == RSA_PUBLIC )
- ? rsa_public( ctx, output, output )
- : rsa_private( ctx, output, output ) );
- }
- /*
- * Do an RSA operation, then remove the message padding
- */
- int rsa_pkcs1_decrypt( rsa_context *ctx,
- int mode, int *olen,
- unsigned char *input,
- unsigned char *output )
- {
- int ret, ilen;
- unsigned char *p;
- unsigned char buf[512];
- ilen = ctx->len;
- if( ilen < 16 || ilen > (int) sizeof( buf ) )
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- ret = ( mode == RSA_PUBLIC )
- ? rsa_public( ctx, input, buf )
- : rsa_private( ctx, input, buf );
- if( ret != 0 )
- return( ret );
- p = buf;
- switch( ctx->padding )
- {
- case RSA_PKCS_V15:
- if( *p++ != 0 || *p++ != RSA_CRYPT )
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- while( *p != 0 )
- {
- if( p >= buf + ilen - 1 )
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- p++;
- }
- p++;
- break;
- default:
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- }
- *olen = ilen - (int)(p - buf);
- memcpy( output, p, *olen );
- return( 0 );
- }
- /*
- * Do an RSA operation to sign the message digest
- */
- int rsa_pkcs1_sign( rsa_context *ctx,
- int mode,
- int hash_id,
- int hashlen,
- unsigned char *hash,
- unsigned char *sig )
- {
- int nb_pad, olen;
- unsigned char *p = sig;
- olen = ctx->len;
- switch( ctx->padding )
- {
- case RSA_PKCS_V15:
- switch( hash_id )
- {
- case RSA_RAW:
- nb_pad = olen - 3 - hashlen;
- break;
- case RSA_MD2:
- case RSA_MD4:
- case RSA_MD5:
- nb_pad = olen - 3 - 34;
- break;
- case RSA_SHA1:
- nb_pad = olen - 3 - 35;
- break;
- default:
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- }
- if( nb_pad < 8 )
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- *p++ = 0;
- *p++ = RSA_SIGN;
- memset( p, 0xFF, nb_pad );
- p += nb_pad;
- *p++ = 0;
- break;
- default:
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- }
- switch( hash_id )
- {
- case RSA_RAW:
- memcpy( p, hash, hashlen );
- break;
- case RSA_MD2:
- memcpy( p, ASN1_HASH_MDX, 18 );
- memcpy( p + 18, hash, 16 );
- p[13] = 2; break;
- case RSA_MD4:
- memcpy( p, ASN1_HASH_MDX, 18 );
- memcpy( p + 18, hash, 16 );
- p[13] = 4; break;
- case RSA_MD5:
- memcpy( p, ASN1_HASH_MDX, 18 );
- memcpy( p + 18, hash, 16 );
- p[13] = 5; break;
- case RSA_SHA1:
- memcpy( p, ASN1_HASH_SHA1, 15 );
- memcpy( p + 15, hash, 20 );
- break;
- default:
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- }
- return( ( mode == RSA_PUBLIC )
- ? rsa_public( ctx, sig, sig )
- : rsa_private( ctx, sig, sig ) );
- }
- /*
- * Do an RSA operation and check the message digest
- */
- int rsa_pkcs1_verify( rsa_context *ctx,
- int mode,
- int hash_id,
- int hashlen,
- unsigned char *hash,
- unsigned char *sig )
- {
- int ret, len, siglen;
- unsigned char *p, c;
- unsigned char buf[512];
- siglen = ctx->len;
- if( siglen < 16 || siglen > (int) sizeof( buf ) )
- return( XYSSL_ERR_RSA_BAD_INPUT_DATA );
- ret = ( mode == RSA_PUBLIC )
- ? rsa_public( ctx, sig, buf )
- : rsa_private( ctx, sig, buf );
- if( ret != 0 )
- return( ret );
- p = buf;
- switch( ctx->padding )
- {
- case RSA_PKCS_V15:
- if( *p++ != 0 || *p++ != RSA_SIGN )
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- while( *p != 0 )
- {
- if( p >= buf + siglen - 1 || *p != 0xFF )
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- p++;
- }
- p++;
- break;
- default:
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- }
- len = siglen - (int)( p - buf );
- if( len == 34 )
- {
- c = p[13];
- p[13] = 0;
- if( memcmp( p, ASN1_HASH_MDX, 18 ) != 0 )
- return( XYSSL_ERR_RSA_VERIFY_FAILED );
- if( ( c == 2 && hash_id == RSA_MD2 ) ||
- ( c == 4 && hash_id == RSA_MD4 ) ||
- ( c == 5 && hash_id == RSA_MD5 ) )
- {
- if( memcmp( p + 18, hash, 16 ) == 0 )
- return( 0 );
- else
- return( XYSSL_ERR_RSA_VERIFY_FAILED );
- }
- }
- if( len == 35 && hash_id == RSA_SHA1 )
- {
- if( memcmp( p, ASN1_HASH_SHA1, 15 ) == 0 &&
- memcmp( p + 15, hash, 20 ) == 0 )
- return( 0 );
- else
- return( XYSSL_ERR_RSA_VERIFY_FAILED );
- }
- if( len == hashlen && hash_id == RSA_RAW )
- {
- if( memcmp( p, hash, hashlen ) == 0 )
- return( 0 );
- else
- return( XYSSL_ERR_RSA_VERIFY_FAILED );
- }
- return( XYSSL_ERR_RSA_INVALID_PADDING );
- }
- /*
- * Free the components of an RSA key
- */
- void rsa_free( rsa_context *ctx )
- {
- mpi_free( &ctx->RQ, &ctx->RP, &ctx->RN,
- &ctx->QP, &ctx->DQ, &ctx->DP,
- &ctx->Q, &ctx->P, &ctx->D,
- &ctx->E, &ctx->N, NULL );
- }
- #if defined(XYSSL_SELF_TEST)
- #include "xyssl/sha1.h"
- /*
- * Example RSA-1024 keypair, for test purposes
- */
- #define KEY_LEN 128
- #define RSA_N "9292758453063D803DD603D5E777D788" \
- "8ED1D5BF35786190FA2F23EBC0848AEA" \
- "DDA92CA6C3D80B32C4D109BE0F36D6AE" \
- "7130B9CED7ACDF54CFC7555AC14EEBAB" \
- "93A89813FBF3C4F8066D2D800F7C38A8" \
- "1AE31942917403FF4946B0A83D3D3E05" \
- "EE57C6F5F5606FB5D4BC6CD34EE0801A" \
- "5E94BB77B07507233A0BC7BAC8F90F79"
- #define RSA_E "10001"
- #define RSA_D "24BF6185468786FDD303083D25E64EFC" \
- "66CA472BC44D253102F8B4A9D3BFA750" \
- "91386C0077937FE33FA3252D28855837" \
- "AE1B484A8A9A45F7EE8C0C634F99E8CD" \
- "DF79C5CE07EE72C7F123142198164234" \
- "CABB724CF78B8173B9F880FC86322407" \
- "AF1FEDFDDE2BEB674CA15F3E81A1521E" \
- "071513A1E85B5DFA031F21ECAE91A34D"
- #define RSA_P "C36D0EB7FCD285223CFB5AABA5BDA3D8" \
- "2C01CAD19EA484A87EA4377637E75500" \
- "FCB2005C5C7DD6EC4AC023CDA285D796" \
- "C3D9E75E1EFC42488BB4F1D13AC30A57"
- #define RSA_Q "C000DF51A7C77AE8D7C7370C1FF55B69" \
- "E211C2B9E5DB1ED0BF61D0D9899620F4" \
- "910E4168387E3C30AA1E00C339A79508" \
- "8452DD96A9A5EA5D9DCA68DA636032AF"
- #define RSA_DP "C1ACF567564274FB07A0BBAD5D26E298" \
- "3C94D22288ACD763FD8E5600ED4A702D" \
- "F84198A5F06C2E72236AE490C93F07F8" \
- "3CC559CD27BC2D1CA488811730BB5725"
- #define RSA_DQ "4959CBF6F8FEF750AEE6977C155579C7" \
- "D8AAEA56749EA28623272E4F7D0592AF" \
- "7C1F1313CAC9471B5C523BFE592F517B" \
- "407A1BD76C164B93DA2D32A383E58357"
- #define RSA_QP "9AE7FBC99546432DF71896FC239EADAE" \
- "F38D18D2B2F0E2DD275AA977E2BF4411" \
- "F5A3B2A5D33605AEBBCCBA7FEB9F2D2F" \
- "A74206CEC169D74BF5A8C50D6F48EA08"
- #define PT_LEN 24
- #define RSA_PT "\xAA\xBB\xCC\x03\x02\x01\x00\xFF\xFF\xFF\xFF\xFF" \
- "\x11\x22\x33\x0A\x0B\x0C\xCC\xDD\xDD\xDD\xDD\xDD"
- /*
- * Checkup routine
- */
- int rsa_self_test( int verbose )
- {
- int len;
- rsa_context rsa;
- unsigned char sha1sum[20];
- unsigned char rsa_plaintext[PT_LEN];
- unsigned char rsa_decrypted[PT_LEN];
- unsigned char rsa_ciphertext[KEY_LEN];
- memset( &rsa, 0, sizeof( rsa_context ) );
- rsa.len = KEY_LEN;
- mpi_read_string( &rsa.N , 16, RSA_N );
- mpi_read_string( &rsa.E , 16, RSA_E );
- mpi_read_string( &rsa.D , 16, RSA_D );
- mpi_read_string( &rsa.P , 16, RSA_P );
- mpi_read_string( &rsa.Q , 16, RSA_Q );
- mpi_read_string( &rsa.DP, 16, RSA_DP );
- mpi_read_string( &rsa.DQ, 16, RSA_DQ );
- mpi_read_string( &rsa.QP, 16, RSA_QP );
- if( verbose != 0 )
- printf( " RSA key validation: " );
- if( rsa_check_pubkey( &rsa ) != 0 ||
- rsa_check_privkey( &rsa ) != 0 )
- {
- if( verbose != 0 )
- printf( "failed\n" );
- return( 1 );
- }
- if( verbose != 0 )
- printf( "passed\n PKCS#1 encryption : " );
- memcpy( rsa_plaintext, RSA_PT, PT_LEN );
- if( rsa_pkcs1_encrypt( &rsa, RSA_PUBLIC, PT_LEN,
- rsa_plaintext, rsa_ciphertext ) != 0 )
- {
- if( verbose != 0 )
- printf( "failed\n" );
- return( 1 );
- }
- if( verbose != 0 )
- printf( "passed\n PKCS#1 decryption : " );
- if( rsa_pkcs1_decrypt( &rsa, RSA_PRIVATE, &len,
- rsa_ciphertext, rsa_decrypted ) != 0 )
- {
- if( verbose != 0 )
- printf( "failed\n" );
- return( 1 );
- }
- if( memcmp( rsa_decrypted, rsa_plaintext, len ) != 0 )
- {
- if( verbose != 0 )
- printf( "failed\n" );
- return( 1 );
- }
- if( verbose != 0 )
- printf( "passed\n PKCS#1 data sign : " );
- sha1( rsa_plaintext, PT_LEN, sha1sum );
- if( rsa_pkcs1_sign( &rsa, RSA_PRIVATE, RSA_SHA1, 20,
- sha1sum, rsa_ciphertext ) != 0 )
- {
- if( verbose != 0 )
- printf( "failed\n" );
- return( 1 );
- }
- if( verbose != 0 )
- printf( "passed\n PKCS#1 sig. verify: " );
- if( rsa_pkcs1_verify( &rsa, RSA_PUBLIC, RSA_SHA1, 20,
- sha1sum, rsa_ciphertext ) != 0 )
- {
- if( verbose != 0 )
- printf( "failed\n" );
- return( 1 );
- }
- if( verbose != 0 )
- printf( "passed\n\n" );
- rsa_free( &rsa );
- return( 0 );
- }
- #endif
- #endif[edit] java code
- // RSAPublicKey: RSA public key
- import java.math.*; // for BigInteger
- public class RSAPublicKey {
- public BigInteger n; // public modulus
- public BigInteger e = new BigInteger("3"); // encryption exponent
- public String userName; // attach name to each public/private key pair
- public RSAPublicKey(String name) {
- userName = name;
- }
- // setN: to give n a value in case only have public key
- public void setN(BigInteger newN) {
- n = newN;
- }
- // getN: provide n
- public BigInteger getN() {
- return n;
- }
- // RSAEncrypt: just raise m to power e (3) mod n
- public BigInteger RSAEncrypt(BigInteger m) {
- return m.modPow(e, n);
- }
- // RSAVerify: same as encryption, since RSA is symmetric
- public BigInteger RSAVerify(BigInteger s) {
- return s.modPow(e, n);
- }
- }
- // RSAPrivateKey: RSA private key
- import java.math.*; // for BigInteger
- import java.util.*; // for Random
- public class RSAPrivateKey extends RSAPublicKey{
- private final BigInteger TWO = new BigInteger("2");
- private final BigInteger THREE = new BigInteger("3");
- private BigInteger p; // first prime
- private BigInteger q; // second prime
- private BigInteger d; // decryption exponent
- public RSAPrivateKey(int size, Random rnd, String name) {
- super(name); generateKeyPair(size, rnd);
- }
- public void generateKeyPair(int size, Random rnd) { // size = n in bits
- // want sizes of primes close, but not too close. Here 10-20 bits apart.
- int size1 = size/2;
- int size2 = size1;
- int offset1 = (int)(5.0*(rnd.nextDouble()) + 5.0);
- int offset2 = -offset1;
- if (rnd.nextDouble() < 0.5) {
- offset1 = -offset1; offset2 = -offset2;
- }
- size1 += offset1; size2 += offset2;
- // generate two random primes, so that p*q = n has size bits
- BigInteger p1 = new BigInteger(size1, rnd); // random int
- p = nextPrime(p1);
- BigInteger pM1 = p.subtract(BigInteger.ONE);
- BigInteger q1 = new BigInteger(size2, rnd);
- q = nextPrime(q1);
- BigInteger qM1 = q.subtract(BigInteger.ONE);
- n = p.multiply(q);
- BigInteger phiN = pM1.multiply(qM1); // (p-1)*(q-1)
- BigInteger e = THREE;
- d = e.modInverse(phiN);
- }
- // nextPrime: next prime p after x, with p-1 and 3 relatively prime
- public BigInteger nextPrime(BigInteger x) {
- if ((x.remainder(TWO)).equals(BigInteger.ZERO))
- x = x.add(BigInteger.ONE);
- while(true) {
- BigInteger xM1 = x.subtract(BigInteger.ONE);
- if (!(xM1.remainder(THREE)).equals(BigInteger.ZERO))
- if (x.isProbablePrime(10)) break;
- x = x.add(TWO);
- }
- return x;
- }
- // RSADecrypt: decryption function
- public BigInteger RSADecrypt(BigInteger c) {
- return c.modPow(d, n);
- }
- // RSASign: same as decryption for RSA (since it is a symmetric PKC)
- public BigInteger RSASign(BigInteger m) {
- return m.modPow(d, n);
- }
- public BigInteger RSASignAndEncrypt(BigInteger m, RSAPublicKey other) {
- // two ways to go, depending on sizes of n and other.getN()
- if (n.compareTo(other.getN()) > 0)
- return RSASign(other.RSAEncrypt(m));
- else
- return other.RSAEncrypt(RSASign(m));
- }
- public BigInteger RSADecryptAndVerify(BigInteger c,
- RSAPrivateKey other) {
- // two ways to go, depending on sizes of n and other.getN()
- if (n.compareTo(other.getN()) > 0)
- return other.RSAVerify(RSADecrypt(c));
- else
- return RSADecrypt(other.RSAVerify(c));
- }
- }
- --------------------------------------------------------------------------------
- // RSATest: Test RSA Implementation
- import java.math.*; // for BigInteger
- import java.util.*; // for Random
- public class RSATest {
- public static void main(String[] args) {
- Random rnd = new Random();
- BigInteger m, m1, m2, m3, c, s, s1;
- RSAPrivateKey alice = new RSAPrivateKey(1024, rnd, "Alice");
- RSAPrivateKey bob = new RSAPrivateKey(1024, rnd, "Bob ");
- m = new BigInteger(
- "1234567890987654321012345678909876543210" +
- "1234567890987654321012345678909876543210" +
- "1234567890987654321012345678909876543210" +
- "1234567890987654321012345678909876543210" +
- "1234567890987654321012345678909876543210" +
- "1234567890987654321012345678909876543210");
- System.out.println("Message m:\n" + m + "\n");
- System.out.println("ALICE ENCRYPTS m FOR BOB; BOB DECRYPTS IT:");
- c = bob.RSAEncrypt(m); // Using Bob's public key
- System.out.println("Message encrypted with Bob's public key:\n" +
- c + "\n");
- m1 = bob.RSADecrypt(c); // Using Bob's private key
- System.out.println("Original message back, decrypted:\n" + m1 + "\n");
- System.out.println("ALICE SIGNS m FOR BOB; BOB VERIFIES SIGNATURE:");
- s = alice.RSASign(m); // Using Alice's private key
- System.out.println("Message signed with Alice's private key:\n" +
- c + "\n");
- m2 = alice.RSAVerify(s); // Using Alice's public key
- System.out.println("Original message back, verified:\n" + m2 + "\n");
- System.out.println("BOB SIGNS AND ENCRYPTS m FOR ALICE;" +
- "\n ALICE VERIFIES SIGNATURE AND DECRYPTS:");
- c = bob.RSASignAndEncrypt(m, alice);
- System.out.println("Message signed and encrypted," +
- "\n using Bob's secret key and Alice's public key:\n" + c + "\n");
- m3 = alice.RSADecryptAndVerify(c, bob);
- System.out.println("Original message back, verified and decrypted," +
- "\n using Alice's secret key and Bob's public key:\n" + m1);
- }
- }
复制代码 |
|