draconisplusplus/include/rfl/internal/get_field_names.hpp

160 lines
4.4 KiB
C++
Raw Normal View History

2024-05-31 22:59:00 -04:00
#ifndef RFL_INTERNAL_GETFIELDNAMES_HPP_
#define RFL_INTERNAL_GETFIELDNAMES_HPP_
#include <array>
#include <iostream>
#include <memory>
#include <source_location>
#include <string>
#include <tuple>
#include <type_traits>
#include <utility>
#include "../Literal.hpp"
#include "bind_fake_object_to_tuple.hpp"
#include "get_fake_object.hpp"
#include "is_flatten_field.hpp"
#include "is_rename.hpp"
#include "num_fields.hpp"
#if __GNUC__
#ifndef __clang__
#pragma GCC system_header
#endif
#endif
namespace rfl::internal {
2024-06-08 14:10:59 -04:00
template <class T>
struct Wrapper {
using Type = T;
T v;
};
template <class T>
Wrapper(T) -> Wrapper<T>;
// This workaround is necessary for clang.
template <class T>
constexpr auto wrap(const T& arg) noexcept {
2024-06-16 00:13:15 -04:00
return Wrapper { arg };
2024-06-08 14:10:59 -04:00
}
template <class T, auto ptr>
consteval auto get_field_name_str_view() {
// Unfortunately, we cannot avoid the use of a compiler-specific macro for
// Clang on Windows. For all other compilers, function_name works as
// intended.
2024-05-31 22:59:00 -04:00
#if defined(__clang__) && defined(_MSC_VER)
2024-06-16 00:13:15 -04:00
const auto func_name = std::string_view { __PRETTY_FUNCTION__ };
2024-05-31 22:59:00 -04:00
#else
2024-06-16 00:13:15 -04:00
const auto func_name = std::string_view { std::source_location::current().function_name() };
2024-05-31 22:59:00 -04:00
#endif
#if defined(__clang__)
2024-06-08 14:10:59 -04:00
const auto split = func_name.substr(0, func_name.size() - 2);
return split.substr(split.find_last_of(":.") + 1);
2024-05-31 22:59:00 -04:00
#elif defined(__GNUC__)
2024-06-08 14:10:59 -04:00
const auto split = func_name.substr(0, func_name.size() - 2);
return split.substr(split.find_last_of(":") + 1);
2024-05-31 22:59:00 -04:00
#elif defined(_MSC_VER)
2024-06-08 14:10:59 -04:00
const auto split = func_name.substr(0, func_name.size() - 7);
return split.substr(split.rfind("->") + 2);
2024-05-31 22:59:00 -04:00
#else
2024-06-08 14:10:59 -04:00
static_assert(
2024-06-16 00:13:15 -04:00
false,
"You are using an unsupported compiler. Please use GCC, Clang "
"or MSVC or switch to the rfl::Field-syntax."
2024-06-08 15:53:06 -04:00
);
2024-05-31 22:59:00 -04:00
#endif
2024-06-08 14:10:59 -04:00
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <class T, auto ptr>
consteval auto get_field_name_str_lit() {
2024-06-08 15:53:06 -04:00
constexpr auto name = get_field_name_str_view<T, ptr>();
const auto to_str_lit = [&]<auto... Ns>(std::index_sequence<Ns...>) {
2024-06-16 00:13:15 -04:00
return StringLiteral<sizeof...(Ns) + 1> { name[Ns]... };
2024-06-08 14:10:59 -04:00
};
return to_str_lit(std::make_index_sequence<name.size()> {});
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <class T>
auto get_field_names();
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <class T, auto ptr>
auto get_field_name() {
2024-05-31 22:59:00 -04:00
#if defined(__clang__)
2024-06-16 00:13:15 -04:00
using Type = std::remove_cvref_t<
std::remove_pointer_t<typename std::remove_pointer_t<decltype(ptr)>::Type>>;
2024-05-31 22:59:00 -04:00
#else
2024-06-08 14:10:59 -04:00
using Type = std::remove_cvref_t<std::remove_pointer_t<decltype(ptr)>>;
2024-05-31 22:59:00 -04:00
#endif
2024-06-08 14:10:59 -04:00
if constexpr (is_rename_v<Type>) {
using Name = typename Type::Name;
return Name();
} else if constexpr (is_flatten_field_v<Type>) {
return get_field_names<std::remove_cvref_t<typename Type::Type>>();
} else {
return rfl::Literal<get_field_name_str_lit<T, ptr>()>();
}
2024-05-31 22:59:00 -04:00
}
2024-06-08 14:10:59 -04:00
template <StringLiteral... _names1, StringLiteral... _names2>
2024-06-08 15:53:06 -04:00
auto concat_two_literals(
2024-06-16 00:13:15 -04:00
const rfl::Literal<_names1...>& _lit1,
const rfl::Literal<_names2...>& _lit2
2024-06-08 15:53:06 -04:00
) {
2024-06-08 14:10:59 -04:00
return rfl::Literal<_names1..., _names2...>::template from_value<0>();
2024-05-31 22:59:00 -04:00
}
2024-06-08 14:10:59 -04:00
template <class Head, class... Tail>
auto concat_literals(const Head& _head, const Tail&... _tail) {
if constexpr (sizeof...(_tail) == 0) {
return _head;
} else {
return concat_two_literals(_head, concat_literals(_tail...));
}
}
inline auto concat_literals() { return rfl::Literal<>(); }
2024-05-31 22:59:00 -04:00
#ifdef __clang__
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wundefined-var-template"
#pragma clang diagnostic ignored "-Wundefined-inline"
#endif
2024-06-08 14:10:59 -04:00
template <class T>
2024-05-31 22:59:00 -04:00
#if __GNUC__
#ifndef __clang__
2024-06-08 14:10:59 -04:00
[[gnu::no_sanitize_undefined]]
2024-05-31 22:59:00 -04:00
#endif
#endif
2024-06-08 14:10:59 -04:00
auto get_field_names() {
using Type = std::remove_cvref_t<T>;
if constexpr (std::is_pointer_v<Type>) {
return get_field_names<std::remove_pointer_t<T>>();
} else {
2024-05-31 22:59:00 -04:00
#if defined(__clang__)
2024-06-08 14:10:59 -04:00
const auto get = []<std::size_t... Is>(std::index_sequence<Is...>) {
return concat_literals(
2024-06-16 00:13:15 -04:00
get_field_name<Type, wrap(std::get<Is>(bind_fake_object_to_tuple<T>()))>()...
2024-06-08 15:53:06 -04:00
);
2024-06-08 14:10:59 -04:00
};
2024-05-31 22:59:00 -04:00
#else
2024-06-08 14:10:59 -04:00
const auto get = []<std::size_t... Is>(std::index_sequence<Is...>) {
2024-06-16 00:13:15 -04:00
return concat_literals(get_field_name<Type, std::get<Is>(bind_fake_object_to_tuple<T>())>(
)...);
2024-06-08 14:10:59 -04:00
};
2024-05-31 22:59:00 -04:00
#endif
2024-06-08 14:10:59 -04:00
return get(std::make_index_sequence<num_fields<T>>());
}
2024-05-31 22:59:00 -04:00
}
#ifdef __clang__
#pragma clang diagnostic pop
#endif
2024-06-08 14:10:59 -04:00
} // namespace rfl::internal
2024-05-31 22:59:00 -04:00
#endif