/* This file is part of TON Blockchain Library. TON Blockchain Library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. TON Blockchain Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with TON Blockchain Library. If not, see . Copyright 2017-2019 Telegram Systems LLP */ #pragma once #include "td/utils/port/config.h" #include "td/utils/common.h" #include "td/utils/Destructor.h" #include #include namespace td { // clang-format off #if TD_GCC || TD_CLANG #define TD_THREAD_LOCAL __thread #elif TD_INTEL || TD_MSVC #define TD_THREAD_LOCAL thread_local #else #warning "TD_THREAD_LOCAL is not defined, trying 'thread_local'" #define TD_THREAD_LOCAL thread_local #endif // clang-format on // If raw_ptr is not nullptr, allocate T as in std::make_unique(args...) and store pointer into raw_ptr template bool init_thread_local(P &raw_ptr, ArgsT &&... args); // Destroy all thread locals, and store nullptr into corresponding pointers void clear_thread_locals(); void set_thread_id(int32 id); int32 get_thread_id(); namespace detail { void add_thread_local_destructor(unique_ptr destructor); template void do_init_thread_local(P &raw_ptr, ArgsT &&... args) { auto ptr = std::make_unique(std::forward(args)...); raw_ptr = ptr.get(); detail::add_thread_local_destructor(create_destructor([ptr = std::move(ptr), &raw_ptr]() mutable { ptr.reset(); raw_ptr = nullptr; })); } } // namespace detail template bool init_thread_local(P &raw_ptr, ArgsT &&... args) { if (likely(raw_ptr != nullptr)) { return false; } detail::do_init_thread_local(raw_ptr, std::forward(args)...); return true; } } // namespace td