2010-11-02 07:06:38 +01:00
|
|
|
/** @file
|
|
|
|
Diffie-Hellman Wrapper Implementation over OpenSSL.
|
|
|
|
|
2018-06-27 11:32:13 +02:00
|
|
|
Copyright (c) 2010 - 2018, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:03:30 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "InternalCryptLib.h"
|
2015-10-29 15:15:53 +01:00
|
|
|
#include <openssl/bn.h>
|
2010-11-02 07:06:38 +01:00
|
|
|
#include <openssl/dh.h>
|
|
|
|
|
|
|
|
/**
|
|
|
|
Allocates and Initializes one Diffie-Hellman Context for subsequent use.
|
|
|
|
|
|
|
|
@return Pointer to the Diffie-Hellman Context that has been initialized.
|
|
|
|
If the allocations fails, DhNew() returns NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID *
|
|
|
|
EFIAPI
|
|
|
|
DhNew (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Allocates & Initializes DH Context by OpenSSL DH_new()
|
|
|
|
//
|
2012-07-27 09:49:42 +02:00
|
|
|
return (VOID *) DH_new ();
|
2010-11-02 07:06:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Release the specified DH context.
|
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
If DhContext is NULL, then return FALSE.
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
@param[in] DhContext Pointer to the DH context to be released.
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
DhFree (
|
|
|
|
IN VOID *DhContext
|
|
|
|
)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Free OpenSSL DH Context
|
|
|
|
//
|
2012-07-27 09:49:42 +02:00
|
|
|
DH_free ((DH *) DhContext);
|
2010-11-02 07:06:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Generates DH parameter.
|
|
|
|
|
|
|
|
Given generator g, and length of prime number p in bits, this function generates p,
|
|
|
|
and sets DH context according to value of g and p.
|
2018-06-27 11:32:13 +02:00
|
|
|
|
2010-11-02 07:06:38 +01:00
|
|
|
Before this function can be invoked, pseudorandom number generator must be correctly
|
|
|
|
initialized by RandomSeed().
|
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
If DhContext is NULL, then return FALSE.
|
|
|
|
If Prime is NULL, then return FALSE.
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
@param[in, out] DhContext Pointer to the DH context.
|
|
|
|
@param[in] Generator Value of generator.
|
|
|
|
@param[in] PrimeLength Length in bits of prime to be generated.
|
|
|
|
@param[out] Prime Pointer to the buffer to receive the generated prime number.
|
|
|
|
|
2016-10-19 09:01:10 +02:00
|
|
|
@retval TRUE DH parameter generation succeeded.
|
2010-11-02 07:06:38 +01:00
|
|
|
@retval FALSE Value of Generator is not supported.
|
|
|
|
@retval FALSE PRNG fails to generate random prime number with PrimeLength.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
DhGenerateParameter (
|
|
|
|
IN OUT VOID *DhContext,
|
|
|
|
IN UINTN Generator,
|
|
|
|
IN UINTN PrimeLength,
|
|
|
|
OUT UINT8 *Prime
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BOOLEAN RetVal;
|
2017-03-21 15:58:07 +01:00
|
|
|
BIGNUM *BnP;
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
//
|
|
|
|
// Check input parameters.
|
|
|
|
//
|
2012-08-02 04:49:24 +02:00
|
|
|
if (DhContext == NULL || Prime == NULL || PrimeLength > INT_MAX) {
|
2012-03-19 06:52:16 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-11-02 07:06:38 +01:00
|
|
|
if (Generator != DH_GENERATOR_2 && Generator != DH_GENERATOR_5) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
RetVal = (BOOLEAN) DH_generate_parameters_ex (DhContext, (UINT32) PrimeLength, (UINT32) Generator, NULL);
|
|
|
|
if (!RetVal) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-03-21 15:58:07 +01:00
|
|
|
DH_get0_pqg (DhContext, (const BIGNUM **)&BnP, NULL, NULL);
|
|
|
|
BN_bn2bin (BnP, Prime);
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Sets generator and prime parameters for DH.
|
|
|
|
|
|
|
|
Given generator g, and prime number p, this function and sets DH
|
|
|
|
context accordingly.
|
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
If DhContext is NULL, then return FALSE.
|
|
|
|
If Prime is NULL, then return FALSE.
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
@param[in, out] DhContext Pointer to the DH context.
|
|
|
|
@param[in] Generator Value of generator.
|
|
|
|
@param[in] PrimeLength Length in bits of prime to be generated.
|
|
|
|
@param[in] Prime Pointer to the prime number.
|
|
|
|
|
2016-10-19 09:01:10 +02:00
|
|
|
@retval TRUE DH parameter setting succeeded.
|
2010-11-02 07:06:38 +01:00
|
|
|
@retval FALSE Value of Generator is not supported.
|
|
|
|
@retval FALSE Value of Generator is not suitable for the Prime.
|
|
|
|
@retval FALSE Value of Prime is not a prime number.
|
|
|
|
@retval FALSE Value of Prime is not a safe prime number.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
DhSetParameter (
|
|
|
|
IN OUT VOID *DhContext,
|
|
|
|
IN UINTN Generator,
|
|
|
|
IN UINTN PrimeLength,
|
|
|
|
IN CONST UINT8 *Prime
|
|
|
|
)
|
|
|
|
{
|
2012-08-02 04:49:24 +02:00
|
|
|
DH *Dh;
|
2017-03-21 15:58:07 +01:00
|
|
|
BIGNUM *BnP;
|
|
|
|
BIGNUM *BnG;
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
//
|
|
|
|
// Check input parameters.
|
|
|
|
//
|
2012-08-02 04:49:24 +02:00
|
|
|
if (DhContext == NULL || Prime == NULL || PrimeLength > INT_MAX) {
|
2012-03-19 06:52:16 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
2017-03-21 15:58:07 +01:00
|
|
|
|
2010-11-02 07:06:38 +01:00
|
|
|
if (Generator != DH_GENERATOR_2 && Generator != DH_GENERATOR_5) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-03-21 15:58:07 +01:00
|
|
|
//
|
|
|
|
// Set the generator and prime parameters for DH object.
|
|
|
|
//
|
|
|
|
Dh = (DH *)DhContext;
|
|
|
|
BnP = BN_bin2bn ((const unsigned char *)Prime, (int)(PrimeLength / 8), NULL);
|
|
|
|
BnG = BN_bin2bn ((const unsigned char *)&Generator, 1, NULL);
|
|
|
|
if ((BnP == NULL) || (BnG == NULL) || !DH_set0_pqg (Dh, BnP, NULL, BnG)) {
|
2012-08-02 04:49:24 +02:00
|
|
|
goto Error;
|
|
|
|
}
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
return TRUE;
|
2012-08-02 04:49:24 +02:00
|
|
|
|
|
|
|
Error:
|
2017-03-21 15:58:07 +01:00
|
|
|
BN_free (BnP);
|
|
|
|
BN_free (BnG);
|
2012-08-02 04:49:24 +02:00
|
|
|
|
|
|
|
return FALSE;
|
2010-11-02 07:06:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Generates DH public key.
|
|
|
|
|
2018-06-27 11:32:13 +02:00
|
|
|
This function generates random secret exponent, and computes the public key, which is
|
2010-11-02 07:06:38 +01:00
|
|
|
returned via parameter PublicKey and PublicKeySize. DH context is updated accordingly.
|
|
|
|
If the PublicKey buffer is too small to hold the public key, FALSE is returned and
|
|
|
|
PublicKeySize is set to the required buffer size to obtain the public key.
|
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
If DhContext is NULL, then return FALSE.
|
|
|
|
If PublicKeySize is NULL, then return FALSE.
|
|
|
|
If PublicKeySize is large enough but PublicKey is NULL, then return FALSE.
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
@param[in, out] DhContext Pointer to the DH context.
|
|
|
|
@param[out] PublicKey Pointer to the buffer to receive generated public key.
|
|
|
|
@param[in, out] PublicKeySize On input, the size of PublicKey buffer in bytes.
|
|
|
|
On output, the size of data returned in PublicKey buffer in bytes.
|
|
|
|
|
|
|
|
@retval TRUE DH public key generation succeeded.
|
|
|
|
@retval FALSE DH public key generation failed.
|
|
|
|
@retval FALSE PublicKeySize is not large enough.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
DhGenerateKey (
|
|
|
|
IN OUT VOID *DhContext,
|
|
|
|
OUT UINT8 *PublicKey,
|
|
|
|
IN OUT UINTN *PublicKeySize
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BOOLEAN RetVal;
|
|
|
|
DH *Dh;
|
2017-03-21 15:58:07 +01:00
|
|
|
BIGNUM *DhPubKey;
|
2012-08-02 04:49:24 +02:00
|
|
|
INTN Size;
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
//
|
|
|
|
// Check input parameters.
|
|
|
|
//
|
|
|
|
if (DhContext == NULL || PublicKeySize == NULL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PublicKey == NULL && *PublicKeySize != 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2017-03-21 15:58:07 +01:00
|
|
|
|
2010-11-02 07:06:38 +01:00
|
|
|
Dh = (DH *) DhContext;
|
|
|
|
|
|
|
|
RetVal = (BOOLEAN) DH_generate_key (DhContext);
|
|
|
|
if (RetVal) {
|
2017-03-21 15:58:07 +01:00
|
|
|
DH_get0_key (Dh, (const BIGNUM **)&DhPubKey, NULL);
|
|
|
|
Size = BN_num_bytes (DhPubKey);
|
|
|
|
if ((Size > 0) && (*PublicKeySize < (UINTN) Size)) {
|
2012-08-02 04:49:24 +02:00
|
|
|
*PublicKeySize = Size;
|
|
|
|
return FALSE;
|
|
|
|
}
|
2017-03-21 15:58:07 +01:00
|
|
|
|
2017-05-19 09:22:10 +02:00
|
|
|
if (PublicKey != NULL) {
|
|
|
|
BN_bn2bin (DhPubKey, PublicKey);
|
|
|
|
}
|
2012-08-02 04:49:24 +02:00
|
|
|
*PublicKeySize = Size;
|
2010-11-02 07:06:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return RetVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Computes exchanged common key.
|
|
|
|
|
|
|
|
Given peer's public key, this function computes the exchanged common key, based on its own
|
2018-06-27 11:32:13 +02:00
|
|
|
context including value of prime modulus and random secret exponent.
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
If DhContext is NULL, then return FALSE.
|
|
|
|
If PeerPublicKey is NULL, then return FALSE.
|
|
|
|
If KeySize is NULL, then return FALSE.
|
2012-08-02 04:49:24 +02:00
|
|
|
If Key is NULL, then return FALSE.
|
|
|
|
If KeySize is not large enough, then return FALSE.
|
2010-11-02 07:06:38 +01:00
|
|
|
|
|
|
|
@param[in, out] DhContext Pointer to the DH context.
|
|
|
|
@param[in] PeerPublicKey Pointer to the peer's public key.
|
|
|
|
@param[in] PeerPublicKeySize Size of peer's public key in bytes.
|
|
|
|
@param[out] Key Pointer to the buffer to receive generated key.
|
|
|
|
@param[in, out] KeySize On input, the size of Key buffer in bytes.
|
|
|
|
On output, the size of data returned in Key buffer in bytes.
|
|
|
|
|
|
|
|
@retval TRUE DH exchanged key generation succeeded.
|
|
|
|
@retval FALSE DH exchanged key generation failed.
|
|
|
|
@retval FALSE KeySize is not large enough.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
DhComputeKey (
|
|
|
|
IN OUT VOID *DhContext,
|
|
|
|
IN CONST UINT8 *PeerPublicKey,
|
|
|
|
IN UINTN PeerPublicKeySize,
|
|
|
|
OUT UINT8 *Key,
|
|
|
|
IN OUT UINTN *KeySize
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BIGNUM *Bn;
|
2012-08-02 04:49:24 +02:00
|
|
|
INTN Size;
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
//
|
|
|
|
// Check input parameters.
|
|
|
|
//
|
2012-08-02 04:49:24 +02:00
|
|
|
if (DhContext == NULL || PeerPublicKey == NULL || KeySize == NULL || Key == NULL) {
|
2012-03-19 06:52:16 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-08-02 04:49:24 +02:00
|
|
|
if (PeerPublicKeySize > INT_MAX) {
|
2012-03-19 06:52:16 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
2018-06-27 11:32:13 +02:00
|
|
|
|
2010-11-02 07:06:38 +01:00
|
|
|
Bn = BN_bin2bn (PeerPublicKey, (UINT32) PeerPublicKeySize, NULL);
|
2012-08-02 04:49:24 +02:00
|
|
|
if (Bn == NULL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-08-02 04:49:24 +02:00
|
|
|
Size = DH_compute_key (Key, Bn, DhContext);
|
|
|
|
if (Size < 0) {
|
|
|
|
BN_free (Bn);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-08-02 04:49:24 +02:00
|
|
|
if (*KeySize < (UINTN) Size) {
|
|
|
|
*KeySize = Size;
|
|
|
|
BN_free (Bn);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2010-11-02 07:06:38 +01:00
|
|
|
|
2012-08-02 04:49:24 +02:00
|
|
|
*KeySize = Size;
|
|
|
|
BN_free (Bn);
|
2010-11-02 07:06:38 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|