From c6109def1d288baefed7d0c301a0efb113314e94 Mon Sep 17 00:00:00 2001 From: Simon Krajewski Date: Wed, 22 Nov 2023 07:07:25 +0100 Subject: [PATCH] [analyzer] deal with unreachable block in binops closes #11402 --- src/optimization/analyzerTexprTransformer.ml | 29 +++++--- tests/unit/src/unit/issues/Issue11402.hx | 78 ++++++++++++++++++++ 2 files changed, 97 insertions(+), 10 deletions(-) create mode 100644 tests/unit/src/unit/issues/Issue11402.hx diff --git a/src/optimization/analyzerTexprTransformer.ml b/src/optimization/analyzerTexprTransformer.ml index 5be0fafca2f..888ded14df5 100644 --- a/src/optimization/analyzerTexprTransformer.ml +++ b/src/optimization/analyzerTexprTransformer.ml @@ -118,11 +118,15 @@ let rec func ctx bb tf t p = let bb,e2 = value bb e2 in bb,{e with eexpr = TBinop(op,e1,e2)} | TBinop(op,e1,e2) -> - let bb,e1,e2 = match ordered_value_list bb [e1;e2] with - | bb,[e1;e2] -> bb,e1,e2 - | _ -> die "" __LOC__ - in - bb,{e with eexpr = TBinop(op,e1,e2)} + begin match ordered_value_list bb [e1;e2] with + | bb,[e1;e2] -> + bb,{e with eexpr = TBinop(op,e1,e2)} + | bb,[e] -> + assert(bb == g.g_unreachable); + bb,e + | _ -> + die "" __LOC__ + end | TUnop(op,flag,e1) -> let bb,e1 = value bb e1 in bb,{e with eexpr = TUnop(op,flag,e1)} @@ -139,11 +143,16 @@ let rec func ctx bb tf t p = let bb,e1 = value bb e1 in bb,{e with eexpr = TField(e1,fa)} | TArray(e1,e2) -> - let bb,e1,e2 = match ordered_value_list bb [e1;e2] with - | bb,[e1;e2] -> bb,e1,e2 - | _ -> die "" __LOC__ - in - bb,{e with eexpr = TArray(e1,e2)} + begin match ordered_value_list bb [e1;e2] with + | bb,[e1;e2] -> + bb,{e with eexpr = TArray(e1,e2)} + + | bb,[e] -> + assert(bb == g.g_unreachable); + bb,e + | _ -> + die "" __LOC__ + end | TMeta(m,e1) -> let bb,e1 = value bb e1 in bb,{e with eexpr = TMeta(m,e1)} diff --git a/tests/unit/src/unit/issues/Issue11402.hx b/tests/unit/src/unit/issues/Issue11402.hx new file mode 100644 index 00000000000..dd1b3e9e495 --- /dev/null +++ b/tests/unit/src/unit/issues/Issue11402.hx @@ -0,0 +1,78 @@ +package unit.issues; + +class Issue11402 extends Test { + static function binopWithReturn() { + var ans = Std.string(return "") + ""; + } + + static function binopWithThrow() { + var ans = Std.string(throw "") + ""; + } + + static function binopWithBreak() { + while (true) { + var ans = Std.string(break) + ""; + return "b"; + } + return "a"; + } + + static function binopWithContinue() { + var a = 0; + while (true) { + if (a++ > 0) { + return "a"; + } + var ans = Std.string(continue) + ""; + return "b"; + } + } + + static function arrayWithReturn() { + var ans = [Std.string(return "")][0]; + } + + static function arrayWithThrow() { + var ans = [Std.string(throw "")][0]; + } + + static function arrayWithBreak() { + while (true) { + var ans = [Std.string(break)][0]; + return "b"; + } + return "a"; + } + + static function arrayWithContinue() { + var a = 0; + while (true) { + if (a++ > 0) { + return "a"; + } + var ans = [Std.string(continue)][0]; + return "b"; + } + } + + static function originalExample() { + var cells = [1, 2, 3]; + var ans = ""; + ans += '${cells.length} cell${if (cells.length == 1) {return "";} else {return "s";}}.\n'; + return ans; + } + + function test() { + eq("", binopWithReturn()); + exc(binopWithThrow); + eq("a", binopWithBreak()); + eq("a", binopWithContinue()); + + eq("", arrayWithReturn()); + exc(arrayWithThrow); + eq("a", arrayWithBreak()); + eq("a", arrayWithContinue()); + + eq("s", originalExample()); + } +}