-
Notifications
You must be signed in to change notification settings - Fork 122
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Refactor TLS 1.3 cipher selection and fix SSL_get_ciphers #2092
base: main
Are you sure you want to change the base?
Changes from 36 commits
bc7b951
e80cb6e
c49f32d
bfb369f
ef0f455
aacb978
b6f297b
749513a
1e818d5
425e9a5
656b710
2622e1e
fe430f4
72603cc
eb28574
5fcdd4a
d4e5047
6707995
ff11f99
849bf37
3f467ff
dbbd45c
94d2189
7669f46
fc0c20c
9ee29c8
e38bfa3
dbd660a
272ef8f
0991f2f
0d7b302
2c809d3
7798ef6
8442723
ae84e27
0e66ad6
a755968
2c6d018
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1690,10 +1690,9 @@ OPENSSL_EXPORT size_t SSL_get_all_standard_cipher_names(const char **out, | |
// Once an equal-preference group is used, future directives must be | ||
// opcode-less. Inside an equal-preference group, spaces are not allowed. | ||
// | ||
// TLS 1.3 ciphers do not participate in this mechanism and instead have a | ||
// built-in preference order. Functions to set cipher lists do not affect TLS | ||
// 1.3, and functions to query the cipher list do not include TLS 1.3 | ||
// ciphers. | ||
// Note: TLS 1.3 ciphersuites are only configurable via | ||
// |SSL_CTX_set_ciphersuites| or |SSL_set_ciphersuites|. Other setter functions have | ||
// no impact on TLS 1.3 ciphersuites. | ||
Comment on lines
+1693
to
+1695
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do we have tests that attempt to set 1.3 ciphers and ensure they are rejected? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do you mean use the TLS 1.2 setter functions to set TLS 1.3 ciphers? If so yes you get a no cipher match error. |
||
|
||
// SSL_DEFAULT_CIPHER_LIST is the default cipher suite configuration. It is | ||
// substituted when a cipher string starts with 'DEFAULT'. | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -224,6 +224,7 @@ static bool collect_cipher_protocol_ids(STACK_OF(SSL_CIPHER) *ciphers, | |
CBB *cbb, uint32_t mask_k, uint32_t mask_a, uint16_t max_version, | ||
uint16_t min_version, bool *any_enabled) { | ||
*any_enabled = false; | ||
|
||
for (const SSL_CIPHER *cipher : ciphers) { | ||
// Skip disabled ciphers | ||
if ((cipher->algorithm_mkey & mask_k) || | ||
|
@@ -259,49 +260,31 @@ static bool ssl_write_client_cipher_list(const SSL_HANDSHAKE *hs, CBB *out, | |
return false; | ||
} | ||
|
||
// Add TLS 1.3 ciphers. | ||
if (hs->max_version >= TLS1_3_VERSION && ssl->ctx->tls13_cipher_list) { | ||
// Use the configured TLSv1.3 ciphers list. | ||
STACK_OF(SSL_CIPHER) *ciphers = ssl->ctx->tls13_cipher_list->ciphers.get(); | ||
// Add all ciphers unless TLS 1.3 only connection | ||
if (hs->min_version < TLS1_3_VERSION && type != ssl_client_hello_inner) { | ||
andrewhop marked this conversation as resolved.
Show resolved
Hide resolved
|
||
bool any_enabled = false; | ||
if (!collect_cipher_protocol_ids(ciphers, &child, mask_k, | ||
if (!collect_cipher_protocol_ids(SSL_get_ciphers(ssl), &child, mask_k, | ||
mask_a, hs->max_version, hs->min_version, &any_enabled)) { | ||
return false; | ||
Comment on lines
+266
to
268
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Double checking my understanding: this is in an if/else block so only There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes if min is less than 1.3, we want to add min to max (where max could also be anything). We pass this in to collect_cipher_protocol_ids. SSL_get_ciphers will return the SSL level (if configured) and otherwise SSL_CTX main cipher list which now holds all ciphersuites. |
||
} | ||
|
||
// If all ciphers were disabled, return the error to the caller. | ||
if (!any_enabled) { | ||
OPENSSL_PUT_ERROR(SSL, SSL_R_NO_CIPHERS_AVAILABLE); | ||
return false; | ||
} | ||
} else if (hs->max_version >= TLS1_3_VERSION) { | ||
// Use the built in TLSv1.3 ciphers. Order ChaCha20-Poly1305 relative to | ||
// AES-GCM based on hardware support. | ||
const bool has_aes_hw = ssl->config->aes_hw_override | ||
? ssl->config->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
|
||
if (!has_aes_hw && | ||
!CBB_add_u16(&child, TLS1_CK_CHACHA20_POLY1305_SHA256 & 0xffff)) { | ||
return false; | ||
} | ||
if (!CBB_add_u16(&child, TLS1_3_CK_AES_128_GCM_SHA256 & 0xffff) || | ||
!CBB_add_u16(&child, TLS1_3_CK_AES_256_GCM_SHA384 & 0xffff)) { | ||
return false; | ||
} | ||
if (has_aes_hw && | ||
!CBB_add_u16(&child, TLS1_3_CK_CHACHA20_POLY1305_SHA256 & 0xffff)) { | ||
return false; | ||
} | ||
} | ||
|
||
if (hs->min_version < TLS1_3_VERSION && type != ssl_client_hello_inner) { | ||
} else if (hs->max_version >= TLS1_3_VERSION && ssl->ctx->tls13_cipher_list) { | ||
// Only TLS 1.3 ciphers | ||
STACK_OF(SSL_CIPHER) *ciphers = ssl->ctx->tls13_cipher_list->ciphers.get(); | ||
bool any_enabled = false; | ||
if (!collect_cipher_protocol_ids(SSL_get_ciphers(ssl), &child, mask_k, | ||
|
||
if (!collect_cipher_protocol_ids(ciphers, &child, mask_k, | ||
mask_a, hs->max_version, hs->min_version, &any_enabled)) { | ||
return false; | ||
} | ||
Comment on lines
+282
to
286
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why are you filtering the TLS 1.3 ciphers again here? From line 276 we already know we want 1.3 ciphers and tls13_cipher_list only has 1.3 ciphers in it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Actually can these two blocks be combined into a single flow:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Cannot combine the two blocks. Different cases for ECH (discussed offline) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I agree you can't combine the two blocks into one shared block, but I'm still curious about the filtering, can this just use There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Are you saying use the same SSL_get_ciphers call below as well? The reason we can't do this is if you are a ECH ClientHelloInner, and you support TLS versions less than 1.3, by RFC we still only send TLS 1.3 ciphersuites. Since collect_ciphers takes min max version from the handshake, using the above ciphers var would erroneously pull in TLS 1.2 and below ciphersuites. Therefore, we specifically pass in only the tls13_ciphersuites in the else block |
||
|
||
// If all ciphers were disabled, return the error to the caller. | ||
if (!any_enabled && hs->max_version < TLS1_3_VERSION) { | ||
if (!any_enabled) { | ||
OPENSSL_PUT_ERROR(SSL, SSL_R_NO_CIPHERS_AVAILABLE); | ||
return false; | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1234,6 +1234,86 @@ static bool is_known_default_alias_keyword_filter_rule(const char *rule, | |
return false; | ||
} | ||
|
||
// update_cipher_list updates |ctx->cipher_list| by: | ||
smittals2 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// 1. Removing any existing TLS 1.3 ciphersuites | ||
// 2. Adding configured ciphersuites from |ctx->tls13_cipher_list| | ||
// 3. Configuring a new |ctx->cipher_list->in_group_flags| | ||
// This function maintains the ordering of ciphersuites and places TLS 1.3 | ||
// ciphersuites at the front of the list. | ||
smittals2 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// Returns one on success and zero on failure. | ||
int update_cipher_list(UniquePtr<SSLCipherPreferenceList> &dst, UniquePtr<SSLCipherPreferenceList> &ciphers, UniquePtr<SSLCipherPreferenceList> &tls13_ciphers) { | ||
bssl::UniquePtr<STACK_OF(SSL_CIPHER)> tmp_cipher_list; | ||
int num_removed_tls13_ciphers = 0, num_added_tls13_ciphers = 0; | ||
Array<bool> updated_in_group_flags; | ||
|
||
if (ciphers && ciphers->ciphers) { | ||
tmp_cipher_list.reset(sk_SSL_CIPHER_dup(ciphers->ciphers.get())); | ||
} else { | ||
tmp_cipher_list.reset(sk_SSL_CIPHER_new_null()); | ||
} | ||
|
||
if (tmp_cipher_list == NULL) { | ||
smittals2 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
return 0; | ||
} | ||
|
||
// Delete any existing TLSv1.3 ciphersuites. These will be first in the list | ||
smittals2 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
while (sk_SSL_CIPHER_num(tmp_cipher_list.get()) > 0 && | ||
SSL_CIPHER_get_min_version(sk_SSL_CIPHER_value(tmp_cipher_list.get(), 0)) | ||
== TLS1_3_VERSION) { | ||
sk_SSL_CIPHER_delete(tmp_cipher_list.get(), 0); | ||
num_removed_tls13_ciphers++; | ||
} | ||
|
||
int num_updated_tls12_ciphers = sk_SSL_CIPHER_num(tmp_cipher_list.get()); | ||
|
||
if (tls13_ciphers && tls13_ciphers->ciphers) { | ||
STACK_OF(SSL_CIPHER) *tls13_cipher_stack = tls13_ciphers->ciphers.get(); | ||
num_added_tls13_ciphers = sk_SSL_CIPHER_num(tls13_cipher_stack); | ||
for (int i = sk_SSL_CIPHER_num(tls13_cipher_stack) - 1; i >= 0; i--) { | ||
const SSL_CIPHER *tls13_cipher = sk_SSL_CIPHER_value(tls13_cipher_stack, i); | ||
if (!sk_SSL_CIPHER_unshift(tmp_cipher_list.get(), tls13_cipher)) { | ||
return 0; | ||
} | ||
} | ||
} | ||
|
||
|
||
if (!updated_in_group_flags.Init(num_added_tls13_ciphers + num_updated_tls12_ciphers)) { | ||
return 0; | ||
} | ||
std::fill(updated_in_group_flags.begin(), updated_in_group_flags.end(), false); | ||
|
||
// Copy in_group_flags from |ctx->tls13_cipher_list| | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What does There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. |
||
if (tls13_ciphers && tls13_ciphers->in_group_flags) { | ||
const auto& tls13_flags = tls13_ciphers->in_group_flags; | ||
// Ensure value of last element in |in_group_flags| is 0. The last cipher | ||
// in a list must be the end of any group in that list. | ||
if (tls13_flags[num_added_tls13_ciphers - 1] != 0) { | ||
tls13_flags[num_added_tls13_ciphers - 1] = false; | ||
} | ||
Comment on lines
+1287
to
+1291
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should this fail here instead? If this happens does that mean something happened up above and the wrong number of things got added? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can a group include a 1.3 and 1.2 cipher suite? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No, you can only group ciphersuites within the same TLS version |
||
for (int i = 0; i < num_added_tls13_ciphers; i++) { | ||
updated_in_group_flags[i] = tls13_flags[i]; | ||
} | ||
} | ||
|
||
// Copy remaining in_group_flags from |ctx->cipher_list| | ||
if (ciphers && ciphers->in_group_flags) { | ||
for (int i = 0; i < num_updated_tls12_ciphers; i++) { | ||
updated_in_group_flags[i + num_added_tls13_ciphers] = | ||
ciphers->in_group_flags[i + num_removed_tls13_ciphers]; | ||
} | ||
} | ||
|
||
Span<const bool> flags_span(updated_in_group_flags.data(), updated_in_group_flags.size()); | ||
UniquePtr<SSLCipherPreferenceList> new_list = MakeUnique<SSLCipherPreferenceList>(); | ||
if (!new_list || !new_list->Init(std::move(tmp_cipher_list), flags_span)) { | ||
return 0; | ||
} | ||
|
||
dst = std::move(new_list); | ||
return 1; | ||
} | ||
|
||
bool ssl_create_cipher_list(UniquePtr<SSLCipherPreferenceList> *out_cipher_list, | ||
const bool has_aes_hw, const char *rule_str, | ||
bool strict, bool config_tls13) { | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -586,7 +586,16 @@ SSL_CTX *SSL_CTX_new(const SSL_METHOD *method) { | |
return nullptr; | ||
} | ||
|
||
if (!SSL_CTX_set_strict_cipher_list(ret.get(), SSL_DEFAULT_CIPHER_LIST) || | ||
const bool has_aes_hw = EVP_has_aes_hardware(); | ||
const char *cipher_rule; | ||
if (has_aes_hw) { | ||
cipher_rule = TLS13_DEFAULT_CIPHER_LIST_AES_HW; | ||
} else { | ||
cipher_rule = TLS13_DEFAULT_CIPHER_LIST_NO_AES_HW; | ||
} | ||
|
||
if (!SSL_CTX_set_ciphersuites(ret.get(), cipher_rule) || | ||
!SSL_CTX_set_strict_cipher_list(ret.get(), SSL_DEFAULT_CIPHER_LIST) || | ||
smittals2 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// Lock the SSL_CTX to the specified version, for compatibility with | ||
// legacy uses of SSL_METHOD. | ||
!SSL_CTX_set_max_proto_version(ret.get(), method->version) || | ||
|
@@ -2178,17 +2187,25 @@ const char *SSL_get_cipher_list(const SSL *ssl, int n) { | |
int SSL_CTX_set_cipher_list(SSL_CTX *ctx, const char *str) { | ||
const bool has_aes_hw = ctx->aes_hw_override ? ctx->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
return ssl_create_cipher_list(&ctx->cipher_list, has_aes_hw, str, | ||
if (!ssl_create_cipher_list(&ctx->cipher_list, has_aes_hw, str, | ||
false /* not strict */, | ||
false /* don't configure TLSv1.3 ciphers */); | ||
false /* don't configure TLSv1.3 ciphers */)) { | ||
return 0; | ||
} | ||
|
||
return update_cipher_list(ctx->cipher_list, ctx->cipher_list, ctx->tls13_cipher_list); | ||
Comment on lines
+2188
to
+2194
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we just pass the 1.3 ciphers to There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The same ssl_create_cipher_list function is used to create tls13_cipher_lists and the main cipher_list. Therefore, I have chosen to do update's separately to properly distinguish what is being set. |
||
} | ||
|
||
int SSL_CTX_set_strict_cipher_list(SSL_CTX *ctx, const char *str) { | ||
const bool has_aes_hw = ctx->aes_hw_override ? ctx->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
return ssl_create_cipher_list(&ctx->cipher_list, has_aes_hw, str, | ||
if (!ssl_create_cipher_list(&ctx->cipher_list, has_aes_hw, str, | ||
true /* strict */, | ||
false /* don't configure TLSv1.3 ciphers */); | ||
false /* don't configure TLSv1.3 ciphers */)) { | ||
return 0; | ||
} | ||
|
||
return update_cipher_list(ctx->cipher_list, ctx->cipher_list, ctx->tls13_cipher_list); | ||
} | ||
|
||
int SSL_set_cipher_list(SSL *ssl, const char *str) { | ||
|
@@ -2198,17 +2215,29 @@ int SSL_set_cipher_list(SSL *ssl, const char *str) { | |
const bool has_aes_hw = ssl->config->aes_hw_override | ||
? ssl->config->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
return ssl_create_cipher_list(&ssl->config->cipher_list, has_aes_hw, str, | ||
if (!ssl_create_cipher_list(&ssl->config->cipher_list, has_aes_hw, str, | ||
false /* not strict */, | ||
false /* don't configure TLSv1.3 ciphers */); | ||
false /* don't configure TLSv1.3 ciphers */)) { | ||
return 0; | ||
} | ||
|
||
UniquePtr<SSLCipherPreferenceList> &tls13_ciphers = ssl->config->tls13_cipher_list ? ssl->config->tls13_cipher_list : | ||
ssl->ctx->tls13_cipher_list; | ||
|
||
return update_cipher_list(ssl->config->cipher_list, ssl->config->cipher_list, tls13_ciphers); | ||
} | ||
|
||
int SSL_CTX_set_ciphersuites(SSL_CTX *ctx, const char *str) { | ||
const bool has_aes_hw = ctx->aes_hw_override ? ctx->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
return ssl_create_cipher_list(&ctx->tls13_cipher_list, has_aes_hw, str, | ||
|
||
if (!ssl_create_cipher_list(&ctx->tls13_cipher_list, has_aes_hw, str, | ||
false /* not strict */, | ||
true /* only configure TLSv1.3 ciphers */); | ||
true /* only configure TLSv1.3 ciphers */)) { | ||
return 0; | ||
} | ||
|
||
return update_cipher_list(ctx->cipher_list, ctx->cipher_list, ctx->tls13_cipher_list); | ||
} | ||
|
||
int SSL_set_ciphersuites(SSL *ssl, const char *str) { | ||
|
@@ -2218,9 +2247,16 @@ int SSL_set_ciphersuites(SSL *ssl, const char *str) { | |
const bool has_aes_hw = ssl->config->aes_hw_override | ||
? ssl->config->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
return ssl_create_cipher_list(&ssl->config->cipher_list, has_aes_hw, str, | ||
false /* not strict */, | ||
true /* configure TLSv1.3 ciphers */); | ||
if (!ssl_create_cipher_list(&ssl->config->tls13_cipher_list, | ||
has_aes_hw, str, false /* not strict */, | ||
true /* configure TLSv1.3 ciphers */)) { | ||
return 0; | ||
} | ||
|
||
UniquePtr<SSLCipherPreferenceList> &ciphers = ssl->config->cipher_list ? ssl->config->cipher_list : | ||
ssl->ctx->cipher_list; | ||
|
||
return update_cipher_list(ssl->config->cipher_list, ciphers, ssl->config->tls13_cipher_list); | ||
} | ||
|
||
int SSL_set_strict_cipher_list(SSL *ssl, const char *str) { | ||
|
@@ -2230,9 +2266,16 @@ int SSL_set_strict_cipher_list(SSL *ssl, const char *str) { | |
const bool has_aes_hw = ssl->config->aes_hw_override | ||
? ssl->config->aes_hw_override_value | ||
: EVP_has_aes_hardware(); | ||
return ssl_create_cipher_list(&ssl->config->cipher_list, has_aes_hw, str, | ||
true /* strict */, | ||
false /* don't configure TLSv1.3 ciphers */); | ||
if (!ssl_create_cipher_list(&ssl->config->cipher_list, | ||
has_aes_hw, str, true /* strict */, | ||
false /* don't configure TLSv1.3 ciphers */)) { | ||
return 0; | ||
} | ||
|
||
UniquePtr<SSLCipherPreferenceList> &tls13_ciphers = ssl->config->tls13_cipher_list ? ssl->config->tls13_cipher_list : | ||
ssl->ctx->tls13_cipher_list; | ||
|
||
return update_cipher_list(ssl->config->cipher_list, ssl->config->cipher_list, tls13_ciphers); | ||
} | ||
|
||
const char *SSL_get_servername(const SSL *ssl, const int type) { | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please update SSL_get_ciphers documentation to note it returns all ciphers 1.2 and 1.3
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The current documentation covers this case ("cipher list for |ssl|") but I'll make it more explicit. We just weren't living up to this contract initially.