diff options
author | Thomas Lively <tlively@google.com> | 2023-11-07 00:43:29 +0100 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-11-06 15:43:29 -0800 |
commit | ba2ebea5fbd4fef2c95e234198cb3edec68eb8f5 (patch) | |
tree | b0f6612d7270b4b3f9bd0d36ab3b33b715ee3f90 /test/gtest | |
parent | e3d27166370da202bef6c012014604beac41d43f (diff) | |
download | binaryen-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.cpp | 59 |
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"}; |