summaryrefslogtreecommitdiff
path: root/src/tools/wasm-fuzz-lattices.cpp
diff options
context:
space:
mode:
authorBruce He <44327446+zm2he@users.noreply.github.com>2023-08-03 17:45:27 +0000
committerGitHub <noreply@github.com>2023-08-03 13:45:27 -0400
commitbcdfa72afe41746ac13d27e6d02866c7d11e7fb5 (patch)
tree0df76ff435ce39e291bd93914eb9afdaccba6a38 /src/tools/wasm-fuzz-lattices.cpp
parent8bd418cc44705f1c272bbbe1ad65186dcd5b0fa9 (diff)
downloadbinaryen-bcdfa72afe41746ac13d27e6d02866c7d11e7fb5.tar.gz
binaryen-bcdfa72afe41746ac13d27e6d02866c7d11e7fb5.tar.bz2
binaryen-bcdfa72afe41746ac13d27e6d02866c7d11e7fb5.zip
Lattice to model Stack (#5849)
This change introduces StackLattice, a lattice to model stack-related behavior. It is templated on a separate lattice whose elements model some property of individual values on the stack. The StackLattice allows users to access the top of the stack, push abstract values, and pop them. Comparisons and least upper bound operations are done on a value by value basis starting from the top of the stack and moving toward the bottom. This is because it allows stacks from different scopes to be joined easily. An application of StackLattice is to model the wasm value stack. The goal is to organize lattice elements representing individual stack values in a natural way which mirrors the wasm value stack. Transfer functions operate on each stack value individually. The stack lattice is an intermediate structure which is not intended to be directly operated on. Rather, it simulates the push and pop behavior of instructions.
Diffstat (limited to 'src/tools/wasm-fuzz-lattices.cpp')
-rw-r--r--src/tools/wasm-fuzz-lattices.cpp70
1 files changed, 68 insertions, 2 deletions
diff --git a/src/tools/wasm-fuzz-lattices.cpp b/src/tools/wasm-fuzz-lattices.cpp
index c31d64290..6bb79b106 100644
--- a/src/tools/wasm-fuzz-lattices.cpp
+++ b/src/tools/wasm-fuzz-lattices.cpp
@@ -21,6 +21,7 @@
#include "analysis/lattice.h"
#include "analysis/liveness-transfer-function.h"
#include "analysis/reaching-definitions-transfer-function.h"
+#include "analysis/stack-lattice.h"
#include "support/command-line.h"
#include "tools/fuzzing.h"
@@ -363,6 +364,65 @@ struct ReachingDefinitionsChecker {
}
};
+// Struct to set up and check the stack lattice. Uses the
+// FiniteIntPowersetLattice as to model stack contents.
+struct StackLatticeChecker {
+ FiniteIntPowersetLattice contentLattice;
+ StackLattice<FiniteIntPowersetLattice> stackLattice;
+
+ // Here only as a placeholder, since the checker utility currently wants a
+ // transfer function.
+ LivenessTransferFunction transferFunction;
+
+ AnalysisChecker<StackLattice<FiniteIntPowersetLattice>,
+ LivenessTransferFunction>
+ checker;
+
+ StackLatticeChecker(Function* func,
+ uint64_t latticeElementSeed,
+ Name funcName)
+ : contentLattice(func->getNumLocals()), stackLattice(contentLattice),
+ checker(stackLattice,
+ transferFunction,
+ "StackLattice<FiniteIntPowersetLattice>",
+ "LivenessTransferFunction",
+ latticeElementSeed,
+ funcName) {}
+
+ StackLattice<FiniteIntPowersetLattice>::Element
+ getRandomElement(Random& rand) {
+ StackLattice<FiniteIntPowersetLattice>::Element result =
+ stackLattice.getBottom();
+
+ // Randomly decide on a stack height. A max height of 15 stack elements is
+ // arbitrarily chosen.
+ size_t stackHeight = rand.upTo(15);
+
+ for (size_t j = 0; j < stackHeight; ++j) {
+ // Randomly generate a FiniteIntPowersetLattice and push it onto the
+ // stack.
+ FiniteIntPowersetLattice::Element content = contentLattice.getBottom();
+ for (size_t i = 0; i < contentLattice.getSetSize(); ++i) {
+ content.set(i, rand.oneIn(2));
+ }
+ result.push(std::move(content));
+ }
+ return result;
+ }
+
+ // Runs all checks for the StackLattice analysis.
+ void runChecks(Random& rand, bool verbose) {
+ StackLattice<FiniteIntPowersetLattice>::Element x = getRandomElement(rand);
+ StackLattice<FiniteIntPowersetLattice>::Element y = getRandomElement(rand);
+ StackLattice<FiniteIntPowersetLattice>::Element z = getRandomElement(rand);
+ if (verbose) {
+ checker.printVerboseFunctionCase(std::cout, x, y, z);
+ }
+
+ checker.checkLatticeElements(x, y, z);
+ }
+};
+
struct Fuzzer {
bool verbose;
@@ -384,16 +444,22 @@ struct Fuzzer {
CFG cfg = CFG::fromFunction(func);
- switch (rand.upTo(2)) {
+ switch (rand.upTo(3)) {
case 0: {
LivenessChecker livenessChecker(func, latticeElementSeed, func->name);
livenessChecker.runChecks(cfg, rand, verbose);
break;
}
- default: {
+ case 1: {
ReachingDefinitionsChecker reachingDefinitionsChecker(
func, latticeElementSeed, func->name);
reachingDefinitionsChecker.runChecks(cfg, rand, verbose);
+ break;
+ }
+ default: {
+ StackLatticeChecker stackLatticeChecker(
+ func, latticeElementSeed, func->name);
+ stackLatticeChecker.runChecks(rand, verbose);
}
}
}