From a6b62ed0beae3b3f25244132c484c6d0d3dca821 Mon Sep 17 00:00:00 2001 From: Alexander McCord Date: Mon, 15 Jul 2024 16:28:07 -0700 Subject: [PATCH] Move Variant to Common and simplify Language. --- {Analysis => Common}/include/Luau/Variant.h | 0 EqSat/include/Luau/Language.h | 140 ++++---------------- Sources.cmake | 2 +- 3 files changed, 28 insertions(+), 114 deletions(-) rename {Analysis => Common}/include/Luau/Variant.h (100%) diff --git a/Analysis/include/Luau/Variant.h b/Common/include/Luau/Variant.h similarity index 100% rename from Analysis/include/Luau/Variant.h rename to Common/include/Luau/Variant.h diff --git a/EqSat/include/Luau/Language.h b/EqSat/include/Luau/Language.h index 9e180383..2a84a303 100644 --- a/EqSat/include/Luau/Language.h +++ b/EqSat/include/Luau/Language.h @@ -4,6 +4,7 @@ #include "Luau/Id.h" #include "Luau/LanguageHash.h" #include "Luau/Slice.h" +#include "Luau/Variant.h" #include #include @@ -222,151 +223,62 @@ public: template class Language final { - static constexpr size_t bufferSize = std::max({sizeof(Ts)...}); - static constexpr size_t bufferAlign = std::max({alignof(Ts)...}); + Variant v; - const char* tag; - alignas(bufferAlign) char buffer[bufferSize]; - -private: template using WithinDomain = std::disjunction, Ts>...>; - using FnCopy = void (*)(void*, const void*); - using FnMove = void (*)(void*, void*); - using FnDtor = void (*)(void*); - using FnPred = bool (*)(const void*, const void*); - using FnHash = size_t (*)(const void*); - - template - using FnSlice = Slice (*)(std::conditional_t, const void*, void*>); - - template - static void fnCopy(void* dst, const void* src) noexcept - { - new (dst) T(*static_cast(src)); - } - - template - static void fnMove(void* dst, void* src) noexcept - { - new (dst) T(static_cast(*static_cast(src))); - } - - template - static void fnDtor(void* dst) noexcept - { - static_cast(dst)->~T(); - } - - template - static bool fnPred(const void* lhs, const void* rhs) noexcept - { - return *static_cast(lhs) == *static_cast(rhs); - } - - template - static size_t fnHash(const void* buffer) noexcept - { - return typename T::Hash{}(*static_cast(buffer)); - } - - template - static Slice fnOperands(std::conditional_t, const void*, void*> buffer) noexcept - { - return static_cast, const T*, T*>>(buffer)->operands(); - } - - static constexpr FnCopy tableCopy[sizeof...(Ts)] = {&fnCopy...}; - static constexpr FnMove tableMove[sizeof...(Ts)] = {&fnMove...}; - static constexpr FnDtor tableDtor[sizeof...(Ts)] = {&fnDtor...}; - static constexpr FnPred tablePred[sizeof...(Ts)] = {&fnPred...}; - static constexpr FnHash tableHash[sizeof...(Ts)] = {&fnHash...}; - static constexpr FnSlice tableSliceId[sizeof...(Ts)] = {&fnOperands...}; - static constexpr FnSlice tableSliceConstId[sizeof...(Ts)] = {&fnOperands...}; - - static constexpr int getIndexFromTag(const char* tag) noexcept - { - constexpr int N = sizeof...(Ts); - constexpr const char* is[N] = {Ts::tag...}; - - for (int i = 0; i < N; ++i) - if (is[i] == tag) - return i; - - return -1; - } - public: template Language(T&& t, std::enable_if_t::value>* = 0) noexcept + : v(std::forward(t)) { - tag = std::decay_t::tag; - new (&buffer) std::decay_t(std::forward(t)); } - Language(const Language& other) noexcept - { - tag = other.tag; - tableCopy[getIndexFromTag(tag)](&buffer, &other.buffer); - } + Language(const Language&) noexcept = default; + Language& operator=(const Language&) noexcept = default; - Language(Language&& other) noexcept - { - tag = other.tag; - tableMove[getIndexFromTag(tag)](&buffer, &other.buffer); - } - - ~Language() noexcept - { - tableDtor[getIndexFromTag(tag)](&buffer); - } - - Language& operator=(const Language& other) noexcept - { - Language copy{other}; - *this = static_cast(copy); - return *this; - } - - Language& operator=(Language&& other) noexcept - { - if (this != &other) - { - tableDtor[getIndexFromTag(tag)](&buffer); - tag = other.tag; - tableMove[getIndexFromTag(tag)](&buffer, &other.buffer); // nothrow - } - return *this; - } + Language(Language&&) noexcept = default; + Language& operator=(Language&&) noexcept = default; int index() const noexcept { - return getIndexFromTag(tag); + return v.index(); } /// You should never call this function with the intention of mutating the `Id`. /// Reading is ok, but you should also never assume that these `Id`s are stable. Slice operands() noexcept { - return tableSliceId[getIndexFromTag(tag)](&buffer); + return visit([](auto&& v) -> Slice { + return v.operands(); + }, v); } Slice operands() const noexcept { - return tableSliceConstId[getIndexFromTag(tag)](&buffer); + return visit([](auto&& v) -> Slice { + return v.operands(); + }, v); + } + + template + T* get() noexcept + { + static_assert(WithinDomain::value); + return v.template get_if(); } template const T* get() const noexcept { static_assert(WithinDomain::value); - return tag == T::tag ? reinterpret_cast(&buffer) : nullptr; + return v.template get_if(); } bool operator==(const Language& rhs) const noexcept { - return tag == rhs.tag && tablePred[getIndexFromTag(tag)](&buffer, &rhs.buffer); + return v == rhs.v; } bool operator!=(const Language& rhs) const noexcept @@ -379,8 +291,10 @@ public: { size_t operator()(const Language& language) const { - size_t seed = std::hash{}(language.tag); - hashCombine(seed, tableHash[getIndexFromTag(language.tag)](&language.buffer)); + size_t seed = std::hash{}(language.index()); + hashCombine(seed, visit([](auto&& v) { + return typename std::decay_t::Hash{}(v); + }, language.v)); return seed; } }; diff --git a/Sources.cmake b/Sources.cmake index dee795f1..97d7e9ee 100644 --- a/Sources.cmake +++ b/Sources.cmake @@ -7,6 +7,7 @@ if(NOT ${CMAKE_VERSION} VERSION_LESS "3.19") Common/include/Luau/BytecodeUtils.h Common/include/Luau/DenseHash.h Common/include/Luau/ExperimentalFlags.h + Common/include/Luau/Variant.h Common/include/Luau/VecDeque.h ) endif() @@ -232,7 +233,6 @@ target_sources(Luau.Analysis PRIVATE Analysis/include/Luau/Unifier.h Analysis/include/Luau/Unifier2.h Analysis/include/Luau/UnifierSharedState.h - Analysis/include/Luau/Variant.h Analysis/include/Luau/VisitType.h Analysis/src/Anyification.cpp