Skip to content

Commit 07c469f

Browse files
authored
[release/8.0]: Light up support for OpenSSL ENGINEs only if they are available.
Some Linux distributions are phasing out support for OpenSSL 1.x ENGINEs. They are doing this by moving the header, `engine.h`, to a separate package or removing the header entirely. The actual OpenSSL shared libraries still contain the engine APIs. This makes the change an API, not ABI, break. We react to this by disabling OpenSSL engine support on non-portable builds that are missing the engine header. Portable builds will continue to probe the loaded OpenSSL library for support, and non-portable builds will only support ENGINEs if the header is present.
1 parent 9b9f7ed commit 07c469f

File tree

7 files changed

+110
-30
lines changed

7 files changed

+110
-30
lines changed

src/libraries/Common/src/Interop/Unix/System.Security.Cryptography.Native/Interop.EvpPkey.cs

+18-4
Original file line numberDiff line numberDiff line change
@@ -217,7 +217,8 @@ internal static ArraySegment<byte> RentEncodeSubjectPublicKeyInfo(SafeEvpPKeyHan
217217
[LibraryImport(Libraries.CryptoNative, StringMarshalling = StringMarshalling.Utf8)]
218218
private static partial SafeEvpPKeyHandle CryptoNative_LoadPrivateKeyFromEngine(
219219
string engineName,
220-
string keyName);
220+
string keyName,
221+
[MarshalAs(UnmanagedType.Bool)] out bool haveEngine);
221222

222223
internal static SafeEvpPKeyHandle LoadPrivateKeyFromEngine(
223224
string engineName,
@@ -226,7 +227,13 @@ internal static SafeEvpPKeyHandle LoadPrivateKeyFromEngine(
226227
Debug.Assert(engineName is not null);
227228
Debug.Assert(keyName is not null);
228229

229-
SafeEvpPKeyHandle pkey = CryptoNative_LoadPrivateKeyFromEngine(engineName, keyName);
230+
SafeEvpPKeyHandle pkey = CryptoNative_LoadPrivateKeyFromEngine(engineName, keyName, out bool haveEngine);
231+
232+
if (!haveEngine)
233+
{
234+
pkey.Dispose();
235+
throw new CryptographicException(SR.Cryptography_EnginesNotSupported);
236+
}
230237

231238
if (pkey.IsInvalid)
232239
{
@@ -240,7 +247,8 @@ internal static SafeEvpPKeyHandle LoadPrivateKeyFromEngine(
240247
[LibraryImport(Libraries.CryptoNative, StringMarshalling = StringMarshalling.Utf8)]
241248
private static partial SafeEvpPKeyHandle CryptoNative_LoadPublicKeyFromEngine(
242249
string engineName,
243-
string keyName);
250+
string keyName,
251+
[MarshalAs(UnmanagedType.Bool)] out bool haveEngine);
244252

245253
internal static SafeEvpPKeyHandle LoadPublicKeyFromEngine(
246254
string engineName,
@@ -249,7 +257,13 @@ internal static SafeEvpPKeyHandle LoadPublicKeyFromEngine(
249257
Debug.Assert(engineName is not null);
250258
Debug.Assert(keyName is not null);
251259

252-
SafeEvpPKeyHandle pkey = CryptoNative_LoadPublicKeyFromEngine(engineName, keyName);
260+
SafeEvpPKeyHandle pkey = CryptoNative_LoadPublicKeyFromEngine(engineName, keyName, out bool haveEngine);
261+
262+
if (!haveEngine)
263+
{
264+
pkey.Dispose();
265+
throw new CryptographicException(SR.Cryptography_EnginesNotSupported);
266+
}
253267

254268
if (pkey.IsInvalid)
255269
{

src/libraries/System.Security.Cryptography/src/Resources/Strings.resx

+3
Original file line numberDiff line numberDiff line change
@@ -327,6 +327,9 @@
327327
<data name="Cryptography_EncryptedIncorrectLength" xml:space="preserve">
328328
<value>{0} unexpectedly produced a ciphertext with the incorrect length.</value>
329329
</data>
330+
<data name="Cryptography_EnginesNotSupported" xml:space="preserve">
331+
<value>OpenSSL ENGINE is not available on this platform.</value>
332+
</data>
330333
<data name="Cryptography_ExceedKdfExtractLimit" xml:space="preserve">
331334
<value>The total number of bytes extracted cannot exceed UInt32.MaxValue * hash length.</value>
332335
</data>

src/native/libs/System.Security.Cryptography.Native/configure.cmake

+7
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
11
include(CheckLibraryExists)
22
include(CheckFunctionExists)
3+
include(CheckSourceCompiles)
34

45
set(CMAKE_REQUIRED_INCLUDES ${OPENSSL_INCLUDE_DIR})
56
set(CMAKE_REQUIRED_LIBRARIES ${OPENSSL_CRYPTO_LIBRARY} ${OPENSSL_SSL_LIBRARY})
7+
set(CMAKE_REQUIRED_DEFINITIONS -DOPENSSL_API_COMPAT=0x10100000L)
68

79
check_function_exists(
810
EC_GF2m_simple_method
@@ -22,6 +24,11 @@ check_function_exists(
2224
HAVE_OPENSSL_SHA3
2325
)
2426

27+
check_source_compiles(C "
28+
#include <openssl/engine.h>
29+
int main(void) { ENGINE_init(NULL); return 1; }"
30+
HAVE_OPENSSL_ENGINE)
31+
2532
configure_file(
2633
${CMAKE_CURRENT_SOURCE_DIR}/pal_crypto_config.h.in
2734
${CMAKE_CURRENT_BINARY_DIR}/pal_crypto_config.h)

src/native/libs/System.Security.Cryptography.Native/opensslshim.h

+29-7
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,6 @@
1717
#include <openssl/dsa.h>
1818
#include <openssl/ec.h>
1919
#include <openssl/ecdsa.h>
20-
#include <openssl/engine.h>
2120
#include <openssl/err.h>
2221
#include <openssl/evp.h>
2322
#include <openssl/hmac.h>
@@ -46,6 +45,11 @@
4645
#include <openssl/provider.h>
4746
#endif
4847

48+
#if HAVE_OPENSSL_ENGINE
49+
// Some Linux distributions build without engine support.
50+
#include <openssl/engine.h>
51+
#endif
52+
4953
#if OPENSSL_VERSION_NUMBER >= OPENSSL_VERSION_1_1_1_RTM
5054
#define HAVE_OPENSSL_SET_CIPHERSUITES 1
5155
#else
@@ -168,6 +172,24 @@ const EVP_MD *EVP_shake256(void);
168172
int EVP_DigestFinalXOF(EVP_MD_CTX *ctx, unsigned char *md, size_t len);
169173
#endif
170174

175+
#if !HAVE_OPENSSL_ENGINE
176+
#undef HAVE_OPENSSL_ENGINE
177+
#define HAVE_OPENSSL_ENGINE 1
178+
179+
ENGINE *ENGINE_by_id(const char *id);
180+
int ENGINE_init(ENGINE *e);
181+
int ENGINE_finish(ENGINE *e);
182+
ENGINE *ENGINE_new(void);
183+
int ENGINE_free(ENGINE *e);
184+
typedef EVP_PKEY *(*ENGINE_LOAD_KEY_PTR)(ENGINE *, const char *,
185+
UI_METHOD *ui_method,
186+
void *callback_data);
187+
EVP_PKEY *ENGINE_load_private_key(ENGINE *e, const char *key_id,
188+
UI_METHOD *ui_method, void *callback_data);
189+
EVP_PKEY *ENGINE_load_public_key(ENGINE *e, const char *key_id,
190+
UI_METHOD *ui_method, void *callback_data);
191+
#endif
192+
171193
#define API_EXISTS(fn) (fn != NULL)
172194

173195
// List of all functions from the libssl that are used in the System.Security.Cryptography.Native.
@@ -298,12 +320,12 @@ int EVP_DigestFinalXOF(EVP_MD_CTX *ctx, unsigned char *md, size_t len);
298320
REQUIRED_FUNCTION(EC_POINT_mul) \
299321
REQUIRED_FUNCTION(EC_POINT_new) \
300322
REQUIRED_FUNCTION(EC_POINT_set_affine_coordinates_GFp) \
301-
REQUIRED_FUNCTION(ENGINE_by_id) \
302-
REQUIRED_FUNCTION(ENGINE_finish) \
303-
REQUIRED_FUNCTION(ENGINE_free) \
304-
REQUIRED_FUNCTION(ENGINE_init) \
305-
REQUIRED_FUNCTION(ENGINE_load_public_key) \
306-
REQUIRED_FUNCTION(ENGINE_load_private_key) \
323+
LIGHTUP_FUNCTION(ENGINE_by_id) \
324+
LIGHTUP_FUNCTION(ENGINE_finish) \
325+
LIGHTUP_FUNCTION(ENGINE_free) \
326+
LIGHTUP_FUNCTION(ENGINE_init) \
327+
LIGHTUP_FUNCTION(ENGINE_load_public_key) \
328+
LIGHTUP_FUNCTION(ENGINE_load_private_key) \
307329
REQUIRED_FUNCTION(ERR_clear_error) \
308330
REQUIRED_FUNCTION(ERR_error_string_n) \
309331
REQUIRED_FUNCTION(ERR_get_error) \

src/native/libs/System.Security.Cryptography.Native/pal_crypto_config.h.in

+1
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,4 @@
44
#cmakedefine01 HAVE_OPENSSL_ALPN
55
#cmakedefine01 HAVE_OPENSSL_CHACHA20POLY1305
66
#cmakedefine01 HAVE_OPENSSL_SHA3
7+
#cmakedefine01 HAVE_OPENSSL_ENGINE

src/native/libs/System.Security.Cryptography.Native/pal_evp_pkey.c

+48-17
Original file line numberDiff line numberDiff line change
@@ -511,41 +511,72 @@ int32_t CryptoNative_EncodeSubjectPublicKeyInfo(EVP_PKEY* pkey, uint8_t* buf)
511511
return i2d_PUBKEY(pkey, &buf);
512512
}
513513

514+
#if HAVE_OPENSSL_ENGINE
514515
static EVP_PKEY* LoadKeyFromEngine(
515516
const char* engineName,
516517
const char* keyName,
517-
ENGINE_LOAD_KEY_PTR load_func)
518+
ENGINE_LOAD_KEY_PTR load_func,
519+
int32_t* haveEngine)
518520
{
521+
assert(haveEngine);
519522
ERR_clear_error();
520523

521-
EVP_PKEY* ret = NULL;
522-
ENGINE* engine = NULL;
524+
if (API_EXISTS(ENGINE_by_id) && API_EXISTS(ENGINE_init) && API_EXISTS(ENGINE_finish) && API_EXISTS(ENGINE_free))
525+
{
526+
*haveEngine = 1;
527+
EVP_PKEY* ret = NULL;
528+
ENGINE* engine = NULL;
523529

524-
// Per https://github.com/openssl/openssl/discussions/21427
525-
// using EVP_PKEY after freeing ENGINE is correct.
526-
engine = ENGINE_by_id(engineName);
530+
// Per https://github.com/openssl/openssl/discussions/21427
531+
// using EVP_PKEY after freeing ENGINE is correct.
532+
engine = ENGINE_by_id(engineName);
527533

528-
if (engine != NULL)
529-
{
530-
if (ENGINE_init(engine))
534+
if (engine != NULL)
531535
{
532-
ret = load_func(engine, keyName, NULL, NULL);
536+
if (ENGINE_init(engine))
537+
{
538+
ret = load_func(engine, keyName, NULL, NULL);
539+
540+
ENGINE_finish(engine);
541+
}
533542

534-
ENGINE_finish(engine);
543+
ENGINE_free(engine);
535544
}
536545

537-
ENGINE_free(engine);
546+
return ret;
538547
}
539548

540-
return ret;
549+
*haveEngine = 0;
550+
return NULL;
541551
}
552+
#endif
542553

543-
EVP_PKEY* CryptoNative_LoadPrivateKeyFromEngine(const char* engineName, const char* keyName)
554+
EVP_PKEY* CryptoNative_LoadPrivateKeyFromEngine(const char* engineName, const char* keyName, int32_t* haveEngine)
544555
{
545-
return LoadKeyFromEngine(engineName, keyName, ENGINE_load_private_key);
556+
#if HAVE_OPENSSL_ENGINE
557+
if (API_EXISTS(ENGINE_load_private_key))
558+
{
559+
return LoadKeyFromEngine(engineName, keyName, ENGINE_load_private_key, haveEngine);
560+
}
561+
#endif
562+
(void)engineName;
563+
(void)keyName;
564+
(void)haveEngine;
565+
*haveEngine = 0;
566+
return NULL;
546567
}
547568

548-
EVP_PKEY* CryptoNative_LoadPublicKeyFromEngine(const char* engineName, const char* keyName)
569+
EVP_PKEY* CryptoNative_LoadPublicKeyFromEngine(const char* engineName, const char* keyName, int32_t* haveEngine)
549570
{
550-
return LoadKeyFromEngine(engineName, keyName, ENGINE_load_public_key);
571+
#if HAVE_OPENSSL_ENGINE
572+
if (API_EXISTS(ENGINE_load_private_key))
573+
{
574+
return LoadKeyFromEngine(engineName, keyName, ENGINE_load_public_key, haveEngine);
575+
}
576+
#endif
577+
(void)engineName;
578+
(void)keyName;
579+
(void)haveEngine;
580+
*haveEngine = 0;
581+
return NULL;
551582
}

src/native/libs/System.Security.Cryptography.Native/pal_evp_pkey.h

+4-2
Original file line numberDiff line numberDiff line change
@@ -93,12 +93,14 @@ PALEXPORT int32_t CryptoNative_EncodeSubjectPublicKeyInfo(EVP_PKEY* pkey, uint8_
9393
Load a named key, via ENGINE_load_private_key, from the named engine.
9494
9595
Returns a valid EVP_PKEY* on success, NULL on failure.
96+
haveEngine is 1 if OpenSSL ENGINE's are supported, otherwise 0.
9697
*/
97-
PALEXPORT EVP_PKEY* CryptoNative_LoadPrivateKeyFromEngine(const char* engineName, const char* keyName);
98+
PALEXPORT EVP_PKEY* CryptoNative_LoadPrivateKeyFromEngine(const char* engineName, const char* keyName, int32_t* haveEngine);
9899

99100
/*
100101
Load a named key, via ENGINE_load_public_key, from the named engine.
101102
102103
Returns a valid EVP_PKEY* on success, NULL on failure.
104+
haveEngine is 1 if OpenSSL ENGINE's are supported, otherwise 0.
103105
*/
104-
PALEXPORT EVP_PKEY* CryptoNative_LoadPublicKeyFromEngine(const char* engineName, const char* keyName);
106+
PALEXPORT EVP_PKEY* CryptoNative_LoadPublicKeyFromEngine(const char* engineName, const char* keyName, int32_t* haveEngine);

0 commit comments

Comments
 (0)