diff --git a/CMakeLists.txt b/CMakeLists.txt index 29946ef..3be174d 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -27,7 +27,7 @@ macro(compile_options target_name) sanitizers(${target_name}) endmacro() -project(blt-gp VERSION 0.3.9) +project(blt-gp VERSION 0.3.10) include(CTest) diff --git a/include/blt/gp/operations.h b/include/blt/gp/operations.h index 1847bbb..59be19a 100644 --- a/include/blt/gp/operations.h +++ b/include/blt/gp/operations.h @@ -29,30 +29,29 @@ namespace blt::gp { - - template + template struct call_with { - template - [[nodiscard]] inline constexpr static blt::size_t getByteOffset() + template + [[nodiscard]] constexpr static size_t getByteOffset() { - blt::size_t offset = 0; - blt::size_t current_index = 0; + size_t offset = 0; + size_t current_index = 0; ((offset += (current_index++ > index ? stack_allocator::aligned_size>() : 0)), ...); -// BLT_INFO("offset %ld for argument %ld", offset, index); + (void) current_index; return offset; } - - template - void print_args(std::integer_sequence) + + template + void print_args(std::integer_sequence) { BLT_INFO("Arguments:"); (BLT_INFO("%ld: %s", indices, blt::type_string().c_str()), ...); } - - template - inline static constexpr Return exec_sequence_to_indices(Func&& func, stack_allocator& allocator, std::integer_sequence, - ExtraArgs&& ... args) + + template + static constexpr Return exec_sequence_to_indices(Func&& func, stack_allocator& allocator, std::integer_sequence, + ExtraArgs&&... args) { //blt::size_t arg_size = (stack_allocator::aligned_size>() + ...); //BLT_TRACE(arg_size); @@ -60,8 +59,8 @@ namespace blt::gp return std::forward(func)(std::forward(args)..., allocator.from>(getByteOffset())...); } - - template + + template void call_destructors_without_first(stack_allocator& read_allocator) const { if constexpr (sizeof...(NoCtxArgs) > 0) @@ -69,9 +68,9 @@ namespace blt::gp read_allocator.call_destructors...>(); } } - - template - Return operator()(const bool has_context, Func&& func, stack_allocator& read_allocator, ExtraArgs&& ... args) + + template + Return operator()(const bool has_context, Func&& func, stack_allocator& read_allocator, ExtraArgs&&... args) { constexpr auto seq = std::make_integer_sequence(); #if BLT_DEBUG_LEVEL > 0 @@ -94,126 +93,132 @@ namespace blt::gp #endif } }; - - template + + template struct call_without_first : public call_with { using call_with::call_with; }; - - template + + template class operation_t; - - template + + template class operation_t { - public: - using function_t = RawFunction; - using First_Arg = typename blt::meta::arg_helper::First; - - constexpr operation_t(const operation_t& copy) = default; - - constexpr operation_t(operation_t&& move) = default; - - template - constexpr explicit operation_t(const Functor& functor, std::optional name = {}): func(functor), name(name) - {} - - [[nodiscard]] constexpr inline Return operator()(stack_allocator& read_allocator) const - { - if constexpr (sizeof...(Args) == 0) - { - return func(); - } else - { - return call_with()(false, func, read_allocator); - } - } - - [[nodiscard]] constexpr inline Return operator()(void* context, stack_allocator& read_allocator) const - { - // should be an impossible state - if constexpr (sizeof...(Args) == 0) - { - BLT_ABORT("Cannot pass context to function without arguments!"); - } - auto& ctx_ref = *static_cast::type>*>(context); - if constexpr (sizeof...(Args) == 1) - { - return func(ctx_ref); - } else - { - return call_without_first()(true, func, read_allocator, ctx_ref); - } - } - - template - [[nodiscard]] detail::operator_func_t make_callable() const - { - return [this](void* context, stack_allocator& read_allocator, stack_allocator& write_allocator) { - if constexpr (detail::is_same_v::type>>) - { - // first arg is context - write_allocator.push(this->operator()(context, read_allocator)); - } else - { - // first arg isn't context - write_allocator.push(this->operator()(read_allocator)); - } - }; - } - - [[nodiscard]] inline constexpr std::optional get_name() const - { - return name; - } - - inline constexpr const auto& get_function() const - { - return func; - } - - auto set_ephemeral() - { - is_ephemeral_ = true; - return *this; - } - - bool is_ephemeral() const - { - return is_ephemeral_; - } + public: + using function_t = RawFunction; + using First_Arg = typename blt::meta::arg_helper::First; - bool return_has_drop() const + constexpr operation_t(const operation_t& copy) = default; + + constexpr operation_t(operation_t&& move) = default; + + template + constexpr explicit operation_t(const Functor& functor, std::optional name = {}): func(functor), name(name) + { + } + + [[nodiscard]] constexpr inline Return operator()(stack_allocator& read_allocator) const + { + if constexpr (sizeof...(Args) == 0) { - return detail::has_func_drop_v; + return func(); } - - operator_id id = -1; - private: - function_t func; - std::optional name; - bool is_ephemeral_ = false; + else + { + return call_with()(false, func, read_allocator); + } + } + + [[nodiscard]] constexpr inline Return operator()(void* context, stack_allocator& read_allocator) const + { + // should be an impossible state + if constexpr (sizeof...(Args) == 0) + { + BLT_ABORT("Cannot pass context to function without arguments!"); + } + auto& ctx_ref = *static_cast::type>*>(context); + if constexpr (sizeof...(Args) == 1) + { + return func(ctx_ref); + } + else + { + return call_without_first()(true, func, read_allocator, ctx_ref); + } + } + + template + [[nodiscard]] detail::operator_func_t make_callable() const + { + return [this](void* context, stack_allocator& read_allocator, stack_allocator& write_allocator) + { + if constexpr (detail::is_same_v::type>>) + { + // first arg is context + write_allocator.push(this->operator()(context, read_allocator)); + } + else + { + // first arg isn't context + write_allocator.push(this->operator()(read_allocator)); + } + }; + } + + [[nodiscard]] inline constexpr std::optional get_name() const + { + return name; + } + + inline constexpr const auto& get_function() const + { + return func; + } + + auto set_ephemeral() + { + is_ephemeral_ = true; + return *this; + } + + bool is_ephemeral() const + { + return is_ephemeral_; + } + + bool return_has_drop() const + { + return detail::has_func_drop_v; + } + + operator_id id = -1; + + private: + function_t func; + std::optional name; + bool is_ephemeral_ = false; }; - - template + + template class operation_t : public operation_t { - public: - using operation_t::operation_t; + public: + using operation_t::operation_t; }; - - template + + template operation_t(Lambda) -> operation_t; - - template - operation_t(Return(*)(Args...)) -> operation_t; - - template + + template + operation_t(Return (*)(Args...)) -> operation_t; + + template operation_t(Lambda, std::optional) -> operation_t; - - template - operation_t(Return(*)(Args...), std::optional) -> operation_t; + + template + operation_t(Return (*)(Args...), std::optional) -> operation_t; } #endif //BLT_GP_OPERATIONS_H diff --git a/include/blt/gp/stack.h b/include/blt/gp/stack.h index 93de54d..f189108 100644 --- a/include/blt/gp/stack.h +++ b/include/blt/gp/stack.h @@ -229,7 +229,8 @@ namespace blt::gp if constexpr (sizeof...(Args) > 0) { size_t offset = (aligned_size>() + ...) - aligned_size::First>>(); - ((call_drop(offset), offset -= aligned_size>()), ...); + ((call_drop(offset + (gp::detail::has_func_drop_v ? sizeof(u64*) : 0)), offset -= aligned_size>()), ...); + (void) offset; } }