/** * Copyright (c) 2020-2022 Paul-Louis Ageneau * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ #include "init.hpp" #include "certificate.hpp" #include "dtlstransport.hpp" #include "icetransport.hpp" #include "internals.hpp" #include "pollservice.hpp" #include "sctptransport.hpp" #include "threadpool.hpp" #include "tls.hpp" #include "utils.hpp" #if RTC_ENABLE_WEBSOCKET #include "tlstransport.hpp" #endif #if RTC_ENABLE_MEDIA #include "dtlssrtptransport.hpp" #endif #ifdef _WIN32 #include #endif #include namespace rtc::impl { struct Init::TokenPayload { TokenPayload(std::shared_future *cleanupFuture) { Init::Instance().doInit(); if (cleanupFuture) *cleanupFuture = cleanupPromise.get_future().share(); } ~TokenPayload() { std::thread t( [](std::promise promise) { utils::this_thread::set_name("RTC cleanup"); try { Init::Instance().doCleanup(); promise.set_value(); } catch (const std::exception &e) { PLOG_WARNING << e.what(); promise.set_exception(std::make_exception_ptr(e)); } }, std::move(cleanupPromise)); t.detach(); } std::promise cleanupPromise; }; Init &Init::Instance() { static Init *instance = new Init; return *instance; } Init::Init() { std::promise p; p.set_value(); mCleanupFuture = p.get_future(); // make it ready } Init::~Init() {} init_token Init::token() { std::lock_guard lock(mMutex); if (auto locked = mWeak.lock()) return locked; mGlobal = std::make_shared(&mCleanupFuture); mWeak = *mGlobal; return *mGlobal; } void Init::preload() { std::lock_guard lock(mMutex); if (!mGlobal) { mGlobal = std::make_shared(&mCleanupFuture); mWeak = *mGlobal; } } std::shared_future Init::cleanup() { std::lock_guard lock(mMutex); mGlobal.reset(); return mCleanupFuture; } void Init::setSctpSettings(SctpSettings s) { std::lock_guard lock(mMutex); if (mGlobal) SctpTransport::SetSettings(s); mCurrentSctpSettings = std::move(s); // store for next init } void Init::doInit() { // mMutex needs to be locked if (std::exchange(mInitialized, true)) return; PLOG_DEBUG << "Global initialization"; #ifdef _WIN32 WSADATA wsaData; if (WSAStartup(MAKEWORD(2, 2), &wsaData)) throw std::runtime_error("WSAStartup failed, error=" + std::to_string(WSAGetLastError())); #endif int concurrency = std::thread::hardware_concurrency(); int count = std::max(concurrency, MIN_THREADPOOL_SIZE); PLOG_DEBUG << "Spawning " << count << " threads"; ThreadPool::Instance().spawn(count); #if RTC_ENABLE_WEBSOCKET PollService::Instance().start(); #endif #if USE_GNUTLS // Nothing to do #elif USE_MBEDTLS // Nothing to do #else openssl::init(); #endif SctpTransport::Init(); SctpTransport::SetSettings(mCurrentSctpSettings); DtlsTransport::Init(); #if RTC_ENABLE_WEBSOCKET TlsTransport::Init(); #endif #if RTC_ENABLE_MEDIA DtlsSrtpTransport::Init(); #endif IceTransport::Init(); } void Init::doCleanup() { std::lock_guard lock(mMutex); if (mGlobal) return; if (!std::exchange(mInitialized, false)) return; PLOG_DEBUG << "Global cleanup"; ThreadPool::Instance().join(); ThreadPool::Instance().clear(); #if RTC_ENABLE_WEBSOCKET PollService::Instance().join(); #endif SctpTransport::Cleanup(); DtlsTransport::Cleanup(); #if RTC_ENABLE_WEBSOCKET TlsTransport::Cleanup(); #endif #if RTC_ENABLE_MEDIA DtlsSrtpTransport::Cleanup(); #endif IceTransport::Cleanup(); #ifdef _WIN32 WSACleanup(); #endif } } // namespace rtc::impl