From 864b0d77791e861bf1c8ecc75a99e8c9ddba178b Mon Sep 17 00:00:00 2001 From: Leonardo de Moura Date: Sun, 25 Aug 2024 17:10:17 -0700 Subject: [PATCH] fix: PANIC at `Lean.MVarId.falseOrByContra` closes #4985 closes #4984 --- src/Lean/Elab/Tactic/FalseOrByContra.lean | 23 +++++++++++++---------- src/Lean/Elab/Tactic/Omega/Frontend.lean | 2 +- tests/lean/run/4985.lean | 3 +++ 3 files changed, 17 insertions(+), 11 deletions(-) create mode 100644 tests/lean/run/4985.lean diff --git a/src/Lean/Elab/Tactic/FalseOrByContra.lean b/src/Lean/Elab/Tactic/FalseOrByContra.lean index 7e03d23244ea..2b903d99c8bd 100644 --- a/src/Lean/Elab/Tactic/FalseOrByContra.lean +++ b/src/Lean/Elab/Tactic/FalseOrByContra.lean @@ -31,16 +31,16 @@ open Lean Meta Elab Tactic -- but fall back to a classical instance. When it is `some true`, we always use the classical instance. -- When it is `some false`, if there is no `Decidable` instance we don't introduce the double negation, -- and fall back to `False.elim`. -partial def falseOrByContra (g : MVarId) (useClassical : Option Bool := none) : MetaM MVarId := do +partial def falseOrByContra (g : MVarId) (useClassical : Option Bool := none) : MetaM (Option MVarId) := do let ty ← whnfR (← g.getType) match ty with - | .const ``False _ => pure g - | .forallE _ _ _ _ + | .const ``False _ => return g + | .forallE .. | .app (.const ``Not _) _ => -- We set the transparency back to default; otherwise this breaks when run by a `simp` discharger. falseOrByContra (← withTransparency default g.intro1P).2 useClassical | _ => - let gs ← if ← isProp ty then + let gs ← if (← isProp ty) then match useClassical with | some true => some <$> g.applyConst ``Classical.byContradiction | some false => @@ -51,12 +51,15 @@ partial def falseOrByContra (g : MVarId) (useClassical : Option Bool := none) : catch _ => some <$> g.applyConst ``Classical.byContradiction else pure none - if let some gs := gs then - let [g] := gs | panic! "expected one subgoal" - pure (← g.intro1).2 - else - let [g] ← g.applyConst ``False.elim | panic! "expected one sugoal" - pure g + match gs with + | some [] => return none + | some [g] => return some (← g.intro1).2 + | some _ => panic! "expected at most one sugoal" + | none => + match (← g.applyConst ``False.elim) with + | [] => return none + | [g] => return some g + | _ => panic! "expected at most one sugoal" @[builtin_tactic Lean.Parser.Tactic.falseOrByContra] def elabFalseOrByContra : Tactic diff --git a/src/Lean/Elab/Tactic/Omega/Frontend.lean b/src/Lean/Elab/Tactic/Omega/Frontend.lean index 9c20975a69b1..a8b60edfed0f 100644 --- a/src/Lean/Elab/Tactic/Omega/Frontend.lean +++ b/src/Lean/Elab/Tactic/Omega/Frontend.lean @@ -682,7 +682,7 @@ open Lean Elab Tactic Parser.Tactic /-- The `omega` tactic, for resolving integer and natural linear arithmetic problems. -/ def omegaTactic (cfg : OmegaConfig) : TacticM Unit := do liftMetaFinishingTactic fun g => do - let g ← g.falseOrByContra + let some g ← g.falseOrByContra | return () g.withContext do let hyps := (← getLocalHyps).toList trace[omega] "analyzing {hyps.length} hypotheses:\n{← hyps.mapM inferType}" diff --git a/tests/lean/run/4985.lean b/tests/lean/run/4985.lean new file mode 100644 index 000000000000..b33128803353 --- /dev/null +++ b/tests/lean/run/4985.lean @@ -0,0 +1,3 @@ +example : id (False → Nat) := by false_or_by_contra + +example : id ((¬True → False) → True) := by false_or_by_contra