2018-12-31 22:04:05 +03:00
|
|
|
//
|
2022-01-01 03:35:39 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2022
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// 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
|
2018-07-03 22:29:04 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/utils/port/config.h"
|
|
|
|
|
|
|
|
#ifdef TD_THREAD_STL
|
|
|
|
|
|
|
|
#include "td/utils/common.h"
|
|
|
|
#include "td/utils/invoke.h"
|
|
|
|
#include "td/utils/port/detail/ThreadIdGuard.h"
|
|
|
|
#include "td/utils/port/thread_local.h"
|
2019-07-06 13:29:15 +02:00
|
|
|
#include "td/utils/Slice.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
#include <thread>
|
|
|
|
#include <tuple>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
namespace td {
|
|
|
|
namespace detail {
|
2022-08-22 00:46:22 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
class ThreadStl {
|
|
|
|
public:
|
|
|
|
ThreadStl() = default;
|
|
|
|
ThreadStl(const ThreadStl &other) = delete;
|
|
|
|
ThreadStl &operator=(const ThreadStl &other) = delete;
|
|
|
|
ThreadStl(ThreadStl &&) = default;
|
|
|
|
ThreadStl &operator=(ThreadStl &&) = default;
|
2019-08-12 14:45:57 +03:00
|
|
|
~ThreadStl() {
|
|
|
|
join();
|
|
|
|
}
|
2022-08-22 00:46:22 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
template <class Function, class... Args>
|
2021-12-10 00:27:13 +03:00
|
|
|
explicit ThreadStl(Function &&f, Args &&...args) {
|
2018-12-31 22:04:05 +03:00
|
|
|
thread_ = std::thread([args = std::make_tuple(decay_copy(std::forward<Function>(f)),
|
|
|
|
decay_copy(std::forward<Args>(args))...)]() mutable {
|
|
|
|
ThreadIdGuard thread_id_guard;
|
|
|
|
invoke_tuple(std::move(args));
|
|
|
|
clear_thread_locals();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void join() {
|
2019-08-12 14:45:57 +03:00
|
|
|
if (thread_.joinable()) {
|
|
|
|
thread_.join();
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
2022-08-22 00:46:22 +03:00
|
|
|
|
2018-09-11 17:55:00 +03:00
|
|
|
void detach() {
|
2019-08-12 14:45:57 +03:00
|
|
|
if (thread_.joinable()) {
|
|
|
|
thread_.detach();
|
|
|
|
}
|
2018-09-11 17:55:00 +03:00
|
|
|
}
|
2022-08-22 00:46:22 +03:00
|
|
|
|
2019-07-06 13:29:15 +02:00
|
|
|
void set_name(CSlice name) {
|
2022-08-22 00:46:22 +03:00
|
|
|
// not supported
|
2019-07-06 13:29:15 +02:00
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
static unsigned hardware_concurrency() {
|
|
|
|
return std::thread::hardware_concurrency();
|
|
|
|
}
|
|
|
|
|
|
|
|
using id = std::thread::id;
|
|
|
|
|
2022-08-22 00:46:22 +03:00
|
|
|
static void send_real_time_signal(id thread_id, int real_time_signal_number) {
|
|
|
|
// not supported
|
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
private:
|
|
|
|
std::thread thread_;
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
std::decay_t<T> decay_copy(T &&v) {
|
|
|
|
return std::forward<T>(v);
|
|
|
|
}
|
|
|
|
};
|
2022-08-22 00:46:22 +03:00
|
|
|
|
2022-07-14 16:15:17 +04:00
|
|
|
namespace this_thread_stl {
|
|
|
|
using std::this_thread::get_id;
|
|
|
|
} // namespace this_thread_stl
|
2022-08-22 00:46:22 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
} // namespace detail
|
|
|
|
} // namespace td
|
|
|
|
|
|
|
|
#endif
|