summaryrefslogtreecommitdiff
path: root/src/analysis/powerset-lattice-impl.h
diff options
context:
space:
mode:
authorBruce He <44327446+zm2he@users.noreply.github.com>2023-06-29 20:16:40 +0000
committerGitHub <noreply@github.com>2023-06-29 16:16:40 -0400
commit7b2e06aa3a0b6cef9568c3eab010a7187b9b6803 (patch)
tree4408c424ab640f9534362784f3355db4981480b7 /src/analysis/powerset-lattice-impl.h
parent84316e8fc3448932c4d62d1e749047aaacf02ef2 (diff)
downloadbinaryen-7b2e06aa3a0b6cef9568c3eab010a7187b9b6803.tar.gz
binaryen-7b2e06aa3a0b6cef9568c3eab010a7187b9b6803.tar.bz2
binaryen-7b2e06aa3a0b6cef9568c3eab010a7187b9b6803.zip
Dynamic Sized Bitvector Powerset Lattice for Static Analysis (#5784)
This change introduces a finite-height powerset lattice FinitePowersetLattice where each element's height is determined when constructed in runtime. This is implemented using a vector of `bools. This change also modifies BlockState and MonotoneCFGAnalyzer to be templated on a generic lattice type instead of using a hardcoded lattice. It additionally fixes an iterator bug in MonotoneCFGAnalyzer::fromCFG which assigned a temporary iterator object as predecessor and successor pointers to BlockStates instead of pointers to actual BlockState objects.
Diffstat (limited to 'src/analysis/powerset-lattice-impl.h')
-rw-r--r--src/analysis/powerset-lattice-impl.h106
1 files changed, 63 insertions, 43 deletions
diff --git a/src/analysis/powerset-lattice-impl.h b/src/analysis/powerset-lattice-impl.h
index 6207c2bec..24931618a 100644
--- a/src/analysis/powerset-lattice-impl.h
+++ b/src/analysis/powerset-lattice-impl.h
@@ -5,61 +5,81 @@
namespace wasm::analysis {
-template<size_t N>
-inline BitsetPowersetLattice<N> BitsetPowersetLattice<N>::getBottom() {
- // Return the empty set as the bottom lattice element.
- BitsetPowersetLattice<N> result{0};
- return result;
-}
+inline LatticeComparison
+FinitePowersetLattice::compare(const FinitePowersetLattice::Element& left,
+ const FinitePowersetLattice::Element& right) {
+ // Both must be from the powerset lattice of the same set.
+ assert(left.bitvector.size() == right.bitvector.size());
-template<size_t N>
-inline bool
-BitsetPowersetLattice<N>::isTop(const BitsetPowersetLattice<N>& element) {
- // Top lattice element is the set containing all possible elements.
- return element.value.all();
-}
+ // True in left, false in right.
+ bool leftNotRight = false;
-template<size_t N>
-inline bool
-BitsetPowersetLattice<N>::isBottom(const BitsetPowersetLattice<N>& element) {
- // Bottom lattice element is the empty set.
- return element.value.none();
-}
+ // True in right, false in left.
+ bool rightNotLeft = false;
-template<size_t N>
-inline LatticeComparison
-BitsetPowersetLattice<N>::compare(const BitsetPowersetLattice<N>& left,
- const BitsetPowersetLattice<N>& right) {
- size_t leftCount = left.value.count();
- size_t rightCount = right.value.count();
-
- // If left has more elements, left might be a superset of right.
- if (leftCount > rightCount) {
- if ((left.value | right.value) == left.value) {
- return GREATER;
+ size_t size = left.bitvector.size();
+
+ for (size_t i = 0; i < size; ++i) {
+ leftNotRight |= (left.bitvector[i] && !right.bitvector[i]);
+ rightNotLeft |= (right.bitvector[i] && !left.bitvector[i]);
+
+ // We can end early if we know neither is a subset of the other.
+ if (leftNotRight && rightNotLeft) {
+ return NO_RELATION;
}
- // If right has more elements, right might be a superset of left.
- } else if (leftCount < rightCount) {
- if ((left.value | right.value) == right.value) {
- return LESS;
+ }
+
+ if (!leftNotRight) {
+ if (!rightNotLeft) {
+ return EQUAL;
}
- } else if (left.value == right.value) {
- return EQUAL;
+ return LESS;
+ } else if (!rightNotLeft) {
+ return GREATER;
}
return NO_RELATION;
}
-// Implement LUB as an OR of the two bitsets.
-template<size_t N>
-inline void BitsetPowersetLattice<N>::getLeastUpperBound(
- const BitsetPowersetLattice<N>& right) {
- value |= right.value;
+inline FinitePowersetLattice::Element FinitePowersetLattice::getBottom() {
+ FinitePowersetLattice::Element result(setSize);
+ return result;
+}
+
+// We count the number of element members present in the element by counting the
+// trues in the bitvector.
+inline size_t FinitePowersetLattice::Element::count() {
+ size_t count = 0;
+ for (auto it : bitvector) {
+ count += it;
+ }
+ return count;
+}
+
+// Least upper bound is implemented as a logical OR between the bitvectors on
+// both sides. We return true if a bit is flipped in-place on the left so the
+// worklist algorithm will know if when to enqueue more work.
+inline bool FinitePowersetLattice::Element::makeLeastUpperBound(
+ const FinitePowersetLattice::Element& other) {
+ // Both must be from powerset lattice of the same set.
+ assert(other.bitvector.size() == bitvector.size());
+
+ bool modified = false;
+ for (size_t i = 0; i < bitvector.size(); ++i) {
+ // Bit is flipped on self only if self is false and other is true when self
+ // and other are OR'ed together.
+ modified |= (!bitvector[i] && other.bitvector[i]);
+ bitvector[i] = bitvector[i] || other.bitvector[i];
+ }
+
+ return modified;
}
-template<size_t N>
-inline void BitsetPowersetLattice<N>::print(std::ostream& os) {
- os << value;
+inline void FinitePowersetLattice::Element::print(std::ostream& os) {
+ // Element member 0 is on the left, element member N is on the right.
+ for (auto it : bitvector) {
+ os << it;
+ }
}
}; // namespace wasm::analysis