diff options
author | Thomas Lively <tlively@google.com> | 2023-10-20 21:14:20 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-10-20 21:14:20 +0200 |
commit | b83d8679a7641312f09380e1cef0e56ceb341d18 (patch) | |
tree | ab7bbfc9b98a50e4cbf5517d5a4dbec9bcea00d0 /src/analysis | |
parent | be6a3393c36ccc1a0cb0d79b116cbe48e169f93b (diff) | |
download | binaryen-b83d8679a7641312f09380e1cef0e56ceb341d18.tar.gz binaryen-b83d8679a7641312f09380e1cef0e56ceb341d18.tar.bz2 binaryen-b83d8679a7641312f09380e1cef0e56ceb341d18.zip |
[analysis][NFC] Move powerset lattices to their own header (#6028)
Move the powerset lattices out of lattice.h, which now only contains the Lattice
concept, to their own dedicated header in a new analysis/lattices directory.
Diffstat (limited to 'src/analysis')
-rw-r--r-- | src/analysis/CMakeLists.txt | 2 | ||||
-rw-r--r-- | src/analysis/lattice.h | 149 | ||||
-rw-r--r-- | src/analysis/lattices/powerset-impl.h (renamed from src/analysis/powerset-lattice-impl.h) | 28 | ||||
-rw-r--r-- | src/analysis/lattices/powerset.h | 158 | ||||
-rw-r--r-- | src/analysis/liveness-transfer-function.h | 1 |
5 files changed, 199 insertions, 139 deletions
diff --git a/src/analysis/CMakeLists.txt b/src/analysis/CMakeLists.txt index 2fc4be305..7895e5af8 100644 --- a/src/analysis/CMakeLists.txt +++ b/src/analysis/CMakeLists.txt @@ -1,4 +1,4 @@ -file(GLOB analysis_HEADERS *.h) +file(GLOB analysis_HEADERS *.h lattices/*.h) set(analysis_SOURCES cfg.cpp sign-lattice.cpp diff --git a/src/analysis/lattice.h b/src/analysis/lattice.h index 354447efc..f3dd6122e 100644 --- a/src/analysis/lattice.h +++ b/src/analysis/lattice.h @@ -1,12 +1,22 @@ +/* + * 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_lattice_h #define wasm_analysis_lattice_h -#include <iostream> -#include <unordered_map> -#include <vector> - -#include "wasm.h" - namespace wasm::analysis { enum LatticeComparison { NO_RELATION, EQUAL, LESS, GREATER }; @@ -51,133 +61,6 @@ concept Lattice = requires(const L& lattice, #endif // __cplusplus >= 202002L -// Represents a powerset lattice constructed from a finite set of consecutive -// integers from 0 to n which can be represented by a bitvector. Set elements -// are represented by FiniteIntPowersetLattice::Element, which represents -// members present in each element by bits in the bitvector. -class FiniteIntPowersetLattice { - // The size of the set that the powerset lattice was created from. This is - // equivalent to the size of the Top lattice element. - size_t setSize; - -public: - FiniteIntPowersetLattice(size_t setSize) : setSize(setSize) {} - - // Returns the size of the set that the powerset lattices was created from. - size_t getSetSize() { return setSize; } - - // This represents an element of a powerset lattice. The element is itself a - // set which has set members. The bitvector tracks which possible members of - // the element are actually present. - class Element { - // If bitvector[i] is true, then member i is present in the lattice element, - // otherwise it isn't. - std::vector<bool> bitvector; - - // This constructs a bottom element, given the lattice set size. Used by the - // lattice's getBottom function. - Element(size_t latticeSetSize) : bitvector(latticeSetSize) {} - - public: - Element(Element&& source) = default; - Element(const Element& source) = default; - - Element& operator=(Element&& source) = default; - Element& operator=(const Element& source) = default; - - // Counts the number of members present the element itself. For instance, if - // we had {true, false, true}, the count would be 2. O(N) operation which - // iterates through the bitvector. - size_t count() const; - - bool get(size_t index) { return bitvector[index]; } - void set(size_t index, bool value) { bitvector[index] = value; } - - bool isTop() const { return count() == bitvector.size(); } - bool isBottom() const { return count() == 0; } - - // Calculates the LUB of this element with some other element and sets - // this element to the LUB in place. Returns true if this element before - // this method call was different than the LUB. - bool makeLeastUpperBound(const Element& other) noexcept; - - // Prints out the bits in the bitvector for a lattice element. - void print(std::ostream& os); - - friend FiniteIntPowersetLattice; - }; - - // Compares two lattice elements and returns a result indicating the - // left element's relation to the right element. - LatticeComparison compare(const Element& left, - const Element& right) const noexcept; - - // Returns an instance of the bottom lattice element. - Element getBottom() const noexcept; -}; - -// A layer of abstraction over FiniteIntPowersetLattice which maps -// set members of some type T to indices in the bitvector. Allows -// the finite powerset lattice to be generalized to arbitrary types. -template<typename T> class FinitePowersetLattice { - FiniteIntPowersetLattice intLattice; - - // Maps a bitvector index to some element member of type T. - // Used to produce initial ordering of element members. - std::vector<T> members; - - // Maps an element member of type T to a bitvector index. - std::unordered_map<T, size_t> memberIndices; - -public: - using Element = FiniteIntPowersetLattice::Element; - - // Takes in an ordered list of all elements of the set to create - // the powerset lattice from (i.e. the powerset lattice top element). This - // is used for mapping the elements to bitvector indices. - FinitePowersetLattice(std::vector<T>&& setMembers) - : intLattice(setMembers.size()), members(std::move(setMembers)) { - for (size_t i = 0; i < members.size(); ++i) { - memberIndices[members[i]] = i; - } - } - - // Iterator to access the list of element members. - using membersIterator = typename std::vector<T>::const_iterator; - membersIterator membersBegin() { return members.cbegin(); } - membersIterator membersEnd() { return members.cend(); } - size_t getSetSize() { return intLattice.getSetSize(); } - - T indexToMember(size_t index) { return members[index]; } - - size_t memberToIndex(T member) { return memberIndices[member]; } - - // Adds member to a powerset lattice element. - void add(Element* element, T member) { - element->set(memberIndices[member], true); - } - - // Removes member from a powerset lattice element. - void remove(Element* element, T member) { - element->set(memberIndices[member], false); - } - - // Checks if member is included in the element set. - bool exists(Element* element, T member) { - return element->get(memberIndices[member]); - } - - // We use implementations from FiniteIntPowersetLattice here. - LatticeComparison compare(const Element& left, - const Element& right) const noexcept { - return intLattice.compare(left, right); - } - - Element getBottom() const noexcept { return intLattice.getBottom(); } -}; - } // namespace wasm::analysis -#include "powerset-lattice-impl.h" - #endif // wasm_analysis_lattice_h diff --git a/src/analysis/powerset-lattice-impl.h b/src/analysis/lattices/powerset-impl.h index 33ebd312f..3390934d3 100644 --- a/src/analysis/powerset-lattice-impl.h +++ b/src/analysis/lattices/powerset-impl.h @@ -1,7 +1,25 @@ -#ifndef wasm_analysis_lattice_impl_h -#define wasm_analysis_lattice_impl_h - -#include "lattice.h" +/* + * 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_powerset_impl_h +#define wasm_analysis_lattices_powerset_impl_h + +#include <iostream> + +#include "powerset.h" namespace wasm::analysis { @@ -85,4 +103,4 @@ inline void FiniteIntPowersetLattice::Element::print(std::ostream& os) { }; // namespace wasm::analysis -#endif // wasm_analysis_lattice_impl_h +#endif // wasm_analysis_lattices_powerset_impl_h diff --git a/src/analysis/lattices/powerset.h b/src/analysis/lattices/powerset.h new file mode 100644 index 000000000..92159da12 --- /dev/null +++ b/src/analysis/lattices/powerset.h @@ -0,0 +1,158 @@ +/* + * 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. + */ + +#include <unordered_map> +#include <vector> + +#include "wasm.h" + +#include "../lattice.h" + +#ifndef wasm_analysis_lattices_powerset_h +#define wasm_analysis_lattices_powerset_h + +namespace wasm::analysis { + +// Represents a powerset lattice constructed from a finite set of consecutive +// integers from 0 to n which can be represented by a bitvector. Set elements +// are represented by FiniteIntPowersetLattice::Element, which represents +// members present in each element by bits in the bitvector. +class FiniteIntPowersetLattice { + // The size of the set that the powerset lattice was created from. This is + // equivalent to the size of the Top lattice element. + size_t setSize; + +public: + FiniteIntPowersetLattice(size_t setSize) : setSize(setSize) {} + + // Returns the size of the set that the powerset lattices was created from. + size_t getSetSize() { return setSize; } + + // This represents an element of a powerset lattice. The element is itself a + // set which has set members. The bitvector tracks which possible members of + // the element are actually present. + class Element { + // If bitvector[i] is true, then member i is present in the lattice element, + // otherwise it isn't. + std::vector<bool> bitvector; + + // This constructs a bottom element, given the lattice set size. Used by the + // lattice's getBottom function. + Element(size_t latticeSetSize) : bitvector(latticeSetSize) {} + + public: + Element(Element&& source) = default; + Element(const Element& source) = default; + + Element& operator=(Element&& source) = default; + Element& operator=(const Element& source) = default; + + // Counts the number of members present the element itself. For instance, if + // we had {true, false, true}, the count would be 2. O(N) operation which + // iterates through the bitvector. + size_t count() const; + + bool get(size_t index) { return bitvector[index]; } + void set(size_t index, bool value) { bitvector[index] = value; } + + bool isTop() const { return count() == bitvector.size(); } + bool isBottom() const { return count() == 0; } + + // Calculates the LUB of this element with some other element and sets + // this element to the LUB in place. Returns true if this element before + // this method call was different than the LUB. + bool makeLeastUpperBound(const Element& other) noexcept; + + // Prints out the bits in the bitvector for a lattice element. + void print(std::ostream& os); + + friend FiniteIntPowersetLattice; + }; + + // Compares two lattice elements and returns a result indicating the + // left element's relation to the right element. + LatticeComparison compare(const Element& left, + const Element& right) const noexcept; + + // Returns an instance of the bottom lattice element. + Element getBottom() const noexcept; +}; + +// A layer of abstraction over FiniteIntPowersetLattice which maps +// set members of some type T to indices in the bitvector. Allows +// the finite powerset lattice to be generalized to arbitrary types. +template<typename T> class FinitePowersetLattice { + FiniteIntPowersetLattice intLattice; + + // Maps a bitvector index to some element member of type T. + // Used to produce initial ordering of element members. + std::vector<T> members; + + // Maps an element member of type T to a bitvector index. + std::unordered_map<T, size_t> memberIndices; + +public: + using Element = FiniteIntPowersetLattice::Element; + + // Takes in an ordered list of all elements of the set to create + // the powerset lattice from (i.e. the powerset lattice top element). This + // is used for mapping the elements to bitvector indices. + FinitePowersetLattice(std::vector<T>&& setMembers) + : intLattice(setMembers.size()), members(std::move(setMembers)) { + for (size_t i = 0; i < members.size(); ++i) { + memberIndices[members[i]] = i; + } + } + + // Iterator to access the list of element members. + using membersIterator = typename std::vector<T>::const_iterator; + membersIterator membersBegin() { return members.cbegin(); } + membersIterator membersEnd() { return members.cend(); } + size_t getSetSize() { return intLattice.getSetSize(); } + + T indexToMember(size_t index) { return members[index]; } + + size_t memberToIndex(T member) { return memberIndices[member]; } + + // Adds member to a powerset lattice element. + void add(Element* element, T member) { + element->set(memberIndices[member], true); + } + + // Removes member from a powerset lattice element. + void remove(Element* element, T member) { + element->set(memberIndices[member], false); + } + + // Checks if member is included in the element set. + bool exists(Element* element, T member) { + return element->get(memberIndices[member]); + } + + // We use implementations from FiniteIntPowersetLattice here. + LatticeComparison compare(const Element& left, + const Element& right) const noexcept { + return intLattice.compare(left, right); + } + + Element getBottom() const noexcept { return intLattice.getBottom(); } +}; + +} // namespace wasm::analysis + +#include "powerset-impl.h" + +#endif // wasm_analysis_lattices_powerset_h diff --git a/src/analysis/liveness-transfer-function.h b/src/analysis/liveness-transfer-function.h index c6d564659..802f7f3fc 100644 --- a/src/analysis/liveness-transfer-function.h +++ b/src/analysis/liveness-transfer-function.h @@ -1,6 +1,7 @@ #ifndef wasm_analysis_liveness_transfer_function_h #define wasm_analysis_liveness_transfer_function_h +#include "lattices/powerset.h" #include "visitor-transfer-function.h" namespace wasm::analysis { |