2023-01-16 14:59:33 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <BAN/Errors.h>
|
2023-01-17 11:16:46 +02:00
|
|
|
#include <BAN/Move.h>
|
2024-01-31 23:55:41 +02:00
|
|
|
#include <BAN/PlacementNew.h>
|
2023-01-16 14:59:33 +02:00
|
|
|
|
|
|
|
namespace BAN
|
|
|
|
{
|
|
|
|
|
|
|
|
template<typename>
|
|
|
|
class Function;
|
|
|
|
template<typename Ret, typename... Args>
|
|
|
|
class Function<Ret(Args...)>
|
|
|
|
{
|
|
|
|
public:
|
2023-02-01 21:41:18 +02:00
|
|
|
Function() = default;
|
2023-01-16 14:59:33 +02:00
|
|
|
Function(Ret(*function)(Args...))
|
|
|
|
{
|
|
|
|
static_assert(sizeof(CallablePointer) <= m_size);
|
|
|
|
new (m_storage) CallablePointer(function);
|
|
|
|
}
|
|
|
|
template<typename Own>
|
|
|
|
Function(Ret(Own::*function)(Args...), Own* owner)
|
|
|
|
{
|
|
|
|
static_assert(sizeof(CallableMember<Own>) <= m_size);
|
|
|
|
new (m_storage) CallableMember<Own>(function, owner);
|
|
|
|
}
|
|
|
|
template<typename Own>
|
|
|
|
Function(Ret(Own::*function)(Args...) const, const Own* owner)
|
|
|
|
{
|
|
|
|
static_assert(sizeof(CallableMemberConst<Own>) <= m_size);
|
|
|
|
new (m_storage) CallableMemberConst<Own>(function, owner);
|
|
|
|
}
|
2023-01-18 14:39:31 +02:00
|
|
|
template<typename Lambda>
|
2024-06-27 00:33:50 +03:00
|
|
|
Function(Lambda lambda) requires requires(Lambda lamda, Args&&... args) { { lambda(forward<Args>(args)...) } -> BAN::same_as<Ret>; }
|
2023-01-18 14:39:31 +02:00
|
|
|
{
|
|
|
|
static_assert(sizeof(CallableLambda<Lambda>) <= m_size);
|
|
|
|
new (m_storage) CallableLambda<Lambda>(lambda);
|
|
|
|
}
|
2023-01-16 14:59:33 +02:00
|
|
|
|
2023-02-01 21:41:18 +02:00
|
|
|
~Function()
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
2023-08-22 11:30:40 +03:00
|
|
|
Ret operator()(Args... args) const
|
2023-01-16 14:59:33 +02:00
|
|
|
{
|
2023-01-17 11:16:46 +02:00
|
|
|
ASSERT(*this);
|
2023-08-22 11:30:40 +03:00
|
|
|
return reinterpret_cast<const CallableBase*>(m_storage)->call(forward<Args>(args)...);
|
2023-01-16 14:59:33 +02:00
|
|
|
}
|
|
|
|
|
2023-01-17 11:16:46 +02:00
|
|
|
operator bool() const
|
2023-01-16 14:59:33 +02:00
|
|
|
{
|
|
|
|
for (size_t i = 0; i < m_size; i++)
|
|
|
|
if (m_storage[i])
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2024-01-24 14:43:46 +02:00
|
|
|
|
2023-02-01 21:41:18 +02:00
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
if (*this)
|
|
|
|
reinterpret_cast<CallableBase*>(m_storage)->~CallableBase();
|
|
|
|
memset(m_storage, 0, m_size);
|
|
|
|
}
|
|
|
|
|
2023-02-02 23:19:18 +02:00
|
|
|
static constexpr size_t size() { return m_size; }
|
|
|
|
|
2023-01-16 14:59:33 +02:00
|
|
|
private:
|
|
|
|
struct CallableBase
|
|
|
|
{
|
|
|
|
virtual ~CallableBase() {}
|
2023-08-22 11:30:40 +03:00
|
|
|
virtual Ret call(Args...) const = 0;
|
2023-01-16 14:59:33 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct CallablePointer : public CallableBase
|
|
|
|
{
|
|
|
|
CallablePointer(Ret(*function)(Args...))
|
|
|
|
: m_function(function)
|
|
|
|
{ }
|
|
|
|
|
2023-08-22 11:30:40 +03:00
|
|
|
virtual Ret call(Args... args) const override
|
2023-01-16 14:59:33 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
return m_function(forward<Args>(args)...);
|
2023-01-16 14:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Ret(*m_function)(Args...) = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename Own>
|
|
|
|
struct CallableMember : public CallableBase
|
|
|
|
{
|
|
|
|
CallableMember(Ret(Own::*function)(Args...), Own* owner)
|
|
|
|
: m_owner(owner)
|
|
|
|
, m_function(function)
|
|
|
|
{ }
|
|
|
|
|
2023-08-22 11:30:40 +03:00
|
|
|
virtual Ret call(Args... args) const override
|
2023-01-16 14:59:33 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
return (m_owner->*m_function)(forward<Args>(args)...);
|
2023-01-16 14:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Own* m_owner = nullptr;
|
|
|
|
Ret(Own::*m_function)(Args...) = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename Own>
|
|
|
|
struct CallableMemberConst : public CallableBase
|
|
|
|
{
|
|
|
|
CallableMemberConst(Ret(Own::*function)(Args...) const, const Own* owner)
|
|
|
|
: m_owner(owner)
|
|
|
|
, m_function(function)
|
|
|
|
{ }
|
|
|
|
|
2023-08-22 11:30:40 +03:00
|
|
|
virtual Ret call(Args... args) const override
|
2023-01-16 14:59:33 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
return (m_owner->*m_function)(forward<Args>(args)...);
|
2023-01-16 14:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const Own* m_owner = nullptr;
|
|
|
|
Ret(Own::*m_function)(Args...) const = nullptr;
|
|
|
|
};
|
|
|
|
|
2023-01-18 14:39:31 +02:00
|
|
|
template<typename Lambda>
|
|
|
|
struct CallableLambda : public CallableBase
|
|
|
|
{
|
|
|
|
CallableLambda(Lambda lambda)
|
|
|
|
: m_lambda(lambda)
|
|
|
|
{ }
|
|
|
|
|
2023-08-22 11:30:40 +03:00
|
|
|
virtual Ret call(Args... args) const override
|
2023-01-18 14:39:31 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
return m_lambda(forward<Args>(args)...);
|
2023-01-18 14:39:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Lambda m_lambda;
|
|
|
|
};
|
|
|
|
|
2023-01-16 14:59:33 +02:00
|
|
|
private:
|
2024-05-28 16:01:41 +03:00
|
|
|
static constexpr size_t m_size = sizeof(void*) * 8;
|
2023-03-09 01:42:17 +02:00
|
|
|
alignas(CallableBase) uint8_t m_storage[m_size] { 0 };
|
2023-01-16 14:59:33 +02:00
|
|
|
};
|
|
|
|
|
2024-01-24 14:43:46 +02:00
|
|
|
}
|