@kha I have added support for opaque constants to the old C++ frontend, and made sure the new frontend can still parse `library/init/core.lean`. The kernel should enforce that opaque constants are really opaque, and the following example should fail ``` constant x : nat := 0 theorem foo : x = 0 := rfl ``` If it doesn't, it is a bug. Here are some remaining issues: 1- `environment.mk_empty` is currently an axiom because we cannot create an inhabitant of an opaque type. A possible solution is to use `option environment` instead of `environment`. 2- There is no support for opaque constants in the new frontend. However, I modified it to handle axioms, and fixed the literal values with decl_cmd_kind. I tried to mark some of my changes with comments, but it is probably much easier for you to just check the commit change list. 3- I did not add any support for automatically constructing `e` at `constant x : t := e`. I think we can do this later after we replace the old frontend with the new one. BTW, it took only a few minutes to provide the inhabitants manually.
166 lines
5.5 KiB
Text
166 lines
5.5 KiB
Text
/-
|
|
Copyright (c) 2018 Microsoft Corporation. All rights reserved.
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
Author: Sebastian Ullrich
|
|
|
|
Parsers for commands that declare things
|
|
-/
|
|
|
|
prelude
|
|
import init.lean.parser.term
|
|
|
|
namespace lean
|
|
namespace parser
|
|
|
|
open combinators monad_parsec
|
|
open parser.has_tokens parser.has_view
|
|
|
|
instance term_parser_command_parser_coe : has_coe term_parser command_parser :=
|
|
⟨λ p, adapt_reader coe $ p.run⟩
|
|
|
|
namespace «command»
|
|
|
|
local postfix `?`:10000 := optional
|
|
local postfix *:10000 := combinators.many
|
|
local postfix +:10000 := combinators.many1
|
|
|
|
@[derive has_tokens has_view]
|
|
def doc_comment.parser : command_parser :=
|
|
node! doc_comment ["/--", doc: raw $ many' (not_followed_by (str "-/") *> any), "-/"]
|
|
|
|
@[derive has_tokens has_view]
|
|
def attr_instance.parser : command_parser :=
|
|
-- use `raw_ident` because of attribute names such as `instance`
|
|
node! attr_instance [name: raw_ident.parser, args: (term.parser max_prec)*]
|
|
|
|
@[derive has_tokens has_view]
|
|
def decl_attributes.parser : command_parser :=
|
|
-- TODO(Sebastian): custom attribute parsers
|
|
node! decl_attributes ["@[", attrs: sep_by1 attr_instance.parser (symbol ","), "]"]
|
|
|
|
set_option class.instance_max_depth 300
|
|
@[derive has_tokens has_view]
|
|
def decl_modifiers.parser : command_parser :=
|
|
node! decl_modifiers [
|
|
doc_comment: doc_comment.parser?,
|
|
attrs: decl_attributes.parser?,
|
|
visibility: node_choice! visibility {"private", "protected"}?,
|
|
«noncomputable»: (symbol "noncomputable")?,
|
|
«unsafe»: (symbol "unsafe")?,
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def decl_sig.parser : command_parser :=
|
|
node! decl_sig [
|
|
params: term.bracketed_binders.parser,
|
|
type: term.type_spec.parser,
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def opt_decl_sig.parser : command_parser :=
|
|
node! opt_decl_sig [
|
|
params: term.bracketed_binders.parser,
|
|
type: term.opt_type.parser,
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def equation.parser : command_parser :=
|
|
node! equation ["|", lhs: (term.parser max_prec)+, ":=", rhs: term.parser]
|
|
|
|
@[derive has_tokens has_view]
|
|
def decl_val.parser : command_parser :=
|
|
node_choice! decl_val {
|
|
simple: node! simple_decl_val [":=", body: term.parser],
|
|
empty_match: symbol ".",
|
|
«match»: equation.parser+
|
|
}
|
|
|
|
@[derive has_tokens has_view]
|
|
def infer_modifier.parser : command_parser :=
|
|
node_choice! infer_modifier {
|
|
relaxed: try $ node! relaxed_infer_modifier ["{", "}"],
|
|
strict: try $ node! strict_infer_modifier ["(", ")"],
|
|
}
|
|
|
|
@[derive has_tokens has_view]
|
|
def intro_rule.parser : command_parser :=
|
|
node! intro_rule [
|
|
"|",
|
|
name: ident.parser,
|
|
infer_mod: infer_modifier.parser?,
|
|
sig: opt_decl_sig.parser,
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def struct_binder_content.parser : command_parser :=
|
|
node! struct_binder_content [
|
|
ids: ident.parser+,
|
|
infer_mod: infer_modifier.parser?,
|
|
sig: opt_decl_sig.parser,
|
|
default: term.binder_default.parser?,
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def structure_field_block.parser : command_parser :=
|
|
node_choice! structure_field_block {
|
|
explicit: node! struct_explicit_binder ["(", content: node_choice! struct_explicit_binder_content {
|
|
«notation»: command.notation_like.parser,
|
|
other: struct_binder_content.parser
|
|
}, right: symbol ")"],
|
|
implicit: node! struct_implicit_binder ["{", content: struct_binder_content.parser, "}"],
|
|
strict_implicit: node! strict_implicit_binder ["⦃", content: struct_binder_content.parser, "⦄"],
|
|
inst_implicit: node! inst_implicit_binder ["[", content: struct_binder_content.parser, "]"],
|
|
}
|
|
|
|
@[derive has_tokens has_view]
|
|
def old_univ_params.parser : command_parser :=
|
|
node! old_univ_params ["{", ids: ident.parser+, "}"]
|
|
|
|
@[derive parser.has_tokens parser.has_view]
|
|
def ident_univ_params.parser : command_parser :=
|
|
node! ident_univ_params [
|
|
id: ident.parser,
|
|
univ_params: node! univ_params [".{", params: ident.parser+, "}"]?
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def structure.parser : command_parser :=
|
|
node! «structure» [
|
|
keyword: node_choice! structure_kw {"structure", "class"},
|
|
old_univ_params: old_univ_params.parser?,
|
|
name: ident_univ_params.parser,
|
|
sig: opt_decl_sig.parser,
|
|
«extends»: node! «extends» ["extends", parents: sep_by1 term.parser (symbol ",")]?,
|
|
":=",
|
|
ctor: node! structure_ctor [name: ident.parser, infer_mod: infer_modifier.parser?, "::"]?,
|
|
field_blocks: structure_field_block.parser*,
|
|
]
|
|
|
|
@[derive has_tokens has_view]
|
|
def declaration.parser : command_parser :=
|
|
node! declaration [
|
|
modifiers: decl_modifiers.parser,
|
|
inner: node_choice! declaration.inner {
|
|
«def_like»: node! «def_like» [
|
|
kind: node_choice! def_like.kind {"def", "abbreviation", "abbrev", "theorem"},
|
|
old_univ_params: old_univ_params.parser?,
|
|
name: ident_univ_params.parser, sig: opt_decl_sig.parser, val: decl_val.parser],
|
|
«instance»: node! «instance» ["instance", name: ident_univ_params.parser?, sig: decl_sig.parser, val: decl_val.parser],
|
|
«example»: node! «example» ["example", sig: decl_sig.parser, val: decl_val.parser],
|
|
«axiom»: node! «axiom» [ -- CommentTo(Kha): -- replaced `constant with `axiom
|
|
kw: node_choice! constant_keyword {"axiom"},
|
|
name: ident_univ_params.parser,
|
|
sig: decl_sig.parser],
|
|
«inductive»: node! «inductive» [try [«class»: (symbol "class")?, "inductive"],
|
|
old_univ_params: old_univ_params.parser?,
|
|
name: ident_univ_params.parser,
|
|
sig: opt_decl_sig.parser,
|
|
local_notation: notation_like.parser?,
|
|
intro_rules: intro_rule.parser*],
|
|
«structure»: structure.parser,
|
|
}
|
|
]
|
|
|
|
end «command»
|
|
end parser
|
|
end lean
|