forked from mit-plv/koika
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathDeriveShow.v
117 lines (95 loc) · 3.02 KB
/
DeriveShow.v
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
(*! Utilities | Automatic derivation of Show instances !*)
Require Import Koika.Show.
Require Koika.IdentParsing.
Require Import Ltac2.Ltac2.
Import Ltac2.Init.
Import Ltac2.Notations.
Module Internals.
Import Coq.Lists.List.ListNotations.
Import IdentParsing.Unsafe.
Import IdentParsing.
Ltac2 Type exn ::= [ NotAConstructor (constr) ].
Ltac2 eval_simpl c :=
Std.eval_simpl {
Std.rBeta := true;
Std.rMatch := true;
Std.rFix := true;
Std.rCofix := true;
Std.rZeta := true;
Std.rDelta := true;
Std.rConst := []
} None c.
Ltac2 rec list_map fn l :=
match l with
| [] => []
| h :: t => fn h :: list_map fn t
end.
Ltac2 rec coq_list_of_list type fn ls :=
match ls with
| [] => constr:(@List.nil $type)
| h :: t =>
let ch := fn h in
let ct := coq_list_of_list type fn t in
constr:($ch :: $ct)
end.
Ltac2 path_of_constructor c :=
match Constr.Unsafe.kind c with
| Constr.Unsafe.Constructor cstr _ =>
let path := Env.path (Std.ConstructRef cstr) in
coq_list_of_list constr:(String.string) coq_string_of_ident path
| _ => Control.throw (NotAConstructor c)
end.
Ltac2 unpack_app c :=
let rec loop acc c :=
match! c with
| ?f ?x => loop (x :: acc) f
| ?x => x, acc
end in
loop [] c.
Definition blocked (A: Type) := A.
Ltac not_blocked t :=
lazymatch t with
| blocked _ => fail
| _ => idtac
end.
Inductive Tracked {A} (x: A) := Track.
Definition type_of {A} (a: A) := A.
Fixpoint butlast {A} (l: list A) :=
match l with
| [] => []
| [x] => [x]
| hd :: tl => butlast tl
end.
Ltac2 derive_show_begin () :=
ltac1:(match goal with
| [ |- Show ?type ] =>
econstructor;
let term := fresh in
intro term; pose proof (@Track type term); destruct term
end).
Ltac2 derive_show_recurse arg :=
(* let instance := constr:(ltac:(typeclasses eauto) : Show (type_of $arg)) in *)
constr:(show $arg).
Ltac2 rec derive_show_app () :=
lazy_match! goal with
| [ _: Tracked ?app |- _ ] =>
match unpack_app app with
| (hd, args) => let hd_path := path_of_constructor hd in
let hd_str := constr:(String.concat "_" (butlast $hd_path)) in
let arg_strs := coq_list_of_list constr:(String.string) derive_show_recurse args in
let str := constr:(String.concat "_" ($hd_str :: $arg_strs)) in
let str := eval_simpl str in
exact $str
end
end.
Ltac2 rec derive_show () :=
Control.enter derive_show_begin; Control.enter derive_show_app.
#[global] Hint Extern 2 (Show _) => ltac2:(derive_show ()) : typeclass_instances.
End Internals.
Ltac derive_show :=
ltac2:(Control.enter Internals.derive_show).
Module Examples.
Inductive x := X0 | X1 | X2.
Inductive y := Y0 (_: x) | Y1 .
Definition test (n: nat) : Show y := _.
End Examples.