// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023 // // 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) // #pragma once #include "td/utils/port/config.h" #ifdef TD_THREAD_STL #include "td/utils/common.h" #include "td/utils/invoke.h" #if TD_WINDOWS && TD_MSVC #include "td/utils/port/detail/NativeFd.h" #endif #include "td/utils/port/detail/ThreadIdGuard.h" #include "td/utils/port/thread_local.h" #include "td/utils/Slice.h" #include "td/utils/Status.h" #include #include #include #include #if TD_WINDOWS && TD_MSVC #define TD_HAVE_THREAD_AFFINITY 1 #endif namespace td { namespace detail { class ThreadStl { public: ThreadStl() = default; ThreadStl(const ThreadStl &) = delete; ThreadStl &operator=(const ThreadStl &) = delete; ThreadStl(ThreadStl &&) = default; ThreadStl &operator=(ThreadStl &&) = default; ~ThreadStl() { join(); } template explicit ThreadStl(Function &&f, Args &&...args) { thread_ = std::thread([args = std::make_tuple(decay_copy(std::forward(f)), decay_copy(std::forward(args))...)]() mutable { ThreadIdGuard thread_id_guard; invoke_tuple(std::move(args)); clear_thread_locals(); }); } void join() { if (thread_.joinable()) { thread_.join(); } } void detach() { if (thread_.joinable()) { thread_.detach(); } } void set_name(CSlice name) { // not supported } static unsigned hardware_concurrency() { return std::thread::hardware_concurrency(); } #if TD_WINDOWS && TD_MSVC using id = DWORD; #else using id = std::thread::id; #endif id get_id() noexcept { #if TD_WINDOWS && TD_MSVC static_assert(std::is_same::value, "Expected HANDLE as native thread type"); return GetThreadId(thread_.native_handle()); #else return thread_.get_id(); #endif } static void send_real_time_signal(id thread_id, int real_time_signal_number) { // not supported } #if TD_HAVE_THREAD_AFFINITY static Status set_affinity_mask(id thread_id, uint64 mask) { if (static_cast(mask) != mask) { return Status::Error("Invalid thread affinity mask specified"); } auto handle = OpenThread(THREAD_SET_LIMITED_INFORMATION | THREAD_QUERY_LIMITED_INFORMATION, FALSE, thread_id); if (handle == nullptr) { return Status::Error("Failed to access thread"); } NativeFd thread_handle(handle); if (SetThreadAffinityMask(thread_handle.fd(), static_cast(mask))) { return Status::OK(); } return OS_ERROR("Failed to set thread affinity mask"); } static uint64 get_affinity_mask(id thread_id) { DWORD_PTR process_mask = 0; DWORD_PTR system_mask = 0; if (GetProcessAffinityMask(GetCurrentProcess(), &process_mask, &system_mask)) { auto handle = OpenThread(THREAD_SET_LIMITED_INFORMATION | THREAD_QUERY_LIMITED_INFORMATION, FALSE, thread_id); if (handle == nullptr) { return 0; } NativeFd thread_handle(handle); auto result = SetThreadAffinityMask(thread_handle.fd(), process_mask); if (result != 0 && result != process_mask) { SetThreadAffinityMask(thread_handle.fd(), result); } return result; } return 0; } #endif private: std::thread thread_; template std::decay_t decay_copy(T &&v) { return std::forward(v); } }; namespace this_thread_stl { #if TD_WINDOWS && TD_MSVC inline ThreadStl::id get_id() { return GetCurrentThreadId(); } #else using std::this_thread::get_id; #endif } // namespace this_thread_stl } // namespace detail } // namespace td #endif