diff options
-rw-r--r-- | .github/workflows/ci.yml | 38 | ||||
-rw-r--r-- | CMakeLists.txt | 1 | ||||
-rwxr-xr-x | scripts/clang-format-diff.sh | 9 | ||||
-rw-r--r-- | src/analysis/lattice.h | 68 | ||||
-rw-r--r-- | src/analysis/monotone-analyzer-impl.h | 41 | ||||
-rw-r--r-- | src/analysis/monotone-analyzer.h | 35 | ||||
-rw-r--r-- | src/analysis/powerset-lattice-impl.h | 7 | ||||
-rw-r--r-- | src/analysis/stack-lattice.h | 10 | ||||
-rw-r--r-- | src/analysis/visitor-transfer-function.h | 9 | ||||
-rw-r--r-- | src/tools/wasm-fuzz-lattices.cpp | 55 |
10 files changed, 151 insertions, 122 deletions
diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 889624cd6..b225bad3b 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -15,6 +15,9 @@ jobs: name: lint if: ${{ github.event_name == 'pull_request' }} runs-on: ubuntu-latest + env: + # Keep this in sync with clang-format-diff.sh + LLVM_VERSION: 17 steps: - uses: actions/setup-python@v1 with: @@ -25,7 +28,11 @@ jobs: - name: install tools run: | sudo pip3 install -r requirements-dev.txt - sudo apt-get install clang-format clang-tidy + sudo apt install lsb-release wget software-properties-common gnupg + wget https://apt.llvm.org/llvm.sh + sudo chmod +x llvm.sh + sudo ./llvm.sh ${LLVM_VERSION} + sudo apt-get install clang-format clang-format-${LLVM_VERSION} clang-tidy-${LLVM_VERSION} - run: flake8 - run: ./scripts/clang-format-diff.sh - name: clang-tidy @@ -338,3 +345,32 @@ jobs: uses: codecov/codecov-action@v3 with: gcov: true + + # Duplicates build-asan. Please keep in sync + build-cxx20: + name: c++20 + runs-on: ubuntu-latest + env: + CC: "gcc" + CXX: "g++" + steps: + - uses: actions/setup-python@v1 + with: + python-version: '3.x' + - uses: actions/checkout@v1 + with: + submodules: true + - name: install ninja + run: sudo apt-get install ninja-build + - name: install Python dev dependencies + run: pip3 install -r requirements-dev.txt + - name: cmake + run: | + mkdir -p out + cmake -S . -B out -G Ninja -DCMAKE_INSTALL_PREFIX=out/install -DCMAKE_C_COMPILER="$CC" -DCMAKE_CXX_COMPILER="$CXX" -DCMAKE_BUILD_TYPE=Release -DCMAKE_CXX_STANDARD=20 + - name: build + run: cmake --build out + - name: test + run: | + python check.py --binaryen-bin=out/bin lit + python check.py --binaryen-bin=out/bin gtest diff --git a/CMakeLists.txt b/CMakeLists.txt index ac49355af..2aefeef05 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -269,6 +269,7 @@ else() endif() add_compile_flag("-Wextra") add_compile_flag("-Wno-unused-parameter") + add_compile_flag("-Wno-dangling-pointer") # false positive in gcc add_compile_flag("-fno-omit-frame-pointer") add_compile_flag("-fno-rtti") # TODO(https://github.com/WebAssembly/binaryen/pull/2314): Remove these two diff --git a/scripts/clang-format-diff.sh b/scripts/clang-format-diff.sh index 6c7144f58..d0046e8cd 100755 --- a/scripts/clang-format-diff.sh +++ b/scripts/clang-format-diff.sh @@ -11,14 +11,17 @@ else BRANCH="@{upstream}" fi +LLVM_VERSION=${LLVM_VERSION:=17} + MERGE_BASE=$(git merge-base $BRANCH HEAD) -FORMAT_MSG=$(git clang-format $MERGE_BASE -q --diff) +FORMAT_ARGS="--binary=clang-format-${LLVM_VERSION} ${MERGE_BASE}" +FORMAT_MSG=$(git clang-format ${FORMAT_ARGS} -q --diff) if [ -n "$FORMAT_MSG" -a "$FORMAT_MSG" != "no modified files to format" ] then - echo "Please run git clang-format before committing, or apply this diff:" + echo "Please run git clang-format with clang-format-${LLVM_VERSION} before committing, or apply this diff:" echo # Run git clang-format again, this time without capruting stdout. This way # clang-format format the message nicely and add color. - git clang-format $MERGE_BASE -q --diff + git clang-format ${FORMAT_ARGS} -q --diff exit 1 fi diff --git a/src/analysis/lattice.h b/src/analysis/lattice.h index 1f2669c89..354447efc 100644 --- a/src/analysis/lattice.h +++ b/src/analysis/lattice.h @@ -23,35 +23,33 @@ inline LatticeComparison reverseComparison(LatticeComparison comparison) { } } -template<typename Lattice> -constexpr bool has_getBottom = - std::is_invocable_r<typename Lattice::Element, - decltype(&Lattice::getBottom), - Lattice>::value; -template<typename Lattice> -constexpr bool has_compare = - std::is_invocable_r<LatticeComparison, - decltype(&Lattice::compare), - Lattice, - const typename Lattice::Element&, - const typename Lattice::Element&>::value; -template<typename Element> -constexpr bool has_makeLeastUpperBound = - std::is_invocable_r<void, - decltype(&Element::makeLeastUpperBound), - Element, - const Element&>::value; -template<typename Element> -constexpr bool has_isTop = - std::is_invocable_r<bool, decltype(&Element::isTop), const Element>::value; -template<typename Element> -constexpr bool has_isBottom = - std::is_invocable_r<bool, decltype(&Element::isBottom), const Element>::value; - -template<typename Lattice> -constexpr bool is_lattice = has_getBottom<Lattice>&& has_compare<Lattice>&& - has_makeLeastUpperBound<typename Lattice::Element>&& has_isTop< - typename Lattice::Element>&& has_isBottom<typename Lattice::Element>; +#if __cplusplus >= 202002L + +#include <concepts> + +template<typename L> +concept Lattice = requires(const L& lattice, + const typename L::Element& constElem, + typename L::Element& elem) { + // Lattices must have elements. + typename L::Element; + // We need to be able to get the bottom element. + { lattice.getBottom() } noexcept -> std::same_as<typename L::Element>; + // Elements should be comparable. TODO: use <=> and std::three_way_comparable + // once we support C++20 everywhere. + { + lattice.compare(constElem, constElem) + } noexcept -> std::same_as<LatticeComparison>; + // We need to be able to get the least upper bound of two elements and know + // whether any change was made. + { elem.makeLeastUpperBound(constElem) } noexcept -> std::same_as<bool>; +}; + +#else + +#define Lattice typename + +#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 @@ -101,7 +99,7 @@ public: // 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); + bool makeLeastUpperBound(const Element& other) noexcept; // Prints out the bits in the bitvector for a lattice element. void print(std::ostream& os); @@ -111,10 +109,11 @@ public: // 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); + LatticeComparison compare(const Element& left, + const Element& right) const noexcept; // Returns an instance of the bottom lattice element. - Element getBottom(); + Element getBottom() const noexcept; }; // A layer of abstraction over FiniteIntPowersetLattice which maps @@ -169,11 +168,12 @@ public: } // We use implementations from FiniteIntPowersetLattice here. - LatticeComparison compare(const Element& left, const Element& right) { + LatticeComparison compare(const Element& left, + const Element& right) const noexcept { return intLattice.compare(left, right); } - Element getBottom() { return intLattice.getBottom(); } + Element getBottom() const noexcept { return intLattice.getBottom(); } }; } // namespace wasm::analysis diff --git a/src/analysis/monotone-analyzer-impl.h b/src/analysis/monotone-analyzer-impl.h index 8599bccf2..a87ba4282 100644 --- a/src/analysis/monotone-analyzer-impl.h +++ b/src/analysis/monotone-analyzer-impl.h @@ -10,14 +10,12 @@ namespace wasm::analysis { // All states are set to the bottom lattice element using the lattice in this // constructor. -template<typename Lattice> -inline BlockState<Lattice>::BlockState(const BasicBlock* underlyingBlock, - Lattice& lattice) +template<Lattice L> +inline BlockState<L>::BlockState(const BasicBlock* underlyingBlock, L& lattice) : cfgBlock(underlyingBlock), inputState(lattice.getBottom()) {} // Prints out inforamtion about a CFG node's state, but not intermediate states. -template<typename Lattice> -inline void BlockState<Lattice>::print(std::ostream& os) { +template<Lattice L> inline void BlockState<L>::print(std::ostream& os) { os << "CFG Block: " << cfgBlock->getIndex() << std::endl; os << "Input State: "; inputState.print(os); @@ -32,9 +30,9 @@ inline void BlockState<Lattice>::print(std::ostream& os) { os << std::endl; } -template<typename Lattice, typename TransferFunction> -inline MonotoneCFGAnalyzer<Lattice, TransferFunction>::MonotoneCFGAnalyzer( - Lattice& lattice, TransferFunction& transferFunction, CFG& cfg) +template<Lattice L, typename TransferFunction> +inline MonotoneCFGAnalyzer<L, TransferFunction>::MonotoneCFGAnalyzer( + L& lattice, TransferFunction& transferFunction, CFG& cfg) : lattice(lattice), transferFunction(transferFunction), cfg(cfg) { // Construct BlockStates for each BasicBlock. @@ -43,30 +41,28 @@ inline MonotoneCFGAnalyzer<Lattice, TransferFunction>::MonotoneCFGAnalyzer( } } -template<typename Lattice, typename TransferFunction> -inline void -MonotoneCFGAnalyzer<Lattice, TransferFunction>::evaluateFunctionEntry( +template<Lattice L, typename TransferFunction> +inline void MonotoneCFGAnalyzer<L, TransferFunction>::evaluateFunctionEntry( Function* func) { transferFunction.evaluateFunctionEntry(func, stateBlocks[0].inputState); } -template<typename Lattice, typename TransferFunction> -inline void MonotoneCFGAnalyzer<Lattice, TransferFunction>::evaluate() { +template<Lattice L, typename TransferFunction> +inline void MonotoneCFGAnalyzer<L, TransferFunction>::evaluate() { std::queue<const BasicBlock*> worklist; // Transfer function enqueues the work in some order which is efficient. transferFunction.enqueueWorklist(cfg, worklist); while (!worklist.empty()) { - BlockState<Lattice>& currBlockState = - stateBlocks[worklist.front()->getIndex()]; + BlockState<L>& currBlockState = stateBlocks[worklist.front()->getIndex()]; worklist.pop(); // For each expression, applies the transfer function, using the expression, // on the state of the expression it depends upon (here the next expression) // to arrive at the expression's state. The beginning and end states of the // CFG block will be updated. - typename Lattice::Element outputState = currBlockState.inputState; + typename L::Element outputState = currBlockState.inputState; transferFunction.transfer(currBlockState.cfgBlock, outputState); // Propagate state to dependents of currBlockState. @@ -81,10 +77,10 @@ inline void MonotoneCFGAnalyzer<Lattice, TransferFunction>::evaluate() { } } -template<typename Lattice, typename TransferFunction> -inline void MonotoneCFGAnalyzer<Lattice, TransferFunction>::collectResults() { +template<Lattice L, typename TransferFunction> +inline void MonotoneCFGAnalyzer<L, TransferFunction>::collectResults() { for (BlockState currBlockState : stateBlocks) { - typename Lattice::Element inputStateCopy = currBlockState.inputState; + typename L::Element inputStateCopy = currBlockState.inputState; // The transfer function generates the final set of states and uses it to // produce useful information. For example, in reaching definitions @@ -96,13 +92,12 @@ inline void MonotoneCFGAnalyzer<Lattice, TransferFunction>::collectResults() { // Currently prints both the basic information and intermediate states of each // BlockState. -template<typename Lattice, typename TransferFunction> -inline void -MonotoneCFGAnalyzer<Lattice, TransferFunction>::print(std::ostream& os) { +template<Lattice L, typename TransferFunction> +inline void MonotoneCFGAnalyzer<L, TransferFunction>::print(std::ostream& os) { os << "CFG Analyzer" << std::endl; for (auto state : stateBlocks) { state.print(os); - typename Lattice::Element temp = state.inputState; + typename L::Element temp = state.inputState; transferFunction.print(os, state.cfgBlock, temp); } os << "End" << std::endl; diff --git a/src/analysis/monotone-analyzer.h b/src/analysis/monotone-analyzer.h index 2b0e0ae16..480cf2185 100644 --- a/src/analysis/monotone-analyzer.h +++ b/src/analysis/monotone-analyzer.h @@ -11,17 +11,16 @@ namespace wasm::analysis { // A node which contains all the lattice states for a given CFG node. -template<typename Lattice> struct BlockState { - static_assert(is_lattice<Lattice>); +template<Lattice L> struct BlockState { // CFG node corresponding to this state block. const BasicBlock* cfgBlock; // State at which the analysis flow starts for a CFG. For instance, the ending // state for backward analysis, or the beginning state for forward analysis. - typename Lattice::Element inputState; + typename L::Element inputState; // All states are set to the bottom lattice element in this constructor. - BlockState(const BasicBlock* underlyingBlock, Lattice& lattice); + BlockState(const BasicBlock* underlyingBlock, L& lattice); // Prints out BlockState information, but not any intermediate states. void print(std::ostream& os); @@ -42,13 +41,13 @@ template<typename Lattice> struct BlockState { // propagated to dependents of the CFG BasicBlock by the worklist algorithm in // MonotoneCFGAnalyzer. -template<typename TransferFunction, typename Lattice> +template<typename TransferFunction, Lattice L> constexpr bool has_transfer = std::is_invocable_r<void, decltype(&TransferFunction::transfer), TransferFunction, const BasicBlock*, - typename Lattice::Element&>::value; + typename L::Element&>::value; // void enqueueWorklist(CFG&, std::queue<const BasicBlock*>& value); @@ -58,7 +57,7 @@ constexpr bool has_transfer = // state propagations, and therefore better performance. The opposite is true // for a forward analysis. -template<typename TransferFunction, typename Lattice> +template<typename TransferFunction, Lattice L> constexpr bool has_enqueueWorklist = std::is_invocable_r<void, decltype(&TransferFunction::enqueueWorklist), @@ -80,27 +79,23 @@ constexpr bool has_getDependents = const BasicBlock*>::value; // Combined TransferFunction assertions. -template<typename TransferFunction, typename Lattice> -constexpr bool is_TransferFunction = has_transfer<TransferFunction, Lattice>&& - has_enqueueWorklist<TransferFunction, Lattice>&& - has_getDependents<TransferFunction>; +template<typename TransferFunction, Lattice L> +constexpr bool is_TransferFunction = has_transfer<TransferFunction, L> && + has_enqueueWorklist<TransferFunction, L> && + has_getDependents<TransferFunction>; -template<typename Lattice, typename TransferFunction> -class MonotoneCFGAnalyzer { - static_assert(is_lattice<Lattice>); - static_assert(is_TransferFunction<TransferFunction, Lattice>); +template<Lattice L, typename TransferFunction> class MonotoneCFGAnalyzer { + static_assert(is_TransferFunction<TransferFunction, L>); - Lattice& lattice; + L& lattice; TransferFunction& transferFunction; CFG& cfg; - std::vector<BlockState<Lattice>> stateBlocks; + std::vector<BlockState<L>> stateBlocks; public: // Will constuct BlockState objects corresponding to BasicBlocks from the // given CFG. - MonotoneCFGAnalyzer(Lattice& lattice, - TransferFunction& transferFunction, - CFG& cfg); + MonotoneCFGAnalyzer(L& lattice, TransferFunction& transferFunction, CFG& cfg); // Runs the worklist algorithm to compute the states for the BlockState graph. void evaluate(); diff --git a/src/analysis/powerset-lattice-impl.h b/src/analysis/powerset-lattice-impl.h index 6d710826b..33ebd312f 100644 --- a/src/analysis/powerset-lattice-impl.h +++ b/src/analysis/powerset-lattice-impl.h @@ -7,7 +7,7 @@ namespace wasm::analysis { inline LatticeComparison FiniteIntPowersetLattice::compare( const FiniteIntPowersetLattice::Element& left, - const FiniteIntPowersetLattice::Element& right) { + const FiniteIntPowersetLattice::Element& right) const noexcept { // Both must be from the powerset lattice of the same set. assert(left.bitvector.size() == right.bitvector.size()); @@ -41,7 +41,8 @@ inline LatticeComparison FiniteIntPowersetLattice::compare( return NO_RELATION; } -inline FiniteIntPowersetLattice::Element FiniteIntPowersetLattice::getBottom() { +inline FiniteIntPowersetLattice::Element +FiniteIntPowersetLattice::getBottom() const noexcept { FiniteIntPowersetLattice::Element result(setSize); return result; } @@ -60,7 +61,7 @@ inline size_t FiniteIntPowersetLattice::Element::count() const { // 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 FiniteIntPowersetLattice::Element::makeLeastUpperBound( - const FiniteIntPowersetLattice::Element& other) { + const FiniteIntPowersetLattice::Element& other) noexcept { // Both must be from powerset lattice of the same set. assert(other.bitvector.size() == bitvector.size()); diff --git a/src/analysis/stack-lattice.h b/src/analysis/stack-lattice.h index 069dfa8ab..e27e01a2f 100644 --- a/src/analysis/stack-lattice.h +++ b/src/analysis/stack-lattice.h @@ -65,8 +65,7 @@ namespace wasm::analysis { // produce a stack [b, LUB(a, a')], where LUB(a, a') takes the maximum // of the two maximum bit values. -template<typename StackElementLattice> class StackLattice { - static_assert(is_lattice<StackElementLattice>); +template<Lattice StackElementLattice> class StackLattice { StackElementLattice& stackElementLattice; public: @@ -127,7 +126,7 @@ public: // fits with the conception of the stack starting at the top and having // an infinite bottom, which allows stacks of different height and scope // to be easily joined. - bool makeLeastUpperBound(const Element& other) { + bool makeLeastUpperBound(const Element& other) noexcept { // Top element cases, since top elements don't actually have the stack // value. if (isTop()) { @@ -185,7 +184,8 @@ public: // >= right top or if the left stack is higher and the right top > left top or // they are unrelated, then there is no relation. Same applies for the reverse // relationship. - LatticeComparison compare(const Element& left, const Element& right) { + LatticeComparison compare(const Element& left, + const Element& right) const noexcept { // Handle cases where there are top elements. if (left.isTop()) { if (right.isTop()) { @@ -246,7 +246,7 @@ public: } } - Element getBottom() { return Element{}; } + Element getBottom() const noexcept { return Element{}; } }; } // namespace wasm::analysis diff --git a/src/analysis/visitor-transfer-function.h b/src/analysis/visitor-transfer-function.h index 18ee35392..699840ff3 100644 --- a/src/analysis/visitor-transfer-function.h +++ b/src/analysis/visitor-transfer-function.h @@ -14,10 +14,10 @@ enum class AnalysisDirection { Forward, Backward }; // Utility for visitor-based transfer functions for forward and backward // analysis. Forward analysis is chosen by default unless the template parameter // Backward is true. -template<typename SubType, typename Lattice, AnalysisDirection Direction> +template<typename SubType, Lattice L, AnalysisDirection Direction> struct VisitorTransferFunc : public Visitor<SubType> { protected: - typename Lattice::Element* currState = nullptr; + typename L::Element* currState = nullptr; // There are two distinct phases in the execution of the analyzer. First, // the worklist algorithm will be run to obtain a solution, calling @@ -46,8 +46,7 @@ public: // Executes the transfer function on all the expressions of the corresponding // CFG node, starting with the node's input state, and changes the input state // to the final output state of the node in place. - void transfer(const BasicBlock* cfgBlock, - typename Lattice::Element& inputState) { + void transfer(const BasicBlock* cfgBlock, typename L::Element& inputState) { // If the block is empty, we propagate the state by inputState = // outputState. @@ -87,7 +86,7 @@ public: // This is for collecting results after solving an analysis. Implemented in // the same way as transfer(), but we also set the collectingResults flag. void collectResults(const BasicBlock* cfgBlock, - typename Lattice::Element& inputState) { + typename L::Element& inputState) { collectingResults = true; currState = &inputState; if constexpr (Direction == AnalysisDirection::Backward) { diff --git a/src/tools/wasm-fuzz-lattices.cpp b/src/tools/wasm-fuzz-lattices.cpp index 6bb79b106..730e6d621 100644 --- a/src/tools/wasm-fuzz-lattices.cpp +++ b/src/tools/wasm-fuzz-lattices.cpp @@ -44,15 +44,15 @@ uint64_t getSeed() { // Utility class which provides methods to check properties of the transfer // function and lattice of an analysis. -template<typename Lattice, typename TransferFunction> struct AnalysisChecker { - Lattice& lattice; +template<Lattice L, typename TransferFunction> struct AnalysisChecker { + L& lattice; TransferFunction& transferFunction; std::string latticeName; std::string transferFunctionName; uint64_t latticeElementSeed; Name funcName; - AnalysisChecker(Lattice& lattice, + AnalysisChecker(L& lattice, TransferFunction& transferFunction, std::string latticeName, std::string transferFunctionName, @@ -71,9 +71,9 @@ template<typename Lattice, typename TransferFunction> struct AnalysisChecker { // Prints information about a particular test case consisting of a randomly // generated function and triple of randomly generate lattice elements. void printVerboseFunctionCase(std::ostream& os, - typename Lattice::Element& x, - typename Lattice::Element& y, - typename Lattice::Element& z) { + typename L::Element& x, + typename L::Element& y, + typename L::Element& z) { os << "Using lattice element seed " << latticeElementSeed << "\nGenerated " << latticeName << " elements:\n"; x.print(os); @@ -86,7 +86,7 @@ template<typename Lattice, typename TransferFunction> struct AnalysisChecker { } // Checks reflexivity of a lattice element, i.e. x = x. - void checkReflexivity(typename Lattice::Element& element) { + void checkReflexivity(typename L::Element& element) { LatticeComparison result = lattice.compare(element, element); if (result != LatticeComparison::EQUAL) { std::stringstream ss; @@ -106,8 +106,7 @@ template<typename Lattice, typename TransferFunction> struct AnalysisChecker { // // Instead, we check for a related concept that x < y implies y > x, and // vice versa in this checkAntiSymmetry function. - void checkAntiSymmetry(typename Lattice::Element& x, - typename Lattice::Element& y) { + void checkAntiSymmetry(typename L::Element& x, typename L::Element& y) { LatticeComparison result = lattice.compare(x, y); LatticeComparison reverseResult = lattice.compare(y, x); @@ -127,9 +126,9 @@ private: // Prints the error message when a triple of lattice elements violates // transitivity. void printTransitivityError(std::ostream& os, - typename Lattice::Element& a, - typename Lattice::Element& b, - typename Lattice::Element& c, + typename L::Element& a, + typename L::Element& b, + typename L::Element& c, LatticeComparison ab, LatticeComparison bc, LatticeComparison ac) { @@ -163,9 +162,9 @@ private: public: // Given three lattice elements x, y, and z, checks if transitivity holds // between them. - void checkTransitivity(typename Lattice::Element& x, - typename Lattice::Element& y, - typename Lattice::Element& z) { + void checkTransitivity(typename L::Element& x, + typename L::Element& y, + typename L::Element& z) { LatticeComparison xy = lattice.compare(x, y); LatticeComparison yz = lattice.compare(y, z); LatticeComparison xz = lattice.compare(x, z); @@ -194,10 +193,10 @@ public: // the CFG block input which caused the transfer function to exhibit // non-monotonic behavior. void checkMonotonicity(const BasicBlock* cfgBlock, - typename Lattice::Element& first, - typename Lattice::Element& second, - typename Lattice::Element& firstResult, - typename Lattice::Element& secondResult) { + typename L::Element& first, + typename L::Element& second, + typename L::Element& firstResult, + typename L::Element& secondResult) { LatticeComparison beforeCmp = lattice.compare(first, second); LatticeComparison afterCmp = lattice.compare(firstResult, secondResult); @@ -241,9 +240,9 @@ public: } // Checks lattice-only properties for a triple of lattices. - void checkLatticeElements(typename Lattice::Element x, - typename Lattice::Element y, - typename Lattice::Element z) { + void checkLatticeElements(typename L::Element x, + typename L::Element y, + typename L::Element z) { checkReflexivity(x); checkReflexivity(y); checkReflexivity(z); @@ -258,16 +257,16 @@ public: // using the same three input states each time and then checking properties on // the inputs and outputs. void checkTransferFunction(CFG& cfg, - typename Lattice::Element x, - typename Lattice::Element y, - typename Lattice::Element z) { + typename L::Element x, + typename L::Element y, + typename L::Element z) { for (auto cfgIter = cfg.begin(); cfgIter != cfg.end(); ++cfgIter) { // Apply transfer function on each lattice element. - typename Lattice::Element xResult = x; + typename L::Element xResult = x; transferFunction.transfer(&(*cfgIter), xResult); - typename Lattice::Element yResult = y; + typename L::Element yResult = y; transferFunction.transfer(&(*cfgIter), yResult); - typename Lattice::Element zResult = z; + typename L::Element zResult = z; transferFunction.transfer(&(*cfgIter), zResult); // Check monotonicity for every pair of transfer function outputs. |