-
Notifications
You must be signed in to change notification settings - Fork 84
/
pan_simpScript.sml
138 lines (124 loc) · 4.09 KB
/
pan_simpScript.sml
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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
(*
Simplification of panLang.
*)
open preamble panLangTheory
val _ = new_theory "pan_simp"
val _ = set_grammar_ancestry ["panLang","backend_common"];
val _ = patternMatchesLib.ENABLE_PMATCH_CASES();
Definition SmartSeq_def[simp]:
SmartSeq p q =
if p = Skip then q else Seq p q
End
Definition seq_assoc_def:
(seq_assoc p Skip = p) /\
(seq_assoc p (Dec v e q) =
SmartSeq p (Dec v e (seq_assoc Skip q))) /\
(seq_assoc p (Seq q r) = seq_assoc (seq_assoc p q) r) /\
(seq_assoc p (If e q r) =
SmartSeq p (If e (seq_assoc Skip q) (seq_assoc Skip r))) /\
(seq_assoc p (While e q) =
SmartSeq p (While e (seq_assoc Skip q))) /\
(seq_assoc p (Call NONE name args) =
SmartSeq p (Call NONE name args)) /\
(seq_assoc p (Call (SOME (rv , exp)) name args) =
SmartSeq p (Call
(dtcase exp of
| NONE => SOME (rv , NONE)
| SOME (eid , ev , ep) =>
SOME (rv , (SOME (eid , ev , (seq_assoc Skip ep)))))
name args)) /\
(seq_assoc p (DecCall v s e es q) =
SmartSeq p (DecCall v s e es (seq_assoc Skip q))) /\
(seq_assoc p (Annot _ _) = p) /\
(seq_assoc p q = SmartSeq p q)
End
Definition seq_call_ret_def:
seq_call_ret prog =
dtcase prog of
| Seq (AssignCall rv1 NONE trgt args) (Return (Var (rv2:mlstring))) =>
if rv1 = rv2 then (TailCall trgt args) else prog
| other => other
End
Definition ret_to_tail_def:
(ret_to_tail Skip = Skip) /\
(ret_to_tail (Dec v e q) = Dec v e (ret_to_tail q)) /\
(ret_to_tail (Seq p q) =
seq_call_ret (Seq (ret_to_tail p) (ret_to_tail q))) /\
(ret_to_tail (If e p q) = If e (ret_to_tail p) (ret_to_tail q)) /\
(ret_to_tail (While e p) = While e (ret_to_tail p)) /\
(ret_to_tail (Call NONE name args) = Call NONE name args) /\
(ret_to_tail (Call (SOME (rv , exp)) name args) =
Call (SOME (rv, (dtcase exp of
| NONE => NONE
| SOME (eid , ev , ep) =>
SOME (eid, ev, (ret_to_tail ep)))))
name args) /\
(ret_to_tail (DecCall v s e es q) = DecCall v s e es (ret_to_tail q)) /\
(ret_to_tail p = p)
End
Definition compile_def:
compile p =
let p = seq_assoc Skip p in
ret_to_tail p
End
Definition compile_prog_def:
compile_prog prog =
MAP (λ(name, params, body).
(name,
params,
compile body)) prog
End
Theorem seq_assoc_pmatch:
!p prog.
seq_assoc p prog =
case prog of
| Skip => p
| (Dec v e q) => SmartSeq p (Dec v e (seq_assoc Skip q))
| (Seq q r) => seq_assoc (seq_assoc p q) r
| (If e q r) =>
SmartSeq p (If e (seq_assoc Skip q) (seq_assoc Skip r))
| (While e q) =>
SmartSeq p (While e (seq_assoc Skip q))
| (Call rtyp name args) =>
SmartSeq p (Call
(dtcase rtyp of
| NONE => NONE
| SOME (rv , NONE) => SOME (rv, NONE)
| SOME (rv , (SOME (eid , ev , ep))) =>
SOME (rv , (SOME (eid , ev , (seq_assoc Skip ep)))))
name args)
| (DecCall v s e es q) => SmartSeq p (DecCall v s e es (seq_assoc Skip q))
| (Annot _ _) => p
| q => SmartSeq p q
Proof
rpt strip_tac >>
CONV_TAC(RAND_CONV patternMatchesLib.PMATCH_ELIM_CONV) >>
every_case_tac >> fs[seq_assoc_def]
QED
Theorem ret_to_tail_pmatch:
!p.
ret_to_tail p =
case p of
| Skip => Skip
| (Dec v e q) => Dec v e (ret_to_tail q)
| (Seq q r) => seq_call_ret (Seq (ret_to_tail q) (ret_to_tail r))
| (If e q r) =>
If e (ret_to_tail q) (ret_to_tail r)
| (While e q) =>
While e (ret_to_tail q)
| (Call rtyp name args) =>
Call
(dtcase rtyp of
| NONE => NONE
| SOME (rv , NONE) => SOME (rv , NONE)
| SOME (rv , (SOME (eid , ev , ep))) =>
SOME (rv , (SOME (eid , ev , (ret_to_tail ep)))))
name args
| (DecCall v s e es q) => DecCall v s e es (ret_to_tail q)
| p => p
Proof
rpt strip_tac >>
CONV_TAC(RAND_CONV patternMatchesLib.PMATCH_ELIM_CONV) >>
every_case_tac >> fs[ret_to_tail_def]
QED
val _ = export_theory();