|
|
|
@ -11,19 +11,19 @@ def congruenceArg (mvarId: MVarId): MetaM (List MVarId) := mvarId.withContext do
|
|
|
|
|
let userName := (← mvarId.getDecl).userName
|
|
|
|
|
|
|
|
|
|
let u ← Meta.mkFreshLevelMVar
|
|
|
|
|
let α ← Meta.mkFreshExprSyntheticOpaqueMVar (mkSort u)
|
|
|
|
|
(tag := userName ++ `α)
|
|
|
|
|
let f ← Meta.mkFreshExprSyntheticOpaqueMVar (.forallE .anonymous α β .default)
|
|
|
|
|
(tag := userName ++ `f)
|
|
|
|
|
let a₁ ← Meta.mkFreshExprSyntheticOpaqueMVar α
|
|
|
|
|
(tag := userName ++ `a₁)
|
|
|
|
|
let a₂ ← Meta.mkFreshExprSyntheticOpaqueMVar α
|
|
|
|
|
(tag := userName ++ `a₂)
|
|
|
|
|
let h ← Meta.mkFreshExprSyntheticOpaqueMVar (← Meta.mkEq a₁ a₂)
|
|
|
|
|
(tag := userName ++ `h)
|
|
|
|
|
let α ← Meta.mkFreshExprMVar (.some $ mkSort u)
|
|
|
|
|
.natural (userName := userName ++ `α)
|
|
|
|
|
let f ← Meta.mkFreshExprMVar (.some <| .forallE .anonymous α β .default)
|
|
|
|
|
.synthetic (userName := userName ++ `f)
|
|
|
|
|
let a₁ ← Meta.mkFreshExprMVar (.some α)
|
|
|
|
|
.synthetic (userName := userName ++ `a₁)
|
|
|
|
|
let a₂ ← Meta.mkFreshExprMVar (.some α)
|
|
|
|
|
.synthetic (userName := userName ++ `a₂)
|
|
|
|
|
let h ← Meta.mkFreshExprMVar (.some $ ← Meta.mkEq a₁ a₂)
|
|
|
|
|
.synthetic (userName := userName ++ `h)
|
|
|
|
|
let conduitType ← Meta.mkEq (← Meta.mkEq (.app f a₁) (.app f a₂)) target
|
|
|
|
|
let conduit ← Meta.mkFreshExprSyntheticOpaqueMVar conduitType
|
|
|
|
|
(tag := userName ++ `conduit)
|
|
|
|
|
let conduit ← Meta.mkFreshExprMVar conduitType
|
|
|
|
|
.synthetic (userName := userName ++ `conduit)
|
|
|
|
|
mvarId.assign $ ← Meta.mkEqMP conduit (← Meta.mkCongrArg f h)
|
|
|
|
|
let result := [α, a₁, a₂, f, h, conduit]
|
|
|
|
|
return result.map (·.mvarId!)
|
|
|
|
@ -39,20 +39,20 @@ def congruenceFun (mvarId: MVarId): MetaM (List MVarId) := mvarId.withContext do
|
|
|
|
|
let .some (β, _, _) := (← instantiateMVars target).eq? | throwError "Goal is not an Eq"
|
|
|
|
|
let userName := (← mvarId.getDecl).userName
|
|
|
|
|
let u ← Meta.mkFreshLevelMVar
|
|
|
|
|
let α ← Meta.mkFreshExprSyntheticOpaqueMVar (mkSort u)
|
|
|
|
|
(tag := userName ++ `α)
|
|
|
|
|
let α ← Meta.mkFreshExprMVar (.some $ mkSort u)
|
|
|
|
|
.natural (userName := userName ++ `α)
|
|
|
|
|
let fType := .forallE .anonymous α β .default
|
|
|
|
|
let f₁ ← Meta.mkFreshExprSyntheticOpaqueMVar fType
|
|
|
|
|
(tag := userName ++ `f₁)
|
|
|
|
|
let f₂ ← Meta.mkFreshExprSyntheticOpaqueMVar fType
|
|
|
|
|
(tag := userName ++ `f₂)
|
|
|
|
|
let a ← Meta.mkFreshExprSyntheticOpaqueMVar α
|
|
|
|
|
(tag := userName ++ `a)
|
|
|
|
|
let h ← Meta.mkFreshExprSyntheticOpaqueMVar (← Meta.mkEq f₁ f₂)
|
|
|
|
|
(tag := userName ++ `h)
|
|
|
|
|
let f₁ ← Meta.mkFreshExprMVar (.some fType)
|
|
|
|
|
.synthetic (userName := userName ++ `f₁)
|
|
|
|
|
let f₂ ← Meta.mkFreshExprMVar (.some fType)
|
|
|
|
|
.synthetic (userName := userName ++ `f₂)
|
|
|
|
|
let a ← Meta.mkFreshExprMVar (.some α)
|
|
|
|
|
.synthetic (userName := userName ++ `a)
|
|
|
|
|
let h ← Meta.mkFreshExprMVar (.some $ ← Meta.mkEq f₁ f₂)
|
|
|
|
|
.synthetic (userName := userName ++ `h)
|
|
|
|
|
let conduitType ← Meta.mkEq (← Meta.mkEq (.app f₁ a) (.app f₂ a)) target
|
|
|
|
|
let conduit ← Meta.mkFreshExprSyntheticOpaqueMVar conduitType
|
|
|
|
|
(tag := userName ++ `conduit)
|
|
|
|
|
let conduit ← Meta.mkFreshExprMVar conduitType
|
|
|
|
|
.synthetic (userName := userName ++ `conduit)
|
|
|
|
|
mvarId.assign $ ← Meta.mkEqMP conduit (← Meta.mkCongrFun h a)
|
|
|
|
|
let result := [α, f₁, f₂, h, a, conduit]
|
|
|
|
|
return result.map (·.mvarId!)
|
|
|
|
@ -68,24 +68,24 @@ def congruence (mvarId: MVarId): MetaM (List MVarId) := mvarId.withContext do
|
|
|
|
|
let .some (β, _, _) := (← instantiateMVars target).eq? | throwError "Goal is not an Eq"
|
|
|
|
|
let userName := (← mvarId.getDecl).userName
|
|
|
|
|
let u ← Meta.mkFreshLevelMVar
|
|
|
|
|
let α ← Meta.mkFreshExprSyntheticOpaqueMVar (mkSort u)
|
|
|
|
|
(tag := userName ++ `α)
|
|
|
|
|
let α ← Meta.mkFreshExprMVar (.some $ mkSort u)
|
|
|
|
|
.natural (userName := userName ++ `α)
|
|
|
|
|
let fType := .forallE .anonymous α β .default
|
|
|
|
|
let f₁ ← Meta.mkFreshExprSyntheticOpaqueMVar fType
|
|
|
|
|
(tag := userName ++ `f₁)
|
|
|
|
|
let f₂ ← Meta.mkFreshExprSyntheticOpaqueMVar fType
|
|
|
|
|
(tag := userName ++ `f₂)
|
|
|
|
|
let a₁ ← Meta.mkFreshExprSyntheticOpaqueMVar α
|
|
|
|
|
(tag := userName ++ `a₁)
|
|
|
|
|
let a₂ ← Meta.mkFreshExprSyntheticOpaqueMVar α
|
|
|
|
|
(tag := userName ++ `a₂)
|
|
|
|
|
let h₁ ← Meta.mkFreshExprSyntheticOpaqueMVar (← Meta.mkEq f₁ f₂)
|
|
|
|
|
(tag := userName ++ `h₁)
|
|
|
|
|
let h₂ ← Meta.mkFreshExprSyntheticOpaqueMVar (← Meta.mkEq a₁ a₂)
|
|
|
|
|
(tag := userName ++ `h₂)
|
|
|
|
|
let f₁ ← Meta.mkFreshExprMVar (.some fType)
|
|
|
|
|
.synthetic (userName := userName ++ `f₁)
|
|
|
|
|
let f₂ ← Meta.mkFreshExprMVar (.some fType)
|
|
|
|
|
.synthetic (userName := userName ++ `f₂)
|
|
|
|
|
let a₁ ← Meta.mkFreshExprMVar (.some α)
|
|
|
|
|
.synthetic (userName := userName ++ `a₁)
|
|
|
|
|
let a₂ ← Meta.mkFreshExprMVar (.some α)
|
|
|
|
|
.synthetic (userName := userName ++ `a₂)
|
|
|
|
|
let h₁ ← Meta.mkFreshExprMVar (.some $ ← Meta.mkEq f₁ f₂)
|
|
|
|
|
.synthetic (userName := userName ++ `h₁)
|
|
|
|
|
let h₂ ← Meta.mkFreshExprMVar (.some $ ← Meta.mkEq a₁ a₂)
|
|
|
|
|
.synthetic (userName := userName ++ `h₂)
|
|
|
|
|
let conduitType ← Meta.mkEq (← Meta.mkEq (.app f₁ a₁) (.app f₂ a₂)) target
|
|
|
|
|
let conduit ← Meta.mkFreshExprSyntheticOpaqueMVar conduitType
|
|
|
|
|
(tag := userName ++ `conduit)
|
|
|
|
|
let conduit ← Meta.mkFreshExprMVar conduitType
|
|
|
|
|
.synthetic (userName := userName ++ `conduit)
|
|
|
|
|
mvarId.assign $ ← Meta.mkEqMP conduit (← Meta.mkCongr h₁ h₂)
|
|
|
|
|
let result := [α, f₁, f₂, a₁, a₂, h₁, h₂, conduit]
|
|
|
|
|
return result.map (·.mvarId!)
|
|
|
|
|