Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

(kernel) constant has already been declared during simp #6067

Closed
3 tasks done
TwoFX opened this issue Nov 13, 2024 · 2 comments · Fixed by #6180
Closed
3 tasks done

(kernel) constant has already been declared during simp #6067

TwoFX opened this issue Nov 13, 2024 · 2 comments · Fixed by #6180
Labels
bug Something isn't working P-medium We may work on this issue if we find the time

Comments

@TwoFX
Copy link
Member

TwoFX commented Nov 13, 2024

Prerequisites

Please put an X between the brackets as you perform the following steps:

Description

inductive Impl where
  | inner (l r : Impl)
  | leaf

namespace Impl

inductive Balanced : Impl → Prop where
  | leaf : Balanced leaf

@[inline]
def balanceLErase (r : Impl) (hrb : Balanced r) : Impl :=
  match r with
  | leaf => .leaf
  | l@(inner _ _) =>
    match l with
    | leaf => .leaf
    | r@(inner ll lr) =>
        if true then
          match ll, lr with
          | inner _ _, inner _ _ => .leaf
          | _, _ => .leaf
        else .leaf

/-- error: (kernel) constant has already been declared 'Impl.balanceLErase.match_1.eq_1' -/
#guard_msgs in
theorem size_balanceLErase {r : Impl} {hrb} : (balanceLErase r hrb) = .leaf := by
  simp only [balanceLErase.eq_def]

Expected behavior: No error

Actual behavior: See above

Versions

4.15.0-nightly-2024-11-13 on live.lean-lang.org

Impact

Add 👍 to issues you consider important. If others are impacted by this issue, please ask them to add 👍 to it.

@TwoFX TwoFX added the bug Something isn't working label Nov 13, 2024
@nomeata
Copy link
Collaborator

nomeata commented Nov 13, 2024

One of the matches cannot be split:

set_option trace.split.failure true
/--
error: tactic 'split' failed, consider using `set_option trace.split.failure true`
case h_2.h_2
r✝¹ : Impl
hrb✝² : r✝¹.Balanced
l✝¹ r✝ l✝ : Impl
h✝¹ : l✝ = l✝¹.inner r✝
hrb✝¹ : (namedPattern l✝ (l✝¹.inner r✝) h✝¹).Balanced
ll✝ lr✝ : Impl
h✝ : ll✝.inner lr✝ = l✝¹.inner r✝
hrb✝ : (l✝¹.inner r✝).Balanced
heq✝¹ : l✝¹.inner r✝ = ll✝.inner lr✝
heq✝ : HEq ⋯ h✝
⊢ (match ll✝, lr✝, ⋯, h✝, hrb✝ with
    | l.inner r, l_1.inner r_1, h, h_1, hrb => leaf
    | x, x_1, h, h_1, hrb => leaf) =
    leaf
---
info: [split.failure] `split` tactic failed at
      match ll✝, lr✝, ⋯, h✝, hrb✝ with
      | l.inner r, l_1.inner r_1, h, h_1, hrb => Impl.leaf
      | x, x_1, h, h_1, hrb => Impl.leaf
    tactic 'contradiction' failed, metavariable has already been assigned
    l✝¹ r✝¹ l✝ r✝ : Impl
    hrb✝ : ((l✝¹.inner r✝¹).inner (l✝.inner r✝)).Balanced
    motive :
      (ll lr : Impl) →
        (h : (l✝¹.inner r✝¹).inner (l✝.inner r✝) = ll.inner lr) →
          (h_3 :
              namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) (ll.inner lr) h =
                (l✝¹.inner r✝¹).inner (l✝.inner r✝)) →
            (namedPattern (namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) (ll.inner lr) h)
                  ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) h_3).Balanced →
              Sort u_1
    h✝ :
      namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) ⋯ =
        (l✝¹.inner r✝¹).inner (l✝.inner r✝)
    hrb :
      (namedPattern (namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) ⋯)
          ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) h✝).Balanced
    h_1 :
      (l r l_1 r_1 : Impl) →
        (h : (l✝¹.inner r✝¹).inner (l✝.inner r✝) = (l.inner r).inner (l_1.inner r_1)) →
          (h_3 :
              namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) ((l.inner r).inner (l_1.inner r_1)) h =
                (l✝¹.inner r✝¹).inner (l✝.inner r✝)) →
            (hrb :
                (namedPattern (namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) ((l.inner r).inner (l_1.inner r_1)) h)
                    ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) h_3).Balanced) →
              motive (l.inner r) (l_1.inner r_1) h h_3 hrb
    h_2 :
      (x x_1 : Impl) →
        (h : (l✝¹.inner r✝¹).inner (l✝.inner r✝) = x.inner x_1) →
          (h_3 :
              namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) (x.inner x_1) h =
                (l✝¹.inner r✝¹).inner (l✝.inner r✝)) →
            (hrb :
                (namedPattern (namedPattern ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) (x.inner x_1) h)
                    ((l✝¹.inner r✝¹).inner (l✝.inner r✝)) h_3).Balanced) →
              motive x x_1 h h_3 hrb
    l_eq✝² : l✝¹ = l✝¹
    r_eq✝³ : r✝¹ = r✝¹
    l_eq✝¹ : l✝ = l✝
    r_eq✝² : r✝ = r✝
    r_eq✝¹ : l✝.inner r✝ = l✝.inner r✝
    l_eq✝ : l✝¹ = l✝¹
    r_eq✝ : r✝¹ = r✝¹
    ⊢ False
-/
#guard_msgs in
theorem size_balanceLErase {r : Impl} {hrb} : (balanceLErase r hrb) = .leaf := by
  rw [balanceLErase.eq_def]
  split
  · rfl
  · split
    · rfl
    · dsimp
      split      

My assumption for this bug is that when simp tries to rewrite with the equations, and thus creates the splitter and equations, then this process fails when adding .eq_1, but the environment isn’t rolled back, so simp tries again, but now .eq_1 already exists.

Didn’t dig deeper, but if that’s true, then

  • we should move the splitter generation into the same machinery as other realizable constants, just for uniformity (I believe @Kha has that on a branch already?)
  • implemented RFC: Realizalbe constants error handling #5944 so that we do not try to generate the splitter over and over
  • make sure that if it fails, we roll back the environment changes.

@Kha
Copy link
Member

Kha commented Nov 15, 2024

we should move the splitter generation into the same machinery as other realizable constants, just for uniformity (I believe @Kha has that on a branch already?)

Yep, I can land that independently once the general Environment.realizeConst machinery has landed

@leanprover-bot leanprover-bot added the P-medium We may work on this issue if we find the time label Nov 15, 2024
leodemoura added a commit that referenced this issue Nov 22, 2024
…essions

This PR fixes a non-termination bug that occurred when generating the
match-expression equation theorems. The bug was triggered when the proof
automation for the equation theorem repeatedly applied `injection(` to
the same local declaration, as it could not be removed due to forward
dependencies. See issue #6067 for an example that reproduces this issue.

closes #6067
github-merge-queue bot pushed a commit that referenced this issue Nov 23, 2024
…essions (#6180)

This PR fixes a non-termination bug that occurred when generating the
match-expression equation theorems. The bug was triggered when the proof
automation for the equation theorem repeatedly applied `injection(` to
the same local declaration, as it could not be removed due to forward
dependencies. See issue #6067 for an example that reproduces this issue.

closes #6067
JovanGerb pushed a commit to JovanGerb/lean4 that referenced this issue Jan 21, 2025
…essions (leanprover#6180)

This PR fixes a non-termination bug that occurred when generating the
match-expression equation theorems. The bug was triggered when the proof
automation for the equation theorem repeatedly applied `injection(` to
the same local declaration, as it could not be removed due to forward
dependencies. See issue leanprover#6067 for an example that reproduces this issue.

closes leanprover#6067
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working P-medium We may work on this issue if we find the time
Projects
None yet
Development

Successfully merging a pull request may close this issue.

4 participants