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

485 lines
11 KiB
C++

#ifndef EFL_EOLIAN_INTEROP_HH
#define EFL_EOLIAN_INTEROP_HH
#include <string>
#include <tuple>
#include <utility>
#include <type_traits>
#include <initializer_list>
#include <Eina.hh>
#include <Eo.hh>
#include "eo_concrete.hh"
namespace efl { namespace eolian {
//// From C++ to C
inline const char*
to_c(std::string const& x)
{
return x.c_str();
}
inline const char*
to_c(eina::optional<std::string> const& x)
{
if (!x)
return nullptr;
return x->c_str();
}
inline const char*
to_c(eina::string_view const& x)
{
return x.data();
}
inline const char*
to_c(eina::optional<eina::string_view> const& x)
{
if (!x)
return nullptr;
return x->data();
}
inline const char*
to_c(efl::eina::stringshare const& x)
{
return x.c_str();
}
inline const char*
to_c(eina::optional<efl::eina::stringshare> const& x)
{
if (!x)
return nullptr;
return x->c_str();
}
inline Eina_Bool
to_c(bool x)
{
return x ? EINA_TRUE : EINA_FALSE;
}
inline Eina_Bool*
to_c(bool* x)
{
static_assert(sizeof(bool) == sizeof(Eina_Bool), "");
return reinterpret_cast<Eina_Bool*>(x);
}
template <typename T>
T to_c(T const& v, typename std::enable_if<!std::is_convertible<T*, ::efl::eo::concrete*>::value>::type* = 0)
{
return v;
}
template <typename T>
Eo* to_c(T const& v, typename std::enable_if<std::is_convertible<T*, ::efl::eo::concrete*>::value>::type* = 0)
{
return v._eo_ptr();
}
template <typename T>
Eo* to_c(eina::optional<T> const& v, typename std::enable_if<std::is_convertible<T*, ::efl::eo::concrete*>::value>::type* = 0)
{
if (!v)
return nullptr;
return v->_eo_ptr();
}
template <typename T>
Eo** to_c(T* v, typename std::enable_if<std::is_convertible<T*, ::efl::eo::concrete*>::value>::type* = 0)
{
static_assert(sizeof(T) == sizeof(Eo*), "");
return static_cast<Eo**>(static_cast<void*>(v));
}
template <typename R, typename T>
R to_native(T& v)
{
static_assert(sizeof(T) == sizeof(R), "");
return v.native_handle();
}
template <typename R, typename T>
R to_native(eina::optional<T>& v)
{
static_assert(sizeof(T) == sizeof(R), "");
if (!v)
return nullptr;
return v->native_handle();
}
template <typename R, typename T>
R to_native(T* v)
{
static_assert(sizeof(T) == sizeof(typename std::remove_pointer<R>::type), "");
return static_cast<R>(static_cast<void*>(v));
}
//// From C to C++
template <typename T>
struct tag {};
template <typename T, typename U, typename O>
T to_cxx(U object, O o);
// XXX
inline void*
to_cxx(void *x, std::tuple<std::false_type>, tag<void*>)
{
return x;
}
// XXX
inline const void*
to_cxx(const void *x, std::tuple<std::false_type>, tag<const void*>)
{
return x;
}
template <typename T>
inline T
to_cxx(Eo* x, std::tuple<std::true_type>, tag<T>)
{
return T(x);
}
template <typename T>
inline T
to_cxx(Eo* x, std::tuple<std::false_type>, tag<T>)
{
return T(::eo_ref(x));
}
template <typename T>
inline eina::optional<T>
to_cxx(Eo* x, std::tuple<std::true_type>, tag<eina::optional<T> >)
{
if (!x)
return nullptr;
return T(x);
}
template <typename T>
inline eina::optional<T>
to_cxx(Eo* x, std::tuple<std::false_type>, tag<eina::optional<T> >)
{
if (!x)
return nullptr;
return T(::eo_ref(x));
}
template <typename T>
inline T
to_cxx(Eo** x, std::tuple<std::false_type>, tag<T>)
{
static_assert(sizeof(Eo*) == sizeof(typename std::remove_pointer<T>::type), "");
return static_cast<T>((static_cast<void*>(x)));
}
#ifdef _EVAS_H
template <typename T>
Evas_Object_Textblock_Node_Format *
to_cxx(Evas_Object_Textblock_Node_Format* x, std::tuple<std::false_type>, tag<T>)
{
return x; // XXX
}
#endif
inline bool
to_cxx(Eina_Bool x, std::tuple<std::false_type>, tag<bool>)
{
return !!x;
}
inline std::string
to_cxx(const char* x, std::tuple<std::false_type>, tag<std::string>)
{
return std::string(x);
}
inline eina::optional<std::string>
to_cxx(const char* x, std::tuple<std::false_type>, tag<eina::optional<std::string> >)
{
if (!x)
return nullptr;
return std::string(x);
}
inline eina::string_view
to_cxx(const char* x, std::tuple<std::false_type>, tag<eina::string_view>)
{
return eina::string_view(x);
}
inline eina::optional<eina::string_view>
to_cxx(const char* x, std::tuple<std::false_type>, tag<eina::optional<eina::string_view> >)
{
if (!x)
return nullptr;
return eina::string_view(x);
}
template <typename T, typename Enable = void>
struct traits
{
typedef T type;
};
template <typename T>
struct traits
<T, typename std::enable_if<std::is_base_of<::efl::eo::concrete, T>::value>::type>
{
typedef Eo* type;
};
template <typename T>
struct traits
<T, typename std::enable_if<std::is_base_of<std::basic_string<char>, T>::value>::type>
{
typedef const char* type;
};
template <typename T>
struct traits
<T, typename std::enable_if<std::is_base_of<::efl::eina::basic_string_view<char>, T>::value>::type>
{
typedef const char* type;
};
template <typename T, typename ...Args>
inline efl::eina::range_list<T const>
to_cxx(const Eina_List* x, std::tuple<std::false_type, Args...>, tag< efl::eina::range_list<T> >)
{
return efl::eina::range_list<T const> {x};
}
template <typename T, typename ...Args>
inline eina::optional<efl::eina::range_list<T const> >
to_cxx(const Eina_List* x, std::tuple<std::false_type, Args...>, tag< eina::optional<efl::eina::range_list<T> > >)
{
if (!x)
return nullptr;
return efl::eina::range_list<T const> {x};
}
template <typename T, typename ...Args>
inline efl::eina::range_list<T>
to_cxx(Eina_List* x, std::tuple<std::false_type, Args...>, tag< efl::eina::range_list<T> >)
{
return efl::eina::range_list<T>{x};
}
template <typename T, typename ...Args>
inline eina::optional<efl::eina::range_list<T> >
to_cxx(Eina_List* x, std::tuple<std::false_type, Args...>, tag< eina::optional<efl::eina::range_list<T> > >)
{
if (!x)
return nullptr;
return efl::eina::range_list<T>{x};
}
template <typename T, typename ...Args>
inline efl::eina::list<T>
to_cxx(Eina_List* x, std::tuple<std::true_type, Args...>, tag< efl::eina::list<T> >)
{
return efl::eina::list<T> {x};
}
template <typename T, typename ...Args>
inline eina::optional<efl::eina::list<T> >
to_cxx(Eina_List* x, std::tuple<std::true_type, Args...>, tag< eina::optional<efl::eina::list<T> > >)
{
if (!x)
return nullptr;
return efl::eina::list<T> {x};
}
inline eina::stringshare
to_cxx(Eina_Stringshare const* x, const std::false_type, tag<eina::stringshare>)
{
return ::eina_stringshare_ref(x);
}
inline eina::optional<eina::stringshare>
to_cxx(Eina_Stringshare const* x, const std::false_type, tag<eina::optional<eina::stringshare> >)
{
if (!x)
return nullptr;
return eina::stringshare(::eina_stringshare_ref(x));
}
template <typename T, typename ...Args>
inline efl::eina::accessor<T>
to_cxx(Eina_Accessor* x, std::tuple<std::false_type, Args...>, tag< efl::eina::accessor<T> >)
{
return efl::eina::accessor<T>(x);
}
template <typename T, typename ...Args>
inline eina::optional<efl::eina::accessor<T> >
to_cxx(Eina_Accessor* x, std::tuple<std::false_type, Args...>, tag< eina::optional<efl::eina::accessor<T> > >)
{
if (!x)
return nullptr;
return efl::eina::accessor<T>(x);
}
template <typename T, typename ...Args>
inline efl::eina::iterator<T>
to_cxx(Eina_Iterator* x, std::tuple<std::false_type, Args...>, tag< efl::eina::iterator<T> >)
{
return efl::eina::iterator<T>(x);
}
template <typename T, typename ...Args>
inline eina::optional<efl::eina::iterator<T> >
to_cxx(Eina_Iterator* x, std::tuple<std::false_type, Args...>, tag< eina::optional<efl::eina::iterator<T> > >)
{
if (!x)
return nullptr;
return efl::eina::iterator<T>(x);
}
template <typename T, typename ...Args>
T
to_cxx(Eo const* x, std::tuple<std::false_type, Args...>, tag< T >
, typename std::enable_if<std::is_base_of<::efl::eo::concrete, T>::value>* = 0)
{
// Workaround for erroneous constness
return T{ ::eo_ref(const_cast<Eo*>(x))};
}
template <typename T, typename ...Args>
eina::optional<T>
to_cxx(Eo const* x, std::tuple<std::false_type, Args...>, tag< eina::optional<T> >
, typename std::enable_if<std::is_base_of<::efl::eo::concrete, T>::value>* = 0)
{
// Workaround for erroneous constness
if (!x)
return nullptr;
return T{ ::eo_ref(const_cast<Eo*>(x))};
}
template <typename T, typename ...Args>
T
to_cxx(Eo const* x, std::tuple<std::true_type, Args...>, tag< T >
, typename std::enable_if<std::is_base_of<::efl::eo::concrete, T>::value>* = 0)
{
// Workaround for erroneous constness
return T{const_cast<Eo*>(x)};
}
template <typename T, typename ...Args>
eina::optional<T>
to_cxx(Eo const* x, std::tuple<std::true_type, Args...>, tag< eina::optional<T> >
, typename std::enable_if<std::is_base_of<::efl::eo::concrete, T>::value>* = 0)
{
// Workaround for erroneous constness
if (!x)
return nullptr;
return T{const_cast<Eo*>(x)};
}
template <typename T, typename U, typename O>
T to_cxx(U object, O o)
{
return to_cxx(object, o, tag<T>());
}
//// Callbacks
template <typename F, typename R, typename V, typename... Args>
R funcall(V* data, Args... args)
{
F const* f = static_cast<F const*>(data);
return (*f)(args...);
}
template <typename T>
struct callback_result_type;
template <typename R, typename... Args>
struct callback_result_type<R(*)(Args...)>
{
typedef R type;
};
template <typename R>
struct callback_args_type;
template <typename R, typename... Args>
struct callback_args_type<R(*)(Args...)>
{
typedef std::tuple<Args...> type;
};
template <typename C, typename F, typename R, typename V, typename... Args>
C get_callback_impl(tag<std::tuple<V*, Args...> >)
{
static_assert(std::is_same<void, typename std::remove_cv<V>::type>::value,
"First argument of callback should be void* or const void*");
return static_cast<C>(&eolian::funcall<F, R, V, Args...>);
}
template <typename C, typename F>
C get_callback()
{
return get_callback_impl<C, F, typename callback_result_type<C>::type>
(tag<typename callback_args_type<C>::type>());
}
template <typename F>
Eina_Bool free_callback_callback(void* data, ::Eo_Event const*)
{
delete static_cast<F*>(data);
return EO_CALLBACK_CONTINUE;
}
template <typename... Fs>
inline
void register_ev_del_free_callback(Eo* eoptr, Fs&&... fs)
{
std::initializer_list<int const> const v {(fs.register_ev_del_free_callback(eoptr), 0)...};
(void) v; (void) eoptr;
}
template <typename F>
inline
std::vector<F>& get_static_callback_vector()
{
static std::vector<F> vec;
return vec;
}
template <typename F>
inline
F* alloc_static_callback(F&& f)
{
get_static_callback_vector<F>().push_back(std::forward<F>(f));
return &(get_static_callback_vector<F>().back());
}
/// Miscellaneous
template <typename... Fs>
inline
void call_ctors(Eo* _obj_eo_self, Fs&&... fs)
{
std::initializer_list<int const> const v {(fs(_obj_eo_self), 0)...};
(void) v;
(void) _obj_eo_self;
}
} } // namespace efl { namespace eolian {
#endif // EFL_EOLIAN_INTEROP_HH