|  | // Copyright (c) 2019 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "quic/core/crypto/tls_connection.h" | 
|  |  | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "third_party/boringssl/src/include/openssl/ssl.h" | 
|  | #include "quic/platform/api/quic_bug_tracker.h" | 
|  |  | 
|  | namespace quic { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // BoringSSL allows storing extra data off of some of its data structures, | 
|  | // including the SSL struct. To allow for multiple callers to store data, each | 
|  | // caller can use a different index for setting and getting data. These indices | 
|  | // are globals handed out by calling SSL_get_ex_new_index. | 
|  | // | 
|  | // SslIndexSingleton calls SSL_get_ex_new_index on its construction, and then | 
|  | // provides this index to be used in calls to SSL_get_ex_data/SSL_set_ex_data. | 
|  | // This is used to store in the SSL struct a pointer to the TlsConnection which | 
|  | // owns it. | 
|  | class SslIndexSingleton { | 
|  | public: | 
|  | static SslIndexSingleton* GetInstance() { | 
|  | static SslIndexSingleton* instance = new SslIndexSingleton(); | 
|  | return instance; | 
|  | } | 
|  |  | 
|  | int ssl_ex_data_index_connection() const { | 
|  | return ssl_ex_data_index_connection_; | 
|  | } | 
|  |  | 
|  | private: | 
|  | SslIndexSingleton() { | 
|  | CRYPTO_library_init(); | 
|  | ssl_ex_data_index_connection_ = | 
|  | SSL_get_ex_new_index(0, nullptr, nullptr, nullptr, nullptr); | 
|  | QUICHE_CHECK_LE(0, ssl_ex_data_index_connection_); | 
|  | } | 
|  |  | 
|  | SslIndexSingleton(const SslIndexSingleton&) = delete; | 
|  | SslIndexSingleton& operator=(const SslIndexSingleton&) = delete; | 
|  |  | 
|  | // The index to supply to SSL_get_ex_data/SSL_set_ex_data for getting/setting | 
|  | // the TlsConnection pointer. | 
|  | int ssl_ex_data_index_connection_; | 
|  | }; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | // static | 
|  | EncryptionLevel TlsConnection::QuicEncryptionLevel( | 
|  | enum ssl_encryption_level_t level) { | 
|  | switch (level) { | 
|  | case ssl_encryption_initial: | 
|  | return ENCRYPTION_INITIAL; | 
|  | case ssl_encryption_early_data: | 
|  | return ENCRYPTION_ZERO_RTT; | 
|  | case ssl_encryption_handshake: | 
|  | return ENCRYPTION_HANDSHAKE; | 
|  | case ssl_encryption_application: | 
|  | return ENCRYPTION_FORWARD_SECURE; | 
|  | default: | 
|  | QUIC_BUG(quic_bug_10698_1) | 
|  | << "Invalid ssl_encryption_level_t " << static_cast<int>(level); | 
|  | return ENCRYPTION_INITIAL; | 
|  | } | 
|  | } | 
|  |  | 
|  | // static | 
|  | enum ssl_encryption_level_t TlsConnection::BoringEncryptionLevel( | 
|  | EncryptionLevel level) { | 
|  | switch (level) { | 
|  | case ENCRYPTION_INITIAL: | 
|  | return ssl_encryption_initial; | 
|  | case ENCRYPTION_HANDSHAKE: | 
|  | return ssl_encryption_handshake; | 
|  | case ENCRYPTION_ZERO_RTT: | 
|  | return ssl_encryption_early_data; | 
|  | case ENCRYPTION_FORWARD_SECURE: | 
|  | return ssl_encryption_application; | 
|  | default: | 
|  | QUIC_BUG(quic_bug_10698_2) | 
|  | << "Invalid encryption level " << static_cast<int>(level); | 
|  | return ssl_encryption_initial; | 
|  | } | 
|  | } | 
|  |  | 
|  | TlsConnection::TlsConnection(SSL_CTX* ssl_ctx, | 
|  | TlsConnection::Delegate* delegate, | 
|  | QuicSSLConfig ssl_config) | 
|  | : delegate_(delegate), | 
|  | ssl_(SSL_new(ssl_ctx)), | 
|  | ssl_config_(std::move(ssl_config)) { | 
|  | SSL_set_ex_data( | 
|  | ssl(), SslIndexSingleton::GetInstance()->ssl_ex_data_index_connection(), | 
|  | this); | 
|  | if (ssl_config_.early_data_enabled.has_value()) { | 
|  | const int early_data_enabled = *ssl_config_.early_data_enabled ? 1 : 0; | 
|  | SSL_set_early_data_enabled(ssl(), early_data_enabled); | 
|  | } | 
|  | if (ssl_config_.signing_algorithm_prefs.has_value()) { | 
|  | SSL_set_signing_algorithm_prefs( | 
|  | ssl(), ssl_config_.signing_algorithm_prefs->data(), | 
|  | ssl_config_.signing_algorithm_prefs->size()); | 
|  | } | 
|  | if (ssl_config_.disable_ticket_support.has_value()) { | 
|  | if (*ssl_config_.disable_ticket_support) { | 
|  | SSL_set_options(ssl(), SSL_OP_NO_TICKET); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void TlsConnection::EnableInfoCallback() { | 
|  | SSL_set_info_callback( | 
|  | ssl(), +[](const SSL* ssl, int type, int value) { | 
|  | ConnectionFromSsl(ssl)->delegate_->InfoCallback(type, value); | 
|  | }); | 
|  | } | 
|  |  | 
|  | void TlsConnection::DisableTicketSupport() { | 
|  | ssl_config_.disable_ticket_support = true; | 
|  | SSL_set_options(ssl(), SSL_OP_NO_TICKET); | 
|  | } | 
|  |  | 
|  | // static | 
|  | bssl::UniquePtr<SSL_CTX> TlsConnection::CreateSslCtx() { | 
|  | CRYPTO_library_init(); | 
|  | bssl::UniquePtr<SSL_CTX> ssl_ctx(SSL_CTX_new(TLS_with_buffers_method())); | 
|  | SSL_CTX_set_min_proto_version(ssl_ctx.get(), TLS1_3_VERSION); | 
|  | SSL_CTX_set_max_proto_version(ssl_ctx.get(), TLS1_3_VERSION); | 
|  | SSL_CTX_set_quic_method(ssl_ctx.get(), &kSslQuicMethod); | 
|  | return ssl_ctx; | 
|  | } | 
|  |  | 
|  | // static | 
|  | TlsConnection* TlsConnection::ConnectionFromSsl(const SSL* ssl) { | 
|  | return reinterpret_cast<TlsConnection*>(SSL_get_ex_data( | 
|  | ssl, SslIndexSingleton::GetInstance()->ssl_ex_data_index_connection())); | 
|  | } | 
|  |  | 
|  | // static | 
|  | enum ssl_verify_result_t TlsConnection::VerifyCallback(SSL* ssl, | 
|  | uint8_t* out_alert) { | 
|  | return ConnectionFromSsl(ssl)->delegate_->VerifyCert(out_alert); | 
|  | } | 
|  |  | 
|  | const SSL_QUIC_METHOD TlsConnection::kSslQuicMethod{ | 
|  | TlsConnection::SetReadSecretCallback, TlsConnection::SetWriteSecretCallback, | 
|  | TlsConnection::WriteMessageCallback, TlsConnection::FlushFlightCallback, | 
|  | TlsConnection::SendAlertCallback}; | 
|  |  | 
|  | // static | 
|  | int TlsConnection::SetReadSecretCallback(SSL* ssl, | 
|  | enum ssl_encryption_level_t level, | 
|  | const SSL_CIPHER* cipher, | 
|  | const uint8_t* secret, | 
|  | size_t secret_length) { | 
|  | // TODO(nharper): replace this vector with a span (which unfortunately doesn't | 
|  | // yet exist in quic/platform/api). | 
|  | std::vector<uint8_t> secret_vec(secret, secret + secret_length); | 
|  | TlsConnection::Delegate* delegate = ConnectionFromSsl(ssl)->delegate_; | 
|  | if (!delegate->SetReadSecret(QuicEncryptionLevel(level), cipher, | 
|  | secret_vec)) { | 
|  | return 0; | 
|  | } | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | // static | 
|  | int TlsConnection::SetWriteSecretCallback(SSL* ssl, | 
|  | enum ssl_encryption_level_t level, | 
|  | const SSL_CIPHER* cipher, | 
|  | const uint8_t* secret, | 
|  | size_t secret_length) { | 
|  | // TODO(nharper): replace this vector with a span (which unfortunately doesn't | 
|  | // yet exist in quic/platform/api). | 
|  | std::vector<uint8_t> secret_vec(secret, secret + secret_length); | 
|  | TlsConnection::Delegate* delegate = ConnectionFromSsl(ssl)->delegate_; | 
|  | delegate->SetWriteSecret(QuicEncryptionLevel(level), cipher, secret_vec); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | // static | 
|  | int TlsConnection::WriteMessageCallback(SSL* ssl, | 
|  | enum ssl_encryption_level_t level, | 
|  | const uint8_t* data, | 
|  | size_t len) { | 
|  | ConnectionFromSsl(ssl)->delegate_->WriteMessage( | 
|  | QuicEncryptionLevel(level), | 
|  | absl::string_view(reinterpret_cast<const char*>(data), len)); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | // static | 
|  | int TlsConnection::FlushFlightCallback(SSL* ssl) { | 
|  | ConnectionFromSsl(ssl)->delegate_->FlushFlight(); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | // static | 
|  | int TlsConnection::SendAlertCallback(SSL* ssl, | 
|  | enum ssl_encryption_level_t level, | 
|  | uint8_t desc) { | 
|  | ConnectionFromSsl(ssl)->delegate_->SendAlert(QuicEncryptionLevel(level), | 
|  | desc); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | }  // namespace quic |