/* * Copyright 2023 WebAssembly Community Group participants * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef wasm_analysis_lattices_tuple_h #define wasm_analysis_lattices_tuple_h #include #include #include "bool.h" #include "support/utilities.h" namespace wasm::analysis { template struct Tuple { using Element = std::tuple; std::tuple lattices; Tuple(Ls&&... lattices) : lattices({std::move(lattices)...}) {} private: template Element getBottomImpl(std::index_sequence) const noexcept { return {std::get(lattices).getBottom()...}; } template Element getTopImpl(std::index_sequence) const noexcept { return {std::get(lattices).getTop()...}; } LatticeComparison compareImpl(const Element& a, const Element& b, LatticeComparison result, std::index_sequence<>) const noexcept { // Base case: there is nothing left to compare. return result; } template LatticeComparison compareImpl(const Element& a, const Element& b, LatticeComparison result, std::index_sequence) const noexcept { // Recursive case: compare the current elements, update `result`, and // recurse to the next elements if necessary. switch (std::get(lattices).compare(std::get(a), std::get(b))) { case EQUAL: return compareImpl(a, b, result, std::index_sequence{}); case LESS: if (result == GREATER) { // Cannot be both less and greater. return NO_RELATION; } return compareImpl(a, b, LESS, std::index_sequence{}); case GREATER: if (result == LESS) { // Cannot be both greater and less. return NO_RELATION; } return compareImpl(a, b, GREATER, std::index_sequence{}); case NO_RELATION: return NO_RELATION; } WASM_UNREACHABLE("unexpected comparison"); } int joinImpl(Element& joinee, const Element& joiner, std::index_sequence<>) const noexcept { // Base case: there is nothing left to join. return false; } template int joinImpl(Element& joinee, const Element& joiner, std::index_sequence) const noexcept { // Recursive case: join the current element and recurse to the next // elements. return std::get(lattices).join(std::get(joinee), std::get(joiner)) | joinImpl(joinee, joiner, std::index_sequence{}); } int meetImpl(Element& meetee, const Element& meeter, std::index_sequence<>) const noexcept { // Base case: there is nothing left to mee. return false; } template int meetImpl(Element& meetee, const Element& meeter, std::index_sequence) const noexcept { // Recursive case: meet the current element and recurse to the next // elements. return (std::get(lattices).meet(std::get(meetee), std::get(meeter))) | meetImpl(meetee, meeter, std::index_sequence{}); } public: Element getBottom() const noexcept { return getBottomImpl(std::index_sequence_for()); } Element getTop() const noexcept { return getTopImpl(std::index_sequence_for()); } LatticeComparison compare(const Element& a, const Element& b) const noexcept { return compareImpl(a, b, EQUAL, std::index_sequence_for()); } bool join(Element& joinee, const Element& joiner) const noexcept { return joinImpl(joinee, joiner, std::index_sequence_for()); } bool meet(Element& meetee, const Element& meeter) const noexcept { return meetImpl(meetee, meeter, std::index_sequence_for()); } }; #if __cplusplus >= 202002L static_assert(FullLattice>); static_assert(FullLattice>); #endif } // namespace wasm::analysis #endif // wasm_analysis_lattices_tuple_h