File ‹Lens_Statespace.ML›
signature LENS_STATESPACE =
sig
val mk_statespace_elements: bstring -> (bstring * typ) list -> xstring list -> theory -> Element.context_i list
val compile_statespace: bstring -> (bstring * typ) list -> xstring list -> theory -> theory
val statespace_cmd: bstring -> xstring list -> (bstring * string) list -> theory -> theory
end
structure Lens_Statespace =
struct
fun mk_statespace_elements vds exts thy =
let
val ctx = Named_Target.theory_init thy
val vns = map fst vds
val vfixes = map (fn (n, t) => (Binding.name n, SOME (Lens_Lib.lensT t Lens_Lib.astateT), NoSyn)) vds
val vwbs = map (Lens_Lib.mk_vwb_lens o Syntax.free) vns;
val simp = Attrib.check_src ctx (Token.make_src ("simp", Position.none) [])
val imps = map (fn e => Locale.check thy (e, Position.none)) exts
val ilnsls = map (filter (Lens_Lib.isLensT o snd) o map fst o Locale.params_of thy) imps
val impind = List.concat (map (Lens_Lib.pairsWith vns) (map (map fst) ilnsls))
val indeps = map (fn (x, y) => Lens_Lib.mk_indep (Syntax.free x) (Syntax.free y))
(Lens_Lib.pairings (vns) @ impind)
in
[ Element.Fixes vfixes
, Element.Assumes [((Binding.name "vwbs", [simp])
, map (fn vwb => (vwb, [])) vwbs),
((Binding.name "indeps", [simp])
, map (fn vwb => (vwb, [])) indeps)]
]
end
fun compile_statespace ssn vds exts thy =
let
val imps = map (fn e => Locale.check thy (e, Position.none)) exts
val locexs = map (fn i => (i, (("", false), (Expression.Positional [], [])))) imps
in
(Local_Theory.exit_global o snd o
Expression.add_locale (Binding.name ssn) Binding.empty [] (locexs,[])
(mk_statespace_elements vds exts thy)) thy
end
fun statespace_cmd n exts vds thy =
let
val ctx = Named_Target.theory_init thy in
compile_statespace n (map (fn (n, t) => (n, Syntax.read_typ ctx t)) vds) exts thy
end
val _ =
Outer_Syntax.command @{command_keyword statespace} "define locale-based statespace with lenses"
((Parse.name --
(@{keyword "="} |-- Scan.repeat (Parse.name --| @{keyword "+"}) --
Scan.repeat (Parse.name -- (Parse.$$$ "::" |-- Parse.!!! Parse.typ))))
>> (fn (n, (exts, vds)) => Toplevel.theory (statespace_cmd n exts vds)))
end;