author | wenzelm |
Sun, 13 Dec 2020 13:16:07 +0100 | |
changeset 72893 | fbdadf5760c2 |
parent 69505 | cc2d676d5395 |
child 80914 | d97fdabd9e2b |
permissions | -rw-r--r-- |
58889 | 1 |
section "Small-Step Semantics of Commands" |
43141 | 2 |
|
3 |
theory Small_Step imports Star Big_Step begin |
|
4 |
||
5 |
subsection "The transition relation" |
|
6 |
||
7 |
inductive |
|
8 |
small_step :: "com * state \<Rightarrow> com * state \<Rightarrow> bool" (infix "\<rightarrow>" 55) |
|
9 |
where |
|
10 |
Assign: "(x ::= a, s) \<rightarrow> (SKIP, s(x := aval a s))" | |
|
11 |
||
53015
a1119cf551e8
standardized symbols via "isabelle update_sub_sup", excluding src/Pure and src/Tools/WWW_Find;
wenzelm
parents:
52046
diff
changeset
|
12 |
Seq1: "(SKIP;;c\<^sub>2,s) \<rightarrow> (c\<^sub>2,s)" | |
a1119cf551e8
standardized symbols via "isabelle update_sub_sup", excluding src/Pure and src/Tools/WWW_Find;
wenzelm
parents:
52046
diff
changeset
|
13 |
Seq2: "(c\<^sub>1,s) \<rightarrow> (c\<^sub>1',s') \<Longrightarrow> (c\<^sub>1;;c\<^sub>2,s) \<rightarrow> (c\<^sub>1';;c\<^sub>2,s')" | |
43141 | 14 |
|
53015
a1119cf551e8
standardized symbols via "isabelle update_sub_sup", excluding src/Pure and src/Tools/WWW_Find;
wenzelm
parents:
52046
diff
changeset
|
15 |
IfTrue: "bval b s \<Longrightarrow> (IF b THEN c\<^sub>1 ELSE c\<^sub>2,s) \<rightarrow> (c\<^sub>1,s)" | |
a1119cf551e8
standardized symbols via "isabelle update_sub_sup", excluding src/Pure and src/Tools/WWW_Find;
wenzelm
parents:
52046
diff
changeset
|
16 |
IfFalse: "\<not>bval b s \<Longrightarrow> (IF b THEN c\<^sub>1 ELSE c\<^sub>2,s) \<rightarrow> (c\<^sub>2,s)" | |
43141 | 17 |
|
50054 | 18 |
While: "(WHILE b DO c,s) \<rightarrow> |
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
19 |
(IF b THEN c;; WHILE b DO c ELSE SKIP,s)" |
43141 | 20 |
|
21 |
||
50054 | 22 |
abbreviation |
23 |
small_steps :: "com * state \<Rightarrow> com * state \<Rightarrow> bool" (infix "\<rightarrow>*" 55) |
|
43141 | 24 |
where "x \<rightarrow>* y == star small_step x y" |
25 |
||
67406 | 26 |
subsection\<open>Executability\<close> |
43141 | 27 |
|
28 |
code_pred small_step . |
|
29 |
||
30 |
values "{(c',map t [''x'',''y'',''z'']) |c' t. |
|
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
31 |
(''x'' ::= V ''z'';; ''y'' ::= V ''x'', |
44036 | 32 |
<''x'' := 3, ''y'' := 7, ''z'' := 5>) \<rightarrow>* (c',t)}" |
43141 | 33 |
|
34 |
||
67406 | 35 |
subsection\<open>Proof infrastructure\<close> |
43141 | 36 |
|
67406 | 37 |
subsubsection\<open>Induction rules\<close> |
43141 | 38 |
|
67406 | 39 |
text\<open>The default induction rule @{thm[source] small_step.induct} only works |
69505 | 40 |
for lemmas of the form \<open>a \<rightarrow> b \<Longrightarrow> \<dots>\<close> where \<open>a\<close> and \<open>b\<close> are |
41 |
not already pairs \<open>(DUMMY,DUMMY)\<close>. We can generate a suitable variant |
|
43141 | 42 |
of @{thm[source] small_step.induct} for pairs by ``splitting'' the arguments |
69505 | 43 |
\<open>\<rightarrow>\<close> into pairs:\<close> |
43141 | 44 |
lemmas small_step_induct = small_step.induct[split_format(complete)] |
45 |
||
46 |
||
67406 | 47 |
subsubsection\<open>Proof automation\<close> |
43141 | 48 |
|
49 |
declare small_step.intros[simp,intro] |
|
50 |
||
67406 | 51 |
text\<open>Rule inversion:\<close> |
43141 | 52 |
|
53 |
inductive_cases SkipE[elim!]: "(SKIP,s) \<rightarrow> ct" |
|
54 |
thm SkipE |
|
55 |
inductive_cases AssignE[elim!]: "(x::=a,s) \<rightarrow> ct" |
|
56 |
thm AssignE |
|
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
57 |
inductive_cases SeqE[elim]: "(c1;;c2,s) \<rightarrow> ct" |
47818 | 58 |
thm SeqE |
43141 | 59 |
inductive_cases IfE[elim!]: "(IF b THEN c1 ELSE c2,s) \<rightarrow> ct" |
60 |
inductive_cases WhileE[elim]: "(WHILE b DO c, s) \<rightarrow> ct" |
|
61 |
||
62 |
||
67406 | 63 |
text\<open>A simple property:\<close> |
43141 | 64 |
lemma deterministic: |
65 |
"cs \<rightarrow> cs' \<Longrightarrow> cs \<rightarrow> cs'' \<Longrightarrow> cs'' = cs'" |
|
45015 | 66 |
apply(induction arbitrary: cs'' rule: small_step.induct) |
43141 | 67 |
apply blast+ |
68 |
done |
|
69 |
||
70 |
||
71 |
subsection "Equivalence with big-step semantics" |
|
72 |
||
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
73 |
lemma star_seq2: "(c1,s) \<rightarrow>* (c1',s') \<Longrightarrow> (c1;;c2,s) \<rightarrow>* (c1';;c2,s')" |
45015 | 74 |
proof(induction rule: star_induct) |
43141 | 75 |
case refl thus ?case by simp |
76 |
next |
|
77 |
case step |
|
47818 | 78 |
thus ?case by (metis Seq2 star.step) |
43141 | 79 |
qed |
80 |
||
47818 | 81 |
lemma seq_comp: |
43141 | 82 |
"\<lbrakk> (c1,s1) \<rightarrow>* (SKIP,s2); (c2,s2) \<rightarrow>* (SKIP,s3) \<rbrakk> |
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
83 |
\<Longrightarrow> (c1;;c2, s1) \<rightarrow>* (SKIP,s3)" |
47818 | 84 |
by(blast intro: star.step star_seq2 star_trans) |
43141 | 85 |
|
67406 | 86 |
text\<open>The following proof corresponds to one on the board where one would |
69505 | 87 |
show chains of \<open>\<rightarrow>\<close> and \<open>\<rightarrow>*\<close> steps.\<close> |
43141 | 88 |
|
89 |
lemma big_to_small: |
|
90 |
"cs \<Rightarrow> t \<Longrightarrow> cs \<rightarrow>* (SKIP,t)" |
|
45015 | 91 |
proof (induction rule: big_step.induct) |
43141 | 92 |
fix s show "(SKIP,s) \<rightarrow>* (SKIP,s)" by simp |
93 |
next |
|
94 |
fix x a s show "(x ::= a,s) \<rightarrow>* (SKIP, s(x := aval a s))" by auto |
|
95 |
next |
|
96 |
fix c1 c2 s1 s2 s3 |
|
97 |
assume "(c1,s1) \<rightarrow>* (SKIP,s2)" and "(c2,s2) \<rightarrow>* (SKIP,s3)" |
|
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
98 |
thus "(c1;;c2, s1) \<rightarrow>* (SKIP,s3)" by (rule seq_comp) |
43141 | 99 |
next |
100 |
fix s::state and b c0 c1 t |
|
101 |
assume "bval b s" |
|
102 |
hence "(IF b THEN c0 ELSE c1,s) \<rightarrow> (c0,s)" by simp |
|
45218 | 103 |
moreover assume "(c0,s) \<rightarrow>* (SKIP,t)" |
104 |
ultimately |
|
105 |
show "(IF b THEN c0 ELSE c1,s) \<rightarrow>* (SKIP,t)" by (metis star.simps) |
|
43141 | 106 |
next |
107 |
fix s::state and b c0 c1 t |
|
108 |
assume "\<not>bval b s" |
|
109 |
hence "(IF b THEN c0 ELSE c1,s) \<rightarrow> (c1,s)" by simp |
|
45218 | 110 |
moreover assume "(c1,s) \<rightarrow>* (SKIP,t)" |
111 |
ultimately |
|
112 |
show "(IF b THEN c0 ELSE c1,s) \<rightarrow>* (SKIP,t)" by (metis star.simps) |
|
43141 | 113 |
next |
114 |
fix b c and s::state |
|
115 |
assume b: "\<not>bval b s" |
|
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
116 |
let ?if = "IF b THEN c;; WHILE b DO c ELSE SKIP" |
43141 | 117 |
have "(WHILE b DO c,s) \<rightarrow> (?if, s)" by blast |
45218 | 118 |
moreover have "(?if,s) \<rightarrow> (SKIP, s)" by (simp add: b) |
45265 | 119 |
ultimately show "(WHILE b DO c,s) \<rightarrow>* (SKIP,s)" by(metis star.refl star.step) |
43141 | 120 |
next |
121 |
fix b c s s' t |
|
122 |
let ?w = "WHILE b DO c" |
|
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
123 |
let ?if = "IF b THEN c;; ?w ELSE SKIP" |
43141 | 124 |
assume w: "(?w,s') \<rightarrow>* (SKIP,t)" |
125 |
assume c: "(c,s) \<rightarrow>* (SKIP,s')" |
|
126 |
assume b: "bval b s" |
|
127 |
have "(?w,s) \<rightarrow> (?if, s)" by blast |
|
52046
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
128 |
moreover have "(?if, s) \<rightarrow> (c;; ?w, s)" by (simp add: b) |
bc01725d7918
replaced `;' by `;;' to disambiguate syntax; unexpected slight increase in build time
nipkow
parents:
50054
diff
changeset
|
129 |
moreover have "(c;; ?w,s) \<rightarrow>* (SKIP,t)" by(rule seq_comp[OF c w]) |
45218 | 130 |
ultimately show "(WHILE b DO c,s) \<rightarrow>* (SKIP,t)" by (metis star.simps) |
43141 | 131 |
qed |
132 |
||
67406 | 133 |
text\<open>Each case of the induction can be proved automatically:\<close> |
43141 | 134 |
lemma "cs \<Rightarrow> t \<Longrightarrow> cs \<rightarrow>* (SKIP,t)" |
45015 | 135 |
proof (induction rule: big_step.induct) |
43141 | 136 |
case Skip show ?case by blast |
137 |
next |
|
138 |
case Assign show ?case by blast |
|
139 |
next |
|
47818 | 140 |
case Seq thus ?case by (blast intro: seq_comp) |
43141 | 141 |
next |
45265 | 142 |
case IfTrue thus ?case by (blast intro: star.step) |
43141 | 143 |
next |
45265 | 144 |
case IfFalse thus ?case by (blast intro: star.step) |
43141 | 145 |
next |
146 |
case WhileFalse thus ?case |
|
45265 | 147 |
by (metis star.step star_step1 small_step.IfFalse small_step.While) |
43141 | 148 |
next |
149 |
case WhileTrue |
|
150 |
thus ?case |
|
47818 | 151 |
by(metis While seq_comp small_step.IfTrue star.step[of small_step]) |
43141 | 152 |
qed |
153 |
||
154 |
lemma small1_big_continue: |
|
155 |
"cs \<rightarrow> cs' \<Longrightarrow> cs' \<Rightarrow> t \<Longrightarrow> cs \<Rightarrow> t" |
|
45015 | 156 |
apply (induction arbitrary: t rule: small_step.induct) |
43141 | 157 |
apply auto |
158 |
done |
|
159 |
||
55834 | 160 |
lemma small_to_big: |
161 |
"cs \<rightarrow>* (SKIP,t) \<Longrightarrow> cs \<Rightarrow> t" |
|
55836 | 162 |
apply (induction cs "(SKIP,t)" rule: star.induct) |
43141 | 163 |
apply (auto intro: small1_big_continue) |
164 |
done |
|
165 |
||
67406 | 166 |
text \<open> |
43141 | 167 |
Finally, the equivalence theorem: |
67406 | 168 |
\<close> |
43141 | 169 |
theorem big_iff_small: |
170 |
"cs \<Rightarrow> t = cs \<rightarrow>* (SKIP,t)" |
|
171 |
by(metis big_to_small small_to_big) |
|
172 |
||
173 |
||
174 |
subsection "Final configurations and infinite reductions" |
|
175 |
||
67613 | 176 |
definition "final cs \<longleftrightarrow> \<not>(\<exists>cs'. cs \<rightarrow> cs')" |
43141 | 177 |
|
178 |
lemma finalD: "final (c,s) \<Longrightarrow> c = SKIP" |
|
179 |
apply(simp add: final_def) |
|
45015 | 180 |
apply(induction c) |
43141 | 181 |
apply blast+ |
182 |
done |
|
183 |
||
184 |
lemma final_iff_SKIP: "final (c,s) = (c = SKIP)" |
|
185 |
by (metis SkipE finalD final_def) |
|
186 |
||
69505 | 187 |
text\<open>Now we can show that \<open>\<Rightarrow>\<close> yields a final state iff \<open>\<rightarrow>\<close> |
67406 | 188 |
terminates:\<close> |
43141 | 189 |
|
190 |
lemma big_iff_small_termination: |
|
67613 | 191 |
"(\<exists>t. cs \<Rightarrow> t) \<longleftrightarrow> (\<exists>cs'. cs \<rightarrow>* cs' \<and> final cs')" |
43141 | 192 |
by(simp add: big_iff_small final_iff_SKIP) |
193 |
||
67406 | 194 |
text\<open>This is the same as saying that the absence of a big step result is |
43141 | 195 |
equivalent with absence of a terminating small step sequence, i.e.\ with |
69505 | 196 |
nontermination. Since \<open>\<rightarrow>\<close> is determininistic, there is no difference |
67406 | 197 |
between may and must terminate.\<close> |
43141 | 198 |
|
199 |
end |