1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
|
#pragma once
#include <cstdint>
#include <tuple>
#include <type_traits>
#include <variant>
#include <openvic-dataloader/detail/Concepts.hpp>
namespace ovdl::detail {
[[noreturn]] inline void unreachable() {
// Uses compiler specific extensions if possible.
// Even if no extension is used, undefined behavior is still raised by
// an empty function body and the noreturn attribute.
#ifdef __GNUC__ // GCC, Clang, ICC
__builtin_unreachable();
#elif defined(_MSC_VER) // MSVC
__assume(false);
#endif
}
template<typename EnumT>
requires std::is_enum_v<EnumT>
constexpr std::underlying_type_t<EnumT> to_underlying(EnumT e) {
return static_cast<std::underlying_type_t<EnumT>>(e);
}
template<typename EnumT>
requires std::is_enum_v<EnumT>
constexpr EnumT from_underlying(std::underlying_type_t<EnumT> ut) {
return static_cast<EnumT>(ut);
}
template<typename Type, typename... Types>
struct TypeRegister {
using tuple_type = std::tuple<Type, Types...>;
using variant_type = std::variant<Type, Types...>;
template<typename QueriedType>
struct _id_getter {
static constexpr std::uint32_t type_id() {
static_assert(any_of<QueriedType, Type, Types...>, "Cannot query an non-registered type");
if constexpr (std::is_same_v<Type, QueriedType>) return 0;
else return 1 + TypeRegister<Types...>::template _id_getter<QueriedType>::type_id();
};
};
template<typename QueriedType>
static constexpr std::uint32_t type_id() {
return _id_getter<QueriedType>::type_id();
}
template<std::uint32_t Id>
using type_by_id = std::tuple_element_t<Id, tuple_type>;
};
template<typename...>
struct type_concat;
template<typename... Ts, template<typename...> typename TT, typename... TTs>
struct type_concat<TT<TTs...>, Ts...> {
using type = TT<TTs..., Ts...>;
};
template<typename... Ts>
using type_concat_t = type_concat<Ts...>::type;
template<typename...>
struct type_prepend;
template<typename... Ts, template<typename...> typename TT, typename... TTs>
struct type_prepend<TT<TTs...>, Ts...> {
using type = TT<Ts..., TTs...>;
};
template<typename... Ts>
using type_prepend_t = type_prepend<Ts...>::type;
template<typename Type, template<typename...> typename Template>
struct is_instance_of : std::false_type {};
template<typename... Ts, template<typename...> typename Template>
struct is_instance_of<Template<Ts...>, Template> : std::true_type {};
template<typename Type, template<typename...> typename Template>
static constexpr auto is_instance_of_v = is_instance_of<Type, Template>::value;
}
|