2020-08-14 18:58:22 +02:00
|
|
|
#ifndef TGCALLS_THREAD_LOCAL_OBJECT_H
|
|
|
|
#define TGCALLS_THREAD_LOCAL_OBJECT_H
|
|
|
|
|
|
|
|
#include "rtc_base/thread.h"
|
|
|
|
|
|
|
|
#include <functional>
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
namespace tgcalls {
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class ThreadLocalObject {
|
|
|
|
public:
|
|
|
|
template <
|
|
|
|
typename Generator,
|
|
|
|
typename = std::enable_if_t<std::is_same<T*, decltype(std::declval<Generator>()())>::value>>
|
|
|
|
ThreadLocalObject(rtc::Thread *thread, Generator &&generator) :
|
|
|
|
_thread(thread),
|
|
|
|
_valueHolder(std::make_unique<ValueHolder>()) {
|
|
|
|
assert(_thread != nullptr);
|
2023-02-18 22:24:25 +01:00
|
|
|
_thread->PostTask([valueHolder = _valueHolder.get(), generator = std::forward<Generator>(generator)]() mutable {
|
2020-08-14 18:58:22 +02:00
|
|
|
valueHolder->_value.reset(generator());
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
~ThreadLocalObject() {
|
2023-02-18 22:24:25 +01:00
|
|
|
_thread->PostTask([valueHolder = std::move(_valueHolder)](){
|
2020-08-14 18:58:22 +02:00
|
|
|
valueHolder->_value.reset();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename FunctorT>
|
2023-02-18 22:24:25 +01:00
|
|
|
void perform(FunctorT &&functor) {
|
|
|
|
_thread->PostTask([valueHolder = _valueHolder.get(), f = std::forward<FunctorT>(functor)]() mutable {
|
2020-08-14 18:58:22 +02:00
|
|
|
assert(valueHolder->_value != nullptr);
|
|
|
|
f(valueHolder->_value.get());
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
T *getSyncAssumingSameThread() {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
assert(_valueHolder->_value != nullptr);
|
|
|
|
return _valueHolder->_value.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct ValueHolder {
|
|
|
|
std::shared_ptr<T> _value;
|
|
|
|
};
|
|
|
|
|
|
|
|
rtc::Thread *_thread = nullptr;
|
|
|
|
std::unique_ptr<ValueHolder> _valueHolder;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace tgcalls
|
|
|
|
|
|
|
|
#endif
|