summaryrefslogtreecommitdiff
path: root/src/tools
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2024-10-07 16:57:56 -0700
committerGitHub <noreply@github.com>2024-10-07 16:57:56 -0700
commit1eb01260efdcb65828c81cf5f72fb358b03d2328 (patch)
tree564262cd79a2499f5782744aadca7bced9905644 /src/tools
parenta1b88267bb977cb5fdba614b5f61fa7c84f51bf6 (diff)
downloadbinaryen-1eb01260efdcb65828c81cf5f72fb358b03d2328.tar.gz
binaryen-1eb01260efdcb65828c81cf5f72fb358b03d2328.tar.bz2
binaryen-1eb01260efdcb65828c81cf5f72fb358b03d2328.zip
Fuzzer: Generate TryTables (#6987)
Also make Try/TryTables with type none, and not just concrete types as before.
Diffstat (limited to 'src/tools')
-rw-r--r--src/tools/fuzzing.h1
-rw-r--r--src/tools/fuzzing/fuzzing.cpp68
2 files changed, 69 insertions, 0 deletions
diff --git a/src/tools/fuzzing.h b/src/tools/fuzzing.h
index 75e3a2a9a..f4400e39e 100644
--- a/src/tools/fuzzing.h
+++ b/src/tools/fuzzing.h
@@ -293,6 +293,7 @@ private:
Expression* buildIf(const struct ThreeArgs& args, Type type);
Expression* makeIf(Type type);
Expression* makeTry(Type type);
+ Expression* makeTryTable(Type type);
Expression* makeBreak(Type type);
Expression* makeCall(Type type);
Expression* makeCallIndirect(Type type);
diff --git a/src/tools/fuzzing/fuzzing.cpp b/src/tools/fuzzing/fuzzing.cpp
index a8b8f7855..abc6a63d5 100644
--- a/src/tools/fuzzing/fuzzing.cpp
+++ b/src/tools/fuzzing/fuzzing.cpp
@@ -1366,6 +1366,7 @@ Expression* TranslateToFuzzReader::_makeConcrete(Type type) {
&Self::makeCall,
&Self::makeCallIndirect)
.add(FeatureSet::ExceptionHandling, &Self::makeTry)
+ .add(FeatureSet::ExceptionHandling, &Self::makeTryTable)
.add(FeatureSet::GC | FeatureSet::ReferenceTypes, &Self::makeCallRef);
}
if (type.isSingle()) {
@@ -1451,6 +1452,8 @@ Expression* TranslateToFuzzReader::_makenone() {
&Self::makeGlobalSet)
.add(FeatureSet::BulkMemory, &Self::makeBulkMemory)
.add(FeatureSet::Atomics, &Self::makeAtomic)
+ .add(FeatureSet::ExceptionHandling, &Self::makeTry)
+ .add(FeatureSet::ExceptionHandling, &Self::makeTryTable)
.add(FeatureSet::GC | FeatureSet::ReferenceTypes, &Self::makeCallRef)
.add(FeatureSet::GC | FeatureSet::ReferenceTypes, &Self::makeStructSet)
.add(FeatureSet::GC | FeatureSet::ReferenceTypes, &Self::makeArraySet)
@@ -1688,6 +1691,71 @@ Expression* TranslateToFuzzReader::makeTry(Type type) {
return builder.makeTry(body, catchTags, catchBodies);
}
+Expression* TranslateToFuzzReader::makeTryTable(Type type) {
+ auto* body = make(type);
+
+ if (funcContext->breakableStack.empty()) {
+ // Nothing to break to, emit a trivial TryTable.
+ // TODO: Perhaps generate a block wrapping us?
+ return builder.makeTryTable(body, {}, {}, {});
+ }
+
+ if (wasm.tags.empty()) {
+ addTag();
+ }
+
+ // Add catches of specific tags, and possibly a catch_all at the end. We use
+ // the last iteration of the loop for that.
+ std::vector<Name> catchTags;
+ std::vector<Name> catchDests;
+ std::vector<bool> catchRefs;
+ auto numCatches = upTo(MAX_TRY_CATCHES);
+ for (Index i = 0; i <= numCatches; i++) {
+ Name tagName;
+ Type tagType;
+ if (i < numCatches) {
+ // Look for a specific tag.
+ auto& tag = pick(wasm.tags);
+ tagName = tag->name;
+ tagType = tag->sig.params;
+ } else {
+ // Add a catch_all at the end, some of the time (but all of the time if we
+ // have nothing else).
+ if (!catchTags.empty() && oneIn(2)) {
+ break;
+ }
+ tagType = Type::none;
+ }
+
+ // We need to find a proper target to break to, which means a target that
+ // has the type of the tag, or the tag + an exnref at the end.
+ std::vector<Type> vec(tagType.begin(), tagType.end());
+ // Use a non-nullable exnref here, and then the subtyping check below will
+ // also accept a target that is nullable.
+ vec.push_back(Type(HeapType::exn, NonNullable));
+ auto tagTypeWithExn = Type(vec);
+ int tries = TRIES;
+ while (tries-- > 0) {
+ auto* target = pick(funcContext->breakableStack);
+ auto dest = getTargetName(target);
+ auto valueType = getTargetType(target);
+ auto subOfTagType = Type::isSubType(tagType, valueType);
+ auto subOfTagTypeWithExn = Type::isSubType(tagTypeWithExn, valueType);
+ if (subOfTagType || subOfTagTypeWithExn) {
+ catchTags.push_back(tagName);
+ catchDests.push_back(dest);
+ catchRefs.push_back(subOfTagTypeWithExn);
+ break;
+ }
+ }
+ // TODO: Perhaps generate a block wrapping us, if we fail to find a target?
+ // TODO: It takes a bit of luck to find a target with an exnref - perhaps
+ // generate those?
+ }
+
+ return builder.makeTryTable(body, catchTags, catchDests, catchRefs);
+}
+
Expression* TranslateToFuzzReader::makeBreak(Type type) {
if (funcContext->breakableStack.empty()) {
return makeTrivial(type);