(* Title: Pure/Syntax/simple_syntax.ML
Author: Makarius
Simple syntax for types and terms --- for bootstrapping Pure.
*)
signature SIMPLE_SYNTAX =
sig
val read_typ: string -> typ
val read_term: string -> term
val read_prop: string -> term
end;
structure Simple_Syntax: SIMPLE_SYNTAX =
struct
(* scanning tokens *)
val lexicon = Scan.make_lexicon
(map Symbol.explode ["!!", "%", "(", ")", ".", "::", "==", "==>", "=?=", "=>", "&&&", "CONST"]);
fun read scan s =
(case
Symbol_Pos.explode (s, Position.none) |>
Lexicon.tokenize lexicon false |>
filter Lexicon.is_proper |>
Scan.read Lexicon.stopper scan of
SOME x => x
| NONE => error ("Malformed input: " ^ quote s));
(* basic scanners *)
fun $$ s = Scan.some (fn Lexicon.Token (Lexicon.Literal, s', _) =>
if s = s' then SOME s else NONE | _ => NONE);
fun enum1 s scan = scan ::: Scan.repeat ($$ s |-- scan);
fun enum2 s scan = scan ::: Scan.repeat1 ($$ s |-- scan);
val tfree = Scan.some (fn Lexicon.Token (Lexicon.TFreeSy, s, _) => SOME s | _ => NONE);
val ident = Scan.some (fn Lexicon.Token (Lexicon.IdentSy, s, _) => SOME s | _ => NONE);
val var = Scan.some (fn Lexicon.Token (Lexicon.VarSy, s, _) =>
SOME (Lexicon.read_indexname (unprefix "?" s)) | _ => NONE);
val long_ident = Scan.some (fn Lexicon.Token (Lexicon.LongIdentSy, s, _) => SOME s | _ => NONE);
val const = long_ident || ident;
(* types *)
(*
typ = typ1 => ... => typ1
| typ1
typ1 = typ2 const ... const
| typ2
typ2 = tfree
| const
| ( typ )
*)
fun typ x =
(enum1 "=>" typ1 >> (op ---> o split_last)) x
and typ1 x =
(typ2 -- Scan.repeat const >> (fn (T, cs) => fold (fn c => fn U => Type (c, [U])) cs T)) x
and typ2 x =
(tfree >> (fn a => TFree (a, [])) ||
const >> (fn c => Type (c, [])) ||
$$ "(" |-- typ --| $$ ")") x;
val read_typ = read typ;
(* terms *)
(*
term = !!ident :: typ. term
| term1
term1 = term2 ==> ... ==> term2
| term2
term2 = term3 == term2
| term3 =?= term2
| term3 &&& term2
| term3
term3 = ident :: typ
| var :: typ
| CONST const :: typ
| %ident :: typ. term3
| term4
term4 = term5 ... term5
| term5
term5 = ident
| var
| CONST const
| ( term )
*)
local
val constraint = $$ "::" |-- typ;
val idt = ident -- constraint;
val bind = idt --| $$ ".";
fun term env T x =
($$ "!!" |-- bind :|-- (fn v =>
term (v :: env) propT >> (fn B => Term.all (#2 v) $ lambda (Free v) B)) ||
term1 env T) x
and term1 env T x =
(enum2 "==>" (term2 env propT) >> foldr1 Logic.mk_implies ||
term2 env T) x
and term2 env T x =
(equal env "==" || equal env "=?=" ||
term3 env propT -- ($$ "&&&" |-- term2 env propT) >> Logic.mk_conjunction ||
term3 env T) x
and equal env eq x =
(term3 env dummyT -- ($$ eq |-- term2 env dummyT) >> (fn (t, u) =>
Const (eq, Term.fastype_of t --> Term.fastype_of u --> propT) $ t $ u)) x
and term3 env T x =
(idt >> Free ||
var -- constraint >> Var ||
$$ "CONST" |-- const -- constraint >> Const ||
$$ "%" |-- bind :|-- (fn v => term3 (v :: env) dummyT >> lambda (Free v)) ||
term4 env T) x
and term4 env T x =
(term5 env dummyT -- Scan.repeat1 (term5 env dummyT) >> Term.list_comb ||
term5 env T) x
and term5 env T x =
(ident >> (fn a => Free (a, the_default T (AList.lookup (op =) env a))) ||
var >> (fn xi => Var (xi, T)) ||
$$ "CONST" |-- const >> (fn c => Const (c, T)) ||
$$ "(" |-- term env T --| $$ ")") x;
fun read_tm T s =
let val t = read (term [] T) s in
if can (Term.map_types Term.no_dummyT) t then t
else error ("Unspecified types in input: " ^ quote s)
end;
in
val read_term = read_tm dummyT;
val read_prop = read_tm propT;
end;
end;