openssl_init.ipp 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. //
  2. // ssl/detail/impl/openssl_init.ipp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2005 Voipster / Indrek dot Juhani at voipster dot com
  6. // Copyright (c) 2005-2023 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  7. //
  8. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  9. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  10. //
  11. #ifndef ASIO_SSL_DETAIL_IMPL_OPENSSL_INIT_IPP
  12. #define ASIO_SSL_DETAIL_IMPL_OPENSSL_INIT_IPP
  13. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  14. # pragma once
  15. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  16. #include "asio/detail/config.hpp"
  17. #include <vector>
  18. #include "asio/detail/assert.hpp"
  19. #include "asio/detail/mutex.hpp"
  20. #include "asio/detail/tss_ptr.hpp"
  21. #include "asio/ssl/detail/openssl_init.hpp"
  22. #include "asio/ssl/detail/openssl_types.hpp"
  23. #include "asio/detail/push_options.hpp"
  24. namespace asio {
  25. namespace ssl {
  26. namespace detail {
  27. class openssl_init_base::do_init
  28. {
  29. public:
  30. do_init()
  31. {
  32. #if (OPENSSL_VERSION_NUMBER < 0x10100000L)
  33. ::SSL_library_init();
  34. ::SSL_load_error_strings();
  35. ::OpenSSL_add_all_algorithms();
  36. mutexes_.resize(::CRYPTO_num_locks());
  37. for (size_t i = 0; i < mutexes_.size(); ++i)
  38. mutexes_[i].reset(new asio::detail::mutex);
  39. ::CRYPTO_set_locking_callback(&do_init::openssl_locking_func);
  40. #endif // (OPENSSL_VERSION_NUMBER < 0x10100000L)
  41. #if (OPENSSL_VERSION_NUMBER < 0x10000000L)
  42. ::CRYPTO_set_id_callback(&do_init::openssl_id_func);
  43. #endif // (OPENSSL_VERSION_NUMBER < 0x10000000L)
  44. #if !defined(SSL_OP_NO_COMPRESSION) \
  45. && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  46. null_compression_methods_ = sk_SSL_COMP_new_null();
  47. #endif // !defined(SSL_OP_NO_COMPRESSION)
  48. // && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  49. }
  50. ~do_init()
  51. {
  52. #if !defined(SSL_OP_NO_COMPRESSION) \
  53. && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  54. sk_SSL_COMP_free(null_compression_methods_);
  55. #endif // !defined(SSL_OP_NO_COMPRESSION)
  56. // && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  57. #if (OPENSSL_VERSION_NUMBER < 0x10000000L)
  58. ::CRYPTO_set_id_callback(0);
  59. #endif // (OPENSSL_VERSION_NUMBER < 0x10000000L)
  60. #if (OPENSSL_VERSION_NUMBER < 0x10100000L)
  61. ::CRYPTO_set_locking_callback(0);
  62. ::ERR_free_strings();
  63. ::EVP_cleanup();
  64. ::CRYPTO_cleanup_all_ex_data();
  65. #endif // (OPENSSL_VERSION_NUMBER < 0x10100000L)
  66. #if (OPENSSL_VERSION_NUMBER < 0x10000000L)
  67. ::ERR_remove_state(0);
  68. #elif (OPENSSL_VERSION_NUMBER < 0x10100000L)
  69. ::ERR_remove_thread_state(NULL);
  70. #endif // (OPENSSL_VERSION_NUMBER < 0x10000000L)
  71. #if (OPENSSL_VERSION_NUMBER >= 0x10002000L) \
  72. && (OPENSSL_VERSION_NUMBER < 0x10100000L) \
  73. && !defined(SSL_OP_NO_COMPRESSION)
  74. ::SSL_COMP_free_compression_methods();
  75. #endif // (OPENSSL_VERSION_NUMBER >= 0x10002000L)
  76. // && (OPENSSL_VERSION_NUMBER < 0x10100000L)
  77. // && !defined(SSL_OP_NO_COMPRESSION)
  78. #if !defined(OPENSSL_IS_BORINGSSL) \
  79. && !defined(ASIO_USE_WOLFSSL) \
  80. && (OPENSSL_VERSION_NUMBER < 0x30000000L)
  81. ::CONF_modules_unload(1);
  82. #endif // !defined(OPENSSL_IS_BORINGSSL)
  83. // && !defined(ASIO_USE_WOLFSSL)
  84. // && (OPENSSL_VERSION_NUMBER < 0x30000000L)
  85. #if !defined(OPENSSL_NO_ENGINE) \
  86. && (OPENSSL_VERSION_NUMBER < 0x10100000L)
  87. ::ENGINE_cleanup();
  88. #endif // !defined(OPENSSL_NO_ENGINE)
  89. // && (OPENSSL_VERSION_NUMBER < 0x10100000L)
  90. }
  91. #if !defined(SSL_OP_NO_COMPRESSION) \
  92. && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  93. STACK_OF(SSL_COMP)* get_null_compression_methods() const
  94. {
  95. return null_compression_methods_;
  96. }
  97. #endif // !defined(SSL_OP_NO_COMPRESSION)
  98. // && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  99. private:
  100. #if (OPENSSL_VERSION_NUMBER < 0x10000000L)
  101. static unsigned long openssl_id_func()
  102. {
  103. #if defined(ASIO_WINDOWS) || defined(__CYGWIN__)
  104. return ::GetCurrentThreadId();
  105. #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__)
  106. void* id = &errno;
  107. ASIO_ASSERT(sizeof(unsigned long) >= sizeof(void*));
  108. return reinterpret_cast<unsigned long>(id);
  109. #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__)
  110. }
  111. #endif // (OPENSSL_VERSION_NUMBER < 0x10000000L)
  112. #if (OPENSSL_VERSION_NUMBER < 0x10100000L)
  113. static void openssl_locking_func(int mode, int n,
  114. const char* /*file*/, int /*line*/)
  115. {
  116. if (mode & CRYPTO_LOCK)
  117. instance()->mutexes_[n]->lock();
  118. else
  119. instance()->mutexes_[n]->unlock();
  120. }
  121. // Mutexes to be used in locking callbacks.
  122. std::vector<asio::detail::shared_ptr<
  123. asio::detail::mutex>> mutexes_;
  124. #endif // (OPENSSL_VERSION_NUMBER < 0x10100000L)
  125. #if !defined(SSL_OP_NO_COMPRESSION) \
  126. && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  127. STACK_OF(SSL_COMP)* null_compression_methods_;
  128. #endif // !defined(SSL_OP_NO_COMPRESSION)
  129. // && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  130. };
  131. asio::detail::shared_ptr<openssl_init_base::do_init>
  132. openssl_init_base::instance()
  133. {
  134. static asio::detail::shared_ptr<do_init> init(new do_init);
  135. return init;
  136. }
  137. #if !defined(SSL_OP_NO_COMPRESSION) \
  138. && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  139. STACK_OF(SSL_COMP)* openssl_init_base::get_null_compression_methods()
  140. {
  141. return instance()->get_null_compression_methods();
  142. }
  143. #endif // !defined(SSL_OP_NO_COMPRESSION)
  144. // && (OPENSSL_VERSION_NUMBER >= 0x00908000L)
  145. } // namespace detail
  146. } // namespace ssl
  147. } // namespace asio
  148. #include "asio/detail/pop_options.hpp"
  149. #endif // ASIO_SSL_DETAIL_IMPL_OPENSSL_INIT_IPP