/- Copyright (c) 2022 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura -/ prelude import Lean.Data.AssocList import Lean.HeadIndex import Lean.Meta.Basic namespace Lean.Meta /-- A mapping that identifies definitionally equal expressions. We implement it as a mapping from `HeadIndex` to `AssocList Expr α`. Remark: this map may be quite inefficient if there are many `HeadIndex` collisions. -/ structure KExprMap (α : Type) where map : PHashMap HeadIndex (AssocList Expr α) := {} deriving Inhabited /-- Return `some v` if there is an entry `e ↦ v` in `m`. -/ def KExprMap.find? (m : KExprMap α) (e : Expr) : MetaM (Option α) := do match m.map.find? e.toHeadIndex with | none => return none | some ps => for ⟨e', a⟩ in ps do if (← isDefEq e e') then return some a return none private def updateList (ps : AssocList Expr α) (e : Expr) (v : α) : MetaM (AssocList Expr α) := do match ps with | AssocList.nil => return AssocList.cons e v ps | AssocList.cons e' v' ps => if (← isDefEq e e') then return AssocList.cons e v ps else return AssocList.cons e' v' (← updateList ps e v) /-- Insert `e ↦ v` into `m` -/ def KExprMap.insert (m : KExprMap α) (e : Expr) (v : α) : MetaM (KExprMap α) := let k := e.toHeadIndex match m.map.find? k with | none => return { map := m.map.insert k (AssocList.cons e v AssocList.nil) } | some ps => return { map := m.map.insert k (← updateList ps e v) } end Lean.Meta