summaryrefslogtreecommitdiff
path: root/test/gtest
diff options
context:
space:
mode:
authorThomas Lively <tlively@google.com>2023-11-07 00:43:29 +0100
committerGitHub <noreply@github.com>2023-11-06 15:43:29 -0800
commitba2ebea5fbd4fef2c95e234198cb3edec68eb8f5 (patch)
treeb0f6612d7270b4b3f9bd0d36ab3b33b715ee3f90 /test/gtest
parente3d27166370da202bef6c012014604beac41d43f (diff)
downloadbinaryen-ba2ebea5fbd4fef2c95e234198cb3edec68eb8f5.tar.gz
binaryen-ba2ebea5fbd4fef2c95e234198cb3edec68eb8f5.tar.bz2
binaryen-ba2ebea5fbd4fef2c95e234198cb3edec68eb8f5.zip
Update CFGWalker to generate consolidated exit blocks (#6079)
Previously CFGWalker designated a particular block as the "exit" block, but it was just the block that happened to appear at the end of the function that returned values by implicitly flowing them out. That exit block was not tied in any way to other blocks that might end in returns, so analyses that needed to perform some action at the end of the function would have had to perform that action at the end of the designated exit block but also separately at any return instruction. Update CFGWalker to make the exit block a synthetic empty block that is a successor of all other blocks tthat implicitly or explicitly return from the function in case there are multiple such blocks, or to make the exit block the single returning block if there is only one. This means that analyses will only perform their end-of-function actions at the end of the exit block rather than additionally at every return instruction.
Diffstat (limited to 'test/gtest')
-rw-r--r--test/gtest/cfg.cpp59
1 files changed, 57 insertions, 2 deletions
diff --git a/test/gtest/cfg.cpp b/test/gtest/cfg.cpp
index 583c8c4f0..bf3742bc1 100644
--- a/test/gtest/cfg.cpp
+++ b/test/gtest/cfg.cpp
@@ -44,6 +44,7 @@ TEST_F(CFGTest, Print) {
)wasm";
auto cfgText = R"cfg(;; preds: [], succs: [1, 5]
+;; entry
0:
0: i32.const 0
1: drop
@@ -66,15 +67,19 @@ TEST_F(CFGTest, Print) {
6: i32.const 3
7: block
-;; preds: [0], succs: []
+;; preds: [0], succs: [7]
5:
8: i32.const 4
9: return
-;; preds: [4], succs: []
+;; preds: [4], succs: [7]
6:
10: drop
11: block
+
+;; preds: [5, 6], succs: []
+;; exit
+7:
)cfg";
Module wasm;
@@ -110,12 +115,14 @@ TEST_F(CFGTest, CallBlock) {
)wasm";
auto cfgText = R"cfg(;; preds: [], succs: [1]
+;; entry
0:
0: i32.const 0
1: drop
2: call $bar
;; preds: [0], succs: []
+;; exit
1:
3: i32.const 1
4: drop
@@ -133,6 +140,54 @@ TEST_F(CFGTest, CallBlock) {
EXPECT_EQ(ss.str(), cfgText);
}
+TEST_F(CFGTest, Empty) {
+ // Check that we create a correct CFG for an empty function.
+ auto moduleText = R"wasm(
+ (module (func $foo))
+ )wasm";
+
+ auto cfgText = R"cfg(;; preds: [], succs: []
+;; entry
+;; exit
+0:
+ 0: nop
+)cfg";
+
+ Module wasm;
+ parseWast(wasm, moduleText);
+
+ CFG cfg = CFG::fromFunction(wasm.getFunction("foo"));
+
+ std::stringstream ss;
+ cfg.print(ss);
+
+ EXPECT_EQ(ss.str(), cfgText);
+}
+
+TEST_F(CFGTest, Unreachable) {
+ // Check that we create a correct CFG for a function that does not return. In
+ // particular, it should not have an exit block.
+ auto moduleText = R"wasm(
+ (module (func $foo (unreachable)))
+ )wasm";
+
+ auto cfgText = R"cfg(;; preds: [], succs: []
+;; entry
+0:
+ 0: unreachable
+)cfg";
+
+ Module wasm;
+ parseWast(wasm, moduleText);
+
+ CFG cfg = CFG::fromFunction(wasm.getFunction("foo"));
+
+ std::stringstream ss;
+ cfg.print(ss);
+
+ EXPECT_EQ(ss.str(), cfgText);
+}
+
TEST_F(CFGTest, FinitePowersetLatticeFunctioning) {
std::vector<std::string> initialSet = {"a", "b", "c", "d", "e", "f"};