// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024 // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #include "td/net/SslStream.h" #if !TD_EMSCRIPTEN #include "td/utils/common.h" #include "td/utils/crypto.h" #include "td/utils/logging.h" #include "td/utils/misc.h" #include "td/utils/port/IPAddress.h" #include "td/utils/Status.h" #include "td/utils/Time.h" #include <openssl/bio.h> #include <openssl/err.h> #include <openssl/ssl.h> #include <openssl/x509.h> #include <cstring> #include <memory> namespace td { namespace detail { namespace { #if OPENSSL_VERSION_NUMBER < 0x10100000L void *BIO_get_data(BIO *b) { return b->ptr; } void BIO_set_data(BIO *b, void *ptr) { b->ptr = ptr; } void BIO_set_init(BIO *b, int init) { b->init = init; } int BIO_get_new_index() { return 0; } BIO_METHOD *BIO_meth_new(int type, const char *name) { auto res = new BIO_METHOD(); std::memset(res, 0, sizeof(*res)); return res; } int BIO_meth_set_write(BIO_METHOD *biom, int (*bwrite)(BIO *, const char *, int)) { biom->bwrite = bwrite; return 1; } int BIO_meth_set_read(BIO_METHOD *biom, int (*bread)(BIO *, char *, int)) { biom->bread = bread; return 1; } int BIO_meth_set_ctrl(BIO_METHOD *biom, long (*ctrl)(BIO *, int, long, void *)) { biom->ctrl = ctrl; return 1; } int BIO_meth_set_create(BIO_METHOD *biom, int (*create)(BIO *)) { biom->create = create; return 1; } int BIO_meth_set_destroy(BIO_METHOD *biom, int (*destroy)(BIO *)) { biom->destroy = destroy; return 1; } #endif int strm_create(BIO *b) { BIO_set_init(b, 1); return 1; } int strm_destroy(BIO *b) { return 1; } int strm_read(BIO *b, char *buf, int len); int strm_write(BIO *b, const char *buf, int len); long strm_ctrl(BIO *b, int cmd, long num, void *ptr) { switch (cmd) { case BIO_CTRL_FLUSH: return 1; case BIO_CTRL_PUSH: case BIO_CTRL_POP: return 0; #if OPENSSL_VERSION_NUMBER >= 0x30000000L && !defined(LIBRESSL_VERSION_NUMBER) case BIO_CTRL_GET_KTLS_SEND: case BIO_CTRL_GET_KTLS_RECV: return 0; #endif default: LOG(FATAL) << b << " " << cmd << " " << num << " " << ptr; } return 1; } BIO_METHOD *BIO_s_sslstream() { static BIO_METHOD *result = [] { BIO_METHOD *res = BIO_meth_new(BIO_get_new_index(), "td::SslStream helper bio"); BIO_meth_set_write(res, strm_write); BIO_meth_set_read(res, strm_read); BIO_meth_set_create(res, strm_create); BIO_meth_set_destroy(res, strm_destroy); BIO_meth_set_ctrl(res, strm_ctrl); return res; }(); return result; } struct SslHandleDeleter { void operator()(SSL *ssl_handle) { auto start_time = Time::now(); if (SSL_is_init_finished(ssl_handle)) { clear_openssl_errors("Before SSL_shutdown"); SSL_set_quiet_shutdown(ssl_handle, 1); SSL_shutdown(ssl_handle); clear_openssl_errors("After SSL_shutdown"); } SSL_free(ssl_handle); auto elapsed_time = Time::now() - start_time; if (elapsed_time >= 0.1) { LOG(WARNING) << "SSL_free took " << elapsed_time << " seconds"; } } }; using SslHandle = std::unique_ptr<SSL, SslHandleDeleter>; } // namespace class SslStreamImpl { public: Status init(CSlice host, SslCtx ssl_ctx, bool check_ip_address_as_host) { if (!ssl_ctx) { return Status::Error("Invalid SSL context provided"); } clear_openssl_errors("Before SslFd::init"); auto ssl_handle = SslHandle(SSL_new(static_cast<SSL_CTX *>(ssl_ctx.get_openssl_ctx()))); if (!ssl_handle) { return create_openssl_error(-13, "Failed to create an SSL handle"); } auto r_ip_address = IPAddress::get_ip_address(host); #if OPENSSL_VERSION_NUMBER >= 0x10002000L X509_VERIFY_PARAM *param = SSL_get0_param(ssl_handle.get()); X509_VERIFY_PARAM_set_hostflags(param, 0); if (r_ip_address.is_ok() && !check_ip_address_as_host) { LOG(DEBUG) << "Set verification IP address to " << r_ip_address.ok().get_ip_str(); X509_VERIFY_PARAM_set1_ip_asc(param, r_ip_address.ok().get_ip_str().c_str()); } else { LOG(DEBUG) << "Set verification host to " << host; X509_VERIFY_PARAM_set1_host(param, host.c_str(), 0); } #else #warning DANGEROUS! HTTPS HOST WILL NOT BE CHECKED. INSTALL OPENSSL >= 1.0.2 OR IMPLEMENT HTTPS HOST CHECK MANUALLY #endif auto *bio = BIO_new(BIO_s_sslstream()); BIO_set_data(bio, static_cast<void *>(this)); SSL_set_bio(ssl_handle.get(), bio, bio); #if OPENSSL_VERSION_NUMBER >= 0x0090806fL && !defined(OPENSSL_NO_TLSEXT) if (r_ip_address.is_error()) { // IP address must not be send as SNI LOG(DEBUG) << "Set SNI host name to " << host; auto host_str = host.str(); SSL_set_tlsext_host_name(ssl_handle.get(), MutableCSlice(host_str).begin()); } #endif SSL_set_connect_state(ssl_handle.get()); ssl_handle_ = std::move(ssl_handle); return Status::OK(); } ByteFlowInterface &read_byte_flow() { return read_flow_; } ByteFlowInterface &write_byte_flow() { return write_flow_; } size_t flow_read(MutableSlice slice) { return read_flow_.read(slice); } size_t flow_write(Slice slice) { return write_flow_.write(slice); } private: SslHandle ssl_handle_; friend class SslReadByteFlow; friend class SslWriteByteFlow; Result<size_t> write(Slice slice) { clear_openssl_errors("Before SslFd::write"); auto start_time = Time::now(); auto size = SSL_write(ssl_handle_.get(), slice.data(), static_cast<int>(slice.size())); auto elapsed_time = Time::now() - start_time; if (elapsed_time >= 0.1) { LOG(WARNING) << "SSL_write of size " << slice.size() << " took " << elapsed_time << " seconds and returned " << size << ' ' << SSL_get_error(ssl_handle_.get(), size); } if (size <= 0) { return process_ssl_error(size); } return size; } Result<size_t> read(MutableSlice slice) { clear_openssl_errors("Before SslFd::read"); auto start_time = Time::now(); auto size = SSL_read(ssl_handle_.get(), slice.data(), static_cast<int>(slice.size())); auto elapsed_time = Time::now() - start_time; if (elapsed_time >= 0.1) { LOG(WARNING) << "SSL_read took " << elapsed_time << " seconds and returned " << size << ' ' << SSL_get_error(ssl_handle_.get(), size); } if (size <= 0) { return process_ssl_error(size); } return size; } class SslReadByteFlow final : public ByteFlowBase { public: explicit SslReadByteFlow(SslStreamImpl *stream) : stream_(stream) { } bool loop() final { auto to_read = output_.prepare_append(); auto r_size = stream_->read(to_read); if (r_size.is_error()) { finish(r_size.move_as_error()); return false; } auto size = r_size.move_as_ok(); if (size == 0) { return false; } output_.confirm_append(size); return true; } size_t read(MutableSlice data) { return input_->advance(min(data.size(), input_->size()), data); } private: SslStreamImpl *stream_; }; class SslWriteByteFlow final : public ByteFlowBase { public: explicit SslWriteByteFlow(SslStreamImpl *stream) : stream_(stream) { } bool loop() final { auto to_write = input_->prepare_read(); auto r_size = stream_->write(to_write); if (r_size.is_error()) { finish(r_size.move_as_error()); return false; } auto size = r_size.move_as_ok(); if (size == 0) { return false; } input_->confirm_read(size); return true; } size_t write(Slice data) { output_.append(data); return data.size(); } private: SslStreamImpl *stream_; }; SslReadByteFlow read_flow_{this}; SslWriteByteFlow write_flow_{this}; Result<size_t> process_ssl_error(int ret) { auto os_error = OS_ERROR("SSL_ERROR_SYSCALL"); int error = SSL_get_error(ssl_handle_.get(), ret); switch (error) { case SSL_ERROR_NONE: LOG(ERROR) << "SSL_get_error returned no error"; return 0; case SSL_ERROR_ZERO_RETURN: LOG(DEBUG) << "SSL_ZERO_RETURN"; return 0; case SSL_ERROR_WANT_READ: LOG(DEBUG) << "SSL_WANT_READ"; return 0; case SSL_ERROR_WANT_WRITE: LOG(DEBUG) << "SSL_WANT_WRITE"; return 0; case SSL_ERROR_WANT_CONNECT: case SSL_ERROR_WANT_ACCEPT: case SSL_ERROR_WANT_X509_LOOKUP: LOG(DEBUG) << "SSL: CONNECT ACCEPT LOOKUP"; return 0; case SSL_ERROR_SYSCALL: if (ERR_peek_error() == 0) { if (os_error.code() != 0) { LOG(DEBUG) << "SSL_ERROR_SYSCALL"; return std::move(os_error); } else { LOG(DEBUG) << "SSL_SYSCALL"; return 0; } } /* fallthrough */ default: LOG(DEBUG) << "SSL_ERROR Default"; return create_openssl_error(1, "SSL error "); } } }; namespace { int strm_read(BIO *b, char *buf, int len) { auto *stream = static_cast<SslStreamImpl *>(BIO_get_data(b)); CHECK(stream != nullptr); BIO_clear_retry_flags(b); CHECK(buf != nullptr); auto res = narrow_cast<int>(stream->flow_read(MutableSlice(buf, len))); if (res == 0) { BIO_set_retry_read(b); return -1; } return res; } int strm_write(BIO *b, const char *buf, int len) { auto *stream = static_cast<SslStreamImpl *>(BIO_get_data(b)); CHECK(stream != nullptr); BIO_clear_retry_flags(b); CHECK(buf != nullptr); return narrow_cast<int>(stream->flow_write(Slice(buf, len))); } } // namespace } // namespace detail SslStream::SslStream() = default; SslStream::SslStream(SslStream &&) noexcept = default; SslStream &SslStream::operator=(SslStream &&) noexcept = default; SslStream::~SslStream() = default; Result<SslStream> SslStream::create(CSlice host, SslCtx ssl_ctx, bool use_ip_address_as_host) { auto impl = make_unique<detail::SslStreamImpl>(); TRY_STATUS(impl->init(host, ssl_ctx, use_ip_address_as_host)); return SslStream(std::move(impl)); } SslStream::SslStream(unique_ptr<detail::SslStreamImpl> impl) : impl_(std::move(impl)) { } ByteFlowInterface &SslStream::read_byte_flow() { return impl_->read_byte_flow(); } ByteFlowInterface &SslStream::write_byte_flow() { return impl_->write_byte_flow(); } size_t SslStream::flow_read(MutableSlice slice) { return impl_->flow_read(slice); } size_t SslStream::flow_write(Slice slice) { return impl_->flow_write(slice); } } // namespace td #else namespace td { namespace detail { class SslStreamImpl {}; } // namespace detail SslStream::SslStream() = default; SslStream::SslStream(SslStream &&) noexcept = default; SslStream &SslStream::operator=(SslStream &&) noexcept = default; SslStream::~SslStream() = default; Result<SslStream> SslStream::create(CSlice host, SslCtx ssl_ctx, bool check_ip_address_as_host) { return Status::Error("Not supported in Emscripten"); } SslStream::SslStream(unique_ptr<detail::SslStreamImpl> impl) : impl_(std::move(impl)) { } ByteFlowInterface &SslStream::read_byte_flow() { UNREACHABLE(); } ByteFlowInterface &SslStream::write_byte_flow() { UNREACHABLE(); } size_t SslStream::flow_read(MutableSlice slice) { UNREACHABLE(); } size_t SslStream::flow_write(Slice slice) { UNREACHABLE(); } } // namespace td #endif