efl/src/bindings/cxx/eo_cxx/eo_future.hh

357 lines
11 KiB
C++

///
/// @file eo_future.hh
///
#ifndef EFL_CXX_EO_FUTURE_HH
#define EFL_CXX_EO_FUTURE_HH
#include <Efl.h>
#include <Eina.hh>
#include <Ecore_Manual.hh>
#include <mutex>
#include <condition_variable>
#include <eina_tuple.hh>
#include <eo_promise_meta.hh>
namespace efl {
template <typename...Args>
struct shared_future;
namespace _impl {
template <typename V = char>
struct wait_state
{
bool available = false;
bool has_failed = false;
std::mutex mutex;
std::condition_variable cv;
typename std::aligned_storage<sizeof(V), alignof(V)>::type storage;
Eina_Error error;
};
static void get_error_cb(void* data, Efl_Event const* event)
{
struct wait_state<>* wait_state = static_cast<struct wait_state<>*>(data);
Efl_Future_Event_Failure* info = static_cast<Efl_Future_Event_Failure*>(event->info);
std::unique_lock<std::mutex> l(wait_state->mutex);
wait_state->error = info->error;
wait_state->has_failed = true;
wait_state->available = true;
wait_state->cv.notify_one();
}
struct shared_future_common
{
explicit shared_future_common(Efl_Future* future)
: _future(future) {}
shared_future_common()
: _future(nullptr) {}
~shared_future_common()
{
if(_future)
efl_unref(_future);
}
shared_future_common(shared_future_common const& future)
: _future(efl_ref(future._future))
{
}
shared_future_common& operator=(shared_future_common const& other)
{
_self_type tmp(other);
tmp.swap(*this);
return *this;
}
shared_future_common(shared_future_common&& future)
: _future(future._future)
{
future._future = nullptr;
}
shared_future_common& operator=(shared_future_common&& other)
{
other.swap(*this);
return *this;
}
void swap(shared_future_common& other)
{
std::swap(_future, other._future);
}
bool valid() const noexcept
{
return _future != nullptr;
}
void wait() const
{
if(eina_main_loop_is())
throw std::runtime_error("Deadlock");
struct wait_state<> wait_state;
efl::ecore::main_loop_thread_safe_call_async
([&]
{
efl_future_then(this->_future, &wait_success, &wait_success, nullptr, &wait_state);
});
std::unique_lock<std::mutex> lock(wait_state.mutex);
while(!wait_state.available)
wait_state.cv.wait(lock);
}
static void wait_success(void* data, Efl_Event const*)
{
struct wait_state<>* wait_state = static_cast<struct wait_state<>*>(data);
std::unique_lock<std::mutex> l(wait_state->mutex);
wait_state->available = true;
wait_state->cv.notify_one();
}
typedef Efl_Future* native_handle_type;
native_handle_type native_handle() const noexcept { return _future; }
typedef shared_future_common _self_type;
Efl_Future* _future;
};
template <typename T>
struct shared_future_1_type : private shared_future_common
{
typedef shared_future_common _base_type;
using _base_type::_base_type;
using _base_type::swap;
using _base_type::valid;
using _base_type::native_handle;
using _base_type::wait;
typedef _base_type::native_handle_type native_handle_type;
T get() const
{
if(eina_main_loop_is())
throw std::runtime_error("Deadlock");
struct wait_state<T> wait_state;
efl::ecore::main_loop_thread_safe_call_async
([&]
{
efl_future_then(this->_future, &get_success, &_impl::get_error_cb, nullptr, &wait_state);
});
{
std::unique_lock<std::mutex> lock(wait_state.mutex);
while(!wait_state.available)
wait_state.cv.wait(lock);
}
if(wait_state.has_failed)
EFL_CXX_THROW(eina::system_error(eina::error_code(wait_state.error, eina::eina_error_category()), "EFL Eina Error"));
return *static_cast<T*>(static_cast<void*>(&wait_state.storage));
}
static void get_success(void* data, Efl_Event const* event)
{
struct wait_state<T>* wait_state = static_cast<struct wait_state<T>*>(data);
Efl_Future_Event_Success* info = static_cast<Efl_Future_Event_Success*>(event->info);
std::unique_lock<std::mutex> l(wait_state->mutex);
_impl::future_copy_traits<T>::copy(static_cast<T*>(static_cast<void*>(&wait_state->storage)), info);
wait_state->available = true;
wait_state->cv.notify_one();
}
typedef shared_future_1_type<T> _self_type;
};
template <typename T>
struct shared_race_future_1_type : private shared_future_common
{
typedef shared_future_common _base_type;
using _base_type::_base_type;
using _base_type::swap;
using _base_type::valid;
using _base_type::native_handle;
using _base_type::wait;
typedef _base_type::native_handle_type native_handle_type;
T get() const
{
if(eina_main_loop_is())
throw std::runtime_error("Deadlock");
struct wait_state<T> wait_state;
efl::ecore::main_loop_thread_safe_call_async
([&]
{
efl_future_then(this->_future, &get_success, &_impl::get_error_cb, nullptr, &wait_state);
});
{
std::unique_lock<std::mutex> lock(wait_state.mutex);
while(!wait_state.available)
wait_state.cv.wait(lock);
}
if(wait_state.has_failed)
EFL_CXX_THROW(eina::system_error(eina::error_code(wait_state.error, eina::eina_error_category()), "EFL Eina Error"));
return *static_cast<T*>(static_cast<void*>(&wait_state.storage));
}
static void get_success(void* data, Efl_Event const* event)
{
struct wait_state<T>* wait_state = static_cast<struct wait_state<T>*>(data);
Efl_Future_Event_Success* info = static_cast<Efl_Future_Event_Success*>(event->info);
std::unique_lock<std::mutex> l(wait_state->mutex);
_impl::future_copy_traits<T>::copy_race(static_cast<T*>(static_cast<void*>(&wait_state->storage)), info);
wait_state->available = true;
wait_state->cv.notify_one();
}
typedef shared_future_1_type<T> _self_type;
};
template <typename...Args>
struct shared_future_varargs_type : private shared_future_common
{
typedef shared_future_common _base_type;
using _base_type::_base_type;
using _base_type::swap;
using _base_type::valid;
using _base_type::native_handle;
using _base_type::wait;
typedef _base_type::native_handle_type native_handle_type;
typedef std::tuple<Args...> tuple_type;
std::tuple<Args...> get() const
{
if(eina_main_loop_is())
throw std::runtime_error("Deadlock");
struct wait_state<tuple_type> wait_state;
efl::ecore::main_loop_thread_safe_call_async
([&]
{
efl_future_then(this->_future, &get_success, &_impl::get_error_cb, nullptr, &wait_state);
});
{
std::unique_lock<std::mutex> lock(wait_state.mutex);
while(!wait_state.available)
wait_state.cv.wait(lock);
}
if(wait_state.has_failed)
EFL_CXX_THROW(eina::system_error(eina::error_code(wait_state.error, eina::eina_error_category()), "EFL Eina Error"));
return *static_cast<tuple_type*>(static_cast<void*>(&wait_state.storage));
}
template <std::size_t N>
static void read_accessor(Eina_Accessor* accessor
, std::tuple<typename std::aligned_storage<sizeof(Args), alignof(Args)>::type...>& storage_tuple
, wait_state<tuple_type>* wait_state
, std::false_type)
{
typedef typename std::tuple_element<N, tuple_type>::type type;
void* value;
if(eina_accessor_data_get(accessor, N, &value))
{
eina::copy_from_c_traits<type>::copy_to_unitialized
(static_cast<type*>(static_cast<void*>(&std::get<N>(storage_tuple))), value);
_self_type::read_accessor<N+1>(accessor, storage_tuple, wait_state
, std::integral_constant<bool, (N+1 == sizeof...(Args))>());
}
else
{
std::abort();
// some error
}
}
template <std::size_t N, std::size_t...I>
static void read_accessor_end(std::tuple<typename std::aligned_storage<sizeof(Args), alignof(Args)>::type...>& storage_tuple
, wait_state<tuple_type>* wait_state
, eina::index_sequence<I...>)
{
std::unique_lock<std::mutex> l(wait_state->mutex);
new (&wait_state->storage) tuple_type{(*static_cast<typename std::tuple_element<I, tuple_type>::type*>
(static_cast<void*>(&std::get<I>(storage_tuple))))...};
wait_state->available = true;
wait_state->cv.notify_one();
}
template <std::size_t N>
static void read_accessor(Eina_Accessor*
, std::tuple<typename std::aligned_storage<sizeof(Args), alignof(Args)>::type...>& storage_tuple
, wait_state<tuple_type>* wait_state
, std::true_type)
{
_self_type::read_accessor_end<N>(storage_tuple, wait_state, eina::make_index_sequence<sizeof...(Args)>{});
}
static void get_success(void* data, Efl_Event const* event)
{
struct wait_state<tuple_type>* wait_state = static_cast<struct wait_state<tuple_type>*>(data);
Efl_Future_Event_Success* info = static_cast<Efl_Future_Event_Success*>(event->info);
Eina_Accessor* accessor = static_cast<Eina_Accessor*>(info->value);
std::tuple<typename std::aligned_storage<sizeof(Args), alignof(Args)>::type...> storage_tuple;
_self_type::read_accessor<0u>(accessor, storage_tuple, wait_state, std::false_type());
}
typedef shared_future_varargs_type<Args...> _self_type;
};
}
template <typename...Args>
struct shared_future : private std::conditional<sizeof...(Args) == 1, _impl::shared_future_1_type<typename std::tuple_element<0u, std::tuple<Args...>>::type>, _impl::shared_future_varargs_type<Args...>>::type
{
typedef typename std::conditional<sizeof...(Args) == 1, _impl::shared_future_1_type<typename std::tuple_element<0u, std::tuple<Args...>>::type>, _impl::shared_future_varargs_type<Args...>>::type _base_type;
using _base_type::_base_type;
using _base_type::swap;
using _base_type::valid;
using _base_type::get;
using _base_type::wait;
using _base_type::native_handle;
typedef typename _base_type::native_handle_type native_handle_type;
};
template <typename...Args>
struct shared_race_future : private std::conditional<sizeof...(Args) == 1, _impl::shared_race_future_1_type<typename std::tuple_element<0u, std::tuple<Args...>>::type>, void>::type
{
typedef typename std::conditional<sizeof...(Args) == 1, _impl::shared_race_future_1_type<typename std::tuple_element<0u, std::tuple<Args...>>::type>, void>::type _base_type;
using _base_type::_base_type;
using _base_type::swap;
using _base_type::valid;
using _base_type::get;
using _base_type::wait;
using _base_type::native_handle;
typedef typename _base_type::native_handle_type native_handle_type;
};
namespace _impl {
template <typename T>
struct is_race_future : std::false_type {};
template <typename...Args>
struct is_race_future<shared_race_future<Args...>> : std::true_type {};
}
}
#endif