summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/wasm/wasm-validator.cpp50
-rw-r--r--test/gtest/CMakeLists.txt1
-rw-r--r--test/gtest/validator.cpp50
3 files changed, 75 insertions, 26 deletions
diff --git a/src/wasm/wasm-validator.cpp b/src/wasm/wasm-validator.cpp
index fbb57e9b0..179235fbc 100644
--- a/src/wasm/wasm-validator.cpp
+++ b/src/wasm/wasm-validator.cpp
@@ -2372,38 +2372,36 @@ void FunctionValidator::visitTry(Try* curr) {
auto* tag = getModule()->getTagOrNull(tagName);
if (!shouldBeTrue(tag != nullptr, curr, "")) {
getStream() << "tag name is invalid: " << tagName << "\n";
- }
-
- if (!shouldBeEqual(tag->sig.results, Type(Type::none), curr, "")) {
+ } else if (!shouldBeEqual(tag->sig.results, Type(Type::none), curr, "")) {
getStream()
<< "catch's tag (" << tagName
<< ") has result values, which is not allowed for exception handling";
- }
-
- auto* catchBody = curr->catchBodies[i];
- auto pops = EHUtils::findPops(catchBody);
- if (tag->sig.params == Type::none) {
- if (!shouldBeTrue(pops.empty(), curr, "")) {
- getStream() << "catch's tag (" << tagName
- << ") doesn't have any params, but there are pops";
- }
} else {
- if (shouldBeTrue(pops.size() == 1, curr, "")) {
- auto* pop = *pops.begin();
- if (!shouldBeSubType(tag->sig.params, pop->type, curr, "")) {
- getStream()
- << "catch's tag (" << tagName
- << ")'s pop doesn't have the same type as the tag's params";
- }
- if (!shouldBeTrue(
- EHUtils::containsValidDanglingPop(catchBody), curr, "")) {
- getStream() << "catch's body (" << tagName
- << ")'s pop's location is not valid";
+ auto* catchBody = curr->catchBodies[i];
+ auto pops = EHUtils::findPops(catchBody);
+ if (tag->sig.params == Type::none) {
+ if (!shouldBeTrue(pops.empty(), curr, "")) {
+ getStream() << "catch's tag (" << tagName
+ << ") doesn't have any params, but there are pops";
}
} else {
- getStream() << "catch's tag (" << tagName
- << ") has params, so there should be a single pop within "
- "the catch body";
+ if (shouldBeTrue(pops.size() == 1, curr, "")) {
+ auto* pop = *pops.begin();
+ if (!shouldBeSubType(tag->sig.params, pop->type, curr, "")) {
+ getStream()
+ << "catch's tag (" << tagName
+ << ")'s pop doesn't have the same type as the tag's params";
+ }
+ if (!shouldBeTrue(
+ EHUtils::containsValidDanglingPop(catchBody), curr, "")) {
+ getStream() << "catch's body (" << tagName
+ << ")'s pop's location is not valid";
+ }
+ } else {
+ getStream() << "catch's tag (" << tagName
+ << ") has params, so there should be a single pop within "
+ "the catch body";
+ }
}
}
}
diff --git a/test/gtest/CMakeLists.txt b/test/gtest/CMakeLists.txt
index f7824834a..d50d33ac1 100644
--- a/test/gtest/CMakeLists.txt
+++ b/test/gtest/CMakeLists.txt
@@ -10,6 +10,7 @@ set(unittest_SOURCES
suffix_tree.cpp
type-builder.cpp
wat-lexer.cpp
+ validator.cpp
)
# suffix_tree.cpp includes LLVM header using std::iterator (deprecated in C++17)
diff --git a/test/gtest/validator.cpp b/test/gtest/validator.cpp
new file mode 100644
index 000000000..87d698e56
--- /dev/null
+++ b/test/gtest/validator.cpp
@@ -0,0 +1,50 @@
+/*
+ * Copyright 2023 WebAssembly Community Group participants
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#include <set>
+#include <string>
+#include <vector>
+
+#include "shared-constants.h"
+#include "support/string.h"
+#include "wasm-binary.h"
+#include "wasm-builder.h"
+#include "wasm-validator.h"
+#include "gtest/gtest.h"
+
+using namespace wasm;
+
+// Check that the validator does not crash on missing tags in catch statements
+// (invalid parse, so cannot be reproduced with Python test suite)
+TEST(ValidatorTest, MissingCatchTag) {
+ WasmValidator validator;
+ Module module;
+ module.features.enable(FeatureSet::ExceptionHandling);
+ auto tryExp = module.allocator.alloc<Try>();
+ tryExp->name = "tst1";
+ auto bodyExp = module.allocator.alloc<Const>();
+ bodyExp->value = Literal(1);
+ tryExp->body = bodyExp;
+ tryExp->catchTags.push_back(wasm::Name("foo"));
+ auto catchBodyExp = module.allocator.alloc<Const>();
+ catchBodyExp->value = Literal(2);
+ tryExp->catchBodies.push_back(catchBodyExp);
+ Function function = Function();
+ function.body = tryExp;
+ WasmValidator::Flags flags =
+ WasmValidator::FlagValues::Globally | WasmValidator::FlagValues::Quiet;
+ EXPECT_FALSE(validator.validate(&function, module, flags));
+}