This PR fixes `inferControlInfoSeq` and `ControlInfo.sequence` to keep aggregating `breaks`/`continues`/`returnsEarly`/`reassigns` past elements whose `ControlInfo` reports `numRegularExits := 0`. Previously the analysis short-circuited at such elements, so any trailing `return`/`break`/`continue` was missing from the inferred info. The elaboration framework only skips subsequent doElems syntactically for top-level `return`/`break`/`continue`; for every other `numRegularExits == 0` case (e.g. a `match`/`if`/`try` whose branches all terminate, or a `repeat` without `break`) the elaborator keeps visiting the continuation and the for/match elaborator then tripped its invariant check with `Early returning ... but the info said there is no early return`. With this change the inferred info matches what the elaborator actually sees, which also removes the need for the `numRegularExits := 1` workaround on `repeat` introduced in #13479.
67 lines
1.9 KiB
Text
67 lines
1.9 KiB
Text
set_option backward.do.legacy false
|
|
|
|
-- When an inner doElem's `ControlInfo` has `numRegularExits := 0` (because all branches
|
|
-- `break`/`continue`/`return`, or it is a non-terminating `repeat`), the trailing `return`
|
|
-- inside the enclosing `for` body used to be dropped during inference, and the for elaborator
|
|
-- then threw "Early returning ... but the info said there is no early return" even though the
|
|
-- elaborator does visit the trailing element. `ofSeq`/`ControlInfo.sequence` now aggregate
|
|
-- `breaks`/`continues`/`returnsEarly`/`reassigns` past `numRegularExits == 0` elements, so the
|
|
-- inferred info matches what the elaborator actually sees.
|
|
|
|
/--
|
|
warning: This `do` element and its control-flow region are dead code. Consider refactoring your code to remove it.
|
|
-/
|
|
#guard_msgs in
|
|
example (cond : Bool) : IO Nat := do
|
|
for _ in [1, 2] do
|
|
if cond then break else break
|
|
return 42
|
|
return 1
|
|
|
|
/--
|
|
warning: This `do` element and its control-flow region are dead code. Consider refactoring your code to remove it.
|
|
-/
|
|
#guard_msgs in
|
|
example (cond : Bool) : IO Nat := do
|
|
for _ in [1, 2] do
|
|
if cond then continue else continue
|
|
return 42
|
|
return 1
|
|
|
|
#guard_msgs in
|
|
example (i : Nat) : IO Nat := do
|
|
for _ in [1, 2] do
|
|
match i with
|
|
| 0 => break
|
|
| _ => break
|
|
return 42
|
|
return 1
|
|
|
|
/--
|
|
warning: This `do` element and its control-flow region are dead code. Consider refactoring your code to remove it.
|
|
-/
|
|
#guard_msgs in
|
|
example : IO Nat := do
|
|
for _ in [1, 2] do
|
|
try break catch _ => break
|
|
return 42
|
|
return 1
|
|
|
|
/--
|
|
warning: This `do` element and its control-flow region are dead code. Consider refactoring your code to remove it.
|
|
-/
|
|
#guard_msgs in
|
|
example (cond : Bool) : IO Nat := do
|
|
for _ in [1, 2] do
|
|
unless cond do break
|
|
if cond then break else break
|
|
return 42
|
|
return 1
|
|
|
|
#guard_msgs in
|
|
example : IO Nat := do
|
|
for _ in [1, 2] do
|
|
repeat
|
|
pure ()
|
|
return 42
|
|
return 1
|