lh-l4v/proof/refine/Refine.thy

810 lines
35 KiB
Plaintext

(*
* Copyright 2014, General Dynamics C4 Systems
*
* This software may be distributed and modified according to the terms of
* the GNU General Public License version 2. Note that NO WARRANTY is provided.
* See "LICENSE_GPLv2.txt" for details.
*
* @TAG(GD_GPL)
*)
(*
The main theorem
*)
theory Refine
imports
KernelInit_R
InitLemmas
PageTableDuplicates
begin
text {* User memory content is the same on both levels *}
lemma typ_at_UserDataI:
"\<lbrakk> typ_at (AArch (AIntData sz)) p s; pspace_relation (kheap s) (ksPSpace s');
pspace_aligned' s'; pspace_distinct' s'; n < 2 ^ (pageBitsForSize sz - pageBits) \<rbrakk>
\<Longrightarrow> typ_at' UserDataT (p + n * 2 ^ pageBits) s'"
apply (clarsimp simp add: obj_at_def a_type_def)
apply (simp split: Structures_A.kernel_object.split_asm
split_if_asm arch_kernel_obj.split_asm)
apply (drule(1) pspace_relation_absD)
apply (clarsimp)
apply (drule_tac x = "p + n * 2 ^ pageBits" in spec)
apply (drule_tac x = "\<lambda>_. op = KOUserData" in spec)
apply (clarsimp simp: obj_at'_def typ_at'_def ko_wp_at'_def
projectKOs)
apply (rule exI [where x = KOUserData])
apply (drule mp)
apply (rule exI [where x = n])
apply simp
apply (clarsimp simp: pspace_aligned'_def)
apply (drule (1) bspec [OF _ domI])
apply (clarsimp simp: objBits_simps)
apply (clarsimp dest!: pspace_distinctD' simp: objBits_simps)
done
lemma typ_at_AIntDataI:
"\<lbrakk> typ_at' UserDataT (p && ~~ mask pageBits) s'; pspace_relation (kheap s) (ksPSpace s'); pspace_aligned s \<rbrakk>
\<Longrightarrow> \<exists>sz. typ_at (AArch (AIntData sz)) (p && ~~ mask (pageBitsForSize sz)) s"
apply (clarsimp simp: obj_at'_def typ_at'_def ko_wp_at'_def
projectKOs)
apply (case_tac ko, simp_all)
apply (frule (1) in_related_pspace_dom)
apply (clarsimp simp: pspace_dom_def)
apply (clarsimp simp: pspace_relation_def dom_def)
apply (erule allE, erule impE, blast)
apply clarsimp
apply (drule (1) bspec)
apply clarsimp
apply (subst mask_lower_twice [where n = pageBits, OF pbfs_atleast_pageBits, symmetric])
apply (clarsimp simp: obj_relation_cuts_def2 pte_relation_def
cte_relation_def other_obj_relation_def
pde_relation_def
split: Structures_A.kernel_object.split_asm
split_if_asm arch_kernel_obj.split_asm)
apply (rule_tac x = vmpage_size in exI)
apply (subst conjunct2 [OF is_aligned_add_helper])
apply (drule (1) pspace_alignedD)
apply simp
apply (erule word_less_power_trans2 [OF _ pbfs_atleast_pageBits])
apply (case_tac vmpage_size, simp_all add: word_bits_conv)[1]
apply (simp add: obj_at_def a_type_def)
done
lemma pointerInUserData_relation:
"\<lbrakk> (s,s') \<in> state_relation; valid_state' s'; valid_state s\<rbrakk>
\<Longrightarrow> pointerInUserData p s' = in_user_frame p s"
apply (simp add: pointerInUserData_def in_user_frame_def)
apply (rule iffI)
apply (erule typ_at_AIntDataI, (clarsimp simp: valid_state_def)+)[1]
apply clarsimp
apply (drule_tac sz = sz and
n = "(p && mask (pageBitsForSize sz)) >> pageBits"
in typ_at_UserDataI [where s = s and s' = s'])
apply (fastforce simp: valid_state'_def)+
apply (rule shiftr_less_t2n')
apply (simp add: pbfs_atleast_pageBits mask_twice)
apply (case_tac sz, simp_all)[1]
apply (subgoal_tac "(p && ~~ mask (pageBitsForSize sz)) + (p && mask (pageBitsForSize sz) >> pageBits) * 2 ^ pageBits = (p && ~~ mask pageBits)")
apply simp
apply (subst mult.commute)
apply (subst shiftl_t2n [symmetric])
apply (simp add: shiftr_shiftl1)
apply (subst mask_out_add_aligned)
apply (rule is_aligned_neg_mask)
apply (simp add: pbfs_atleast_pageBits)
apply (subst add.commute)
apply (simp add: word_plus_and_or_coroll2)
done
lemma user_mem_relation:
"\<lbrakk>(s,s') \<in> state_relation; valid_state' s'; valid_state s\<rbrakk>
\<Longrightarrow> user_mem' s' = user_mem s"
apply (rule ext)
apply (clarsimp simp: user_mem_def user_mem'_def pointerInUserData_relation)
apply (simp add: state_relation_def)
done
text {* The top-level invariance *}
lemma set_thread_state_sched_act:
"\<lbrace>(\<lambda>s. runnable state) and (\<lambda>s. P (scheduler_action s))\<rbrace>
set_thread_state thread state
\<lbrace>\<lambda>rs s. P (scheduler_action (s::det_state))\<rbrace>"
apply (simp add: set_thread_state_def)
apply wp
apply (simp add: set_thread_state_ext_def)
apply wp
apply (rule hoare_pre_cont)
apply (rule_tac Q="\<lambda>rv. (\<lambda>s. runnable ts) and (\<lambda>s. P (scheduler_action s))"
in hoare_strengthen_post)
apply wp
apply force
apply (wp gts_st_tcb_at)
apply (rule_tac Q="\<lambda>rv. st_tcb_at (op = state) thread and (\<lambda>s. runnable state) and (\<lambda>s. P (scheduler_action s))" in hoare_strengthen_post)
apply (simp add: st_tcb_at_def)
apply (wp obj_set_prop_at)
apply (force simp: st_tcb_at_def obj_at_def)
apply wp
apply clarsimp
done
lemma activate_thread_sched_act:
"\<lbrace>ct_in_state activatable and (\<lambda>s. P (scheduler_action s))\<rbrace>
activate_thread
\<lbrace>\<lambda>rs s. P (scheduler_action (s::det_state))\<rbrace>"
by (simp add: activate_thread_def set_thread_state_def arch_activate_idle_thread_def | wp set_thread_state_sched_act gts_wp | wpc)+
lemma schedule_sched_act_rct[wp]:
"\<lbrace>\<top>\<rbrace> Schedule_A.schedule
\<lbrace>\<lambda>rs (s::det_state). scheduler_action s = resume_cur_thread\<rbrace>"
apply (simp add: Schedule_A.schedule_def)
apply (wp | wpc | simp add: set_scheduler_action_def)+
done
lemma call_kernel_sched_act_rct[wp]:
"\<lbrace>einvs and (\<lambda>s. e \<noteq> Interrupt \<longrightarrow> ct_running s)
and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>
call_kernel e
\<lbrace>\<lambda>rs (s::det_state). scheduler_action s = resume_cur_thread\<rbrace>"
apply (simp add: call_kernel_def)
apply (wp activate_thread_sched_act | simp)+
apply (clarsimp simp: active_from_running)
done
lemma kernel_entry_invs:
"\<lbrace>einvs and (\<lambda>s. e \<noteq> Interrupt \<longrightarrow> ct_running s)
and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>
kernel_entry e us
\<lbrace>\<lambda>rv. einvs and (\<lambda>s. ct_running s \<or> ct_idle s)
and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>"
apply (rule_tac Q="\<lambda>rv. invs and (\<lambda>s. ct_running s \<or> ct_idle s) and valid_sched and valid_list and (\<lambda>s. scheduler_action s = resume_cur_thread)" in hoare_strengthen_post)
apply (simp add: kernel_entry_def)
apply (wp akernel_invs_det_ext call_kernel_valid_sched thread_set_invs_trivial
thread_set_ct_running thread_set_not_state_valid_sched static_imp_wp
| clarsimp simp add: tcb_cap_cases_def)+
done
definition
"full_invs \<equiv> {((tc, s :: det_ext state), m, e). einvs s \<and>
(ct_running s \<or> ct_idle s) \<and>
(m = KernelMode \<longrightarrow> e \<noteq> None) \<and>
(m = UserMode \<longrightarrow> ct_running s) \<and>
(m = IdleMode \<longrightarrow> ct_idle s) \<and>
(e \<noteq> None \<and> e \<noteq> Some Interrupt \<longrightarrow> ct_running s) \<and>
(scheduler_action s = resume_cur_thread)}"
lemma do_user_op_valid_list:"\<lbrace>valid_list\<rbrace> do_user_op f tc \<lbrace>\<lambda>_. valid_list\<rbrace>"
unfolding do_user_op_def
apply (wp select_wp | simp add: split_def)+
done
lemma do_user_op_valid_sched:"\<lbrace>valid_sched\<rbrace> do_user_op f tc \<lbrace>\<lambda>_. valid_sched\<rbrace>"
unfolding do_user_op_def
apply (wp select_wp | simp add: split_def)+
done
lemma do_user_op_sched_act:
"\<lbrace>\<lambda>s. P (scheduler_action s)\<rbrace> do_user_op f tc \<lbrace>\<lambda>_ s. P (scheduler_action s)\<rbrace>"
unfolding do_user_op_def
apply (wp select_wp | simp add: split_def)+
done
lemma do_user_op_invs2:
"\<lbrace>einvs and ct_running and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>
do_user_op f tc
\<lbrace>\<lambda>_. (einvs and ct_running and (\<lambda>s. scheduler_action s = resume_cur_thread))\<rbrace>"
apply (rule_tac Q="\<lambda>_. valid_list and valid_sched and
(\<lambda>s. scheduler_action s = resume_cur_thread) and (invs and ct_running)"
in hoare_strengthen_post)
apply (wp do_user_op_valid_list do_user_op_valid_sched do_user_op_sched_act
do_user_op_invs |simp | force)+
done
(* FIXME: move *)
lemma thread_set_ct_idle:
"(\<And>tcb. tcb_state (f tcb) = tcb_state tcb) \<Longrightarrow>
\<lbrace>ct_idle\<rbrace> thread_set f t \<lbrace>\<lambda>rv. ct_idle\<rbrace>"
by (clarsimp simp: thread_set_def set_object_def get_tcb_def
ct_in_state_def st_tcb_at_def obj_at_def
| wp)+
lemma ct_running_irq_state_independent[intro!, simp]:
"ct_running (s \<lparr>machine_state := machine_state s \<lparr>irq_state := f (irq_state (machine_state s)) \<rparr> \<rparr>)
= ct_running s"
by (simp add: ct_in_state_def)
lemma ct_idle_irq_state_independent[intro!, simp]:
"ct_idle (s \<lparr>machine_state := machine_state s \<lparr>irq_state := f (irq_state (machine_state s)) \<rparr> \<rparr>)
= ct_idle s"
by (simp add: ct_in_state_def)
(* FIXME: move *)
lemma check_active_irq_invs:
"\<lbrace>invs and (ct_running or ct_idle) and einvs and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>
check_active_irq
\<lbrace>\<lambda>_. invs and (ct_running or ct_idle) and valid_list and valid_sched and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>"
by (wp | simp add: check_active_irq_def | force)+
(* FIXME: move *)
lemma check_active_irq_invs_just_running:
"\<lbrace>invs and ct_running and einvs and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>
check_active_irq
\<lbrace>\<lambda>_. invs and ct_running and valid_list and valid_sched and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>"
by (wp | simp add: check_active_irq_def | force)+
lemma check_active_irq_invs_just_idle:
"\<lbrace>invs and ct_idle and einvs and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>
check_active_irq
\<lbrace>\<lambda>_. invs and ct_idle and valid_list and valid_sched and (\<lambda>s. scheduler_action s = resume_cur_thread)\<rbrace>"
by (wp | simp add: check_active_irq_def | force)+
lemmas ext_init_def = ext_init_det_ext_ext_def ext_init_unit_def
lemma valid_list_init[simp]:
"valid_list init_A_st"
by (simp add: valid_list_2_def init_A_st_def ext_init_def init_cdt_def)
lemmas valid_list_inits[simp] = valid_list_init[simplified]
lemma valid_sched_init[simp]:
"valid_sched init_A_st"
apply (simp add: valid_sched_def init_A_st_def ext_init_def)
apply (clarsimp simp: valid_etcbs_def init_kheap_def st_tcb_at_kh_def obj_at_kh_def
obj_at_def is_etcb_at_def idle_thread_ptr_def init_globals_frame_def
init_global_pd_def valid_queues_2_def ct_not_in_q_def not_queued_def
valid_sched_action_def is_activatable_def
ct_in_cur_domain_2_def valid_blocked_2_def valid_idle_etcb_def etcb_at'_def default_etcb_def)
done
lemma akernel_invariant:
"ADT_A uop \<Turnstile> full_invs"
unfolding full_invs_def
apply (rule invariantI)
apply (clarsimp simp: ADT_A_def subset_iff)
apply (frule bspec[OF akernel_init_invs])
apply (simp add: Let_def Init_A_def)
apply (simp add: init_A_st_def ext_init_def)
apply (clarsimp simp: ADT_A_def global_automaton_def)
apply (elim disjE)
apply ((clarsimp simp: kernel_call_A_def
| drule use_valid[OF _ kernel_entry_invs])+)[2]
apply ((clarsimp simp: do_user_op_A_def monad_to_transition_def
check_active_irq_A_def
| drule use_valid[OF _ do_user_op_invs2]
| drule use_valid[OF _ check_active_irq_invs_just_running])+)[2]
apply ((clarsimp simp add: check_active_irq_A_def
| drule use_valid[OF _ check_active_irq_invs])+)[1]
apply (clarsimp simp: ct_in_state_def st_tcb_at_def obj_at_def)
apply ((clarsimp simp add: do_user_op_A_def check_active_irq_A_def
| drule use_valid[OF _ do_user_op_invs2]
| drule use_valid[OF _ check_active_irq_invs_just_running])+)[1]
apply (clarsimp simp: ct_in_state_def st_tcb_at_def obj_at_def)
apply (clarsimp simp: ct_in_state_def st_tcb_at_def obj_at_def)
apply ((clarsimp simp add: check_active_irq_A_def
| drule use_valid[OF _ check_active_irq_invs])+)[1]
apply ((clarsimp simp add: check_active_irq_A_def
| drule use_valid[OF _ check_active_irq_invs_just_idle])+)[1]
apply ((clarsimp simp add: check_active_irq_A_def
| drule use_valid[OF _ check_active_irq_invs])+)[1]
done
lemma ckernel_invs:
"\<lbrace>invs' and (\<lambda>s. vs_valid_duplicates' (ksPSpace s)) and
(\<lambda>s. e \<noteq> Interrupt \<longrightarrow> ct_running' s) and
(\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>
callKernel e
\<lbrace>\<lambda>rs. (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)
and (invs' and (ct_running' or ct_idle'))\<rbrace>"
apply (simp add: callKernel_def)
apply (rule hoare_pre)
apply (wp activate_invs' activate_sch_act schedule_sch
schedule_sch_act_simple he_invs' schedule_invs'
| simp add: no_irq_getActiveIRQ)+
done
lemma doMachineOp_sch_act_simple:
"\<lbrace>sch_act_simple\<rbrace> doMachineOp f \<lbrace>\<lambda>_. sch_act_simple\<rbrace>"
by (simp add: sch_act_simple_def) wp
lemma doMachineOp_ct_running':
"\<lbrace>ct_running'\<rbrace> doMachineOp f \<lbrace>\<lambda>_. ct_running'\<rbrace>"
apply (simp add: ct_in_state'_def doMachineOp_def split_def)
apply wp
apply (simp add: st_tcb_at'_def o_def)
done
lemma kernelEntry_invs':
"\<lbrace> invs' and (\<lambda>s. e \<noteq> Interrupt \<longrightarrow> ct_running' s) and
(\<lambda>s. vs_valid_duplicates' (ksPSpace s)) and
(\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>
kernelEntry e tc
\<lbrace>\<lambda>rs. (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread) and
(invs' and (ct_running' or ct_idle')) and
(\<lambda>s. vs_valid_duplicates' (ksPSpace s))\<rbrace>"
apply (simp add: kernelEntry_def)
apply (wp ckernel_invs callKernel_valid_duplicates'
threadSet_invs_trivial threadSet_ct_running' select_wp
VSpace_R.doMachineOp_cur_tcb' TcbAcc_R.dmo_invs'
doMachineOp_ct_running' doMachineOp_sch_act_simple static_imp_wp
| clarsimp simp add: user_memory_update_def no_irq_def tcb_at_invs')+
done
lemma absKState_correct':
"\<lbrakk>einvs s; invs' s'; (s,s') \<in> state_relation\<rbrakk>
\<Longrightarrow> absKState s' = s"
using assms
apply (intro state.equality, simp_all add: absKState_def)
apply (rule absHeap_correct)
apply (clarsimp simp: valid_state_def valid_pspace_def)+
apply (clarsimp dest!: state_relationD)
apply (rule absCDT_correct)
apply (clarsimp simp: valid_state_def valid_pspace_def
valid_state'_def valid_pspace'_def)+
apply (rule absIsOriginalCap_correct, clarsimp+)
apply (simp add: state_relation_def)
apply (simp add: state_relation_def)
apply (simp add: state_relation_def)
apply (rule absInterruptIRQNode_correct, simp add: state_relation_def)
apply (rule absInterruptStates_correct, simp add: state_relation_def)
apply (erule absArchState_correct)
apply (rule absExst_correct, simp, assumption+)
done
lemma ptable_rights_imp_UserData:
assumes invs: "einvs s" and invs': "invs' s'"
assumes rel: "(s,s') : state_relation"
assumes rights: "ptable_rights t (absKState s') x \<noteq> {}"
assumes trans:
"ptable_lift t (absKState s') x = Some (Platform.addrFromPPtr y)"
shows "pointerInUserData y s'"
proof -
from invs invs' rel have [simp]: "absKState s' = s"
by - (rule absKState_correct', simp_all)
from invs have valid: "valid_state s" by auto
from invs' have valid': "valid_state' s'" by auto
have "in_user_frame y s"
by (rule ptable_rights_imp_user_frame[OF valid rights[simplified]
trans[simplified]])
thus ?thesis by (simp add: pointerInUserData_relation[OF rel valid' valid])
qed
definition
"ex_abs G \<equiv> \<lambda>s'. \<exists>s. ((s :: (det_ext) state),s') \<in> state_relation \<and> G s"
lemma doUserOp_invs':
"\<lbrace>invs' and ex_abs einvs and
(\<lambda>s. ksSchedulerAction s = ResumeCurrentThread) and ct_running'\<rbrace>
doUserOp f tc
\<lbrace>\<lambda>_. invs' and
(\<lambda>s. ksSchedulerAction s = ResumeCurrentThread) and ct_running'\<rbrace>"
apply (simp add: doUserOp_def split_def ex_abs_def)
apply (wp dmo_invs')
apply (clarsimp simp add: no_irq_modify user_memory_update_def)
apply (wp doMachineOp_ct_running' doMachineOp_sch_act select_wp)
apply (clarsimp simp: user_memory_update_def simpler_modify_def
restrict_map_def
split: option.splits)
apply (erule ptable_rights_imp_UserData[rotated 2], auto)
done
lemma doUserOp_valid_duplicates':
"\<lbrace>\<lambda>s. vs_valid_duplicates' (ksPSpace s)\<rbrace>
doUserOp f tc
\<lbrace>\<lambda>_ s. vs_valid_duplicates' (ksPSpace s)\<rbrace>"
apply (simp add: doUserOp_def split_def)
apply (wp dmo_invs' select_wp)
apply clarsimp
done
text {* The top-level correspondence *}
lemma kernel_corres:
"corres dc (einvs and (\<lambda>s. event \<noteq> Interrupt \<longrightarrow> ct_running s) and
(\<lambda>s. scheduler_action s = resume_cur_thread))
(invs' and (\<lambda>s. event \<noteq> Interrupt \<longrightarrow> ct_running' s) and
(\<lambda>s. ksSchedulerAction s = ResumeCurrentThread) and
(\<lambda>s. vs_valid_duplicates' (ksPSpace s)))
(call_kernel event) (callKernel event)"
apply (simp add: call_kernel_def callKernel_def)
apply (rule corres_guard_imp)
apply (rule corres_split)
prefer 2
apply (rule corres_split_handle [OF _ he_corres])
apply simp
apply (rule corres_split [OF _ corres_machine_op])
prefer 2
apply (rule corres_underlying_trivial)
apply (rule no_fail_getActiveIRQ)
apply clarsimp
apply (rule_tac x=irqa in option_corres)
apply (rule_tac P=\<top> and P'=\<top> in corres_inst)
apply (simp add: when_def)
apply (rule corres_when, simp)
apply simp
apply (rule handle_interrupt_corres)
apply simp
apply (wp hoare_drop_imps)[1]
apply simp
apply (rule_tac Q="\<lambda>irq s. invs' s \<and>
(\<forall>irq'. irq = Some irq' \<longrightarrow>
intStateIRQTable (ksInterruptState s ) irq' \<noteq>
IRQInactive)"
in hoare_post_imp)
apply simp
apply (wp doMachineOp_getActiveIRQ_IRQ_active handle_event_valid_sched | simp)+
apply (rule_tac Q="\<lambda>_. \<top>" and E="\<lambda>_. invs'" in hoare_post_impErr)
apply wp
apply (simp add: invs'_def valid_state'_def)
apply (rule corres_split [OF _ schedule_corres])
apply (rule activate_corres)
apply (wp schedule_invs' hoare_vcg_if_lift2 hoare_drop_imps |simp)+
apply (rule_tac Q="\<lambda>_. valid_sched and invs" and E="\<lambda>_. valid_sched and invs"
in hoare_post_impErr)
apply (wp handle_event_valid_sched |simp)+
apply (clarsimp simp: active_from_running)
apply (clarsimp simp: active_from_running')
done
lemma user_mem_corres:
"corres (op =) invs invs' (gets (\<lambda>x. g (user_mem x))) (gets (\<lambda>x. g (user_mem' x)))"
by (clarsimp simp add: gets_def get_def return_def bind_def
invs_def invs'_def
corres_underlying_def user_mem_relation)
lemma entry_corres:
"corres (op =) (einvs and (\<lambda>s. event \<noteq> Interrupt \<longrightarrow> ct_running s) and
(\<lambda>s. scheduler_action s = resume_cur_thread))
(invs' and (\<lambda>s. event \<noteq> Interrupt \<longrightarrow> ct_running' s) and
(\<lambda>s. ksSchedulerAction s = ResumeCurrentThread) and
(\<lambda>s. vs_valid_duplicates' (ksPSpace s)))
(kernel_entry event tc) (kernelEntry event tc)"
apply (simp add: kernel_entry_def kernelEntry_def)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ gct_corres])
apply (rule corres_split)
prefer 2
apply simp
apply (rule threadset_corresT)
apply (simp add: tcb_relation_def)
apply (clarsimp simp: tcb_cap_cases_def)
apply (clarsimp simp: tcb_cte_cases_def)
apply (simp add: exst_same_def)
apply (rule corres_split [OF _ kernel_corres])
apply (rule corres_split_eqr [OF _ gct_corres])
apply (rule threadget_corres)
apply (simp add: tcb_relation_def)
apply wp
apply (rule hoare_strengthen_post, rule akernel_invs_det_ext, simp add: invs_def cur_tcb_def)
apply (rule hoare_strengthen_post, rule ckernel_invs, simp add: invs'_def cur_tcb'_def)
apply (wp thread_set_invs_trivial thread_set_ct_running
threadSet_invs_trivial threadSet_ct_running'
select_wp thread_set_not_state_valid_sched static_imp_wp
| simp add: tcb_cap_cases_def)+
apply (clarsimp simp: invs_def cur_tcb_def)
apply clarsimp
done
lemma do_user_op_corres:
"corres (op =) (einvs and ct_running)
(invs' and (%s. ksSchedulerAction s = ResumeCurrentThread) and
ct_running')
(do_user_op f tc) (doUserOp f tc)"
apply (simp add: do_user_op_def doUserOp_def split_def)
apply (rule corres_guard_imp)
apply (rule corres_split[OF _ gct_corres])
apply (rule_tac r'="op=" and P=einvs and P'=invs' in corres_split)
prefer 2
apply (fastforce dest: absKState_correct [rotated])
apply (rule_tac r'="op=" and P=einvs and P'=invs' in corres_split)
prefer 2
apply (fastforce dest: absKState_correct [rotated])
apply (rule_tac r'="op=" and P=invs and P'=invs' in corres_split)
prefer 2
apply (rule user_mem_corres)
apply (rule_tac r'="op=" in corres_split[OF _ corres_select])
apply (rule corres_split[OF _ corres_machine_op,
where R="\<top>\<top>" and R'="\<top>\<top>"])
apply simp+
apply (rule corres_underlying_trivial)
apply (simp add: user_memory_update_def)
apply (wp | simp)+
done
lemma ct_running_related:
"\<lbrakk> (a, c) \<in> state_relation; ct_running' c \<rbrakk>
\<Longrightarrow> ct_running a"
apply (clarsimp simp: ct_in_state_def ct_in_state'_def
curthread_relation)
apply (frule(1) st_tcb_at_coerce_abstract)
apply (erule st_tcb_weakenE)
apply (case_tac st, simp_all)[1]
done
lemma ct_idle_related:
"\<lbrakk> (a, c) \<in> state_relation; ct_idle' c \<rbrakk>
\<Longrightarrow> ct_idle a"
apply (clarsimp simp: ct_in_state_def ct_in_state'_def
curthread_relation)
apply (frule(1) st_tcb_at_coerce_abstract)
apply (erule st_tcb_weakenE)
apply (case_tac st, simp_all)[1]
done
(* FIXME: move *)
lemma valid_corres_combined:
assumes "valid P f Q"
assumes "corres_underlying sr nf rr P P' f f'"
assumes "valid (\<lambda>s'. \<exists>s. (s,s')\<in>sr \<and> P s \<and> P' s') f' Q'" (is "valid ?P _ _")
shows "valid ?P f' (\<lambda>r' s'. \<exists>r s. (s,s') \<in> sr \<and> Q r s \<and> Q' r' s' \<and> rr r r')"
using assms
by (fastforce simp: valid_def corres_underlying_def split_def)
(* FIXME: move *)
lemma fw_sim_LI: "fw_sim r C A = LI A C r UNIV" by (simp add: fw_sim_def LI_def)
definition
"full_invs' \<equiv> {((tc,s),m,e). invs' s \<and> vs_valid_duplicates' (ksPSpace s) \<and>
ex_abs (einvs::det_ext state \<Rightarrow> bool) s \<and>
ksSchedulerAction s = ResumeCurrentThread \<and>
(ct_running' s \<or> ct_idle' s) \<and>
(m = KernelMode \<longrightarrow> e \<noteq> None) \<and>
(m = UserMode \<longrightarrow> ct_running' s) \<and>
(m = IdleMode \<longrightarrow> ct_idle' s) \<and>
(e \<noteq> None \<and> e \<noteq> Some Interrupt \<longrightarrow> ct_running' s)}"
lemma checkActiveIRQ_valid_duplicates':
"\<lbrace>\<lambda>s. vs_valid_duplicates' (ksPSpace s)\<rbrace>
checkActiveIRQ
\<lbrace>\<lambda>_ s. vs_valid_duplicates' (ksPSpace s)\<rbrace>"
apply (simp add: checkActiveIRQ_def)
apply wp
done
lemma check_active_irq_corres':
"corres op = \<top> \<top> (check_active_irq) (checkActiveIRQ)"
apply (simp add: check_active_irq_def checkActiveIRQ_def)
apply (rule corres_guard_imp)
apply (rule corres_split[OF _ corres_machine_op[OF corres_underlying_trivial], where R="\<lambda>_. \<top>" and R'="\<lambda>_. \<top>"])
apply simp
apply (rule no_fail_getActiveIRQ)
apply (wp | simp )+
done
lemma check_active_irq_corres:
"corres op = (invs and (ct_running or ct_idle) and einvs and (\<lambda>s. scheduler_action s = resume_cur_thread))
(invs' and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)
and (ct_running' or ct_idle')
and (\<lambda>s. vs_valid_duplicates' (ksPSpace s)))
(check_active_irq) (checkActiveIRQ)"
apply (rule corres_guard_imp)
apply (rule check_active_irq_corres', auto)
done
lemma check_active_irq_corres_just_running:
"corres op = (invs and ct_running and einvs and (\<lambda>s. scheduler_action s = resume_cur_thread))
(invs' and ct_running' and (\<lambda>s. vs_valid_duplicates' (ksPSpace s)) and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread))
(check_active_irq) (checkActiveIRQ)"
apply (rule corres_guard_imp)
apply (rule check_active_irq_corres', auto)
done
lemma check_active_irq_corres_just_idle:
"corres op = (invs and ct_idle and einvs and (\<lambda>s. scheduler_action s = resume_cur_thread))
(invs' and ct_idle' and (\<lambda>s. vs_valid_duplicates' (ksPSpace s)) and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread))
(check_active_irq) (checkActiveIRQ)"
apply (rule corres_guard_imp)
apply (rule check_active_irq_corres', auto)
done
lemma checkActiveIRQ_invs':
"\<lbrace>invs' and ex_abs invs and (ct_running' or ct_idle')
and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>
checkActiveIRQ
\<lbrace>\<lambda>_. invs' and (ct_running' or ct_idle')
and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>"
apply (simp add: checkActiveIRQ_def ex_abs_def)
apply (wp dmo_invs' | simp)+
done
lemma checkActiveIRQ_invs'_just_running:
"\<lbrace>invs' and ex_abs invs and ct_running'
and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>
checkActiveIRQ
\<lbrace>\<lambda>_. invs' and ct_running'
and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>"
apply (simp add: checkActiveIRQ_def)
apply (wp | simp)+
done
lemma checkActiveIRQ_invs'_just_idle:
"\<lbrace>invs' and ex_abs invs and ct_idle'
and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>
checkActiveIRQ
\<lbrace>\<lambda>_. invs' and ct_idle'
and (\<lambda>s. ksSchedulerAction s = ResumeCurrentThread)\<rbrace>"
apply (simp add: checkActiveIRQ_def)
apply (wp | simp)+
done
lemma sched_act_rct_related:
"\<lbrakk> (a, c) \<in> state_relation; ksSchedulerAction c = ResumeCurrentThread\<rbrakk>
\<Longrightarrow> scheduler_action a = resume_cur_thread"
by (case_tac "scheduler_action a", simp_all add: state_relation_def)
lemma ckernel_invariant:
"ADT_H a b c d e f uop \<Turnstile> full_invs'"
unfolding full_invs'_def
apply (rule invariantI)
apply (clarsimp simp add: ADT_H_def)
apply (subst conj_commute, simp)
apply (rule conjI)
apply (drule ckernel_init_valid_duplicates'[rule_format], simp)
apply (rule conjI)
apply (frule init_refinement[simplified subset_eq, THEN bspec])
apply (clarsimp simp: ex_abs_def lift_state_relation_def)
apply (frule akernel_init_invs[THEN bspec])
apply (rule_tac x = s in exI)
apply (clarsimp simp: Init_A_def)
apply (insert ckernel_init_invs[of a b c d e f])[1]
apply clarsimp
apply (frule ckernel_init_sch_norm)
apply (frule ckernel_init_ctr)
apply (fastforce simp: Init_H_def)
apply (clarsimp simp: ADT_A_def ADT_H_def global_automaton_def)
apply (erule_tac P="?a \<and> (\<exists>x. ?b x)" in disjE)
apply ((clarsimp simp: kernel_call_H_def
| drule use_valid[OF _ valid_corres_combined
[OF kernel_entry_invs entry_corres],
OF _ kernelEntry_invs'[THEN hoare_weaken_pre]]
| fastforce simp: ex_abs_def sch_act_simple_def ct_running_related sched_act_rct_related)+)[1]
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> ?d \<and> ?e" in disjE)
apply (clarsimp simp add: do_user_op_H_def monad_to_transition_def)
apply (drule use_valid)
apply (rule hoare_vcg_conj_lift)
apply (rule doUserOp_valid_duplicates')
apply (rule valid_corres_combined[OF do_user_op_invs2 corres_guard_imp2[OF do_user_op_corres]])
apply clarsimp
apply (rule doUserOp_invs'[THEN hoare_weaken_pre])
apply (fastforce simp: ex_abs_def)
apply (fastforce simp: ex_abs_def ct_running_related sched_act_rct_related)
apply (fastforce simp: ex_abs_def)
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> (\<exists>x. ?d x)" in disjE)
apply (clarsimp simp add: do_user_op_H_def monad_to_transition_def)
apply (drule use_valid)
apply (rule hoare_vcg_conj_lift)
apply (rule doUserOp_valid_duplicates')
apply (rule valid_corres_combined[OF do_user_op_invs2 corres_guard_imp2[OF do_user_op_corres]])
apply clarsimp
apply (rule doUserOp_invs'[THEN hoare_weaken_pre])
apply (fastforce simp: ex_abs_def)
apply (fastforce simp: ex_abs_def ct_running_related sched_act_rct_related)
apply (fastforce simp: ex_abs_def)
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> ?d \<and> ?e" in disjE)
apply (clarsimp simp: check_active_irq_H_def)
apply (drule use_valid)
apply (rule hoare_vcg_conj_lift)
apply (rule checkActiveIRQ_valid_duplicates')
apply (rule valid_corres_combined[OF check_active_irq_invs_just_running check_active_irq_corres_just_running])
apply (rule checkActiveIRQ_invs'_just_running[THEN hoare_weaken_pre])
apply (fastforce simp: ex_abs_def)
apply (fastforce simp: ex_abs_def ct_running_related sched_act_rct_related)
apply (fastforce simp: ex_abs_def)
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> ?d \<and> ?e" in disjE)
apply (clarsimp simp: check_active_irq_H_def)
apply (drule use_valid)
apply (rule hoare_vcg_conj_lift)
apply (rule checkActiveIRQ_valid_duplicates')
apply (rule valid_corres_combined[OF check_active_irq_invs_just_idle check_active_irq_corres_just_idle])
apply (rule checkActiveIRQ_invs'_just_idle[THEN hoare_weaken_pre])
apply clarsimp
apply (fastforce simp: ex_abs_def)
apply (fastforce simp: ex_abs_def ct_idle_related sched_act_rct_related)
apply (fastforce simp: ex_abs_def)
apply (clarsimp simp: check_active_irq_H_def)
apply (drule use_valid)
apply (rule hoare_vcg_conj_lift)
apply (rule checkActiveIRQ_valid_duplicates')
apply (rule valid_corres_combined[OF check_active_irq_invs check_active_irq_corres])
apply (rule checkActiveIRQ_invs'[THEN hoare_weaken_pre])
apply clarsimp
apply (fastforce simp: ex_abs_def)
apply (fastforce simp: ex_abs_def ct_running_related ct_idle_related sched_act_rct_related)
apply (fastforce simp: ex_abs_def)
done
text {* The top-level theorem *}
lemma fw_sim_A_H:
"LI (ADT_A uop)
(ADT_H a b c d e f uop)
(lift_state_relation state_relation)
(full_invs \<times> full_invs')"
apply (unfold LI_def full_invs_def full_invs'_def)
apply (simp add: ADT_H_def ADT_A_def)
apply (intro conjI)
apply (rule init_refinement)
apply (clarsimp simp: rel_semi_def relcomp_unfold in_lift_state_relation_eq)
apply (rename_tac tc ak m ev tc' ck' m' ev' ck)
apply (simp add: global_automaton_def)
apply (erule_tac P="?a \<and> (\<exists>x. ?b x)" in disjE)
apply (clarsimp simp add: kernel_call_H_def kernel_call_A_def)
apply (rule rev_mp, rule_tac tc=tc and event=x in entry_corres)
apply (clarsimp simp: corres_underlying_def)
apply (drule (1) bspec)
apply (clarsimp simp: sch_act_simple_def)
apply (drule (1) bspec)
apply clarsimp
apply (rule conjI)
apply clarsimp
apply (rule_tac x=b in exI)
apply (rule conjI)
apply (rule impI, simp)
apply (frule (2) ct_running_related)
apply clarsimp
apply (rule_tac x=b in exI)
apply (drule use_valid, rule kernelEntry_invs')
apply (simp add: sch_act_simple_def)
apply clarsimp
apply (frule (1) ct_idle_related)
apply (clarsimp simp: ct_in_state_def st_tcb_at_def obj_at_def)
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> ?d \<and> ?e" in disjE)
apply (clarsimp simp: do_user_op_H_def do_user_op_A_def monad_to_transition_def)
apply (rule rev_mp, rule_tac tc1=tc and f1=uop and P="ct_running and einvs" in corres_guard_imp2[OF do_user_op_corres])
apply simp
apply (clarsimp simp add: corres_underlying_def)
apply (drule (1) bspec, clarsimp)
apply (drule (1) bspec, clarsimp)
apply fastforce
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> (\<exists>x. ?d x)" in disjE)
apply (clarsimp simp: do_user_op_H_def do_user_op_A_def monad_to_transition_def)
apply (rule rev_mp, rule_tac tc1=tc and f1=uop and P="ct_running and einvs" in corres_guard_imp2[OF do_user_op_corres])
apply simp
apply (clarsimp simp add: corres_underlying_def)
apply (drule (1) bspec, clarsimp)
apply (drule (1) bspec, clarsimp)
apply fastforce
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> ?d \<and> ?e" in disjE)
apply (clarsimp simp: check_active_irq_H_def check_active_irq_A_def)
apply (rule rev_mp, rule check_active_irq_corres)
apply (clarsimp simp: corres_underlying_def)
apply fastforce
apply (erule_tac P="?a \<and> ?b \<and> ?c \<and> ?d \<and> ?e" in disjE)
apply (clarsimp simp: check_active_irq_H_def check_active_irq_A_def)
apply (rule rev_mp, rule check_active_irq_corres)
apply (clarsimp simp: corres_underlying_def)
apply fastforce
apply (clarsimp simp: check_active_irq_H_def check_active_irq_A_def)
apply (rule rev_mp, rule check_active_irq_corres)
apply (clarsimp simp: corres_underlying_def)
apply fastforce
apply (clarsimp simp: absKState_correct dest!: lift_state_relationD)
done
theorem refinement:
"ADT_H a b c d e f uop \<sqsubseteq> ADT_A uop"
apply (rule sim_imp_refines)
apply (rule L_invariantI)
apply (rule akernel_invariant)
apply (rule ckernel_invariant)
apply (rule fw_sim_A_H)
done
end