Telegram-Android/TMessagesProj/jni/voip/tgcalls/ThreadLocalObject.h

59 lines
1.4 KiB
C
Raw Permalink Normal View History

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