Files
meta.hpp/untests/meta_states/function_tests.cpp
2022-01-14 07:35:25 +07:00

233 lines
9.9 KiB
C++

/*******************************************************************************
* This file is part of the "https://github.com/blackmatov/meta.hpp"
* For conditions of distribution and use, see copyright notice in LICENSE.md
* Copyright (C) 2021, by Matvey Cherevko (blackmatov@gmail.com)
******************************************************************************/
#include "../meta_untests.hpp"
namespace
{
struct ivec2 {
int x{};
int y{};
static ivec2 iadd(const ivec2& l, const ivec2& r) noexcept {
return {l.x + r.x, l.y + r.y};
}
static int ilength2(const ivec2& v) noexcept {
return v.x * v.x + v.y * v.y;
}
static int arg_bounded_arr(ivec2 vs[2]) { return vs[0].x + vs[0].y + vs[1].x + vs[1].y; }
static int arg_unbounded_arr(ivec2 vs[]) { return vs[0].x + vs[0].y + vs[1].x + vs[1].y; }
static int arg_bounded_const_arr(const ivec2 vs[2]) { return vs[0].x + vs[0].y + vs[1].x + vs[1].y; }
static int arg_unbounded_const_arr(const ivec2 vs[]) { return vs[0].x + vs[0].y + vs[1].x + vs[1].y; }
};
bool operator==(const ivec2& l, const ivec2& r) noexcept {
return l.x == r.x && l.y == r.y;
}
}
TEST_CASE("meta/meta_states/function") {
namespace meta = meta_hpp;
meta::class_<ivec2>()
.function_("iadd", &ivec2::iadd)
.function_("ilength2", &ivec2::ilength2)
.function_("arg_bounded_arr", &ivec2::arg_bounded_arr)
.function_("arg_unbounded_arr", &ivec2::arg_unbounded_arr)
.function_("arg_bounded_const_arr", &ivec2::arg_bounded_const_arr)
.function_("arg_unbounded_const_arr", &ivec2::arg_unbounded_const_arr);
const meta::class_type ivec2_type = meta::resolve_type<ivec2>();
REQUIRE(ivec2_type);
SUBCASE("") {
const meta::function func;
CHECK_FALSE(func);
CHECK_FALSE(func.is_valid());
CHECK(func == ivec2_type.get_function("non-existent-function"));
}
SUBCASE("operators") {
const meta::function iadd_f = ivec2_type.get_function("iadd");
const meta::function ilength2_f = ivec2_type.get_function("ilength2");
CHECK(iadd_f == iadd_f);
CHECK(iadd_f != ilength2_f);
CHECK((iadd_f < ilength2_f || ilength2_f < iadd_f));
}
SUBCASE("iadd") {
const meta::function func = ivec2_type.get_function("iadd");
REQUIRE(func);
CHECK(func.get_index().type == func.get_type());
CHECK(func.get_index().name == "iadd");
CHECK(func.get_type() == meta::resolve_type(&ivec2::iadd));
CHECK(func.get_name() == "iadd");
CHECK_FALSE(func.is_invocable_with<>());
CHECK_FALSE(func.is_invocable_with<int>());
CHECK_FALSE(func.is_invocable_with<ivec2, int>());
CHECK_FALSE(func.is_invocable_with<int, ivec2>());
CHECK_FALSE(func.is_invocable_with<ivec2, ivec2, int>());
CHECK(func.is_invocable_with<ivec2, ivec2>());
CHECK(func.is_invocable_with<const ivec2&, ivec2&&>());
CHECK_THROWS(func.invoke());
CHECK_THROWS(func.invoke(42));
CHECK_THROWS(func.invoke(ivec2{}, 42));
CHECK_THROWS(func.invoke(42, ivec2{}));
CHECK_THROWS(func.invoke(ivec2{}, ivec2{}, 42));
CHECK(func.invoke(ivec2{1,2}, ivec2{3,4}));
CHECK(func.invoke(ivec2{1,2}, ivec2{3,4}).value() == ivec2{4,6});
}
SUBCASE("ilength2") {
const meta::function func = ivec2_type.get_function("ilength2");
REQUIRE(func);
CHECK(func.get_index().type == func.get_type());
CHECK(func.get_index().name == "ilength2");
CHECK(func.get_type() == meta::resolve_type(&ivec2::ilength2));
CHECK(func.get_name() == "ilength2");
CHECK_FALSE(func.is_invocable_with<>());
CHECK_FALSE(func.is_invocable_with<int>());
CHECK_FALSE(func.is_invocable_with<ivec2, int>());
CHECK(func.is_invocable_with<ivec2>());
CHECK(func.is_invocable_with<const ivec2&>());
CHECK_THROWS(func.invoke());
CHECK_THROWS(func.invoke(42));
CHECK_THROWS(func.invoke(ivec2{}, 42));
CHECK(func.invoke(ivec2{2,3}));
CHECK(func.invoke(ivec2{2,3}).value() == 13);
}
SUBCASE("arg_arr") {
ivec2 bounded_arr[2]{{1,2},{3,4}};
ivec2* unbounded_arr = bounded_arr;
const ivec2 bounded_const_arr[2]{{1,2},{3,4}};
const ivec2* unbounded_const_arr = bounded_const_arr;
{
const meta::function func1 = ivec2_type.get_function("arg_bounded_arr");
REQUIRE(func1);
CHECK(func1.is_invocable_with<ivec2*>());
CHECK(func1.is_invocable_with<ivec2* const>());
CHECK_FALSE(func1.is_invocable_with<const ivec2*>());
CHECK_FALSE(func1.is_invocable_with<const ivec2* const>());
CHECK(func1.invoke(bounded_arr) == 10);
CHECK(func1.invoke(unbounded_arr) == 10);
CHECK_THROWS(func1.invoke(bounded_const_arr));
CHECK_THROWS(func1.invoke(unbounded_const_arr));
CHECK(func1.invoke(meta::value{bounded_arr}) == 10);
CHECK(func1.invoke(meta::value{unbounded_arr}) == 10);
CHECK_THROWS(func1.invoke(meta::value{bounded_const_arr}));
CHECK_THROWS(func1.invoke(meta::value{unbounded_const_arr}));
static_assert(std::is_invocable_v<decltype(&ivec2::arg_bounded_arr), decltype(bounded_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_bounded_arr), decltype(unbounded_arr)>);
static_assert(!std::is_invocable_v<decltype(&ivec2::arg_bounded_arr), decltype(bounded_const_arr)>);
static_assert(!std::is_invocable_v<decltype(&ivec2::arg_bounded_arr), decltype(unbounded_const_arr)>);
}
{
const meta::function func2 = ivec2_type.get_function("arg_unbounded_arr");
REQUIRE(func2);
CHECK(func2.is_invocable_with<ivec2*>());
CHECK(func2.is_invocable_with<ivec2* const>());
CHECK_FALSE(func2.is_invocable_with<const ivec2*>());
CHECK_FALSE(func2.is_invocable_with<const ivec2* const>());
CHECK(func2.invoke(bounded_arr) == 10);
CHECK(func2.invoke(unbounded_arr) == 10);
CHECK_THROWS(func2.invoke(bounded_const_arr));
CHECK_THROWS(func2.invoke(unbounded_const_arr));
CHECK(func2.invoke(meta::value{bounded_arr}) == 10);
CHECK(func2.invoke(meta::value{unbounded_arr}) == 10);
CHECK_THROWS(func2.invoke(meta::value{bounded_const_arr}));
CHECK_THROWS(func2.invoke(meta::value{unbounded_const_arr}));
static_assert(std::is_invocable_v<decltype(&ivec2::arg_unbounded_arr), decltype(bounded_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_unbounded_arr), decltype(unbounded_arr)>);
static_assert(!std::is_invocable_v<decltype(&ivec2::arg_unbounded_arr), decltype(bounded_const_arr)>);
static_assert(!std::is_invocable_v<decltype(&ivec2::arg_unbounded_arr), decltype(unbounded_const_arr)>);
}
}
SUBCASE("arg_const_arr") {
ivec2 bounded_arr[2]{{1,2},{3,4}};
ivec2* unbounded_arr = bounded_arr;
const ivec2 bounded_const_arr[2]{{1,2},{3,4}};
const ivec2* unbounded_const_arr = bounded_const_arr;
{
const meta::function func1 = ivec2_type.get_function("arg_bounded_const_arr");
REQUIRE(func1);
CHECK(func1.is_invocable_with<ivec2*>());
CHECK(func1.is_invocable_with<ivec2* const>());
CHECK(func1.is_invocable_with<const ivec2*>());
CHECK(func1.is_invocable_with<const ivec2* const>());
CHECK(func1.invoke(bounded_arr) == 10);
CHECK(func1.invoke(unbounded_arr) == 10);
CHECK(func1.invoke(bounded_const_arr) == 10);
CHECK(func1.invoke(unbounded_const_arr) == 10);
CHECK(func1.invoke(meta::value{bounded_arr}) == 10);
CHECK(func1.invoke(meta::value{unbounded_arr}) == 10);
CHECK(func1.invoke(meta::value{bounded_const_arr}) == 10);
CHECK(func1.invoke(meta::value{unbounded_const_arr}) == 10);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_bounded_const_arr), decltype(bounded_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_bounded_const_arr), decltype(unbounded_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_bounded_const_arr), decltype(bounded_const_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_bounded_const_arr), decltype(unbounded_const_arr)>);
}
{
const meta::function func2 = ivec2_type.get_function("arg_unbounded_const_arr");
REQUIRE(func2);
CHECK(func2.is_invocable_with<ivec2*>());
CHECK(func2.is_invocable_with<ivec2* const>());
CHECK(func2.is_invocable_with<const ivec2*>());
CHECK(func2.is_invocable_with<const ivec2* const>());
CHECK(func2.invoke(bounded_arr) == 10);
CHECK(func2.invoke(unbounded_arr) == 10);
CHECK(func2.invoke(bounded_const_arr) == 10);
CHECK(func2.invoke(unbounded_const_arr) == 10);
CHECK(func2.invoke(meta::value{bounded_arr}) == 10);
CHECK(func2.invoke(meta::value{unbounded_arr}) == 10);
CHECK(func2.invoke(meta::value{bounded_const_arr}) == 10);
CHECK(func2.invoke(meta::value{unbounded_const_arr}) == 10);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_unbounded_const_arr), decltype(bounded_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_unbounded_const_arr), decltype(unbounded_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_unbounded_const_arr), decltype(bounded_const_arr)>);
static_assert(std::is_invocable_v<decltype(&ivec2::arg_unbounded_const_arr), decltype(unbounded_const_arr)>);
}
}
}