lh-l4v/proof/refine/Tcb_R.thy

2451 lines
111 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)
*)
theory Tcb_R
imports CNodeInv_R
begin
context begin interpretation Arch . (*FIXME: arch_split*)
lemma setNextPCs_corres:
"corres dc (tcb_at t and invs) (tcb_at' t and invs')
(as_user t (setNextPC v)) (asUser t (setNextPC v))"
apply (rule corres_as_user)
apply (rule corres_Id, simp, simp)
apply (rule no_fail_setNextPC)
done
lemma activate_idle_thread_corres:
"corres dc (invs and st_tcb_at idle t)
(invs' and st_tcb_at' idle' t)
(arch_activate_idle_thread t) (activateIdleThread t)"
by (simp add: arch_activate_idle_thread_def activateIdleThread_def)
lemma activate_corres:
"corres dc (invs and ct_in_state activatable) (invs' and ct_in_state' activatable')
activate_thread activateThread"
apply (simp add: activate_thread_def activateThread_def)
apply (rule corres_guard_imp)
apply (rule corres_split_eqr [OF _ gct_corres])
apply (rule_tac R="\<lambda>ts s. valid_tcb_state ts s \<and> (idle ts \<or> runnable ts)
\<and> invs s \<and> st_tcb_at (op = ts) thread s"
and R'="\<lambda>ts s. valid_tcb_state' ts s \<and> (idle' ts \<or> runnable' ts)
\<and> invs' s \<and> st_tcb_at' (\<lambda>ts'. ts' = ts) thread s"
in corres_split [OF _ gts_corres])
apply (rule_tac F="idle rv \<or> runnable rv" in corres_req, simp)
apply (rule_tac F="idle' rv' \<or> runnable' rv'" in corres_req, simp)
apply (case_tac rv, simp_all add:
isRunning_def isRestart_def,
safe, simp_all)[1]
apply (rule corres_guard_imp)
apply (rule corres_split_eqr [OF _ getRestartPCs_corres])
apply (rule corres_split_nor [OF _ setNextPCs_corres])
apply (rule sts_corres)
apply (simp | wp weak_sch_act_wf_lift_linear)+
apply (clarsimp simp: st_tcb_at_tcb_at)
apply fastforce
apply (rule corres_guard_imp)
apply (rule activate_idle_thread_corres)
apply (clarsimp elim!: st_tcb_weakenE)
apply (clarsimp elim!: pred_tcb'_weakenE)
apply (wp gts_st_tcb gts_st_tcb' gts_st_tcb_at)
apply (clarsimp simp: ct_in_state_def tcb_at_invs
elim!: st_tcb_weakenE)
apply (clarsimp simp: tcb_at_invs' ct_in_state'_def
elim!: pred_tcb'_weakenE)
done
lemma bind_notification_corres:
"corres dc
(invs and tcb_at t and ntfn_at a) (invs' and tcb_at' t and ntfn_at' a)
(bind_notification t a) (bindNotification t a)"
apply (simp add: bind_notification_def bindNotification_def)
apply (rule corres_guard_imp)
apply (rule corres_split[OF _ get_ntfn_corres])
apply (rule corres_split[OF _ set_ntfn_corres])
apply (rule sbn_corres)
apply (clarsimp simp: ntfn_relation_def split: Structures_A.ntfn.splits)
apply (wp)
apply auto
done
abbreviation
"ct_idle' \<equiv> ct_in_state' idle'"
lemma gts_st_tcb':
"\<lbrace>tcb_at' t\<rbrace> getThreadState t \<lbrace>\<lambda>rv. st_tcb_at' (\<lambda>st. st = rv) t\<rbrace>"
apply (rule hoare_vcg_precond_imp)
apply (rule hoare_post_imp[where Q="\<lambda>rv s. \<exists>rv'. rv = rv' \<and> st_tcb_at' (\<lambda>st. st = rv') t s"])
apply simp
apply (wp hoare_ex_wp)
apply (clarsimp simp add: pred_tcb_at'_def obj_at'_def)
done
lemma activateIdle_invs:
"\<lbrace>invs' and ct_idle'\<rbrace>
activateIdleThread thread
\<lbrace>\<lambda>rv. invs' and ct_idle'\<rbrace>"
by (simp add: activateIdleThread_def)
lemma activate_invs':
"\<lbrace>invs' and sch_act_simple and ct_in_state' activatable'\<rbrace>
activateThread
\<lbrace>\<lambda>rv. invs' and (ct_running' or ct_idle')\<rbrace>"
apply (simp add: activateThread_def)
apply (rule hoare_seq_ext)
apply (rule_tac B="\<lambda>state s. invs' s \<and> sch_act_simple s
\<and> st_tcb_at' (\<lambda>st. st = state) thread s
\<and> thread = ksCurThread s
\<and> (runnable' state \<or> idle' state)" in hoare_seq_ext)
apply (case_tac x, simp_all add: isTS_defs hoare_pre_cont
split del: if_splits cong: if_cong)
apply (rule hoare_pre, wp)
apply (clarsimp simp: ct_in_state'_def)
apply (rule_tac Q="\<lambda>rv. invs' and ct_idle'" in hoare_post_imp, simp)
apply (wp activateIdle_invs)
apply (clarsimp simp: ct_in_state'_def)
apply (rule_tac Q="\<lambda>rv. invs' and ct_running' and sch_act_simple"
in hoare_post_imp, simp)
apply (rule hoare_weaken_pre)
apply (wp ct_in_state'_set asUser_ct sts_invs_minor'
| wp_once sch_act_simple_lift)+
apply (rule_tac Q="\<lambda>_. st_tcb_at' runnable' thread
and sch_act_simple and invs'
and (\<lambda>s. thread = ksCurThread s)"
in hoare_post_imp, clarsimp)
apply (wp sch_act_simple_lift)
apply (clarsimp simp: valid_idle'_def invs'_def valid_state'_def
pred_tcb_at'_def obj_at'_def
elim!: pred_tcb'_weakenE)
apply (wp gts_st_tcb')
apply (clarsimp simp: tcb_at_invs' ct_in_state'_def
pred_disj_def)
apply simp
done
crunch nosch[wp]: activateIdleThread "\<lambda>s. P (ksSchedulerAction s)"
(ignore: setNextPC)
declare not_psubset_eq[dest!]
lemma setThreadState_runnable_simp:
"runnable' ts \<Longrightarrow> setThreadState ts t =
threadSet (tcbState_update (\<lambda>x. ts)) t"
apply (simp add: setThreadState_def isRunnable_def isBlocked_def liftM_def)
apply (subst bind_return[symmetric], rule bind_cong[OF refl])
apply (drule use_valid[OF _ threadSet_pred_tcb_at_state[where proj="itcbState" and p=t and P="op = ts"]])
apply simp
apply (subst bind_known_operation_eq)
apply wp
apply clarsimp
apply (subst eq_commute, erule conjI[OF _ refl])
apply (rule empty_fail_getThreadState)
apply (simp add: getCurThread_def getSchedulerAction_def exec_gets)
apply (auto simp: when_def split: Structures_H.thread_state.split)
done
lemma activate_sch_act:
"\<lbrace>ct_in_state' activatable' and (\<lambda>s. P (ksSchedulerAction s))\<rbrace>
activateThread \<lbrace>\<lambda>rv s. P (ksSchedulerAction s)\<rbrace>"
apply (simp add: activateThread_def getCurThread_def
cong: if_cong Structures_H.thread_state.case_cong)
apply (rule hoare_seq_ext [OF _ gets_sp])
apply (rule hoare_seq_ext[where B="\<lambda>st s. (runnable' or idle') st
\<and> P (ksSchedulerAction s)"])
apply (rule hoare_pre)
apply (wp | wpc | simp add: setThreadState_runnable_simp)+
apply (clarsimp simp: ct_in_state'_def cur_tcb'_def pred_tcb_at'
elim!: pred_tcb'_weakenE)
done
lemma activate_sch_act_simple:
"\<lbrace>ct_in_state' activatable' and sch_act_simple\<rbrace>
activateThread
\<lbrace>\<lambda>rv. sch_act_simple\<rbrace>"
unfolding sch_act_simple_def
by (rule activate_sch_act)
lemma runnable_tsr:
"thread_state_relation ts ts' \<Longrightarrow> runnable' ts' = runnable ts"
by (case_tac ts, auto)
lemma idle_tsr:
"thread_state_relation ts ts' \<Longrightarrow> idle' ts' = idle ts"
by (case_tac ts, auto)
crunch cur [wp]: cancelIPC cur_tcb'
(wp: crunch_wps simp: crunch_simps)
crunch cur [wp]: setupReplyMaster cur_tcb'
(wp: crunch_wps simp: crunch_simps)
lemma setCTE_weak_sch_act_wf[wp]:
"\<lbrace>\<lambda>s. weak_sch_act_wf (ksSchedulerAction s) s\<rbrace>
setCTE c cte
\<lbrace>\<lambda>rv s. weak_sch_act_wf (ksSchedulerAction s) s\<rbrace>"
apply (simp add: weak_sch_act_wf_def)
apply (wp hoare_vcg_all_lift hoare_convert_imp setCTE_pred_tcb_at' setCTE_tcb_in_cur_domain')
done
lemma setupReplyMaster_weak_sch_act_wf[wp]:
"\<lbrace>\<lambda>s. weak_sch_act_wf (ksSchedulerAction s) s\<rbrace>
setupReplyMaster thread
\<lbrace>\<lambda>rv s. weak_sch_act_wf (ksSchedulerAction s) s\<rbrace>"
apply (simp add: setupReplyMaster_def)
apply (wp)
apply (rule_tac Q="\<lambda>_ s. weak_sch_act_wf (ksSchedulerAction s) s"
in hoare_post_imp, clarsimp)
apply (wp)
done
crunch valid_queues[wp]: setupReplyMaster "Invariants_H.valid_queues"
crunch valid_queues'[wp]: setupReplyMaster "valid_queues'"
lemma restart_corres:
"corres dc (einvs and tcb_at t) (invs' and tcb_at' t)
(Tcb_A.restart t) (ThreadDecls_H.restart t)"
apply (simp add: Tcb_A.restart_def Thread_H.restart_def)
apply (simp add: isBlocked_def2 liftM_def)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ gts_corres])
apply (clarsimp simp add: runnable_tsr idle_tsr when_def)
apply (rule corres_split_nor [OF _ cancel_ipc_corres])
apply (rule corres_split_nor [OF _ setup_reply_master_corres])
apply (rule corres_split_nor [OF _ sts_corres])
apply (rule corres_split [OF switchIfRequiredTo_corres tcbSchedEnqueue_corres])
apply (wp_trace set_thread_state_runnable_weak_valid_sched_action sts_st_tcb_at' sts_valid_queues sts_st_tcb' | clarsimp simp: valid_tcb_state'_def)+
apply (rule_tac Q="\<lambda>rv. valid_sched and cur_tcb" in hoare_strengthen_post)
apply wp
apply (simp add: valid_sched_def valid_sched_action_def)
apply (rule_tac Q="\<lambda>rv. invs' and tcb_at' t" in hoare_strengthen_post)
apply wp
apply (clarsimp simp: invs'_def valid_state'_def sch_act_wf_weak valid_pspace'_def)
apply wp
apply (simp add: valid_sched_def invs_def tcb_at_is_etcb_at)
apply (clarsimp simp add: invs'_def valid_state'_def sch_act_wf_weak)
done
lemma setupReplyMaster_sch_act_simple[wp]:
"\<lbrace>sch_act_simple\<rbrace> setupReplyMaster thread \<lbrace>\<lambda>_. sch_act_simple\<rbrace>"
apply (simp add: setupReplyMaster_def sch_act_simple_def)
apply (wp)
apply (rule_tac Q="\<lambda>_. sch_act_simple" in hoare_post_imp,
clarsimp simp: sch_act_simple_def)
apply (wp)
apply (simp add: sch_act_simple_def)
done
lemma restart_invs':
"\<lbrace>invs' and ex_nonz_cap_to' t and (\<lambda>s. t \<noteq> ksIdleThread s)\<rbrace>
ThreadDecls_H.restart t \<lbrace>\<lambda>rv. invs'\<rbrace>"
apply (simp add: restart_def isBlocked_def2 switchIfRequiredTo_def)
apply (wp setThreadState_nonqueued_state_update
cancelIPC_simple setThreadState_st_tcb
| wp_once sch_act_simple_lift)+
apply (wp hoare_convert_imp)
apply (wp setThreadState_nonqueued_state_update
setThreadState_st_tcb)
apply (clarsimp)
apply (wp hoare_convert_imp)[1]
apply (clarsimp)
apply (wp)
apply (clarsimp simp: comp_def)
apply (rule hoare_strengthen_post, rule gts_sp')
apply (clarsimp simp: pred_tcb_at' invs'_def valid_state'_def
ct_in_state'_def)
apply (fastforce simp: pred_tcb_at'_def obj_at'_def)
done
lemma restart_tcb'[wp]:
"\<lbrace>tcb_at' t'\<rbrace> ThreadDecls_H.restart t \<lbrace>\<lambda>rv. tcb_at' t'\<rbrace>"
apply (simp add: restart_def isBlocked_def2)
apply wp
apply simp
apply wp
done
lemma
no_fail_setRegister: "no_fail \<top> (setRegister r v)"
by (simp add: setRegister_def)
lemma copyRegsToArea_invs[wp]:
"\<lbrace>invs\<rbrace> copyRegsToArea regs a b \<lbrace>\<lambda>rv. invs\<rbrace>"
apply (simp add: copyRegsToArea_def)
apply (wp zipWithM_x_inv)
apply simp
apply wp
done
lemma copyAreaToRegs_invs[wp]:
"\<lbrace>invs and tcb_at b\<rbrace> copyAreaToRegs regs a b \<lbrace>\<lambda>rv. invs\<rbrace>"
apply (simp add: copyAreaToRegs_def)
apply wp
apply (rule thread_set_invs_trivial, (simp add: tcb_cap_cases_def)+)
apply (rule mapM_wp [where S=UNIV, simplified])
apply wp
apply simp
done
lemma suspend_cap_to'[wp]:
"\<lbrace>ex_nonz_cap_to' p\<rbrace> suspend t \<lbrace>\<lambda>rv. ex_nonz_cap_to' p\<rbrace>"
apply (simp add: suspend_def unless_def)
apply (wp threadSet_cap_to' | simp)+
done
declare det_getRegister[simp]
declare det_setRegister[simp]
lemma
no_fail_getRegister[wp]: "no_fail \<top> (getRegister r)"
by (simp add: getRegister_def)
lemma readreg_corres:
"corres (intr \<oplus> op =)
(einvs and tcb_at src and ex_nonz_cap_to src)
(invs' and sch_act_simple and tcb_at' src and ex_nonz_cap_to' src)
(invoke_tcb (tcb_invocation.ReadRegisters src susp n arch))
(invokeTCB (tcbinvocation.ReadRegisters src susp n arch'))"
apply (simp add: invokeTCB_def performTransfer_def genericTake_def
frame_registers_def gp_registers_def
frameRegisters_def gpRegisters_def)
apply (rule corres_guard_imp)
apply (rule corres_split_nor)
apply (rule corres_split [OF _ gct_corres])
apply (simp add: liftM_def[symmetric])
apply (rule corres_as_user)
apply (rule corres_Id)
apply simp
apply simp
apply (rule no_fail_mapM)
apply (simp add: no_fail_getRegister)
apply wp
apply (rule corres_when [OF refl])
apply (rule suspend_corres)
apply wp
apply (clarsimp simp: invs_def valid_state_def valid_pspace_def
dest!: idle_no_ex_cap)
apply (clarsimp simp: invs'_def valid_state'_def dest!: global'_no_ex_cap)
done
crunch sch_act_simple [wp]: asUser "sch_act_simple"
(lift: sch_act_simple_lift)
lemma writereg_corres:
"corres (intr \<oplus> op =) (einvs and tcb_at dest and ex_nonz_cap_to dest)
(invs' and sch_act_simple and tcb_at' dest and ex_nonz_cap_to' dest)
(invoke_tcb (tcb_invocation.WriteRegisters dest resume values arch))
(invokeTCB (tcbinvocation.WriteRegisters dest resume values arch'))"
apply (simp add: invokeTCB_def performTransfer_def
frameRegisters_def gpRegisters_def
sanitiseRegister_def)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ gct_corres])
apply (rule corres_split_nor)
prefer 2
apply (rule corres_as_user)
apply (simp add: zipWithM_mapM getRestartPC_def setNextPC_def)
apply (rule corres_Id, simp+)
apply (rule no_fail_pre, wp no_fail_mapM)
apply clarsimp
apply (wp no_fail_setRegister | simp)+
apply (rule corres_split_nor)
apply (rule_tac P=\<top> and P'=\<top> in corres_inst)
apply simp
apply (rule corres_when [OF refl])
apply (rule restart_corres)
apply (wp static_imp_wp | clarsimp simp: invs'_def valid_state'_def
dest!: global'_no_ex_cap)+
done
crunch it[wp]: suspend "\<lambda>s. P (ksIdleThread s)"
lemma tcbSchedDequeue_ResumeCurrentThread_imp_notct[wp]:
"\<lbrace>\<lambda>s. ksSchedulerAction s = ResumeCurrentThread \<longrightarrow> ksCurThread s \<noteq> t'\<rbrace>
tcbSchedDequeue t
\<lbrace>\<lambda>rv s. ksSchedulerAction s = ResumeCurrentThread \<longrightarrow> ksCurThread s \<noteq> t'\<rbrace>"
by (wp hoare_convert_imp)
lemma suspend_ResumeCurrentThread_imp_notct[wp]:
"\<lbrace>\<lambda>s. ksSchedulerAction s = ResumeCurrentThread \<longrightarrow> ksCurThread s \<noteq> t'\<rbrace>
suspend t
\<lbrace>\<lambda>rv s. ksSchedulerAction s = ResumeCurrentThread \<longrightarrow> ksCurThread s \<noteq> t'\<rbrace>"
by (simp add: suspend_def, wp)
lemma copyreg_corres:
"corres (intr \<oplus> op =)
(einvs and simple_sched_action and tcb_at dest and tcb_at src and ex_nonz_cap_to src and
ex_nonz_cap_to dest)
(invs' and sch_act_simple and tcb_at' dest and tcb_at' src
and ex_nonz_cap_to' src and ex_nonz_cap_to' dest)
(invoke_tcb (tcb_invocation.CopyRegisters dest src susp resume frames ints arch))
(invokeTCB (tcbinvocation.CopyRegisters dest src susp resume frames ints arch'))"
proof -
have Q: "\<And>src src' des des' r r'. \<lbrakk> src = src'; des = des' \<rbrakk> \<Longrightarrow>
corres dc (tcb_at src and tcb_at des and invs)
(tcb_at' src' and tcb_at' des' and invs')
(do v \<leftarrow> as_user src (get_register r);
as_user des (set_register r' v)
od)
(do v \<leftarrow> asUser src' (getRegister r);
asUser des' (setRegister r' v)
od)"
apply clarsimp
apply (rule corres_guard_imp)
apply (rule corres_split_eqr)
apply (simp add: set_register_def setRegister_def)
apply (rule corres_as_user)
apply (rule corres_modify')
apply simp
apply simp
apply (rule user_getreg_corres)
apply (simp | wp)+
done
have R: "\<And>src src' des des' xs ys. \<lbrakk> src = src'; des = des'; xs = ys \<rbrakk> \<Longrightarrow>
corres dc (tcb_at src and tcb_at des and invs)
(tcb_at' src' and tcb_at' des' and invs')
(mapM_x (\<lambda>r. do v \<leftarrow> as_user src (get_register r);
as_user des (set_register r v)
od) xs)
(mapM_x (\<lambda>r'. do v \<leftarrow> asUser src' (getRegister r');
asUser des' (setRegister r' v)
od) ys)"
apply (rule corres_mapM_x [where S=Id])
apply simp
apply (rule Q)
apply (clarsimp simp: set_zip_same | wp)+
done
have S: "get_register = getRegister" "set_register = setRegister"
by (rule ext | simp add: get_register_def getRegister_def
set_register_def setRegister_def)+
have U: "\<And>t. corres dc (tcb_at t and invs) (tcb_at' t and invs')
(do pc \<leftarrow> as_user t getRestartPC; as_user t (setNextPC pc) od)
(do pc \<leftarrow> asUser t getRestartPC; asUser t (setNextPC pc) od)"
apply (rule corres_guard_imp)
apply (rule corres_split_eqr [OF _ getRestartPCs_corres])
apply (rule setNextPCs_corres)
apply wp
apply simp+
done
show ?thesis
apply (simp add: invokeTCB_def performTransfer_def)
apply (rule corres_guard_imp)
apply (rule corres_split_nor)
apply (rule corres_split_nor)
apply (rule corres_split_nor)
apply (simp add: liftM_def[symmetric] o_def dc_def[symmetric])
apply (rule corres_when[OF refl])
apply (rule R[unfolded S, OF refl refl])
apply (simp add: gpRegisters_def)
apply (rule corres_when[OF refl])
apply (rule corres_split_nor)
apply (simp add: getRestartPC_def setNextPC_def)
apply (rule Q[unfolded S, OF refl refl])
apply (rule R[unfolded S, OF refl refl])
apply (simp add: frame_registers_def frameRegisters_def)
apply (wp mapM_x_wp' static_imp_wp | simp)+
apply (rule corres_when [OF refl])
apply (rule restart_corres)
apply (wp restart_invs' static_imp_wp | simp add: if_apply_def2)+
apply (rule corres_when [OF refl])
apply (rule suspend_corres)
apply (wp suspend_nonz_cap_to_tcb static_imp_wp | simp add: if_apply_def2)+
apply (fastforce simp: invs_def valid_state_def valid_pspace_def
dest!: idle_no_ex_cap)
apply (fastforce simp: invs'_def valid_state'_def dest!: global'_no_ex_cap)
done
qed
lemma readreg_invs':
"\<lbrace>invs' and sch_act_simple and tcb_at' src and ex_nonz_cap_to' src\<rbrace>
invokeTCB (tcbinvocation.ReadRegisters src susp n arch)
\<lbrace>\<lambda>rv. invs'\<rbrace>"
by (simp add: invokeTCB_def performTransfer_def | wp
| clarsimp simp: invs'_def valid_state'_def
dest!: global'_no_ex_cap)+
lemma writereg_invs':
"\<lbrace>invs' and sch_act_simple and tcb_at' dest and ex_nonz_cap_to' dest\<rbrace>
invokeTCB (tcbinvocation.WriteRegisters dest resume values arch)
\<lbrace>\<lambda>rv. invs'\<rbrace>"
by (simp add: invokeTCB_def performTransfer_def | wp restart_invs' | rule conjI
| clarsimp
| clarsimp simp: invs'_def valid_state'_def
dest!: global'_no_ex_cap)+
lemma copyreg_invs'':
"\<lbrace>invs' and sch_act_simple and tcb_at' src and tcb_at' dest and ex_nonz_cap_to' src and ex_nonz_cap_to' dest\<rbrace>
invokeTCB (tcbinvocation.CopyRegisters dest src susp resume frames ints arch)
\<lbrace>\<lambda>rv. invs' and tcb_at' dest\<rbrace>"
apply (simp add: invokeTCB_def performTransfer_def if_apply_def2)
apply (wp mapM_x_wp' restart_invs' | simp)+
apply (rule conjI)
apply (wp | clarsimp)+
by (fastforce simp: invs'_def valid_state'_def dest!: global'_no_ex_cap)
lemma copyreg_invs':
"\<lbrace>invs' and sch_act_simple and tcb_at' src and
tcb_at' dest and ex_nonz_cap_to' src and ex_nonz_cap_to' dest\<rbrace>
invokeTCB (tcbinvocation.CopyRegisters dest src susp resume frames ints arch)
\<lbrace>\<lambda>rv. invs'\<rbrace>"
by (rule hoare_strengthen_post, rule copyreg_invs'', simp)
lemma isRunnable_wp':
"\<lbrace>\<lambda>s. Q (st_tcb_at' runnable' t s) s\<rbrace> isRunnable t \<lbrace>Q\<rbrace>"
apply (simp add: isRunnable_def2)
apply wp
apply (simp add: getThreadState_def threadGet_def)
apply wp
apply (clarsimp simp: getObject_def valid_def in_monad
loadObject_default_def projectKOs obj_at'_def
split_def objBits_simps in_magnitude_check)
apply (clarsimp simp: pred_tcb_at'_def obj_at'_def projectKOs objBitsKO_def)
done
lemma threadSet_valid_queues_no_state:
"\<lbrace>Invariants_H.valid_queues and (\<lambda>s. \<forall>p. t \<notin> set (ksReadyQueues s p))\<rbrace>
threadSet f t \<lbrace>\<lambda>_. Invariants_H.valid_queues\<rbrace>"
apply (simp add: threadSet_def)
apply wp
apply (simp add: valid_queues_def valid_queues_no_bitmap_def' pred_tcb_at'_def)
apply (wp setObject_queues_unchanged_tcb
hoare_Ball_helper
hoare_vcg_all_lift
setObject_tcb_strongest)[1]
apply (wp getObject_tcb_wp)
apply (clarsimp simp: valid_queues_def valid_queues_no_bitmap_def' pred_tcb_at'_def)
apply (clarsimp simp: obj_at'_def)
done
lemma threadSet_valid_queues'_no_state:
"(\<And>tcb. tcbQueued tcb = tcbQueued (f tcb))
\<Longrightarrow> \<lbrace>valid_queues' and (\<lambda>s. \<forall>p. t \<notin> set (ksReadyQueues s p))\<rbrace>
threadSet f t \<lbrace>\<lambda>_. valid_queues'\<rbrace>"
apply (simp add: valid_queues'_def threadSet_def obj_at'_real_def
split del: split_if)
apply (simp only: imp_conv_disj)
apply (wp hoare_vcg_all_lift hoare_vcg_disj_lift)
apply (wp setObject_ko_wp_at | simp add: objBits_simps)+
apply (wp getObject_tcb_wp updateObject_default_inv
| simp split del: split_if)+
apply (clarsimp simp: obj_at'_def ko_wp_at'_def projectKOs
objBits_simps addToQs_def
split del: split_if cong: if_cong)
apply (fastforce simp: projectKOs inQ_def split: split_if_asm)
done
lemma gts_isRunnable_corres:
"corres (\<lambda>ts runn. runnable ts = runn) (tcb_at t) (tcb_at' t)
(get_thread_state t) (isRunnable t)"
apply (simp add: isRunnable_def)
apply (subst bind_return[symmetric])
apply (rule corres_guard_imp)
apply (rule corres_split[OF _ gts_corres])
apply (case_tac rv, clarsimp+)
apply (wp hoare_TrueI, simp+)
done
lemma tcbSchedDequeue_not_queued:
"\<lbrace>\<top>\<rbrace> tcbSchedDequeue t
\<lbrace>\<lambda>rv. obj_at' (Not \<circ> tcbQueued) t\<rbrace>"
apply (simp add: tcbSchedDequeue_def)
apply (wp | simp)+
apply (rule_tac Q="\<lambda>rv. obj_at' (\<lambda>obj. tcbQueued obj = rv) t"
in hoare_post_imp)
apply (clarsimp simp: obj_at'_def)
apply (wp tg_sp' [where P=\<top>, simplified] | simp)+
done
lemma tcbSchedDequeue_not_in_queue:
"\<And>p. \<lbrace>Invariants_H.valid_queues and tcb_at' t and valid_objs'\<rbrace> tcbSchedDequeue t
\<lbrace>\<lambda>rv s. t \<notin> set (ksReadyQueues s p)\<rbrace>"
apply (rule_tac Q="\<lambda>rv. Invariants_H.valid_queues and obj_at' (Not \<circ> tcbQueued) t"
in hoare_post_imp)
apply (fastforce simp: valid_queues_def valid_queues_no_bitmap_def obj_at'_def projectKOs inQ_def )
apply (wp tcbSchedDequeue_not_queued tcbSchedDequeue_valid_queues |
simp add: valid_objs'_maxDomain valid_objs'_maxPriority)+
done
lemma threadSet_ct_in_state':
"(\<And>tcb. tcbState (f tcb) = tcbState tcb) \<Longrightarrow>
\<lbrace>ct_in_state' test\<rbrace> threadSet f t \<lbrace>\<lambda>rv. ct_in_state' test\<rbrace>"
apply (simp add: ct_in_state'_def)
apply (rule hoare_lift_Pf [where f=ksCurThread])
apply (wp threadSet_pred_tcb_no_state)
apply simp
apply wp
done
lemma tcbSchedDequeue_ct_in_state'[wp]:
"\<lbrace>ct_in_state' test\<rbrace> tcbSchedDequeue t \<lbrace>\<lambda>rv. ct_in_state' test\<rbrace>"
apply (simp add: ct_in_state'_def)
apply (rule hoare_lift_Pf [where f=ksCurThread])
apply wp
done
lemma valid_tcb'_tcbPriority_update: "\<lbrakk>valid_tcb' tcb s; f (tcbPriority tcb) \<le> maxPriority \<rbrakk> \<Longrightarrow> valid_tcb' (tcbPriority_update f tcb) s"
apply (simp add: valid_tcb'_def tcb_cte_cases_def)
done
lemma threadSet_valid_objs_tcbPriority_update:
"\<lbrace>valid_objs' and (\<lambda>_. x \<le> maxPriority)\<rbrace>
threadSet (tcbPriority_update (\<lambda>_. x)) t
\<lbrace>\<lambda>_. valid_objs'\<rbrace>"
apply (simp add: threadSet_def)
apply wp
prefer 2
apply (rule getObject_tcb_sp)
apply (rule hoare_weaken_pre)
apply (rule setObject_tcb_valid_objs)
apply (clarsimp simp: valid_obj'_def)
apply (frule (1) ko_at_valid_objs')
apply (simp add: projectKOs)
apply (simp add: valid_obj'_def)
apply (subgoal_tac "tcb_at' t s")
apply simp
apply (rule valid_tcb'_tcbPriority_update)
apply (fastforce simp: obj_at'_def)+
done
lemma sp_corres2:
"corres dc (valid_etcbs and weak_valid_sched_action)
(Invariants_H.valid_queues and valid_queues' and tcb_at' t and (\<lambda>s. weak_sch_act_wf (ksSchedulerAction s) s) and valid_objs' and (\<lambda>_. x \<le> maxPriority))
(set_priority t x) (setPriority t x)"
apply (simp add: setPriority_def set_priority_def thread_set_priority_def)
apply (rule stronger_corres_guard_imp)
apply (rule corres_split [OF _ tcbSchedDequeue_corres])
apply (rule corres_split [OF _ ethread_set_corres], simp_all)[1]
apply (rule corres_split [OF _ gts_isRunnable_corres])
apply (rule corres_split [OF _ corres_when[OF _ tcbSchedEnqueue_corres]])
apply (rule corres_split [OF corres_when[OF _ rescheduleRequired_corres] gct_corres])
apply (wp hoare_vcg_imp_lift hoare_vcg_if_lift hoare_vcg_all_lift hoare_vcg_disj_lift
gts_wp isRunnable_wp' threadSet_pred_tcb_no_state threadSet_valid_queues_no_state
threadSet_valid_queues'_no_state threadSet_valid_objs_tcbPriority_update threadSet_weak_sch_act_wf
tcbSchedDequeue_not_in_queue threadSet_ct_in_state'[simplified ct_in_state'_def]
tcbSchedDequeue_valid_queues
tcbSchedDequeue_ct_in_state'[simplified ct_in_state'_def] | simp add: etcb_relation_def)+
apply (force simp: valid_etcbs_def tcb_at_st_tcb_at[symmetric] state_relation_def
dest: pspace_relation_tcb_at)
apply (force simp: state_relation_def elim: valid_objs'_maxDomain valid_objs'_maxPriority)
done
lemma sp_corres: "corres dc (einvs and tcb_at t) (invs' and tcb_at' t and valid_objs' and (\<lambda>_. x \<le> maxPriority))
(set_priority t x) (setPriority t x)"
apply (rule corres_guard_imp)
apply (rule sp_corres2)
apply (simp add: valid_sched_def valid_sched_action_def)
apply (clarsimp simp: invs'_def valid_state'_def sch_act_wf_weak)
done
definition
"out_rel fn fn' v v' \<equiv>
((v = None) = (v' = None)) \<and>
(\<forall>tcb tcb'. tcb_relation tcb tcb' \<longrightarrow>
tcb_relation (case_option id fn v tcb)
(case_option id fn' v' tcb'))"
lemma out_corresT:
assumes x: "\<And>tcb v. \<forall>(getF, setF)\<in>ran tcb_cap_cases. getF (fn v tcb) = getF tcb"
assumes y: "\<And>v. \<forall>tcb. \<forall>(getF, setF)\<in>ran tcb_cte_cases. getF (fn' v tcb) = getF tcb"
assumes e: "\<And>tcb v. exst_same tcb (fn' v tcb)"
shows
"out_rel fn fn' v v' \<Longrightarrow>
corres dc (tcb_at t)
(tcb_at' t)
(option_update_thread t fn v)
(case_option (return ()) (\<lambda>x. threadSet (fn' x) t) v')"
apply (case_tac v, simp_all add: out_rel_def
option_update_thread_def)
apply clarsimp
apply (clarsimp simp add: threadset_corresT [OF _ x y e])
done
lemmas out_corres = out_corresT [OF _ all_tcbI, OF ball_tcb_cap_casesI ball_tcb_cte_casesI]
lemma tcbSchedEnqueue_queued:
"\<lbrace>\<top>\<rbrace> tcbSchedEnqueue t \<lbrace>\<lambda>rv. obj_at' tcbQueued t\<rbrace>"
apply (simp add: tcbSchedEnqueue_def unless_def)
apply (wp | simp)+
apply (rule_tac Q="\<lambda>rv. obj_at' (\<lambda>obj. tcbQueued obj = rv) t"
in hoare_post_imp)
apply (clarsimp simp: obj_at'_def)
apply (wp tg_sp' [where P=\<top>, simplified] | simp)+
done
crunch sch_act[wp]: tcbSchedEnqueue "\<lambda>s. sch_act_wf (ksSchedulerAction s) s"
(simp: unless_def)
lemma setP_vq[wp]:
"\<lbrace>\<lambda>s. Invariants_H.valid_queues s \<and> tcb_at' t s \<and> sch_act_wf (ksSchedulerAction s) s \<and> valid_objs' s \<and> p \<le> maxPriority\<rbrace>
setPriority t p
\<lbrace>\<lambda>rv. Invariants_H.valid_queues\<rbrace>"
apply (simp add: setPriority_def)
apply (wp threadSet_valid_queues threadSet_valid_objs_tcbPriority_update threadSet_weak_sch_act_wf hoare_drop_imps hoare_vcg_all_lift
tcbSchedDequeue_not_in_queue tcbSchedEnqueue_valid_objs' tcbSchedDequeue_valid_queues
| clarsimp simp: pred_tcb_at'_def o_def)+
apply (fastforce elim: valid_objs'_maxDomain valid_objs'_maxPriority
| clarsimp simp: st_tcb_at'_def o_def
| frule sch_act_wf_weak)+
done
lemma ps_clear_ksReadyQueue[simp]:
"ps_clear x n (ksReadyQueues_update f s) = ps_clear x n s"
by (simp add: ps_clear_def)
lemma valid_queues_subsetE':
"\<lbrakk> valid_queues' s; ksPSpace s = ksPSpace s';
\<forall>x. set (ksReadyQueues s x) \<subseteq> set (ksReadyQueues s' x) \<rbrakk>
\<Longrightarrow> valid_queues' s'"
by (simp add: valid_queues'_def obj_at'_def
ps_clear_def subset_iff projectKOs)
crunch vq'[wp]: getCurThread valid_queues'
lemma valid_queues_ksSchedulerAction_update [simp]:
"valid_queues' (ksSchedulerAction_update f s) = valid_queues' s"
by (simp add: valid_queues'_def)
lemma setP_vq'[wp]:
"\<lbrace>\<lambda>s. valid_queues' s \<and> tcb_at' t s \<and> sch_act_wf (ksSchedulerAction s) s \<and> p \<le> maxPriority\<rbrace>
setPriority t p
\<lbrace>\<lambda>rv. valid_queues'\<rbrace>"
apply (rule hoare_strengthen_post[where Q="\<lambda>rv s. valid_queues' s \<and> sch_act_wf (ksSchedulerAction s) s"])
apply (simp add: setPriority_def)
apply (wp | clarsimp simp: pred_tcb_at'_def o_def)+
apply(rule_tac Q="\<lambda>s. valid_queues' s \<and> sch_act_wf (ksSchedulerAction s) s \<and> obj_at' (Not \<circ> tcbQueued) t s" in hoare_pre_imp, assumption)
apply ((wp_trace threadSet_valid_queues' threadSet_sch_act | clarsimp simp: projectKOs inQ_def obj_at'_def)+)[1]
apply (wp tcbSchedDequeue_sch_act_wf tcbSchedDequeue_valid_queues' tcbSchedDequeue_not_queued | simp) +
done
lemma setQueue_invs_bits[wp]:
"\<lbrace>valid_pspace'\<rbrace> setQueue d p q \<lbrace>\<lambda>rv. valid_pspace'\<rbrace>"
"\<lbrace>\<lambda>s. sch_act_wf (ksSchedulerAction s) s\<rbrace> setQueue d p q \<lbrace>\<lambda>rv s. sch_act_wf (ksSchedulerAction s) s\<rbrace>"
"\<lbrace>\<lambda>s. sym_refs (state_refs_of' s)\<rbrace> setQueue d p q \<lbrace>\<lambda>rv s. sym_refs (state_refs_of' s)\<rbrace>"
"\<lbrace>if_live_then_nonz_cap'\<rbrace> setQueue d p q \<lbrace>\<lambda>rv. if_live_then_nonz_cap'\<rbrace>"
"\<lbrace>if_unsafe_then_cap'\<rbrace> setQueue d p q \<lbrace>\<lambda>rv. if_unsafe_then_cap'\<rbrace>"
"\<lbrace>cur_tcb'\<rbrace> setQueue d p q \<lbrace>\<lambda>rv. cur_tcb'\<rbrace>"
"\<lbrace>valid_global_refs'\<rbrace> setQueue d p q \<lbrace>\<lambda>rv. valid_global_refs'\<rbrace>"
"\<lbrace>valid_irq_handlers'\<rbrace> setQueue d p q \<lbrace>\<lambda>rv. valid_irq_handlers'\<rbrace>"
by (simp add: setQueue_def tcb_in_cur_domain'_def
| wp sch_act_wf_lift cur_tcb_lift
| fastforce)+
lemma setQueue_ex_idle_cap[wp]:
"\<lbrace>\<lambda>s. ex_nonz_cap_to' (ksIdleThread s) s\<rbrace>
setQueue d p q
\<lbrace>\<lambda>rv s. ex_nonz_cap_to' (ksIdleThread s) s\<rbrace>"
by (simp add: setQueue_def, wp,
simp add: ex_nonz_cap_to'_def cte_wp_at_pspaceI)
lemma tcbPriority_ts_safe:
"tcbState (tcbPriority_update f tcb) = tcbState tcb"
by (case_tac tcb, simp)
lemma tcbQueued_ts_safe:
"tcbState (tcbQueued_update f tcb) = tcbState tcb"
by (case_tac tcb, simp)
lemma tcbPriority_caps_safe:
"\<forall>tcb. \<forall>x\<in>ran tcb_cte_cases. (\<lambda>(getF, setF). getF (tcbPriority_update f tcb) = getF tcb) x"
by (rule all_tcbI, rule ball_tcb_cte_casesI, simp+)
lemma tcbPriority_Queued_caps_safe:
"\<forall>tcb. \<forall>x\<in>ran tcb_cte_cases. (\<lambda>(getF, setF). getF (tcbPriority_update f (tcbQueued_update g tcb)) = getF tcb) x"
by (rule all_tcbI, rule ball_tcb_cte_casesI, simp+)
lemma setP_invs':
"\<lbrace>invs' and tcb_at' t and K (p \<le> maxPriority)\<rbrace> setPriority t p \<lbrace>\<lambda>rv. invs'\<rbrace>"
proof -
have enq: "\<And>s'. t \<noteq> ksCurThread s' \<Longrightarrow>
\<lbrace>(op = s') and invs' and st_tcb_at' runnable' t\<rbrace>
tcbSchedEnqueue t \<lbrace>\<lambda>_. invs'\<rbrace>"
by (wp, clarsimp)
show ?thesis
apply (rule hoare_gen_asm)
apply (simp add: setPriority_def)
apply (wp rescheduleRequired_all_invs_but_ct_not_inQ)
apply (wps tcbSchedEnqueue_ct')
apply (case_tac "t \<noteq> ksCurThread prev_s")
apply (clarsimp, erule enq)
apply (clarsimp simp add: invs'_def valid_state'_def)
apply (wp valid_irq_node_lift, clarsimp+)
apply (erule(1) st_tcb_ex_cap'')
apply (case_tac st, clarsimp+)[1]
apply (clarsimp)+
apply (rule_tac Q="\<lambda>r s. (r \<longrightarrow> st_tcb_at' runnable' t s) \<and> invs' s"
in hoare_post_imp, clarsimp simp: invs'_def valid_state'_def)
apply (wp)
apply (rule_tac Q="\<lambda>_. invs'" in hoare_post_imp,
clarsimp simp: pred_tcb_at'_def comp_def)
apply (wp threadSet_invs_trivial,
simp_all add: inQ_def cong: conj_cong)
apply (rule_tac Q="\<lambda>_. invs' and obj_at' (Not \<circ> tcbQueued) t
and (\<lambda>s. \<forall>d p. t \<notin> set (ksReadyQueues s (d,p)))"
in hoare_post_imp)
apply (clarsimp dest: obj_at_ko_at' simp: obj_at'_def)
apply (wp_trace tcbSchedDequeue_not_queued)
apply (clarsimp)+
done
qed
crunch typ_at'[wp]: setPriority "\<lambda>s. P (typ_at' T p s)"
(ignore: getObject simp: crunch_simps)
lemmas setPriority_typ_ats [wp] = typ_at_lifts [OF setPriority_typ_at']
crunch valid_cap[wp]: setPriority "valid_cap' c"
(wp: getObject_inv_tcb)
definition
newroot_rel :: "(cap \<times> cslot_ptr) option \<Rightarrow> (capability \<times> machine_word) option \<Rightarrow> bool"
where
"newroot_rel \<equiv> opt_rel (\<lambda>(cap, ptr) (cap', ptr').
cap_relation cap cap'
\<and> ptr' = cte_map ptr)"
function recursive :: "nat \<Rightarrow> ((nat \<times> nat), unit) nondet_monad"
where
"recursive (Suc n) s = (do f \<leftarrow> gets fst; s \<leftarrow> gets snd; put ((f + s), n); recursive n od) s"
| "recursive 0 s = (modify (\<lambda>(a, b). (a, 0))) s"
by (case_tac "fst x", fastforce+)
termination recursive
apply (rule recursive.termination)
apply (rule wf_measure [where f=fst])
apply simp
done
lemma cte_map_tcb_0:
"cte_map (t, tcb_cnode_index 0) = t"
by (simp add: cte_map_def tcb_cnode_index_def)
lemma cte_map_tcb_1:
"cte_map (t, tcb_cnode_index 1) = t + 16"
by (simp add: cte_map_def tcb_cnode_index_def to_bl_1)
lemma sameRegion_corres2:
"\<lbrakk> cap_relation c c'; cap_relation d d' \<rbrakk>
\<Longrightarrow> same_region_as c d = sameRegionAs c' d'"
by (erule(1) same_region_as_relation)
lemma sameObject_corres2:
"\<lbrakk> cap_relation c c'; cap_relation d d' \<rbrakk>
\<Longrightarrow> same_object_as c d = sameObjectAs c' d'"
apply (frule(1) sameRegion_corres2[symmetric, where c=c and d=d])
apply (case_tac c; simp add: sameObjectAs_def same_object_as_def
isCap_simps is_cap_simps bits_of_def)
apply (case_tac d; simp)
apply (case_tac d'; simp)
apply (rename_tac arch_cap)
apply clarsimp
apply (case_tac d, (simp_all split: arch_cap.split)[11])
apply (rename_tac arch_capa)
apply (clarsimp simp add: ARM_H.sameObjectAs_def Let_def)
apply (intro conjI impI)
apply (case_tac arch_cap; simp add: isCap_simps)
apply (case_tac arch_capa; simp)
apply (case_tac arch_cap; simp add: sameRegionAs_def isCap_simps)
apply (case_tac arch_capa; simp)
done
lemma check_cap_at_corres:
assumes r: "cap_relation cap cap'"
assumes c: "corres dc Q Q' f f'"
assumes Q: "\<And>s. P s \<and> cte_wp_at (same_object_as cap) slot s \<Longrightarrow> Q s"
assumes Q': "\<And>s. P' s \<and> cte_wp_at' (sameObjectAs cap' o cteCap) (cte_map slot) s \<Longrightarrow> Q' s"
shows "corres dc (P and cte_at slot and invs) (P' and pspace_aligned' and pspace_distinct')
(check_cap_at cap slot f)
(checkCapAt cap' (cte_map slot) f')" using r c
apply (simp add: check_cap_at_def checkCapAt_def liftM_def when_def)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ get_cap_corres])
apply (rule corres_if [unfolded if_apply_def2])
apply (erule(1) sameObject_corres2)
apply assumption
apply (rule corres_trivial, simp)
apply (wp get_cap_wp getCTE_wp')
apply (fastforce elim: cte_wp_at_weakenE intro: Q)
apply (fastforce elim: cte_wp_at_weakenE' intro: Q')
done
lemma check_cap_at_corres_weak:
assumes r: "cap_relation cap cap'"
assumes c: "corres dc P P' f f'"
shows "corres dc (P and cte_at slot and invs) (P' and pspace_aligned' and pspace_distinct')
(check_cap_at cap slot f)
(checkCapAt cap' (cte_map slot) f')"
apply (rule check_cap_at_corres, rule r, rule c)
apply auto
done
defs
assertDerived_def:
"assertDerived src cap f \<equiv>
do stateAssert (\<lambda>s. cte_wp_at' (is_derived' (ctes_of s) src cap o cteCap) src s) []; f od"
lemma checked_insert_corres:
"cap_relation new_cap newCap \<Longrightarrow>
corres dc (einvs and cte_wp_at (\<lambda>c. c = cap.NullCap) (target, ref)
and cte_at slot and K (is_cnode_or_valid_arch new_cap
\<and> (is_pt_cap new_cap \<or> is_pd_cap new_cap
\<longrightarrow> cap_asid new_cap \<noteq> None))
and cte_wp_at (\<lambda>c. obj_refs c = obj_refs new_cap
\<longrightarrow> table_cap_ref c = table_cap_ref new_cap \<and>
pt_pd_asid c = pt_pd_asid new_cap) src_slot)
(invs' and cte_wp_at' (\<lambda>cte. cteCap cte = NullCap) (cte_map (target, ref))
and valid_cap' newCap)
(check_cap_at new_cap src_slot
(check_cap_at (cap.ThreadCap target) slot
(cap_insert new_cap src_slot (target, ref))))
(checkCapAt newCap (cte_map src_slot)
(checkCapAt (ThreadCap target) (cte_map slot)
(assertDerived (cte_map src_slot) newCap (cteInsert newCap (cte_map src_slot) (cte_map (target, ref))))))"
apply (rule corres_guard_imp)
apply (rule_tac P="cte_wp_at (\<lambda>c. c = cap.NullCap) (target, ref) and
cte_at slot and
cte_wp_at (\<lambda>c. obj_refs c = obj_refs new_cap
\<longrightarrow> table_cap_ref c = table_cap_ref new_cap \<and> pt_pd_asid c = pt_pd_asid new_cap) src_slot
and einvs and K (is_cnode_or_valid_arch new_cap
\<and> (is_pt_cap new_cap \<or> is_pd_cap new_cap
\<longrightarrow> cap_asid new_cap \<noteq> None))"
and
P'="cte_wp_at' (\<lambda>c. cteCap c = NullCap) (cte_map (target, ref))
and invs' and valid_cap' newCap"
in check_cap_at_corres, assumption)
apply (rule check_cap_at_corres_weak, simp)
apply (unfold assertDerived_def)[1]
apply (rule corres_stateAssert_implied [where P'=\<top>])
apply simp
apply (erule cins_corres [OF _ refl refl])
apply clarsimp
apply (drule cte_wp_at_norm [where p=src_slot])
apply (case_tac src_slot)
apply (clarsimp simp: state_relation_def)
apply (drule (1) pspace_relation_cte_wp_at)
apply fastforce
apply fastforce
apply (clarsimp simp: cte_wp_at_ctes_of)
apply (erule (2) is_derived_eq [THEN iffD1])
apply (erule cte_wp_at_weakenE, rule TrueI)
apply assumption
apply clarsimp
apply (rule conjI, fastforce)+
apply (cases src_slot)
apply (clarsimp simp: cte_wp_at_caps_of_state)
apply (rule conjI)
apply (frule same_object_as_cap_master)
apply (clarsimp simp: cap_master_cap_simps is_cnode_or_valid_arch_def
is_cap_simps is_valid_vtable_root_def
dest!: cap_master_cap_eqDs)
apply (erule(1) checked_insert_is_derived)
apply simp
apply simp
apply fastforce
apply (clarsimp simp: cte_wp_at_caps_of_state)
apply clarsimp
apply fastforce
done
lemma capBadgeNone_masters:
"capMasterCap cap = capMasterCap cap'
\<Longrightarrow> (capBadge cap = None) = (capBadge cap' = None)"
apply (rule master_eqI)
apply (auto simp add: capBadge_def capMasterCap_def isCap_simps
split: capability.split)
done
definition
"pt_pd_asid' cap \<equiv> case cap of
ArchObjectCap (PageTableCap _ (Some (asid, _))) \<Rightarrow> Some asid
| ArchObjectCap (PageDirectoryCap _ (Some asid)) \<Rightarrow> Some asid
| _ \<Rightarrow> None"
lemmas pt_pd_asid'_simps [simp] =
pt_pd_asid'_def [split_simps capability.split arch_capability.split option.split prod.split]
lemma checked_insert_tcb_invs'[wp]:
"\<lbrace>invs' and cte_wp_at' (\<lambda>cte. cteCap cte = NullCap) slot
and valid_cap' new_cap
and K (capBadge new_cap = None)
and K (slot \<in> cte_refs' (ThreadCap target) 0)
and K (\<not> isReplyCap new_cap \<and> \<not>isIRQControlCap new_cap)\<rbrace>
checkCapAt new_cap src_slot
(checkCapAt (ThreadCap target) slot'
(assertDerived src_slot new_cap (cteInsert new_cap src_slot slot))) \<lbrace>\<lambda>rv. invs'\<rbrace>"
apply (simp add: checkCapAt_def liftM_def assertDerived_def stateAssert_def)
apply (rule hoare_pre)
apply (wp getCTE_cteCap_wp cteInsert_invs)
apply (clarsimp split: option.splits)
apply (subst(asm) tree_cte_cteCap_eq[unfolded o_def])
apply (clarsimp split: option.splits)
apply (rule conjI)
apply (clarsimp simp: sameObjectAs_def3)
apply (clarsimp simp: tree_cte_cteCap_eq
is_derived'_def
ex_cte_cap_to'_cteCap)
apply (erule sameObjectAsE)+
apply (clarsimp simp: badge_derived'_def)
apply (frule capBadgeNone_masters, simp)
apply (rule conjI)
apply (rule_tac x=slot' in exI)
apply fastforce
apply (clarsimp simp: isCap_simps cteCaps_of_def)
apply (erule(1) valid_irq_handlers_ctes_ofD)
apply (clarsimp simp:invs'_def valid_state'_def)
done
lemma checkCap_inv:
assumes x: "\<lbrace>P\<rbrace> f \<lbrace>\<lambda>rv. P\<rbrace>"
shows "\<lbrace>P\<rbrace> checkCapAt cap slot f \<lbrace>\<lambda>rv. P\<rbrace>"
unfolding checkCapAt_def
by (wp x | simp)+
lemma isValidVTableRootD:
"isValidVTableRoot cap
\<Longrightarrow> isArchObjectCap cap \<and> isPageDirectoryCap (capCap cap)
\<and> capPDMappedASID (capCap cap) \<noteq> None"
by (simp add: isValidVTableRoot_def
ARM_H.isValidVTableRoot_def
isCap_simps
split: capability.split_asm arch_capability.split_asm
option.split_asm)
declare in_image_op_plus[simp]
lemma assertDerived_wp:
"\<lbrace>P and (\<lambda>s. cte_wp_at' (is_derived' (ctes_of s) slot cap o cteCap) slot s)\<rbrace> f \<lbrace>Q\<rbrace> \<Longrightarrow>
\<lbrace>P\<rbrace> assertDerived slot cap f \<lbrace>Q\<rbrace>"
apply (simp add: assertDerived_def)
apply (rule hoare_pre)
apply (wp|assumption)+
apply simp
done
crunch sch_act_simple[wp]: setPriority sch_act_simple
(wp: ssa_sch_act_simple crunch_wps lift: sch_act_simple_lift simp: crunch_simps)
(* For some reason, when this was embedded in a larger expression clarsimp wouldn't remove it. Adding it as a simp rule does *)
lemma inQ_tc_corres_helper:
"(\<forall>d p. (\<exists>tcb. tcbQueued tcb \<and> tcbPriority tcb = p \<and> tcbDomain tcb = d \<and> (tcbQueued tcb \<longrightarrow> tcbDomain tcb \<noteq> d)) \<longrightarrow> a \<notin> set (ksReadyQueues s (d, p))) = True"
by clarsimp
abbreviation "valid_option_prio \<equiv> case_option True (\<lambda>p. p \<le> maxPriority)"
definition valid_tcb_invocation :: "tcbinvocation \<Rightarrow> bool" where
"valid_tcb_invocation i \<equiv> case i of
ThreadControl _ _ _ p _ _ _ \<Rightarrow> valid_option_prio p
| _ \<Rightarrow> True"
lemma tc_corres:
assumes x: "newroot_rel e e'" and y: "newroot_rel f f'" and p: "p = p'"
and z: "(case g of None \<Rightarrow> g' = None
| Some (vptr, g'') \<Rightarrow> \<exists>g'''. g' = Some (vptr, g''')
\<and> newroot_rel g'' g''')"
and sl: "{e, f, option_map undefined g} \<noteq> {None} \<longrightarrow> sl' = cte_map slot"
shows
"corres (intr \<oplus> op =)
(einvs and simple_sched_action and tcb_at a and
(\<lambda>s. {e, f, option_map undefined g} \<noteq> {None} \<longrightarrow> cte_at slot s) and
case_option \<top> (valid_cap o fst) e and
case_option \<top> (cte_at o snd) e and
case_option \<top> (no_cap_to_obj_dr_emp o fst) e and
K (case_option True (is_cnode_cap o fst) e) and
case_option \<top> (valid_cap o fst) f and
case_option \<top> (cte_at o snd) f and
case_option \<top> (no_cap_to_obj_dr_emp o fst) f and
K (case_option True (is_valid_vtable_root o fst) f)
and case_option \<top> (case_option \<top> (cte_at o snd) o snd) g
and case_option \<top> (case_option \<top> (no_cap_to_obj_dr_emp o fst) o snd) g
and case_option \<top> (case_option \<top> (valid_cap o fst) o snd) g
and K (case_option True ((\<lambda>v. is_aligned v msg_align_bits) o fst) g)
and K (case_option True (case_option True (is_cnode_or_valid_arch o fst) o snd) g))
(invs' and sch_act_simple and case_option \<top> (valid_cap' o fst) e' and
(\<lambda>s. {e', f', option_map undefined g'} \<noteq> {None} \<longrightarrow> cte_at' (cte_map slot) s) and
K (case_option True (isCNodeCap o fst) e') and
case_option \<top> (valid_cap' o fst) f' and
K (case_option True (isValidVTableRoot o fst) f') and
case_option \<top> (case_option \<top> (valid_cap' o fst) o snd) g' and
tcb_at' a and ex_nonz_cap_to' a and K (valid_option_prio p'))
(invoke_tcb (tcb_invocation.ThreadControl a slot (option_map to_bl b') p e f g))
(invokeTCB (tcbinvocation.ThreadControl a sl' b' p' e' f' g'))"
proof -
have P: "\<And>t v. corres dc
(tcb_at t)
(tcb_at' t)
(option_update_thread t (tcb_fault_handler_update o (%x _. x))
(option_map to_bl v))
(case v of None \<Rightarrow> return ()
| Some x \<Rightarrow> threadSet (tcbFaultHandler_update (%_. x)) t)"
apply (rule out_corres, simp_all add: exst_same_def)
apply (case_tac v, simp_all add: out_rel_def)
apply (safe, case_tac tcb', simp add: tcb_relation_def split: option.split)
done
have R: "\<And>t v. corres dc
(tcb_at t)
(tcb_at' t)
(option_update_thread t (tcb_ipc_buffer_update o (%x _. x)) v)
(case v of None \<Rightarrow> return ()
| Some x \<Rightarrow> threadSet (tcbIPCBuffer_update (%_. x)) t)"
apply (rule out_corres, simp_all add: exst_same_def)
apply (case_tac v, simp_all add: out_rel_def)
apply (safe, case_tac tcb', simp add: tcb_relation_def)
done
have S: "\<And>t x. corres dc (einvs and tcb_at t) (invs' and tcb_at' t and valid_objs' and K (valid_option_prio p'))
(case p of None \<Rightarrow> return () | Some prio \<Rightarrow> set_priority t prio)
(case_option (return ()) (setPriority t) p')"
apply (case_tac p, simp_all add: sp_corres p)
done
have T: "\<And>x x' ref getfn target.
\<lbrakk> newroot_rel x x'; getfn = return (cte_map (target, ref));
x \<noteq> None \<longrightarrow> {e, f, option_map undefined g} \<noteq> {None} \<rbrakk> \<Longrightarrow>
corres (intr \<oplus> dc)
(einvs and simple_sched_action and cte_at (target, ref) and emptyable (target, ref) and
(\<lambda>s. \<forall>(sl, c) \<in> (case x of None \<Rightarrow> {} | Some (c, sl) \<Rightarrow> {(sl, c), (slot, c)}).
cte_at sl s \<and> no_cap_to_obj_dr_emp c s \<and> valid_cap c s)
and K (case x of None \<Rightarrow> True
| Some (c, sl) \<Rightarrow> is_cnode_or_valid_arch c))
(invs' and sch_act_simple and cte_at' (cte_map (target, ref)) and
(\<lambda>s. \<forall>cp \<in> (case x' of None \<Rightarrow> {} | Some (c, sl) \<Rightarrow> {c}). s \<turnstile>' cp))
(case x of None \<Rightarrow> returnOk ()
| Some pr \<Rightarrow> case_prod (\<lambda>new_cap src_slot.
doE cap_delete (target, ref);
liftE $ check_cap_at new_cap src_slot $
check_cap_at (cap.ThreadCap target) slot $
cap_insert new_cap src_slot (target, ref)
odE) pr)
(case x' of
None \<Rightarrow> returnOk ()
| Some pr \<Rightarrow> (\<lambda>(newCap, srcSlot).
do slot \<leftarrow> getfn;
doE uu \<leftarrow> cteDelete slot True;
liftE (checkCapAt newCap srcSlot
(checkCapAt (capability.ThreadCap target) sl'
(assertDerived srcSlot newCap (cteInsert newCap srcSlot slot))))
odE
od) pr)"
apply (case_tac "x = None")
apply (simp add: newroot_rel_def returnOk_def)
apply (drule(1) mp, drule mp [OF sl])
apply (clarsimp simp add: newroot_rel_def returnOk_def split_def)
apply (rule corres_gen_asm)
apply (rule corres_guard_imp)
apply (rule corres_split_norE [OF _ cap_delete_corres])
apply (simp del: dc_simp)
apply (erule checked_insert_corres)
apply (fold validE_R_def)
apply (wp cap_delete_deletes cap_delete_cte_at cap_delete_valid_cap
| strengthen use_no_cap_to_obj_asid_strg)+
apply (wp cteDelete_invs' cteDelete_deletes cteDelete_aligned'
cteDelete_distinct')
apply (clarsimp dest!: is_cnode_or_valid_arch_cap_asid)
apply clarsimp
done
have U2: "getThreadBufferSlot a = return (cte_map (a, tcb_cnode_index 4))"
by (simp add: getThreadBufferSlot_def locateSlot_conv
cte_map_def tcb_cnode_index_def tcbIPCBufferSlot_def
cte_level_bits_def)
have T2: "corres (intr \<oplus> dc)
(einvs and simple_sched_action and tcb_at a and
(\<lambda>s. \<forall>(sl, c) \<in> (case g of None \<Rightarrow> {} | Some (x, v) \<Rightarrow> {(slot, cap.NullCap)} \<union>
(case v of None \<Rightarrow> {} | Some (c, sl) \<Rightarrow> {(sl, c), (slot, c)})).
cte_at sl s \<and> no_cap_to_obj_dr_emp c s \<and> valid_cap c s)
and K (case g of None \<Rightarrow> True | Some (x, v) \<Rightarrow> (case v of
None \<Rightarrow> True | Some (c, sl) \<Rightarrow> is_cnode_or_valid_arch c
\<and> is_aligned x msg_align_bits)))
(invs' and sch_act_simple and tcb_at' a and
(\<lambda>s. \<forall>cp \<in> (case g' of None \<Rightarrow> {} | Some (x, v) \<Rightarrow> (case v of
None \<Rightarrow> {} | Some (c, sl) \<Rightarrow> {c})). s \<turnstile>' cp))
(case_option (returnOk ())
(case_prod
(\<lambda>ptr frame.
doE cap_delete (a, tcb_cnode_index 4);
do y \<leftarrow> thread_set (tcb_ipc_buffer_update (\<lambda>_. ptr)) a;
liftE $
case_option (return ())
(case_prod
(\<lambda>new_cap src_slot.
check_cap_at new_cap src_slot $
check_cap_at (cap.ThreadCap a) slot $
cap_insert new_cap src_slot (a, tcb_cnode_index 4)))
frame
od
odE))
g)
(case_option (returnOk ())
(\<lambda>(ptr, frame).
do bufferSlot \<leftarrow> getThreadBufferSlot a;
doE y \<leftarrow> cteDelete bufferSlot True;
do y \<leftarrow> threadSet (tcbIPCBuffer_update (\<lambda>_. ptr)) a;
liftE
(case_option (return ())
(case_prod
(\<lambda>newCap srcSlot.
checkCapAt newCap srcSlot $
checkCapAt
(capability.ThreadCap a)
sl' $
assertDerived srcSlot newCap $
cteInsert newCap srcSlot bufferSlot))
frame)
od odE od)
g')"
using z sl
apply -
apply (rule corres_guard_imp[where P=P and P'=P'
and Q="P and cte_at (a, tcb_cnode_index 4)"
and Q'="P' and cte_at' (cte_map (a, cap))" for P P' a cap])
apply (cases g)
apply (simp, simp add: returnOk_def)
apply (clarsimp simp: liftME_def[symmetric] U2 liftE_bindE)
apply (case_tac b, simp_all add: newroot_rel_def)
apply (rule corres_guard_imp)
apply (rule corres_split_norE)
apply (rule corres_split_nor)
apply (rule corres_trivial, simp)
apply (rule threadset_corres,
(simp add: tcb_relation_def), (simp add: exst_same_def)+)[1]
apply wp
apply (rule cap_delete_corres)
apply wp
apply (fastforce simp: emptyable_def)
apply fastforce
apply clarsimp
apply (rule corres_guard_imp)
apply (rule corres_split_norE [OF _ cap_delete_corres])
apply (rule_tac F="is_aligned aa msg_align_bits"
in corres_gen_asm)
apply (rule corres_split_nor)
apply simp
apply (erule checked_insert_corres)
apply (rule threadset_corres,
simp add: tcb_relation_def, (simp add: exst_same_def)+)
apply (wp thread_set_tcb_ipc_buffer_cap_cleared_invs
thread_set_cte_wp_at_trivial thread_set_not_state_valid_sched
| simp add: ran_tcb_cap_cases)+
apply (wp threadSet_invs_trivial
threadSet_cte_wp_at' | simp)+
apply (wp cap_delete_deletes cap_delete_cte_at
cap_delete_valid_cap cteDelete_deletes
cteDelete_invs'
| strengthen use_no_cap_to_obj_asid_strg
| clarsimp simp: inQ_def inQ_tc_corres_helper)+
apply (clarsimp simp: cte_wp_at_caps_of_state
dest!: is_cnode_or_valid_arch_cap_asid)
apply (fastforce simp: emptyable_def)
apply (clarsimp simp: inQ_def)
apply (clarsimp simp: obj_at_def is_tcb)
apply (rule cte_wp_at_tcbI, simp, fastforce, simp)
apply (clarsimp simp: cte_map_def tcb_cnode_index_def obj_at'_def
projectKOs objBits_simps)
apply (erule(2) cte_wp_at_tcbI', fastforce, simp)
done
have U: "getThreadCSpaceRoot a = return (cte_map (a, tcb_cnode_index 0))"
apply (clarsimp simp add: getThreadCSpaceRoot)
apply (simp add: cte_map_def tcb_cnode_index_def
cte_level_bits_def word_bits_def)
done
have V: "getThreadVSpaceRoot a = return (cte_map (a, tcb_cnode_index 1))"
apply (clarsimp simp add: getThreadVSpaceRoot)
apply (simp add: cte_map_def tcb_cnode_index_def to_bl_1
cte_level_bits_def word_bits_def)
done
have X: "\<And>x P Q R M. (\<And>y. x = Some y \<Longrightarrow> \<lbrace>P y\<rbrace> M y \<lbrace>Q\<rbrace>,\<lbrace>R\<rbrace>)
\<Longrightarrow> \<lbrace>case_option (Q ()) P x\<rbrace> case_option (returnOk ()) M x \<lbrace>Q\<rbrace>,\<lbrace>R\<rbrace>"
by (case_tac x, simp_all, wp)
have Y: "\<And>x P Q M. (\<And>y. x = Some y \<Longrightarrow> \<lbrace>P y\<rbrace> M y \<lbrace>Q\<rbrace>,-)
\<Longrightarrow> \<lbrace>case_option (Q ()) P x\<rbrace> case_option (returnOk ()) M x \<lbrace>Q\<rbrace>,-"
by (case_tac x, simp_all, wp)
have Z: "\<And>P f R Q x. \<lbrace>P\<rbrace> f \<lbrace>\<lambda>rv. Q and R\<rbrace> \<Longrightarrow> \<lbrace>P\<rbrace> f \<lbrace>\<lambda>rv. case_option Q (\<lambda>y. R) x\<rbrace>"
apply (rule hoare_post_imp)
defer
apply assumption
apply (case_tac x, simp_all)
done
have A: "\<And>x P Q M. (\<And>y. x = Some y \<Longrightarrow> \<lbrace>P y\<rbrace> M y \<lbrace>Q\<rbrace>)
\<Longrightarrow> \<lbrace>case_option (Q ()) P x\<rbrace> case_option (return ()) M x \<lbrace>Q\<rbrace>"
by (case_tac x, simp_all, wp)
have B: "\<And>t v. \<lbrace>invs' and tcb_at' t\<rbrace> threadSet (tcbFaultHandler_update v) t \<lbrace>\<lambda>rv. invs'\<rbrace>"
by (wp threadSet_invs_trivial | clarsimp simp: inQ_def)+
note stuff = Z B out_invs_trivial hoare_case_option_wp
hoare_vcg_const_Ball_lift hoare_vcg_const_Ball_lift_R
cap_delete_deletes cap_delete_valid_cap out_valid_objs
cap_insert_objs
cteDelete_deletes cteDelete_sch_act_simple
out_valid_cap out_cte_at out_tcb_valid out_emptyable
CSpaceInv_AI.cap_insert_valid_cap cap_insert_cte_at cap_delete_cte_at
cap_delete_tcb cteDelete_invs' checkCap_inv [where P="valid_cap' c0" for c0]
check_cap_inv[where P="tcb_at p0" for p0] checkCap_inv [where P="tcb_at' p0" for p0]
check_cap_inv[where P="cte_at p0" for p0] checkCap_inv [where P="cte_at' p0" for p0]
check_cap_inv[where P="valid_cap c" for c] checkCap_inv [where P="valid_cap' c" for c]
check_cap_inv[where P="tcb_cap_valid c p1" for c p1]
check_cap_inv[where P=valid_sched]
check_cap_inv[where P=simple_sched_action]
checkCap_inv [where P=sch_act_simple]
out_no_cap_to_trivial [OF ball_tcb_cap_casesI]
checked_insert_no_cap_to
note if_cong [cong] option.case_cong [cong]
show ?thesis
apply (simp add: invokeTCB_def liftE_bindE)
apply (rule corres_guard_imp)
apply (rule corres_split_nor [OF _ P])
apply (rule corres_split_nor [OF _ S, simplified])
apply (rule corres_split_norE [OF _ T [OF x U], simplified])
apply (rule corres_split_norE [OF _ T [OF y V], simplified])
apply (simp add: liftME_def[symmetric] o_def dc_def[symmetric])
apply (rule T2[simplified])
apply (wp stuff hoare_vcg_all_lift_R hoare_vcg_all_lift
hoare_vcg_const_imp_lift_R hoare_vcg_const_imp_lift
threadSet_valid_objs'
typ_at_lifts [OF setPriority_typ_at'] setP_invs'
assertDerived_wp threadSet_cap_to' out_pred_tcb_at_preserved
| simp add: ran_tcb_cap_cases split_def U V
emptyable_def
| wpc
| strengthen tcb_cap_always_valid_strg
use_no_cap_to_obj_asid_strg
| wp_once sch_act_simple_lift hoare_drop_imps
| (erule exE, clarsimp simp: word_bits_def))+
apply (clarsimp simp: tcb_at_cte_at_0 tcb_at_cte_at_1[simplified]
tcb_cap_valid_def is_cnode_or_valid_arch_def
tcb_at_st_tcb_at[symmetric] invs_valid_objs
emptyable_def obj_ref_none_no_asid
no_cap_to_obj_with_diff_ref_Null
is_valid_vtable_root_def is_cap_simps
cap_asid_def vs_cap_ref_def
split: option.split_asm)
apply (clarsimp simp: invs'_def valid_state'_def valid_pspace'_def
cte_map_tcb_0 cte_map_tcb_1[simplified]
tcb_at_cte_at' cte_at_tcb_at_16' isCap_simps
domIff valid_tcb'_def tcb_cte_cases_def
split: option.split_asm
dest!: isValidVTableRootD) (* long *)
done
qed
lemma isReplyCapD:
"isReplyCap cap \<Longrightarrow> \<exists>ptr master. cap = capability.ReplyCap ptr master"
by (simp add: isCap_simps)
lemmas threadSet_ipcbuffer_trivial
= threadSet_invs_trivial[where F="tcbIPCBuffer_update F'" for F',
simplified inQ_def, simplified]
crunch cap_to'[wp]: setPriority "ex_nonz_cap_to' a"
(simp: crunch_simps)
lemma cteInsert_sa_simple[wp]:
"\<lbrace>sch_act_simple\<rbrace> cteInsert newCap srcSlot destSlot \<lbrace>\<lambda>_. sch_act_simple\<rbrace>"
by (simp add: sch_act_simple_def, wp)
lemma tc_invs':
"\<lbrace>invs' and sch_act_simple and tcb_at' a and ex_nonz_cap_to' a and
K (valid_option_prio d) and
case_option \<top> (valid_cap' o fst) e' and
K (case_option True (isCNodeCap o fst) e') and
case_option \<top> (valid_cap' o fst) f' and
K (case_option True (isValidVTableRoot o fst) f') and
case_option \<top> (valid_cap') (case_option None (case_option None (Some o fst) o snd) g) and
K (case_option True isArchObjectCap (case_option None (case_option None (Some o fst) o snd) g))
and K (case_option True (swp is_aligned msg_align_bits o fst) g) \<rbrace>
invokeTCB (tcbinvocation.ThreadControl a sl b' d e' f' g)
\<lbrace>\<lambda>rv. invs'\<rbrace>"
apply (rule hoare_gen_asm)
apply (simp add: split_def invokeTCB_def getThreadCSpaceRoot getThreadVSpaceRoot
getThreadBufferSlot_def locateSlot_conv
cong: option.case_cong)
apply (rule hoare_walk_assmsE)
apply (clarsimp simp: pred_conj_def option.splits [where P="\<lambda>x. x s" for s])
apply ((wp case_option_wp threadSet_invs_trivial static_imp_wp
hoare_vcg_all_lift threadSet_cap_to' | clarsimp simp: inQ_def)+)[2]
apply (rule hoare_walk_assmsE)
apply (clarsimp simp: pred_conj_def option.splits [where P="\<lambda>x. x s" for s])
apply ((wp case_option_wp threadSet_invs_trivial setP_invs' static_imp_wp
hoare_vcg_all_lift threadSet_cap_to' | clarsimp simp: inQ_def)+)[2]
apply (rule hoare_pre)
apply ((simp only: simp_thms cases_simp cong: conj_cong
| (wp cteDelete_deletes cteDelete_invs' cteDelete_sch_act_simple
threadSet_ipcbuffer_trivial
checkCap_inv[where P="tcb_at' t" for t]
checkCap_inv[where P="valid_cap' c" for c]
checkCap_inv[where P="\<lambda>s. P (ksReadyQueues s)"]
checkCap_inv[where P=sch_act_simple]
hoare_vcg_const_imp_lift_R
typ_at_lifts [OF setPriority_typ_at']
assertDerived_wp
threadSet_cte_wp_at'
hoare_vcg_all_lift_R
hoare_vcg_all_lift
static_imp_wp
)[1]
| wpc
| simp add: inQ_def
| wp hoare_vcg_conj_liftE1 cteDelete_invs' cteDelete_deletes
hoare_vcg_const_imp_lift
)+)
apply (clarsimp simp: tcb_cte_cases_def cte_level_bits_def
tcbIPCBufferSlot_def)
apply (auto dest!: isCapDs isReplyCapD isValidVTableRootD
simp: isCap_simps)
done
lemma setSchedulerAction_invs'[wp]:
"\<lbrace>invs' and sch_act_wf sa
and (\<lambda>s. sa = ResumeCurrentThread
\<longrightarrow> obj_at' (Not \<circ> tcbQueued) (ksCurThread s) s)
and (\<lambda>s. sa = ResumeCurrentThread
\<longrightarrow> ksCurThread s = ksIdleThread s \<or> tcb_in_cur_domain' (ksCurThread s) s)\<rbrace>
setSchedulerAction sa
\<lbrace>\<lambda>rv. invs'\<rbrace>"
apply (simp add: setSchedulerAction_def)
apply wp
apply (clarsimp simp add: invs'_def valid_state'_def valid_irq_node'_def
valid_queues_def valid_queues_no_bitmap_def bitmapQ_defs cur_tcb'_def
ct_not_inQ_def)
apply (simp add:ct_idle_or_in_cur_domain'_def)
done
end
consts
copyregsets_map :: "arch_copy_register_sets \<Rightarrow> Arch.copy_register_sets"
context begin interpretation Arch . (*FIXME: arch_split*)
primrec
tcbinv_relation :: "tcb_invocation \<Rightarrow> tcbinvocation \<Rightarrow> bool"
where
"tcbinv_relation (tcb_invocation.ReadRegisters a b c d) x
= (x = tcbinvocation.ReadRegisters a b c (copyregsets_map d))"
| "tcbinv_relation (tcb_invocation.WriteRegisters a b c d) x
= (x = tcbinvocation.WriteRegisters a b c (copyregsets_map d))"
| "tcbinv_relation (tcb_invocation.CopyRegisters a b c d e f g) x
= (x = tcbinvocation.CopyRegisters a b c d e f (copyregsets_map g))"
| "tcbinv_relation (tcb_invocation.ThreadControl a sl b d e f g) x
= (\<exists>b' e' f' sl' g'. b = option_map to_bl b' \<and>
newroot_rel e e' \<and> newroot_rel f f' \<and>
({e, f, option_map undefined g} \<noteq> {None}
\<longrightarrow> sl' = cte_map sl) \<and>
(case g of None \<Rightarrow> g' = None
| Some (vptr, g'') \<Rightarrow> \<exists>g'''. g' = Some (vptr, g''')
\<and> newroot_rel g'' g''') \<and>
x = tcbinvocation.ThreadControl a sl' b' d e' f' g')"
| "tcbinv_relation (tcb_invocation.Suspend a) x
= (x = tcbinvocation.Suspend a)"
| "tcbinv_relation (tcb_invocation.Resume a) x
= (x = tcbinvocation.Resume a)"
| "tcbinv_relation (tcb_invocation.NotificationControl t ntfnptr) x
= (x = tcbinvocation.NotificationControl t ntfnptr)"
primrec
tcb_inv_wf' :: "tcbinvocation \<Rightarrow> kernel_state \<Rightarrow> bool"
where
"tcb_inv_wf' (tcbinvocation.Suspend t)
= (tcb_at' t and ex_nonz_cap_to' t)"
| "tcb_inv_wf' (tcbinvocation.Resume t)
= (tcb_at' t and ex_nonz_cap_to' t)"
| "tcb_inv_wf' (tcbinvocation.ThreadControl t sl fe p croot vroot buf)
= (tcb_at' t and ex_nonz_cap_to' t and
K (valid_option_prio p) and
case_option \<top> (valid_cap' o fst) croot and
K (case_option True (isCNodeCap o fst) croot) and
case_option \<top> (valid_cap' o fst) vroot and
K (case_option True (isValidVTableRoot o fst) vroot) and
case_option \<top> (case_option \<top> (valid_cap' o fst) o snd) buf and
case_option \<top> (case_option \<top> (cte_at' o snd) o snd) buf and
K (case_option True (swp is_aligned msg_align_bits o fst) buf) and
K (case_option True (case_option True (isArchObjectCap o fst) o snd) buf) and
(\<lambda>s. {croot, vroot, option_map undefined buf} \<noteq> {None}
\<longrightarrow> cte_at' sl s))"
| "tcb_inv_wf' (tcbinvocation.ReadRegisters src susp n arch)
= (tcb_at' src and ex_nonz_cap_to' src)"
| "tcb_inv_wf' (tcbinvocation.WriteRegisters dest resume values arch)
= (tcb_at' dest and ex_nonz_cap_to' dest)"
| "tcb_inv_wf' (tcbinvocation.CopyRegisters dest src suspend_source resume_target
trans_frame trans_int trans_arch)
= (tcb_at' dest and tcb_at' src and ex_nonz_cap_to' src and ex_nonz_cap_to' dest)"
| "tcb_inv_wf' (tcbinvocation.NotificationControl t ntfn)
= (tcb_at' t and ex_nonz_cap_to' t
and (case ntfn of None \<Rightarrow> \<top>
| Some ntfnptr \<Rightarrow> obj_at' (\<lambda>ko. ntfnBoundTCB ko = None
\<and> (\<forall>q. ntfnObj ko \<noteq> WaitingNtfn q)) ntfnptr
and ex_nonz_cap_to' ntfnptr
and bound_tcb_at' (op = None) t) )"
lemma tcbinv_corres:
"tcbinv_relation ti ti' \<Longrightarrow>
corres (intr \<oplus> op =)
(einvs and simple_sched_action and tcb_inv_wf ti)
(invs' and sch_act_simple and tcb_inv_wf' ti')
(invoke_tcb ti) (invokeTCB ti')"
apply (case_tac ti, simp_all only: tcbinv_relation.simps valid_tcb_invocation_def)
apply (rule corres_guard_imp [OF writereg_corres], simp+)[1]
apply (rule corres_guard_imp [OF readreg_corres], simp+)[1]
apply (rule corres_guard_imp [OF copyreg_corres], simp+)[1]
apply (clarsimp simp del: invoke_tcb.simps)
apply (rename_tac word one t2 t3 t4 t5 t6 t7 t8 t9 t10 t11)
apply (rule_tac F="is_aligned word 5" in corres_req)
apply (clarsimp simp add: is_aligned_weaken [OF tcb_aligned])
apply (rule corres_guard_imp [OF tc_corres], clarsimp+)
apply (clarsimp simp: is_cnode_or_valid_arch_def
split: option.split option.split_asm)
apply clarsimp
apply (auto split: option.split_asm simp: newroot_rel_def)[1]
apply (simp add: invokeTCB_def liftM_def[symmetric]
o_def dc_def[symmetric])
apply (rule corres_guard_imp [OF suspend_corres], simp+)
apply (simp add: invokeTCB_def liftM_def[symmetric]
o_def dc_def[symmetric])
apply (rule corres_guard_imp [OF restart_corres], simp+)
apply (simp add:invokeTCB_def)
apply (rename_tac option)
apply (case_tac option)
apply simp
apply (rule corres_guard_imp)
apply (rule corres_split[OF _ unbind_notification_corres])
apply (rule corres_trivial, simp)
apply wp
apply (clarsimp)
apply clarsimp
apply simp
apply (rule corres_guard_imp)
apply (rule corres_split[OF _ bind_notification_corres])
apply (rule corres_trivial, simp)
apply wp
apply clarsimp
apply (clarsimp simp: obj_at_def is_ntfn)
apply (clarsimp simp: obj_at'_def projectKOs)
done
lemma tcbBoundNotification_caps_safe[simp]:
"\<forall>(getF, setF)\<in>ran tcb_cte_cases.
getF (tcbBoundNotification_update (\<lambda>_. Some ntfnptr) tcb) = getF tcb"
by (case_tac tcb, simp add: tcb_cte_cases_def)
lemma threadSet_tcbBoundNotification_valid_pspace:
"\<lbrace>valid_pspace' and ntfn_at' ntfnptr\<rbrace>
threadSet (tcbBoundNotification_update (\<lambda>_. Some ntfnptr)) t
\<lbrace>\<lambda>rv. valid_pspace'\<rbrace>"
apply (simp add: valid_pspace'_def threadSet_def)
apply (rule hoare_pre)
apply (wp setObject_tcb_valid_objs getObject_tcb_wp)
apply (auto simp: projectKOs obj_at'_def
valid_obj'_def valid_tcb'_def valid_bound_ntfn'_def
tcb_cte_cases_def)
done
lemma threadSet_tcbBoundNotification_state_refs_of':
"\<lbrace>(\<lambda>s. P ((state_refs_of' s)(t := {(ntfnptr, TCBBound)} \<union> state_refs_of' s t)))
and obj_at' (\<lambda>tcb. tcbBoundNotification tcb = None) t\<rbrace>
threadSet (tcbBoundNotification_update (\<lambda>_. Some ntfnptr)) t
\<lbrace>\<lambda>_ s. P (state_refs_of' s)\<rbrace>"
apply (simp add: threadSet_def)
apply (wp getObject_tcb_wp)
apply (auto simp: fun_upd_def obj_at'_def projectKOs state_refs_of'_def cong: if_cong)
done
lemma valid_bound_ntfn_lift:
assumes P: "\<And>P T p. \<lbrace>\<lambda>s. P (typ_at' T p s)\<rbrace> f \<lbrace>\<lambda>rv s. P (typ_at' T p s)\<rbrace>"
shows "\<lbrace>\<lambda>s. valid_bound_ntfn' a s\<rbrace> f \<lbrace>\<lambda>rv s. valid_bound_ntfn' a s\<rbrace>"
apply (simp add: valid_bound_ntfn'_def, case_tac a, simp_all)
apply (wp typ_at_lifts[OF P])
done
lemma bindNotification_invs':
"\<lbrace>bound_tcb_at' (op = None) tcbptr
and ex_nonz_cap_to' ntfnptr
and ex_nonz_cap_to' tcbptr
and obj_at' (\<lambda>ntfn. ntfnBoundTCB ntfn = None \<and> (\<forall>q. ntfnObj ntfn \<noteq> WaitingNtfn q)) ntfnptr
and invs'\<rbrace>
bindNotification tcbptr ntfnptr
\<lbrace>\<lambda>_. invs'\<rbrace>"
apply (simp add: bindNotification_def invs'_def valid_state'_def)
apply (rule hoare_seq_ext[OF _ get_ntfn_sp'])
apply (wp set_ntfn_valid_pspace' sbn_sch_act' sbn_valid_queues valid_irq_node_lift
setBoundNotification_ct_not_inQ valid_bound_ntfn_lift
| clarsimp dest!: global'_no_ex_cap)+
apply simp_all
apply (clarsimp simp: valid_pspace'_def)
apply (frule_tac P="\<lambda>k. k=ntfn" in obj_at_valid_objs', simp)
apply (clarsimp simp: valid_obj'_def projectKOs valid_ntfn'_def obj_at'_def
dest!: pred_tcb_at'
split: ntfn.splits)
apply (rule conjI, rule impI, clarsimp dest!: pred_tcb_at' simp: obj_at'_def projectKOs)
apply (clarsimp, erule delta_sym_refs)
apply (fastforce simp: ntfn_q_refs_of'_def obj_at'_def projectKOs
dest!: symreftype_inverse'
split: ntfn.splits split_if_asm)
apply (clarsimp split: split_if_asm)
apply (fastforce simp: tcb_st_refs_of'_def
dest!: bound_tcb_at_state_refs_ofD'
split: split_if_asm thread_state.splits)
apply (fastforce simp: obj_at'_def projectKOs state_refs_of'_def
dest!: symreftype_inverse')
apply (clarsimp simp: obj_at'_def projectKOs)
apply (clarsimp dest!: pred_tcb_at')
apply (clarsimp simp: valid_pspace'_def)
apply (frule_tac P="\<lambda>k. k=ntfn" in obj_at_valid_objs', simp)
apply (clarsimp simp: valid_obj'_def projectKOs valid_ntfn'_def obj_at'_def
dest!: pred_tcb_at'
split: ntfn.splits)
done
lemma tcbntfn_invs':
"\<lbrace>invs' and tcb_inv_wf' (tcbinvocation.NotificationControl tcb ntfnptr)\<rbrace>
invokeTCB (tcbinvocation.NotificationControl tcb ntfnptr)
\<lbrace>\<lambda>rv. invs'\<rbrace>"
apply (simp add: invokeTCB_def)
apply (case_tac ntfnptr, simp_all)
apply (wp unbindNotification_invs bindNotification_invs' | simp)+
done
lemma tcbinv_invs':
"\<lbrace>invs' and sch_act_simple and ct_in_state' runnable' and tcb_inv_wf' ti\<rbrace>
invokeTCB ti
\<lbrace>\<lambda>rv. invs'\<rbrace>"
apply (case_tac ti, simp_all only:)
apply (simp add: invokeTCB_def)
apply wp
apply (clarsimp simp: invs'_def valid_state'_def
dest!: global'_no_ex_cap)
apply (simp add: invokeTCB_def)
apply (wp restart_invs')
apply (clarsimp simp: invs'_def valid_state'_def
dest!: global'_no_ex_cap)
apply (wp tc_invs')
apply (clarsimp split: option.split dest!: isCapDs)
apply (wp writereg_invs' readreg_invs' copyreg_invs' tcbntfn_invs' | simp)+
done
crunch_ignore (add: setNextPC getRestartPC)
declare assertDerived_wp [wp]
crunch_ignore (add: assertDerived)
lemma copyregsets_map_only[simp]:
"copyregsets_map v = ARMNoExtraRegisters"
by (cases "copyregsets_map v", simp)
lemma decode_readreg_corres:
"corres (ser \<oplus> tcbinv_relation) (invs and tcb_at t) (invs' and tcb_at' t)
(decode_read_registers args (cap.ThreadCap t))
(decodeReadRegisters args (ThreadCap t))"
apply (simp add: decode_read_registers_def decodeReadRegisters_def)
apply (cases args, simp_all)
apply (case_tac list, simp_all)
apply (simp add: decodeTransfer_def)
apply (simp add: range_check_def rangeCheck_def frameRegisters_def gpRegisters_def)
apply (simp add: unlessE_def split del: split_if, simp add: returnOk_def split del: split_if)
apply (rule corres_guard_imp)
apply (rule corres_split_norE)
prefer 2
apply (rule corres_trivial)
apply (fastforce simp: returnOk_def)
apply (simp add: liftE_bindE)
apply (rule corres_split[OF _ gct_corres])
apply (rule corres_trivial)
apply (clarsimp simp: whenE_def)
apply (wp|simp)+
done
lemma decode_writereg_corres:
notes if_cong [cong]
shows
"\<lbrakk> length args < 2 ^ word_bits \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation) (invs and tcb_at t) (invs' and tcb_at' t)
(decode_write_registers args (cap.ThreadCap t))
(decodeWriteRegisters args (ThreadCap t))"
apply (simp add: decode_write_registers_def decodeWriteRegisters_def)
apply (cases args, simp_all)
apply (case_tac list, simp_all)
apply (simp add: decodeTransfer_def genericLength_def)
apply (simp add: word_less_nat_alt unat_of_nat32)
apply (simp add: whenE_def, simp add: returnOk_def)
apply (simp add: genericTake_def)
apply clarsimp
apply (rule corres_guard_imp)
apply (simp add: liftE_bindE)
apply (rule corres_split[OF _ gct_corres])
apply (rule corres_split_norE)
apply (rule corres_trivial, simp)
apply (rule corres_trivial, simp)
apply (wp)
apply simp+
done
lemma decode_copyreg_corres:
"\<lbrakk> list_all2 cap_relation extras extras'; length args < 2 ^ word_bits \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation) (invs and tcb_at t) (invs' and tcb_at' t)
(decode_copy_registers args (cap.ThreadCap t) extras)
(decodeCopyRegisters args (ThreadCap t) extras')"
apply (simp add: decode_copy_registers_def decodeCopyRegisters_def)
apply (cases args, simp_all)
apply (cases extras, simp_all add: decodeTransfer_def null_def)
apply (clarsimp simp: list_all2_Cons1 null_def)
apply (case_tac aa, simp_all)
apply (simp add: returnOk_def)
apply clarsimp
done
lemma decodeReadReg_inv:
"\<lbrace>P\<rbrace> decodeReadRegisters args cap \<lbrace>\<lambda>rv. P\<rbrace>"
apply (rule hoare_pre)
apply (simp add: decodeReadRegisters_def decodeTransfer_def whenE_def
| rule conjI | clarsimp
| wp_once | wpcw)+
done
lemma decodeWriteReg_inv:
"\<lbrace>P\<rbrace> decodeWriteRegisters args cap \<lbrace>\<lambda>rv. P\<rbrace>"
apply (rule hoare_pre)
apply (simp add: decodeWriteRegisters_def whenE_def decodeTransfer_def
split del: split_if
| wp_once | wpcw)+
done
lemma decodeCopyReg_inv:
"\<lbrace>P\<rbrace> decodeCopyRegisters args cap extras \<lbrace>\<lambda>rv. P\<rbrace>"
apply (rule hoare_pre)
apply (simp add: decodeCopyRegisters_def whenE_def decodeTransfer_def
split del: split_if
| wp_once | wpcw)+
done
lemma decodeReadReg_wf:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t\<rbrace>
decodeReadRegisters args (ThreadCap t)
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeReadRegisters_def decodeTransfer_def whenE_def
cong: list.case_cong)
apply (rule hoare_pre)
apply (wp | wpc)+
apply simp
done
lemma decodeWriteReg_wf:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t\<rbrace>
decodeWriteRegisters args (ThreadCap t)
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeWriteRegisters_def whenE_def decodeTransfer_def
cong: list.case_cong)
apply (rule hoare_pre)
apply (wp | wpc)+
apply simp
done
lemma decodeCopyReg_wf:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t
and (\<lambda>s. \<forall>x \<in> set extras. s \<turnstile>' x
\<and> (\<forall>y \<in> zobj_refs' x. ex_nonz_cap_to' y s))\<rbrace>
decodeCopyRegisters args (ThreadCap t) extras
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeCopyRegisters_def whenE_def decodeTransfer_def
cong: list.case_cong capability.case_cong bool.case_cong
split del: split_if)
apply (rule hoare_pre)
apply (wp | wpc)+
apply (clarsimp simp: null_def neq_Nil_conv
valid_cap'_def[where c="ThreadCap t" for t])
done
lemma OR_choice_decode_sp_simp[simp]:
"OR_choice (decode_set_priority_error_choice (ucast a) cur) f g
= do rv \<leftarrow> decode_set_priority_error_choice (ucast a) cur;
if rv then f else g od"
apply (rule ext)
apply (clarsimp simp: OR_choice_def select_f_def decode_set_priority_error_choice_def
wrap_ext_bool_det_ext_ext_def get_def bind_def ethread_get_def split_def
gets_the_def assert_opt_def return_def gets_def fail_def mk_ef_def
split: option.splits)
done
lemma decode_set_priority_corres:
"\<lbrakk> cap_relation cap cap'; is_thread_cap cap \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation) (cur_tcb and valid_etcbs) invs'
(decode_set_priority args cap slot)
(decodeSetPriority args cap')"
apply (simp add: decode_set_priority_def decodeSetPriority_def
split del: split_if)
apply (cases args)
apply simp
apply (simp add: liftE_bindE)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ gct_corres])
apply (simp add: decode_set_priority_error_choice_def)
apply (rule corres_split [OF _ ethreadget_corres[where r="op ="]])
apply (clarsimp simp: returnOk_def newroot_rel_def cap_relation_def)
apply (case_tac cap, clarsimp+)
apply (simp add: etcb_relation_def)+
apply (wp hoare_TrueI)
apply (force simp: cur_tcb_def valid_etcbs_def tcb_at_st_tcb_at)
apply (simp add: invs'_def cur_tcb'_def)
done
lemma threadGet_tcbPriority_maxPriority:
"\<lbrace> valid_objs' \<rbrace> threadGet tcbPriority t \<lbrace> \<lambda>p s. p \<le> maxPriority\<rbrace>"
apply (wp threadGet_const)
apply (clarsimp simp: o_def elim!: valid_objs'_maxPriority)
done
lemma decodeSetPriority_wf[wp]:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t \<rbrace>
decodeSetPriority args (ThreadCap t) \<lbrace>tcb_inv_wf'\<rbrace>,-"
proof -
have prio_maxPrio:
"\<And>P t. \<lbrakk> \<And>s. P s \<Longrightarrow> valid_objs' s \<rbrakk> \<Longrightarrow> \<lbrace> P \<rbrace> threadGet tcbPriority t \<lbrace> \<lambda>p s. P s \<and> p \<le> maxPriority \<rbrace>"
apply (wp threadGet_tcbPriority_maxPriority)
apply (simp)+
done
show ?thesis
apply (simp add: decodeSetPriority_def Let_def split del: split_if)
apply (rule hoare_pre)
apply (case_tac "args")
defer
apply simp
apply (rule validE_validE_R, rule seqE)
apply (rule_tac Q="\<lambda>_. invs' and ex_nonz_cap_to' t and tcb_at' t" in liftE_wp)
apply (wp_trace getCurThread_invs' getCurThread_cap_to')[1]
apply (rule seqE, rule liftE_wp)
apply (rule prio_maxPrio, clarsimp simp: invs'_def valid_state'_def valid_pspace'_def)
apply (wp_trace)[1]
apply (wp whenE_throwError_wp)
apply fastforce+
apply (simp, wp throwError_validE_R)
done
qed
lemma decodeSetPriority_inv[wp]:
"\<lbrace>P\<rbrace> decodeSetPriority args cap \<lbrace>\<lambda>rv. P\<rbrace>"
apply (simp add: decodeSetPriority_def Let_def split del: split_if)
apply (rule hoare_pre)
apply (wp | simp add: whenE_def split del: split_if
| rule hoare_drop_imps
| wpcw)+
done
lemma check_valid_ipc_corres:
"cap_relation cap cap' \<Longrightarrow>
corres (ser \<oplus> dc) \<top> \<top>
(check_valid_ipc_buffer vptr cap)
(checkValidIPCBuffer vptr cap')"
apply (simp add: check_valid_ipc_buffer_def
checkValidIPCBuffer_def
ARM_H.checkValidIPCBuffer_def
unlessE_def Let_def
split: cap_relation_split_asm arch_cap.split_asm)
apply (simp add: capTransferDataSize_def msgMaxLength_def
msg_max_length_def msgMaxExtraCaps_def
cap_transfer_data_size_def word_size
msgLengthBits_def msgExtraCapBits_def msg_align_bits msgAlignBits_def
msg_max_extra_caps_def is_aligned_mask whenE_def)
apply (simp add: returnOk_def )
done
lemma checkValidIPCBuffer_ArchObject_wp:
"\<lbrace>\<lambda>s. isArchObjectCap cap \<and> is_aligned x msg_align_bits \<longrightarrow> P s\<rbrace>
checkValidIPCBuffer x cap
\<lbrace>\<lambda>rv s. P s\<rbrace>,-"
apply (simp add: checkValidIPCBuffer_def
ARM_H.checkValidIPCBuffer_def
whenE_def unlessE_def
cong: capability.case_cong
arch_capability.case_cong
split del: split_if)
apply (rule hoare_pre)
apply (wp | wpc)+
apply (clarsimp simp: isCap_simps is_aligned_mask msg_align_bits msgAlignBits_def)
done
lemma decode_set_ipc_corres:
notes if_cong [cong]
shows
"\<lbrakk> cap_relation cap cap'; is_thread_cap cap;
list_all2 (\<lambda>(c, sl) (c', sl'). cap_relation c c' \<and> sl' = cte_map sl) extras extras' \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation) (\<lambda>s. \<forall>x \<in> set extras. cte_at (snd x) s)
(\<lambda>s. invs' s \<and> (\<forall>x \<in> set extras'. cte_at' (snd x) s))
(decode_set_ipc_buffer args cap slot extras)
(decodeSetIPCBuffer args cap' (cte_map slot) extras')"
apply (simp add: decode_set_ipc_buffer_def decodeSetIPCBuffer_def
split del: split_if)
apply (cases args)
apply simp
apply (cases extras)
apply simp
apply (clarsimp simp: list_all2_Cons1 liftME_def[symmetric]
is_cap_simps
split del: split_if)
apply (clarsimp simp add: returnOk_def newroot_rel_def)
apply (rule corres_guard_imp)
apply (rule corres_splitEE [OF _ derive_cap_corres])
apply (simp add: o_def newroot_rel_def split_def dc_def[symmetric])
apply (erule check_valid_ipc_corres)
apply (wp hoareE_TrueI | simp)+
apply fastforce
done
lemma decodeSetIPC_wf[wp]:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t and cte_at' slot
and (\<lambda>s. \<forall>v \<in> set extras. s \<turnstile>' fst v \<and> cte_at' (snd v) s)\<rbrace>
decodeSetIPCBuffer args (ThreadCap t) slot extras
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeSetIPCBuffer_def Let_def whenE_def
split del: split_if cong: list.case_cong prod.case_cong)
apply (rule hoare_pre)
apply (wp | wpc | simp)+
apply (rule checkValidIPCBuffer_ArchObject_wp)
apply simp
apply (wp hoare_drop_imps)
apply clarsimp
done
lemma decodeSetIPCBuffer_is_tc[wp]:
"\<lbrace>\<top>\<rbrace> decodeSetIPCBuffer args cap slot extras \<lbrace>\<lambda>rv s. isThreadControl rv\<rbrace>,-"
apply (simp add: decodeSetIPCBuffer_def Let_def
split del: split_if cong: list.case_cong prod.case_cong)
apply (rule hoare_pre)
apply (wp | wpc)+
apply (simp only: isThreadControl_def tcbinvocation.simps)
apply wp
apply (clarsimp simp: isThreadControl_def)
done
lemma decodeSetPriority_is_tc[wp]:
"\<lbrace>\<top>\<rbrace> decodeSetPriority args cap \<lbrace>\<lambda>rv s. isThreadControl rv\<rbrace>,-"
apply (simp add: decodeSetPriority_def Let_def
split del: split_if cong: list.case_cong prod.case_cong)
apply (rule hoare_pre)
apply (wp | wpc)+
apply (clarsimp simp: isThreadControl_def tcbinvocation.simps)
done
crunch inv[wp]: decodeSetIPCBuffer "P"
(simp: crunch_simps)
lemma slot_long_running_corres:
"cte_map ptr = ptr' \<Longrightarrow>
corres op = (cte_at ptr and invs) invs'
(slot_cap_long_running_delete ptr)
(slotCapLongRunningDelete ptr')"
apply (clarsimp simp: slot_cap_long_running_delete_def
slotCapLongRunningDelete_def)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ get_cap_corres])
apply (auto split: cap_relation_split_asm arch_cap.split_asm
intro!: corres_rel_imp [OF final_cap_corres[where ptr=ptr]]
simp: liftM_def[symmetric] final_matters'_def
long_running_delete_def
longRunningDelete_def isCap_simps)[1]
apply (wp get_cap_wp getCTE_wp)
apply clarsimp
apply (clarsimp simp: cte_wp_at_ctes_of)
apply fastforce
done
lemma slot_long_running_inv'[wp]:
"\<lbrace>P\<rbrace> slotCapLongRunningDelete ptr \<lbrace>\<lambda>rv. P\<rbrace>"
apply (simp add: slotCapLongRunningDelete_def)
apply (rule hoare_seq_ext [OF _ getCTE_inv])
apply (rule hoare_pre, wpcw, wp isFinalCapability_inv)
apply simp
done
lemma cap_CNode_case_throw:
"(case cap of CNodeCap a b c d \<Rightarrow> m | _ \<Rightarrow> throw x)
= (doE unlessE (isCNodeCap cap) (throw x); m odE)"
by (cases cap, simp_all add: isCap_simps unlessE_def)
lemma decode_set_space_corres:
notes if_cong [cong]
shows
"\<lbrakk> cap_relation cap cap'; list_all2 (\<lambda>(c, sl) (c', sl'). cap_relation c c' \<and> sl' = cte_map sl) extras extras';
is_thread_cap cap \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation)
(invs and valid_cap cap and (\<lambda>s. \<forall>x \<in> set extras. cte_at (snd x) s))
(invs' and valid_cap' cap' and (\<lambda>s. \<forall>x \<in> set extras'. cte_at' (snd x) s))
(decode_set_space args cap slot extras)
(decodeSetSpace args cap' (cte_map slot) extras')"
apply (simp add: decode_set_space_def decodeSetSpace_def
Let_def
split del: split_if)
apply (cases "3 \<le> length args \<and> 2 \<le> length extras'")
apply (clarsimp simp: val_le_length_Cons list_all2_Cons2
split del: split_if)
apply (simp add: liftE_bindE liftM_def
getThreadCSpaceRoot getThreadVSpaceRoot
split del: split_if)
apply (rule corres_guard_imp)
apply (rule corres_split [OF _ slot_long_running_corres])
apply (rule corres_split [OF _ slot_long_running_corres])
apply (rule corres_split_norE)
apply (simp(no_asm) add: split_def unlessE_throwError_returnOk
bindE_assoc cap_CNode_case_throw
split del: split_if)
apply (rule corres_splitEE [OF _ derive_cap_corres])
apply (rule corres_split_norE)
apply (rule corres_splitEE [OF _ derive_cap_corres])
apply (rule corres_split_norE)
apply (rule corres_trivial)
apply (clarsimp simp: returnOk_def newroot_rel_def is_cap_simps
list_all2_conv_all_nth split_def)
apply (unfold unlessE_whenE)
apply (rule corres_whenE)
apply (case_tac vroot_cap', simp_all add:
is_valid_vtable_root_def isValidVTableRoot_def
ARM_H.isValidVTableRoot_def)[1]
apply (rename_tac arch_cap)
apply (clarsimp, case_tac arch_cap, simp_all)[1]
apply (simp split: option.split)
apply (rule corres_trivial, simp)
apply simp
apply wp
apply (clarsimp simp: cap_map_update_data)
apply simp
apply ((simp only: simp_thms pred_conj_def | wp)+)[2]
apply (rule corres_whenE)
apply simp
apply (rule corres_trivial, simp)
apply simp
apply (unfold whenE_def, wp)[2]
apply (fastforce dest: list_all2_nthD2[where p=0] simp: cap_map_update_data)
apply (fastforce dest: list_all2_nthD2[where p=0])
apply ((simp split del: split_if | wp | rule hoare_drop_imps)+)[2]
apply (rule corres_whenE)
apply simp
apply (rule corres_trivial, simp)
apply simp
apply (unfold whenE_def, wp)[2]
apply (clarsimp simp: is_cap_simps get_tcb_vtable_ptr_def cte_map_tcb_1[simplified])
apply simp
apply (wp hoare_drop_imps)
apply (clarsimp simp: is_cap_simps get_tcb_ctable_ptr_def cte_map_tcb_0)
apply wp
apply (clarsimp simp: get_tcb_ctable_ptr_def get_tcb_vtable_ptr_def
is_cap_simps valid_cap_def tcb_at_cte_at_0
tcb_at_cte_at_1[simplified])
apply fastforce
apply (frule list_all2_lengthD)
apply (clarsimp split: list.split)
done
lemma decodeSetSpace_wf[wp]:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t and cte_at' slot
and (\<lambda>s. \<forall>x \<in> set extras. s \<turnstile>' fst x \<and> cte_at' (snd x) s \<and> t \<noteq> snd x \<and> t + 16 \<noteq> snd x)\<rbrace>
decodeSetSpace args (ThreadCap t) slot extras
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeSetSpace_def Let_def split_def
unlessE_def getThreadVSpaceRoot getThreadCSpaceRoot
cap_CNode_case_throw
split del: split_if cong: if_cong list.case_cong)
apply (rule hoare_pre)
apply (wp
| simp add: o_def split_def
split del: split_if
| wpc
| rule hoare_drop_imps)+
apply (clarsimp simp del: length_greater_0_conv
split del: split_if)
apply (simp del: length_greater_0_conv add: valid_updateCapDataI)
done
lemma decodeSetSpace_inv[wp]:
"\<lbrace>P\<rbrace> decodeSetSpace args cap slot extras \<lbrace>\<lambda>rv. P\<rbrace>"
apply (simp add: decodeSetSpace_def Let_def split_def
unlessE_def getThreadVSpaceRoot getThreadCSpaceRoot
split del: split_if cong: if_cong list.case_cong)
apply (rule hoare_pre)
apply (wp hoare_drop_imps
| simp add: o_def split_def split del: split_if
| wpcw)+
done
lemma decodeSetSpace_is_tc[wp]:
"\<lbrace>\<top>\<rbrace> decodeSetSpace args cap slot extras \<lbrace>\<lambda>rv s. isThreadControl rv\<rbrace>,-"
apply (simp add: decodeSetSpace_def Let_def split_def
unlessE_def getThreadVSpaceRoot getThreadCSpaceRoot
split del: split_if cong: list.case_cong)
apply (rule hoare_pre)
apply (wp hoare_drop_imps
| simp only: isThreadControl_def tcbinvocation.simps
| wpcw)+
apply simp
done
lemma decodeSetSpace_tc_target[wp]:
"\<lbrace>\<lambda>s. P (capTCBPtr cap)\<rbrace> decodeSetSpace args cap slot extras \<lbrace>\<lambda>rv s. P (tcThread rv)\<rbrace>,-"
apply (simp add: decodeSetSpace_def Let_def split_def
unlessE_def getThreadVSpaceRoot getThreadCSpaceRoot
split del: split_if cong: list.case_cong)
apply (rule hoare_pre)
apply (wp hoare_drop_imps
| simp only: tcThread.simps
| wpcw)+
apply simp
done
lemma decodeSetSpace_tc_slot[wp]:
"\<lbrace>\<lambda>s. P slot\<rbrace> decodeSetSpace args cap slot extras \<lbrace>\<lambda>rv s. P (tcThreadCapSlot rv)\<rbrace>,-"
apply (simp add: decodeSetSpace_def split_def unlessE_def
getThreadVSpaceRoot getThreadCSpaceRoot
cong: list.case_cong)
apply (rule hoare_pre)
apply (wp hoare_drop_imps | wpcw | simp only: tcThreadCapSlot.simps)+
apply simp
done
lemma decode_tcb_conf_corres:
notes if_cong [cong] option.case_cong [cong]
shows
"\<lbrakk> cap_relation cap cap'; list_all2 (\<lambda>(c, sl) (c', sl'). cap_relation c c' \<and> sl' = cte_map sl) extras extras';
is_thread_cap cap \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation) (einvs and valid_cap cap and (\<lambda>s. \<forall>x \<in> set extras. cte_at (snd x) s))
(invs' and valid_cap' cap' and (\<lambda>s. \<forall>x \<in> set extras'. cte_at' (snd x) s))
(decode_tcb_configure args cap slot extras)
(decodeTCBConfigure args cap' (cte_map slot) extras')"
apply (clarsimp simp add: decode_tcb_configure_def decodeTCBConfigure_def Let_def)
apply (cases "length args < 5")
apply (clarsimp split: list.split)
apply (cases "length extras < 3")
apply (clarsimp split: list.split simp: list_all2_Cons2)
apply (clarsimp simp: linorder_not_less val_le_length_Cons
list_all2_Cons1)
apply (rule corres_guard_imp)
apply (rule corres_splitEE [OF _ decode_set_priority_corres])
apply (rule corres_splitEE [OF _ decode_set_ipc_corres])
apply (rule corres_splitEE [OF _ decode_set_space_corres])
apply (rule_tac F="is_thread_control set_prio" in corres_gen_asm)
apply (rule_tac F="is_thread_control set_params" in corres_gen_asm)
apply (rule_tac F="is_thread_control set_space" in corres_gen_asm)
apply (rule_tac F="tcThreadCapSlot setSpace = cte_map slot" in corres_gen_asm2)
apply (rule corres_trivial)
apply (clarsimp simp: returnOk_def is_thread_control_def2 is_cap_simps)
apply (wp | simp add: invs_def valid_sched_def)+
done
lemma isThreadControl_def2:
"isThreadControl tinv = (\<exists>a b c d e f g. tinv = ThreadControl a b c d e f g)"
by (cases tinv, simp_all add: isThreadControl_def)
lemma decodeSetSpaceSome[wp]:
"\<lbrace>\<top>\<rbrace> decodeSetSpace xs cap y zs
\<lbrace>\<lambda>rv s. tcNewCRoot rv \<noteq> None\<rbrace>,-"
apply (simp add: decodeSetSpace_def split_def cap_CNode_case_throw
cong: list.case_cong if_cong del: not_None_eq)
apply (rule hoare_pre)
apply (wp hoare_drop_imps | wpcw
| simp only: tcNewCRoot.simps option.simps)+
apply simp
done
lemma decodeSetPriority_wf_isThreadControl:
"\<lbrace> invs' and tcb_at' t and ex_nonz_cap_to' t \<rbrace> decodeSetPriority args (ThreadCap t) \<lbrace> \<lambda>tc s. tcb_inv_wf' tc s \<and> isThreadControl tc \<rbrace>, -"
apply (rule validE_validE_R)
apply (wp hoare_vcg_conj_liftE decodeSetPriority_wf)
apply simp
done
lemma decodeTCBConf_wf[wp]:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t and cte_at' slot
and (\<lambda>s. \<forall>x \<in> set extras. s \<turnstile>' fst x \<and> cte_at' (snd x) s \<and> t \<noteq> snd x \<and> t + 16 \<noteq> snd x)\<rbrace>
decodeTCBConfigure args (ThreadCap t) slot extras
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (clarsimp simp add: decodeTCBConfigure_def Let_def
split del: split_if cong: list.case_cong)
apply (rule hoare_pre)
apply (wp | wpc)+
apply (rule_tac Q'="\<lambda>setSpace s. tcb_inv_wf' setSpace s \<and> tcb_inv_wf' setIPCParams s
\<and> isThreadControl setSpace \<and> isThreadControl setIPCParams
\<and> tcThread setSpace = t
\<and> tcNewCRoot setSpace \<noteq> None \<and> valid_option_prio (tcNewPriority setPriority)"
in hoare_post_imp_R)
apply wp
apply (clarsimp simp: isThreadControl_def2 cong: option.case_cong)
apply wp
apply (rule hoare_post_imp_R)
apply (rule decodeSetPriority_wf_isThreadControl)
apply (fastforce simp: isThreadControl_def2)+
done
lemma decodeTCBConf_inv[wp]:
"\<lbrace>P\<rbrace> decodeTCBConfigure args (ThreadCap t) slot extras \<lbrace>\<lambda>rv. P\<rbrace>"
apply (clarsimp simp add: decodeTCBConfigure_def Let_def
split del: split_if cong: list.case_cong)
apply (rule hoare_pre)
apply (wp | wpcw)+
apply simp
done
lemma getSlotCap_valid_option:
"\<lbrace>valid_objs'\<rbrace> getSlotCap a
\<lbrace>\<lambda>rv s. s \<turnstile>' (case rootdata of
None \<Rightarrow> rv
| Some w \<Rightarrow> RetypeDecls_H.updateCapData p w rv)\<rbrace>"
apply (cases rootdata)
apply (simp|wp getSlotCap_valid_cap1)+
apply (rule hoare_strengthen_post)
apply (rule getSlotCap_valid_cap1)
apply (simp add: valid_updateCapDataI)
done
declare hoare_True_E_R [simp del]
declare resolveAddressBits.simps[simp del]
lemma lsft_real_cte:
"\<lbrace>valid_objs'\<rbrace> lookupSlotForThread t x \<lbrace>\<lambda>rv. real_cte_at' rv\<rbrace>, -"
apply (simp add: lookupSlotForThread_def)
apply (wp resolveAddressBits_real_cte_at'|simp add: split_def)+
done
lemma tcb_real_cte_16:
"\<lbrakk> real_cte_at' (t+16) s; tcb_at' t s \<rbrakk> \<Longrightarrow> False"
by (clarsimp simp: obj_at'_def projectKOs objBits_simps
ps_clear_16)
lemma isValidVTableRoot:
"isValidVTableRoot c = (\<exists>p asid. c = ArchObjectCap (PageDirectoryCap p (Some asid)))"
by (simp add: isValidVTableRoot_def ARM_H.isValidVTableRoot_def isCap_simps
split: capability.splits arch_capability.splits option.splits)
lemma corres_splitEE':
assumes y: "\<And>x y x' y'. r' (x, y) (x', y')
\<Longrightarrow> corres_underlying sr nf (f \<oplus> r) (R x y) (R' x' y') (b x y) (d x' y')"
assumes "corres_underlying sr nf (f \<oplus> r') P P' a c"
assumes x: "\<lbrace>Q\<rbrace> a \<lbrace>%(x, y). R x y \<rbrace>,\<lbrace>\<top>\<top>\<rbrace>" "\<lbrace>Q'\<rbrace> c \<lbrace>%(x, y). R' x y\<rbrace>,\<lbrace>\<top>\<top>\<rbrace>"
shows "corres_underlying sr nf (f \<oplus> r) (P and Q) (P' and Q') (a >>=E (\<lambda>(x, y). b x y)) (c >>=E (\<lambda>(x, y). d x y))"
using assms
apply (unfold bindE_def validE_def split_def)
apply (rule corres_split)
defer
apply assumption+
apply (case_tac rv)
apply (clarsimp simp: lift_def y)+
done
lemma decode_bind_notification_corres:
notes if_cong[cong] shows
"\<lbrakk> list_all2 (\<lambda>x y. cap_relation (fst x) (fst y)) extras extras' \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation)
(invs and tcb_at t and (\<lambda>s. \<forall>x \<in> set extras. s \<turnstile> (fst x)))
(invs' and tcb_at' t and (\<lambda>s. \<forall>x \<in> set extras'. s \<turnstile>' (fst x)))
(decode_bind_notification (cap.ThreadCap t) extras)
(decodeBindNotification (capability.ThreadCap t) extras')"
apply (simp add: decode_bind_notification_def decodeBindNotification_def)
apply (simp add: null_def returnOk_def)
apply (rule corres_guard_imp)
apply (rule corres_split_norE)
apply (rule_tac F="extras \<noteq> []" in corres_gen_asm)
apply (rule corres_split_eqrE)
apply (rule corres_split_norE)
apply (rule corres_splitEE'[where r'="\<lambda>rv rv'. ((fst rv) = (fst rv')) \<and> ((snd rv') = (AllowRead \<in> (snd rv)))"])
apply (rule corres_split_norE)
apply (clarsimp split del: split_if)
apply (rule corres_splitEE[where r'=ntfn_relation])
apply (rule corres_trivial, simp split del: split_if)
apply (simp add: ntfn_relation_def
split: Structures_A.ntfn.splits Structures_H.ntfn.splits
option.splits)
apply simp
apply (rule get_ntfn_corres)
apply wp
apply (rule corres_trivial, clarsimp simp: whenE_def returnOk_def)
apply (wp | simp add: whenE_def split del: split_if)+
apply (rule corres_trivial, simp)
apply (case_tac extras, simp, clarsimp simp: list_all2_Cons1)
apply (fastforce split: cap.splits capability.splits simp: returnOk_def)
apply (wp | wpc | simp)+
apply (rule corres_trivial, simp split: option.splits add: returnOk_def)
apply (wp | wpc | simp)+
apply (rule gbn_corres)
apply (simp | wp gbn_wp gbn_wp')+
apply (rule corres_trivial)
apply (auto simp: returnOk_def whenE_def)[1]
apply (simp add: whenE_def split del: split_if | wp)+
apply (fastforce simp: valid_cap_def valid_cap'_def dest: hd_in_set)+
done
lemma decode_unbind_notification_corres:
"corres (ser \<oplus> tcbinv_relation)
(tcb_at t)
(tcb_at' t)
(decode_unbind_notification (cap.ThreadCap t))
(decodeUnbindNotification (capability.ThreadCap t))"
apply (simp add: decode_unbind_notification_def decodeUnbindNotification_def)
apply (rule corres_guard_imp)
apply (rule corres_split_eqrE)
apply (rule corres_trivial)
apply (simp split: option.splits)
apply (simp add: returnOk_def)
apply simp
apply (rule gbn_corres)
apply wp
apply auto
done
lemma decode_tcb_inv_corres:
"\<lbrakk> c = Structures_A.ThreadCap t; cap_relation c c';
list_all2 (\<lambda>(c, sl) (c', sl'). cap_relation c c' \<and> sl' = cte_map sl) extras extras';
length args < 2 ^ word_bits \<rbrakk> \<Longrightarrow>
corres (ser \<oplus> tcbinv_relation) (einvs and tcb_at t and (\<lambda>s. \<forall>x \<in> set extras. s \<turnstile> fst x \<and> cte_at (snd x) s))
(invs' and tcb_at' t and (\<lambda>s. \<forall>x \<in> set extras'. s \<turnstile>' fst x \<and> cte_at' (snd x) s))
(decode_tcb_invocation label args c slot extras)
(decodeTCBInvocation label args c' (cte_map slot) extras')"
apply (rule_tac F="cap_aligned c \<and> capAligned c'" in corres_req)
apply (clarsimp simp: cap_aligned_def capAligned_def objBits_simps word_bits_def)
apply (drule obj_at_aligned', simp_all add: objBits_simps)
apply (clarsimp simp: decode_tcb_invocation_def
decodeTCBInvocation_def
split del: split_if split: invocation_label.split)
apply (simp add: returnOk_def)
apply (intro conjI impI
corres_guard_imp[OF decode_readreg_corres]
corres_guard_imp[OF decode_writereg_corres]
corres_guard_imp[OF decode_copyreg_corres]
corres_guard_imp[OF decode_tcb_conf_corres]
corres_guard_imp[OF decode_set_priority_corres]
corres_guard_imp[OF decode_set_ipc_corres]
corres_guard_imp[OF decode_set_space_corres]
corres_guard_imp[OF decode_bind_notification_corres]
corres_guard_imp[OF decode_unbind_notification_corres],
simp_all add: valid_cap_simps valid_cap_simps' invs_def valid_sched_def)
apply (auto simp: list_all2_map1 list_all2_map2
elim!: list_all2_mono)
done
crunch inv[wp]: decodeTCBInvocation P
(simp: crunch_simps)
lemma real_cte_at_not_tcb_at':
"real_cte_at' x s \<Longrightarrow> \<not> tcb_at' x s"
"real_cte_at' (x + 16) s \<Longrightarrow> \<not> tcb_at' x s"
apply (clarsimp simp: obj_at'_def projectKOs)
apply (clarsimp elim!: tcb_real_cte_16)
done
lemma decodeBindNotification_wf:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t
and (\<lambda>s. \<forall>x \<in> set extras. s \<turnstile>' (fst x) \<and> (\<forall>y \<in> zobj_refs' (fst x). ex_nonz_cap_to' y s))\<rbrace>
decodeBindNotification (capability.ThreadCap t) extras
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeBindNotification_def whenE_def
cong: list.case_cong split del: split_if)
apply (rule hoare_pre)
apply (wp getNotification_wp getObject_tcb_wp
| wpc
| simp add: threadGet_def getBoundNotification_def)+
apply (fastforce simp: valid_cap'_def[where c="capability.ThreadCap t"]
is_ntfn invs_def valid_state'_def valid_pspace'_def
projectKOs null_def pred_tcb_at'_def obj_at'_def
dest!: global'_no_ex_cap hd_in_set)
done
lemma decodeUnbindNotification_wf:
"\<lbrace>invs' and tcb_at' t and ex_nonz_cap_to' t\<rbrace>
decodeUnbindNotification (capability.ThreadCap t)
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeUnbindNotification_def)
apply (wp getObject_tcb_wp | wpc | simp add: threadGet_def getBoundNotification_def)+
apply (auto simp: obj_at'_def pred_tcb_at'_def)
done
lemma decodeTCBInv_wf:
"\<lbrace>invs' and tcb_at' t and cte_at' slot and ex_nonz_cap_to' t
and (\<lambda>s. \<forall>x \<in> set extras. real_cte_at' (snd x) s
\<and> s \<turnstile>' fst x \<and> (\<forall>y \<in> zobj_refs' (fst x). ex_nonz_cap_to' y s))\<rbrace>
decodeTCBInvocation label args (capability.ThreadCap t) slot extras
\<lbrace>tcb_inv_wf'\<rbrace>,-"
apply (simp add: decodeTCBInvocation_def Let_def
cong: if_cong invocation_label.case_cong split del: split_if)
apply (rule hoare_pre)
apply (wpc, wp decodeTCBConf_wf decodeReadReg_wf
decodeWriteReg_wf decodeCopyReg_wf decodeBindNotification_wf decodeUnbindNotification_wf)
apply (clarsimp simp: real_cte_at')
apply (fastforce simp: real_cte_at_not_tcb_at')
done
lemma restart_makes_simple':
"\<lbrace>st_tcb_at' simple' t\<rbrace>
restart t'
\<lbrace>\<lambda>rv. st_tcb_at' simple' t\<rbrace>"
apply (simp add: restart_def)
apply (wp sts_st_tcb_at'_cases cancelIPC_simple
cancelIPC_st_tcb_at static_imp_wp | simp)+
apply (rule hoare_strengthen_post [OF isBlocked_inv])
apply clarsimp
done
lemma setPriority_st_tcb_at'[wp]:
"\<lbrace>st_tcb_at' P t\<rbrace> setPriority t' p \<lbrace>\<lambda>rv. st_tcb_at' P t\<rbrace>"
apply (simp add: setPriority_def)
apply (wp threadSet_pred_tcb_no_state | simp)+
done
lemma setCurThread_st_tcb[wp]:
"\<lbrace>st_tcb_at' P t\<rbrace> setCurThread t' \<lbrace>\<lambda>rv. st_tcb_at' P t\<rbrace>"
unfolding pred_tcb_at'_def
by (rule setCurThread_obj_at)
lemma st_tcb_at_ksArch_update[simp]:
"st_tcb_at' P t (ksArchState_update f s) = st_tcb_at' P t s"
by (simp add: pred_tcb_at'_def obj_at'_def projectKOs ps_clear_def)
lemma cteDelete_makes_simple':
"\<lbrace>st_tcb_at' simple' t\<rbrace> cteDelete slot v \<lbrace>\<lambda>rv. st_tcb_at' simple' t\<rbrace>"
by (wp cteDelete_st_tcb_at' | simp)+
lemma invokeTCB_makes_simple':
"\<lbrace>invs' and st_tcb_at' simple' t\<rbrace>
invokeTCB tinv
\<lbrace>\<lambda>rv. st_tcb_at' simple' t\<rbrace>,-"
apply (cases tinv, simp_all add: invokeTCB_def Let_def split_def
getThreadVSpaceRoot
performTransfer_def split_def
getThreadBufferSlot_def
split del: split_if cong: if_cong option.case_cong)
defer 3 defer 3(* thread control, ntfn control *)
apply ((wp restart_makes_simple' suspend_makes_simple'
mapM_x_wp'
| simp cong: if_cong)+)[5]
apply (rule hoare_pre)
apply ((simp split del: split_if
| wp cteDelete_makes_simple' checkCap_inv [where P="st_tcb_at' simple' t"]
threadSet_pred_tcb_no_state mapM_x_wp' suspend_makes_simple'
| (wp case_options_weak_wp)[1])+)[2]
apply (rule hoare_pre)
apply (wp | wpc | simp)+
done
crunch irq_states'[wp]: getThreadBufferSlot, setPriority
valid_irq_states'
(simp: crunch_simps)
lemma inv_tcb_IRQInactive:
"\<lbrace>valid_irq_states'\<rbrace> invokeTCB tcb_inv
-, \<lbrace>\<lambda>rv s. intStateIRQTable (ksInterruptState s) rv \<noteq> irqstate.IRQInactive\<rbrace>"
apply (simp add: invokeTCB_def)
apply (rule hoare_pre)
apply (wpc |
wp withoutPreemption_R cteDelete_IRQInactive checkCap_inv
hoare_vcg_const_imp_lift_R cteDelete_irq_states'
hoare_vcg_const_imp_lift |
simp add: split_def)+
done
end
end