summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--.github/workflows/ci.yml38
-rw-r--r--CMakeLists.txt1
-rwxr-xr-xscripts/clang-format-diff.sh9
-rw-r--r--src/analysis/lattice.h68
-rw-r--r--src/analysis/monotone-analyzer-impl.h41
-rw-r--r--src/analysis/monotone-analyzer.h35
-rw-r--r--src/analysis/powerset-lattice-impl.h7
-rw-r--r--src/analysis/stack-lattice.h10
-rw-r--r--src/analysis/visitor-transfer-function.h9
-rw-r--r--src/tools/wasm-fuzz-lattices.cpp55
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.