2010-12-31 08:22:48 +01:00
|
|
|
/** @file
|
|
|
|
PEM (Privacy Enhanced Mail) Format Handler Wrapper Implementation over OpenSSL.
|
|
|
|
|
2020-04-22 11:48:02 +02:00
|
|
|
Copyright (c) 2010 - 2020, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:03:30 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2010-12-31 08:22:48 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "InternalCryptLib.h"
|
|
|
|
#include <openssl/pem.h>
|
|
|
|
|
|
|
|
/**
|
|
|
|
Callback function for password phrase conversion used for retrieving the encrypted PEM.
|
|
|
|
|
|
|
|
@param[out] Buf Pointer to the buffer to write the passphrase to.
|
|
|
|
@param[in] Size Maximum length of the passphrase (i.e. the size of Buf).
|
|
|
|
@param[in] Flag A flag which is set to 0 when reading and 1 when writing.
|
|
|
|
@param[in] Key Key data to be passed to the callback routine.
|
|
|
|
|
|
|
|
@retval The number of characters in the passphrase or 0 if an error occurred.
|
|
|
|
|
|
|
|
**/
|
|
|
|
INTN
|
|
|
|
PasswordCallback (
|
2018-06-27 11:32:13 +02:00
|
|
|
OUT CHAR8 *Buf,
|
|
|
|
IN INTN Size,
|
|
|
|
IN INTN Flag,
|
2010-12-31 08:22:48 +01:00
|
|
|
IN VOID *Key
|
|
|
|
)
|
|
|
|
{
|
|
|
|
INTN KeyLength;
|
|
|
|
|
2021-12-05 23:53:54 +01:00
|
|
|
ZeroMem ((VOID *)Buf, (UINTN)Size);
|
2010-12-31 08:22:48 +01:00
|
|
|
if (Key != NULL) {
|
|
|
|
//
|
|
|
|
// Duplicate key phrase directly.
|
|
|
|
//
|
2021-12-05 23:53:54 +01:00
|
|
|
KeyLength = (INTN)AsciiStrLen ((CHAR8 *)Key);
|
|
|
|
KeyLength = (KeyLength > Size) ? Size : KeyLength;
|
|
|
|
CopyMem (Buf, Key, (UINTN)KeyLength);
|
2010-12-31 08:22:48 +01:00
|
|
|
return KeyLength;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Retrieve the RSA Private Key from the password-protected PEM key data.
|
|
|
|
|
|
|
|
@param[in] PemData Pointer to the PEM-encoded key data to be retrieved.
|
|
|
|
@param[in] PemSize Size of the PEM key data in bytes.
|
|
|
|
@param[in] Password NULL-terminated passphrase used for encrypted PEM key data.
|
|
|
|
@param[out] RsaContext Pointer to new-generated RSA context which contain the retrieved
|
|
|
|
RSA private key component. Use RsaFree() function to free the
|
|
|
|
resource.
|
|
|
|
|
2012-03-19 06:52:16 +01:00
|
|
|
If PemData is NULL, then return FALSE.
|
|
|
|
If RsaContext is NULL, then return FALSE.
|
2010-12-31 08:22:48 +01:00
|
|
|
|
|
|
|
@retval TRUE RSA Private Key was retrieved successfully.
|
|
|
|
@retval FALSE Invalid PEM key data or incorrect password.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
RsaGetPrivateKeyFromPem (
|
|
|
|
IN CONST UINT8 *PemData,
|
|
|
|
IN UINTN PemSize,
|
|
|
|
IN CONST CHAR8 *Password,
|
|
|
|
OUT VOID **RsaContext
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BOOLEAN Status;
|
|
|
|
BIO *PemBio;
|
|
|
|
|
|
|
|
//
|
2012-03-19 06:52:16 +01:00
|
|
|
// Check input parameters.
|
2010-12-31 08:22:48 +01:00
|
|
|
//
|
2021-12-05 23:53:54 +01:00
|
|
|
if ((PemData == NULL) || (RsaContext == NULL) || (PemSize > INT_MAX)) {
|
2012-03-19 06:52:16 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
2011-10-28 09:41:26 +02:00
|
|
|
|
2010-12-31 08:22:48 +01:00
|
|
|
//
|
|
|
|
// Add possible block-cipher descriptor for PEM data decryption.
|
2020-04-22 11:48:02 +02:00
|
|
|
// NOTE: Only support most popular ciphers AES for the encrypted PEM.
|
2010-12-31 08:22:48 +01:00
|
|
|
//
|
2012-08-02 04:49:24 +02:00
|
|
|
if (EVP_add_cipher (EVP_aes_128_cbc ()) == 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2021-12-05 23:53:54 +01:00
|
|
|
|
2012-08-02 04:49:24 +02:00
|
|
|
if (EVP_add_cipher (EVP_aes_192_cbc ()) == 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2021-12-05 23:53:54 +01:00
|
|
|
|
2012-08-02 04:49:24 +02:00
|
|
|
if (EVP_add_cipher (EVP_aes_256_cbc ()) == 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = FALSE;
|
2010-12-31 08:22:48 +01:00
|
|
|
|
|
|
|
//
|
|
|
|
// Read encrypted PEM Data.
|
|
|
|
//
|
|
|
|
PemBio = BIO_new (BIO_s_mem ());
|
|
|
|
if (PemBio == NULL) {
|
|
|
|
goto _Exit;
|
|
|
|
}
|
|
|
|
|
2021-12-05 23:53:54 +01:00
|
|
|
if (BIO_write (PemBio, PemData, (int)PemSize) <= 0) {
|
2013-08-07 10:11:14 +02:00
|
|
|
goto _Exit;
|
|
|
|
}
|
|
|
|
|
2010-12-31 08:22:48 +01:00
|
|
|
//
|
|
|
|
// Retrieve RSA Private Key from encrypted PEM data.
|
|
|
|
//
|
2021-12-05 23:53:54 +01:00
|
|
|
*RsaContext = PEM_read_bio_RSAPrivateKey (PemBio, NULL, (pem_password_cb *)&PasswordCallback, (void *)Password);
|
2010-12-31 08:22:48 +01:00
|
|
|
if (*RsaContext != NULL) {
|
|
|
|
Status = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_Exit:
|
|
|
|
//
|
|
|
|
// Release Resources.
|
|
|
|
//
|
|
|
|
BIO_free (PemBio);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
2022-10-12 04:47:58 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
Retrieve the EC Private Key from the password-protected PEM key data.
|
|
|
|
|
|
|
|
@param[in] PemData Pointer to the PEM-encoded key data to be retrieved.
|
|
|
|
@param[in] PemSize Size of the PEM key data in bytes.
|
|
|
|
@param[in] Password NULL-terminated passphrase used for encrypted PEM key data.
|
|
|
|
@param[out] EcContext Pointer to new-generated EC DSA context which contain the retrieved
|
|
|
|
EC private key component. Use EcFree() function to free the
|
|
|
|
resource.
|
|
|
|
|
|
|
|
If PemData is NULL, then return FALSE.
|
|
|
|
If EcContext is NULL, then return FALSE.
|
|
|
|
|
|
|
|
@retval TRUE EC Private Key was retrieved successfully.
|
|
|
|
@retval FALSE Invalid PEM key data or incorrect password.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
EcGetPrivateKeyFromPem (
|
|
|
|
IN CONST UINT8 *PemData,
|
|
|
|
IN UINTN PemSize,
|
|
|
|
IN CONST CHAR8 *Password,
|
|
|
|
OUT VOID **EcContext
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BOOLEAN Status;
|
|
|
|
BIO *PemBio;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check input parameters.
|
|
|
|
//
|
|
|
|
if ((PemData == NULL) || (EcContext == NULL) || (PemSize > INT_MAX)) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add possible block-cipher descriptor for PEM data decryption.
|
|
|
|
// NOTE: Only support most popular ciphers AES for the encrypted PEM.
|
|
|
|
//
|
|
|
|
if (EVP_add_cipher (EVP_aes_128_cbc ()) == 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EVP_add_cipher (EVP_aes_192_cbc ()) == 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EVP_add_cipher (EVP_aes_256_cbc ()) == 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = FALSE;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Read encrypted PEM Data.
|
|
|
|
//
|
|
|
|
PemBio = BIO_new (BIO_s_mem ());
|
|
|
|
if (PemBio == NULL) {
|
|
|
|
goto _Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (BIO_write (PemBio, PemData, (int)PemSize) <= 0) {
|
|
|
|
goto _Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Retrieve EC Private Key from encrypted PEM data.
|
|
|
|
//
|
|
|
|
*EcContext = PEM_read_bio_ECPrivateKey (PemBio, NULL, (pem_password_cb *)&PasswordCallback, (void *)Password);
|
|
|
|
if (*EcContext != NULL) {
|
|
|
|
Status = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_Exit:
|
|
|
|
//
|
|
|
|
// Release Resources.
|
|
|
|
//
|
|
|
|
BIO_free (PemBio);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|