#ifndef ABSL_FUNCTIONAL_INTERNAL_FUNCTION_REF_H_
#define ABSL_FUNCTIONAL_INTERNAL_FUNCTION_REF_H_
#include <cassert>
#include <functional>
#include <type_traits>
#include "absl/functional/any_invocable.h"
#include "absl/meta/type_traits.h"
namespace absl {
ABSL_NAMESPACE_BEGIN
namespace functional_internal {
union VoidPtr {
const void* obj;
void (*fun)();
};
template <typename T,
bool IsLValueReference = std::is_lvalue_reference<T>::value>
struct PassByValue : std::false_type {};
template <typename T>
struct PassByValue<T, false>
: std::integral_constant<bool,
absl::is_trivially_copy_constructible<T>::value &&
absl::is_trivially_copy_assignable<
typename std::remove_cv<T>::type>::value &&
std::is_trivially_destructible<T>::value &&
sizeof(T) <= 2 * sizeof(void*)> {};
template <typename T>
struct ForwardT : std::conditional<PassByValue<T>::value, T, T&&> {};
template <typename R, typename... Args>
using Invoker = R (*)(VoidPtr, typename ForwardT<Args>::type...);
template <typename Obj, typename R, typename... Args>
R InvokeObject(VoidPtr ptr, typename ForwardT<Args>::type... args) {
using T = std::remove_reference_t<Obj>;
return static_cast<R>(std::invoke(
std::forward<Obj>(*const_cast<T*>(static_cast<const T*>(ptr.obj))),
std::forward<typename ForwardT<Args>::type>(args)...));
}
template <typename Obj, typename Fun, Fun F, typename R, typename... Args>
R InvokeObject(VoidPtr ptr, typename ForwardT<Args>::type... args) {
using T = std::remove_reference_t<Obj>;
Obj&& obj =
std::forward<Obj>(*const_cast<T*>(static_cast<const T*>(ptr.obj)));
if constexpr (std::is_member_function_pointer_v<Fun>) {
return static_cast<R>((std::forward<Obj>(obj).*F)(
std::forward<typename ForwardT<Args>::type>(args)...));
} else {
return static_cast<R>(
F(std::forward<Obj>(obj),
std::forward<typename ForwardT<Args>::type>(args)...));
}
}
template <typename T, typename Fun, Fun F, typename R, typename... Args>
R InvokePtr(VoidPtr ptr, typename ForwardT<Args>::type... args) {
T* obj = const_cast<T*>(static_cast<const T*>(ptr.obj));
if constexpr (std::is_member_function_pointer_v<Fun>) {
return static_cast<R>(
(obj->*F)(std::forward<typename ForwardT<Args>::type>(args)...));
} else {
return static_cast<R>(
F(obj, std::forward<typename ForwardT<Args>::type>(args)...));
}
}
template <typename Fun, typename R, typename... Args>
R InvokeFunction(VoidPtr ptr, typename ForwardT<Args>::type... args) {
auto f = reinterpret_cast<Fun>(ptr.fun);
return static_cast<R>(
std::invoke(f, std::forward<typename ForwardT<Args>::type>(args)...));
}
template <typename Fun, Fun F, typename R, typename... Args>
R InvokeFunction(VoidPtr, typename ForwardT<Args>::type... args) {
return static_cast<R>(
F(std::forward<typename ForwardT<Args>::type>(args)...));
}
template <typename Sig>
void AssertNonNull(const std::function<Sig>& f) {
assert(f != nullptr);
(void)f;
}
template <typename Sig>
void AssertNonNull(const AnyInvocable<Sig>& f) {
assert(f != nullptr);
(void)f;
}
template <typename F>
void AssertNonNull(const F&) {}
template <typename F, typename C>
void AssertNonNull(F C::* f) {
assert(f != nullptr);
(void)f;
}
template <bool C>
using EnableIf = typename ::std::enable_if<C, int>::type;
} ABSL_NAMESPACE_END
}
#endif