/- Copyright (c) 2019 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Authors: Leonardo de Moura -/ prelude import init.lean.environment init.lean.syntax namespace Lean /- Scope management -/ structure ScopeManagerState := (allNamespaces : NameSet := {}) /- Stack of namespaces for each each open namespace and section -/ (namespaces : List Name := []) /- Stack of namespace/section headers -/ (headers : List Name := []) (isNamespace : List Bool := []) namespace ScopeManagerState instance : Inhabited ScopeManagerState := ⟨{}⟩ def saveNamespace (s : ScopeManagerState) (n : Name) : ScopeManagerState := { allNamespaces := s.allNamespaces.insert n, .. s } end ScopeManagerState def regScopeManagerExtension : IO (SimplePersistentEnvExtension Name ScopeManagerState) := registerSimplePersistentEnvExtension { name := `scopes, addImportedFn := λ as, mkStateFromImportedEntries ScopeManagerState.saveNamespace {} as, addEntryFn := λ s n, { allNamespaces := s.allNamespaces.insert n, .. s }, } @[init regScopeManagerExtension] constant scopeManagerExt : SimplePersistentEnvExtension Name ScopeManagerState := default _ namespace Environment @[export lean.get_namespaces_core] def getNamespaces (env : Environment) : List Name := (scopeManagerExt.getState env).namespaces def getNamespaceSet (env : Environment) : NameSet := (scopeManagerExt.getState env).allNamespaces @[export lean.is_namespace_core] def isNamespace (env : Environment) (n : Name) : Bool := env.getNamespaceSet.contains n @[export lean.in_section_core] def inSection (env : Environment) : Bool := match (scopeManagerExt.getState env).isNamespace with | (b::_) := !b | _ := false @[export lean.has_open_scopes_core] def hasOpenScopes (env : Environment) : Bool := !env.getNamespaces.isEmpty @[export lean.get_namespace_core] def getNamespace (env : Environment) : Name := match env.getNamespaces with | (n::_) := n | _ := Name.anonymous @[export lean.get_scope_header_core] def getScopeHeader (env : Environment) : Name := match (scopeManagerExt.getState env).headers with | (n::_) := n | _ := Name.anonymous @[export lean.to_valid_namespace_core] def toValidNamespace (env : Environment) (n : Name) : Option Name := let s := scopeManagerExt.getState env in if s.allNamespaces.contains n then some n else s.namespaces.foldl (λ r ns, match r with | some _ := r | none := let c := ns ++ n in if s.allNamespaces.contains c then some c else none) none def registerNamespaceAux (env : Environment) (n : Name) : Environment := if env.getNamespaceSet.contains n then env else scopeManagerExt.addEntry env n @[export lean.register_namespace_core] def registerNamespace : Environment → Name → Environment | env n@(Name.mkString p _) := registerNamespace (registerNamespaceAux env n) p | env _ := env def pushScopeCore (env : Environment) (header : Name) (isNamespace : Bool) : Environment := let ns := env.getNamespace in let newNs := if isNamespace then ns ++ header else ns in let env := env.registerNamespaceAux newNs in let env := scopeManagerExt.modifyState env $ λ s, { headers := header :: s.headers, namespaces := newNs :: s.namespaces, isNamespace := isNamespace :: s.isNamespace, .. s } in env def popScopeCore (env : Environment) : Environment := if env.getNamespaces.isEmpty then env else scopeManagerExt.modifyState env $ λ s, { headers := s.headers.tail, namespaces := s.namespaces.tail, isNamespace := s.isNamespace.tail, .. s } end Environment inductive AttributeApplicationTime | afterTypeChecking | afterCompilation structure AttributeImpl := (name : Name) (descr : String) (add (env : Environment) (decl : Name) (args : Syntax) (persistent : Bool) : IO Environment) (addScoped (env : Environment) (decl : Name) (args : Syntax) : IO Environment := throw (IO.userError ("attribute '" ++ toString name ++ "' does not support scopes"))) (erase (env : Environment) (decl : Name) (persistent : Bool) : IO Environment := throw (IO.userError ("attribute '" ++ toString name ++ "' does not support removal"))) (activateScoped (env : Environment) (scope : Name) : IO Environment := pure env) (pushScope (env : Environment) : IO Environment := pure env) (popScope (env : Environment) : IO Environment := pure env) (applicationTime := AttributeApplicationTime.afterTypeChecking) instance AttributeImpl.inhabited : Inhabited AttributeImpl := ⟨{ name := default _, descr := default _, add := λ env _ _ _, pure env }⟩ def mkAttributeMapRef : IO (IO.Ref (HashMap Name AttributeImpl)) := IO.mkRef {} @[init mkAttributeMapRef] constant attributeMapRef : IO.Ref (HashMap Name AttributeImpl) := default _ def mkAttributeArrayRef : IO (IO.Ref (Array AttributeImpl)) := IO.mkRef Array.empty @[init mkAttributeArrayRef] constant attributeArrayRef : IO.Ref (Array AttributeImpl) := default _ /- Low level attribute registration function. -/ def registerAttribute (attr : AttributeImpl) : IO Unit := do m ← attributeMapRef.get, when (m.contains attr.name) $ throw (IO.userError ("invalid attribute declaration, '" ++ toString attr.name ++ "' has already been used")), initializing ← IO.initializing, unless initializing $ throw (IO.userError ("failed to register attribute, attributes can only be registered during initialization")), attributeMapRef.modify (λ m, m.insert attr.name attr), attributeArrayRef.modify (λ attrs, attrs.push attr) /- Return true iff `n` is the name of a registered attribute. -/ @[export lean.is_attribute_core] def isAttribute (n : Name) : IO Bool := do m ← attributeMapRef.get, pure (m.contains n) /- Return the name of all registered attributes. -/ def getAttributeNames : IO (List Name) := do m ← attributeMapRef.get, pure $ m.fold (λ r n _, n::r) [] def getAttributeImpl (attrName : Name) : IO AttributeImpl := do m ← attributeMapRef.get, match m.find attrName with | some attr := pure attr | none := throw (IO.userError ("unknown attribute '" ++ toString attrName ++ "'")) @[export lean.attribute_application_time_core] def attributeApplicationTime (n : Name) : IO AttributeApplicationTime := do attr ← getAttributeImpl n, pure attr.applicationTime namespace Environment /- Add attribute `attr` to declaration `decl` with arguments `args`. If `persistent == true`, then attribute is saved on .olean file. It throws an error when - `attr` is not the name of an attribute registered in the system. - `attr` does not support `persistent == false`. - `args` is not valid for `attr`. -/ @[export lean.add_attribute_core] def addAttribute (env : Environment) (decl : Name) (attrName : Name) (args : Syntax := Syntax.missing) (persistent := true) : IO Environment := do attr ← getAttributeImpl attrName, attr.add env decl args persistent /- Add a scoped attribute `attr` to declaration `decl` with arguments `args` and scope `decl.getPrefix`. Scoped attributes are always persistent. It returns `Except.error` when - `attr` is not the name of an attribute registered in the system. - `attr` does not support scoped attributes. - `args` is not valid for `attr`. Remark: the attribute will not be activated if `decl` is not inside the current namespace `env.getNamespace`. -/ @[export lean.add_scoped_attribute_core] def addScopedAttribute (env : Environment) (decl : Name) (attrName : Name) (args : Syntax := Syntax.missing) : IO Environment := do attr ← getAttributeImpl attrName, attr.addScoped env decl args /- Remove attribute `attr` from declaration `decl`. The effect is the current scope. It returns `Except.error` when - `attr` is not the name of an attribute registered in the system. - `attr` does not support erasure. - `args` is not valid for `attr`. -/ @[export lean.erase_attribute_core] def eraseAttribute (env : Environment) (decl : Name) (attrName : Name) (persistent := true) : IO Environment := do attr ← getAttributeImpl attrName, attr.erase env decl persistent /- Activate the scoped attribute `attr` for all declarations in scope `scope`. We use this function to implement the command `open foo`. -/ @[export lean.activate_scoped_attribute_core] def activateScopedAttribute (env : Environment) (attrName : Name) (scope : Name) : IO Environment := do attr ← getAttributeImpl attrName, attr.activateScoped env scope /- Activate all scoped attributes at `scope` -/ @[export lean.activate_scoped_attributes_core] def activateScopedAttributes (env : Environment) (scope : Name) : IO Environment := do attrs ← attributeArrayRef.get, attrs.mfoldl (λ env attr, attr.activateScoped env scope) env /- We use this function to implement commands `namespace foo` and `section foo`. It activates scoped attributes in the new resulting namespace. -/ @[export lean.push_scope_core] def pushScope (env : Environment) (header : Name) (isNamespace : Bool) : IO Environment := do let env := env.pushScopeCore header isNamespace, let ns := env.getNamespace, attrs ← attributeArrayRef.get, attrs.mfoldl (λ env attr, do env ← attr.pushScope env, if isNamespace then attr.activateScoped env ns else pure env) env /- We use this function to implement commands `end foo` for closing namespaces and sections. -/ @[export lean.pop_scope_core] def popScope (env : Environment) : IO Environment := do let env := env.popScopeCore, attrs ← attributeArrayRef.get, attrs.mfoldl (λ env attr, attr.popScope env) env end Environment /-- Tag attributes are simple and efficient. They are useful for marking declarations in the modules where they were defined. The startup cost for this kind of attribute is very small since `addImportedFn` is a constant function. They provide the predicate `tagAttr.hasTag env decl` which returns true iff declaration `decl` is tagged in the environment `env`. -/ structure TagAttribute := (attr : AttributeImpl) (ext : PersistentEnvExtension Name NameSet) def registerTagAttribute (name : Name) (descr : String) (validate : Environment → Name → Except String Unit := λ _ _, Except.ok ()) : IO TagAttribute := do ext : PersistentEnvExtension Name NameSet ← registerPersistentEnvExtension { name := name, addImportedFn := λ _, {}, addEntryFn := λ (s : NameSet) n, s.insert n, exportEntriesFn := λ es, let r : Array Name := es.fold (λ a e, a.push e) Array.empty in r.qsort Name.quickLt, statsFn := λ s, "tag attribute" ++ Format.line ++ "number of local entries: " ++ format s.size }, let attrImpl : AttributeImpl := { name := name, descr := descr, add := λ env decl args persistent, do unless args.isMissing $ throw (IO.userError ("invalid attribute '" ++ toString name ++ "', unexpected argument")), unless persistent $ throw (IO.userError ("invalid attribute '" ++ toString name ++ "', must be persistent")), unless (env.getModuleIdxFor decl).isNone $ throw (IO.userError ("invalid attribute '" ++ toString name ++ "', declaration is in an imported module")), match validate env decl with | Except.error msg := throw (IO.userError ("invalid attribute '" ++ toString name ++ "', " ++ msg)) | _ := pure $ ext.addEntry env decl }, registerAttribute attrImpl, pure { attr := attrImpl, ext := ext } namespace TagAttribute instance : Inhabited TagAttribute := ⟨{attr := default _, ext := default _}⟩ def hasTag (attr : TagAttribute) (env : Environment) (decl : Name) : Bool := match env.getModuleIdxFor decl with | some modIdx := (attr.ext.getModuleEntries env modIdx).binSearchContains decl Name.quickLt | none := (attr.ext.getState env).contains decl end TagAttribute /-- A `TagAttribute` variant where we can attach parameters to attributes. It is slightly more expensive and consumes a little bit more memory than `TagAttribute`. They provide the function `pAttr.getParam env decl` which returns `some p` iff declaration `decl` contains the attribute `pAttr` with parameter `p`. -/ structure ParametricAttribute (α : Type) := (attr : AttributeImpl) (ext : PersistentEnvExtension (Name × α) (NameMap α)) def registerParametricAttribute {α : Type} [Inhabited α] (name : Name) (descr : String) (getParam : Environment → Name → Syntax → Except String α) : IO (ParametricAttribute α) := do ext : PersistentEnvExtension (Name × α) (NameMap α) ← registerPersistentEnvExtension { name := name, addImportedFn := λ _, {}, addEntryFn := λ (s : NameMap α) (p : Name × α), s.insert p.1 p.2, exportEntriesFn := λ m, let r : Array (Name × α) := m.fold (λ a n p, a.push (n, p)) Array.empty in r.qsort (λ a b, Name.quickLt a.1 b.1), statsFn := λ s, "parametric attribute" ++ Format.line ++ "number of local entries: " ++ format s.size }, let attrImpl : AttributeImpl := { name := name, descr := descr, add := λ env decl args persistent, do unless persistent $ throw (IO.userError ("invalid attribute '" ++ toString name ++ "', must be persistent")), unless (env.getModuleIdxFor decl).isNone $ throw (IO.userError ("invalid attribute '" ++ toString name ++ "', declaration is in an imported module")), match getParam env decl args with | Except.error msg := throw (IO.userError ("invalid attribute '" ++ toString name ++ "', " ++ msg)) | Except.ok val := pure $ ext.addEntry env (decl, val) }, registerAttribute attrImpl, pure { attr := attrImpl, ext := ext } namespace ParametricAttribute instance {α : Type} : Inhabited (ParametricAttribute α) := ⟨{attr := default _, ext := default _}⟩ def getParam {α : Type} [Inhabited α] (attr : ParametricAttribute α) (env : Environment) (decl : Name) : Option α := match env.getModuleIdxFor decl with | some modIdx := match (attr.ext.getModuleEntries env modIdx).binSearch (decl, default _) (λ a b, Name.quickLt a.1 b.1) with | some (_, val) := some val | none := none | none := (attr.ext.getState env).find decl def setParam {α : Type} (attr : ParametricAttribute α) (env : Environment) (decl : Name) (param : α) : Except String Environment := if (env.getModuleIdxFor decl).isSome then Except.error ("invalid '" ++ toString attr.attr.name ++ "'.setParam, declaration is in an imported module") else if ((attr.ext.getState env).find decl).isSome then Except.error ("invalid '" ++ toString attr.attr.name ++ "'.setParam, attribute has already been set") else Except.ok (attr.ext.addEntry env (decl, param)) end ParametricAttribute end Lean