lh-l4v/proof/access-control/Syscall_AC.thy

1160 lines
55 KiB
Plaintext

(*
* Copyright 2014, NICTA
*
* 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(NICTA_GPL)
*)
theory Syscall_AC
imports
Ipc_AC
Tcb_AC
Interrupt_AC
DomainSepInv
begin
definition
authorised_invocation :: "'a PAS \<Rightarrow> Invocations_A.invocation \<Rightarrow> 'z::state_ext state \<Rightarrow> bool"
where
"authorised_invocation aag i \<equiv> \<lambda>s. case i of
Invocations_A.InvokeUntyped i' \<Rightarrow> valid_untyped_inv i' s \<and> (authorised_untyped_inv aag i' \<and> authorised_untyped_inv_state aag i' s) \<and> ct_active s
| Invocations_A.InvokeEndpoint epptr badge can_grant \<Rightarrow>
\<exists>ep. ko_at (Endpoint ep) epptr s \<and>
(can_grant \<longrightarrow> (\<forall>r \<in> ep_q_refs_of ep. snd r = EPRecv \<longrightarrow> is_subject aag (fst r)) \<and> aag_has_auth_to aag Grant epptr)
\<and> aag_has_auth_to aag SyncSend epptr
| Invocations_A.InvokeNotification ep badge \<Rightarrow> aag_has_auth_to aag Notify ep
| Invocations_A.InvokeReply thread slot \<Rightarrow> is_subject aag thread \<and> is_subject aag (fst slot)
| Invocations_A.InvokeTCB i' \<Rightarrow> tcb_inv_wf i' s \<and> authorised_tcb_inv aag i'
| Invocations_A.InvokeDomain thread slot \<Rightarrow> False
| Invocations_A.InvokeCNode i' \<Rightarrow> authorised_cnode_inv aag i' s \<and> is_subject aag (cur_thread s)
\<and> cnode_inv_auth_derivations i' s
| Invocations_A.InvokeIRQControl i' \<Rightarrow> authorised_irq_ctl_inv aag i'
| Invocations_A.InvokeIRQHandler i' \<Rightarrow> authorised_irq_hdl_inv aag i'
| Invocations_A.InvokeArchObject i' \<Rightarrow> valid_arch_inv i' s \<and> authorised_arch_inv aag i' \<and> ct_active s"
lemma perform_invocation_pas_refined:
"\<lbrace>pas_refined aag and pas_cur_domain aag
and einvs and simple_sched_action and valid_invocation oper
and is_subject aag \<circ> cur_thread
and authorised_invocation aag oper\<rbrace>
perform_invocation blocking calling oper
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (cases oper, simp_all)
apply (simp add: authorised_invocation_def validE_R_def[symmetric] invs_valid_objs
| wp invoke_untyped_pas_refined send_ipc_pas_refined send_signal_pas_refined
do_reply_transfer_pas_refined invoke_tcb_pas_refined invoke_cnode_pas_refined
invoke_irq_control_pas_refined invoke_irq_handler_pas_refined
invoke_arch_pas_refined decode_cnode_invocation_auth_derived
| fastforce)+
done
lemma ntfn_gives_obj_at:
"invs s \<Longrightarrow> (\<exists>ntfn. ko_at (Notification ntfn) ntfnptr s \<and> (\<forall>x\<in>ntfn_q_refs_of (ntfn_obj ntfn). (\<lambda>(t, rt). obj_at (\<lambda>tcb. ko_at tcb t s) t s) x)) = ntfn_at ntfnptr s"
apply (rule iffI)
apply (clarsimp simp: obj_at_def is_ntfn)
apply (clarsimp simp: obj_at_def is_ntfn)
apply (drule (1) ntfn_queued_st_tcb_at [where P = \<top>, unfolded obj_at_def, simplified])
apply clarsimp
apply clarsimp
apply (clarsimp simp: st_tcb_def2 dest!: get_tcb_SomeD)
done
lemma pi_cases:
"perform_invocation block call i =
(case i of
Invocations_A.InvokeUntyped i \<Rightarrow> perform_invocation block call (Invocations_A.InvokeUntyped i)
| Invocations_A.InvokeEndpoint ep badge canGrant
\<Rightarrow> perform_invocation block call (Invocations_A.InvokeEndpoint ep badge canGrant)
| Invocations_A.InvokeNotification ep badge \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeNotification ep badge)
| Invocations_A.InvokeTCB i \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeTCB i)
| Invocations_A.InvokeDomain thread slot \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeDomain thread slot)
| Invocations_A.InvokeReply thread slot \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeReply thread slot)
| Invocations_A.InvokeCNode i \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeCNode i)
| Invocations_A.InvokeIRQControl i \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeIRQControl i)
| Invocations_A.InvokeIRQHandler i \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeIRQHandler i)
| Invocations_A.InvokeArchObject i \<Rightarrow> perform_invocation block call ( Invocations_A.InvokeArchObject i))"
by (cases i, simp_all)
(* (op = st) -- too strong, the thread state of the calling thread changes. *)
lemma perform_invocation_respects:
"\<lbrace>pas_refined aag and integrity aag X st
and einvs and simple_sched_action and valid_invocation oper
and authorised_invocation aag oper
and is_subject aag \<circ> cur_thread
and (\<lambda>s. \<forall>p ko. kheap s p = Some ko \<longrightarrow> \<not> (is_tcb ko \<and> p = cur_thread st) \<longrightarrow> kheap st p = Some ko)
\<rbrace>
perform_invocation blocking calling oper
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (subst pi_cases)
apply (rule hoare_pre)
apply (wpc
| simp
| wp invoke_untyped_integrity send_ipc_integrity_autarch send_signal_respects
do_reply_transfer_respects invoke_tcb_respects invoke_cnode_respects
invoke_arch_respects invoke_irq_control_respects invoke_irq_handler_respects
| wp_once hoare_pre_cont)+
apply (clarsimp simp: authorised_invocation_def split: Invocations_A.invocation.splits)
-- "EP case"
apply (fastforce simp: obj_at_def is_tcb split: split_if_asm)
-- "NTFN case"
apply fastforce
done
declare AllowSend_def[simp] AllowRecv_def[simp]
lemma diminshed_IRQControlCap_eq:
"diminished IRQControlCap = (op = IRQControlCap)"
apply (rule ext)
apply (case_tac x, auto simp: diminished_def mask_cap_def cap_rights_update_def)
done
lemma diminished_DomainCap_eq:
"diminished DomainCap = (op = DomainCap)"
apply (rule ext)
apply (case_tac x, auto simp: diminished_def mask_cap_def cap_rights_update_def)
done
lemma hoare_conjunct1_R:
"\<lbrace> P \<rbrace> f \<lbrace> \<lambda> r s. Q r s \<and> Q' r s\<rbrace>,- \<Longrightarrow> \<lbrace> P \<rbrace> f \<lbrace> Q \<rbrace>,-"
apply(auto intro: hoare_post_imp_R)
done
lemma hoare_conjunct2_R:
"\<lbrace> P \<rbrace> f \<lbrace> \<lambda> r s. Q r s \<and> Q' r s\<rbrace>,- \<Longrightarrow> \<lbrace> P \<rbrace> f \<lbrace> Q' \<rbrace>,-"
apply(auto intro: hoare_post_imp_R)
done
lemma decode_invocation_authorised:
"\<lbrace>pas_refined aag and valid_cap cap and invs and ct_active and cte_wp_at (diminished cap) slot
and ex_cte_cap_to slot
and (\<lambda>s. \<forall>r\<in>zobj_refs cap. ex_nonz_cap_to r s)
and (\<lambda>s. \<forall>r\<in>cte_refs cap (interrupt_irq_node s). ex_cte_cap_to r s)
and (\<lambda>s. \<forall>cap \<in> set excaps. \<forall>r\<in>cte_refs (fst cap) (interrupt_irq_node s). ex_cte_cap_to r s)
and (\<lambda>s. \<forall>x \<in> set excaps. s \<turnstile> (fst x))
and (\<lambda>s. \<forall>x \<in> set excaps. \<forall>r\<in>zobj_refs (fst x). ex_nonz_cap_to r s)
and (\<lambda>s. \<forall>x \<in> set excaps. cte_wp_at (diminished (fst x)) (snd x) s)
and (\<lambda>s. \<forall>x \<in> set excaps. real_cte_at (snd x) s)
and (\<lambda>s. \<forall>x \<in> set excaps. ex_cte_cap_wp_to is_cnode_cap (snd x) s)
and (\<lambda>s. \<forall>x \<in> set excaps. cte_wp_at (interrupt_derived (fst x)) (snd x) s)
and (is_subject aag \<circ> cur_thread) and
K (is_subject aag (fst slot) \<and> pas_cap_cur_auth aag cap
\<and> (\<forall>slot \<in> set excaps. is_subject aag (fst (snd slot)))
\<and> (\<forall>slot \<in> set excaps. pas_cap_cur_auth aag (fst slot)))
and domain_sep_inv (pasMaySendIrqs aag) st'\<rbrace>
decode_invocation info_label args ptr slot cap excaps
\<lbrace>\<lambda>rv. authorised_invocation aag rv\<rbrace>, -"
unfolding decode_invocation_def
apply (rule hoare_pre)
apply (wp decode_untyped_invocation_authorised[THEN hoare_conjunct1_R]
decode_untyped_invocation_authorised[THEN hoare_conjunct2_R]
decode_cnode_invocation_auth_derived
decode_cnode_inv_authorised
decode_tcb_invocation_authorised decode_tcb_inv_wf
decode_arch_invocation_authorised
| strengthen cnode_diminished_strg
| wpc | simp add: comp_def authorised_invocation_def decode_invocation_def
split del: split_if del: hoare_post_taut hoare_True_E_R
| wp_once hoare_FalseE_R)+
apply (clarsimp simp: aag_has_Control_iff_owns split_def aag_cap_auth_def)
apply (cases cap, simp_all)
apply (fastforce simp: cte_wp_at_caps_of_state)
apply (clarsimp simp: valid_cap_def obj_at_def is_ep cap_auth_conferred_def cap_rights_to_auth_def
ball_Un)
apply (fastforce simp: valid_cap_def cap_auth_conferred_def cap_rights_to_auth_def obj_at_def is_ep intro!: owns_ep_owns_receivers)
apply (fastforce simp: cap_auth_conferred_def cap_rights_to_auth_def)
apply (fastforce simp: cap_auth_conferred_def cap_rights_to_auth_def pas_refined_Control [symmetric])
apply ((clarsimp simp: valid_cap_def cte_wp_at_eq_simp
is_cap_simps
ex_cte_cap_wp_to_weakenE[OF _ TrueI]
cap_auth_conferred_def cap_rights_to_auth_def pas_refined_all_auth_is_owns
| rule conjI | (subst split_paired_Ex[symmetric], erule exI)
| erule cte_wp_at_weakenE
| drule(1) bspec
| erule diminished_no_cap_to_obj_with_diff_ref)+)[1]
apply (simp only: domain_sep_inv_def diminished_DomainCap_eq)
apply (rule impI, erule subst, rule pas_refined_sita_mem [OF sita_controlled], auto
simp: cte_wp_at_caps_of_state diminshed_IRQControlCap_eq)[1]
apply (clarsimp simp add: cap_links_irq_def )
apply (drule (1) pas_refined_Control, simp)
apply (clarsimp simp: cap_links_asid_slot_def label_owns_asid_slot_def)
apply (fastforce dest!: pas_refined_Control)
done
lemma in_extended: "(u,a) \<in> fst (do_extended_op f s) \<Longrightarrow> \<exists>e. a = (trans_state (\<lambda>_. e) s)"
apply (clarsimp simp add: do_extended_op_def bind_def gets_def return_def get_def
mk_ef_def modify_def select_f_def put_def trans_state_update')
apply force
done
lemma set_thread_state_authorised_untyped_inv_state:
"\<lbrace>valid_objs and authorised_untyped_inv_state aag ui\<rbrace>
set_thread_state t ts
\<lbrace>\<lambda>_. authorised_untyped_inv_state aag ui\<rbrace>"
unfolding set_thread_state_def
apply(clarsimp simp: authorised_untyped_inv_state_def split: untyped_invocation.splits simp: valid_def)
apply(subgoal_tac "cte_wp_at (op = cap) (a, b) s", fastforce)
apply(clarsimp simp: set_object_def gets_the_def bind_def in_monad dest!: in_extended)
apply(subgoal_tac "obj_at (same_caps (TCB (ab\<lparr>tcb_state := ts\<rparr>))) t s")
apply(drule_tac P="op = cap" and p="(a,b)" in cte_wp_at_after_update)
apply(clarsimp simp: fun_upd_def)
apply(clarsimp simp: get_tcb_def split: option.splits kernel_object.splits)
apply(clarsimp simp: obj_at_def)
apply(clarsimp simp: tcb_cap_cases_def)
apply auto
done
lemma set_thread_state_authorised[wp]:
"\<lbrace>authorised_invocation aag i and (\<lambda>s. thread = cur_thread s) and valid_objs\<rbrace>
set_thread_state thread Structures_A.thread_state.Restart
\<lbrace>\<lambda>rv. authorised_invocation aag i\<rbrace>"
apply (cases i)
apply (simp_all add: authorised_invocation_def)
apply (wp sts_valid_untyped_inv ct_in_state_set
hoare_vcg_ex_lift sts_obj_at_impossible
set_thread_state_authorised_untyped_inv_state
| simp)+
apply (rename_tac tcb_invocation)
apply (case_tac tcb_invocation, simp_all)
apply (wp hoare_case_option_wp sts_typ_ats set_thread_state_cte_wp_at
hoare_vcg_conj_lift static_imp_wp
| simp)+
apply ((clarsimp split: option.splits)+)[3]
apply ((wp
| simp)+)[2]
apply (rename_tac option)
apply (case_tac option, simp_all)[1]
apply (wp set_thread_state_tcb_at sts_obj_at_impossible | simp add: authorised_tcb_inv_def)+
apply (rename_tac cnode_invocation)
apply (case_tac cnode_invocation,
simp_all add: cnode_inv_auth_derivations_def authorised_cnode_inv_def)[1]
apply (wp set_thread_state_cte_wp_at | simp)+
apply (rename_tac arch_invocation)
apply (case_tac arch_invocation, simp_all add: valid_arch_inv_def)[1]
apply (rename_tac page_table_invocation)
apply (case_tac page_table_invocation, simp_all add: valid_pti_def)[1]
apply (wp sts_typ_ats sts_obj_at_impossible ct_in_state_set
hoare_vcg_ex_lift hoare_vcg_conj_lift
| simp add: valid_pdi_def)+
apply (rename_tac asid_control_invocation)
apply (case_tac asid_control_invocation, simp_all add: valid_aci_def)
apply (wp ct_in_state_set | simp)+
apply (rename_tac asid_pool_invocation)
apply (case_tac asid_pool_invocation; simp add: valid_apinv_def)
apply (wp sts_obj_at_impossible ct_in_state_set
hoare_vcg_ex_lift
| simp)+
done
lemma sts_first_restart:
"\<lbrace>op = st and (\<lambda>s. thread = cur_thread s)\<rbrace>
set_thread_state thread Structures_A.thread_state.Restart
\<lbrace>\<lambda>rv s. \<forall>p ko. kheap s p = Some ko \<longrightarrow>
(is_tcb ko \<longrightarrow> p \<noteq> cur_thread st) \<longrightarrow> kheap st p = Some ko\<rbrace>"
unfolding set_thread_state_def set_object_def
apply (wp dxo_wp_weak |simp)+
apply (clarsimp simp: is_tcb)
done
lemma lcs_reply_owns:
"\<lbrace>pas_refined aag and K (is_subject aag thread)\<rbrace>
lookup_cap_and_slot thread ptr
\<lbrace>\<lambda>rv s. \<forall>ep. (\<exists>m. fst rv = cap.ReplyCap ep m) \<longrightarrow> is_subject aag ep\<rbrace>, -"
apply (rule hoare_post_imp_R)
apply (rule hoare_pre)
apply (rule hoare_vcg_conj_lift_R [where S = "K (pas_refined aag)"])
apply (rule lookup_cap_and_slot_cur_auth)
apply (simp | wp lookup_cap_and_slot_inv)+
apply (clarsimp simp: aag_cap_auth_Reply)
done
crunch pas_refined[wp]: reply_from_kernel "pas_refined aag"
(simp: split_def)
lemma lookup_cap_and_slot_valid_fault3:
"\<lbrace>valid_objs\<rbrace> lookup_cap_and_slot thread cptr
-,
\<lbrace>\<lambda>ft s. valid_fault (ExceptionTypes_A.CapFault (of_bl cptr) rp ft)\<rbrace>"
apply (unfold validE_E_def)
apply (rule hoare_post_impErr)
apply (rule lookup_cap_and_slot_valid_fault)
apply auto
done
declare hoare_post_taut [simp del]
crunch pas_cur_domain[wp]: as_user "pas_cur_domain aag"
definition guarded_pas_domain where
"guarded_pas_domain aag \<equiv> \<lambda>s. cur_thread s \<noteq> idle_thread s \<longrightarrow> pasDomainAbs aag (cur_domain s) = pasObjectAbs aag (cur_thread s)"
lemma guarded_pas_domain_lift:
assumes a: "\<And>P. \<lbrace>\<lambda>s. P (cur_thread s)\<rbrace> f \<lbrace>\<lambda>r s. P (cur_thread s)\<rbrace>"
assumes b: "\<And>P. \<lbrace>\<lambda>s. P (cur_domain s)\<rbrace> f \<lbrace>\<lambda>r s. P (cur_domain s)\<rbrace>"
assumes c: "\<And>P. \<lbrace>\<lambda>s. P (idle_thread s)\<rbrace> f \<lbrace>\<lambda>r s. P (idle_thread s)\<rbrace>"
shows "\<lbrace>guarded_pas_domain aag\<rbrace> f \<lbrace>\<lambda>_. guarded_pas_domain aag\<rbrace>"
apply (simp add: guarded_pas_domain_def)
apply (rule hoare_pre)
apply (wps a b c)
apply wp
apply simp
done
lemma guarded_to_cur_domain: "\<lbrakk>invs s; ct_in_state x s; \<not> x IdleThreadState; guarded_pas_domain aag s; is_subject aag (cur_thread s)\<rbrakk> \<Longrightarrow> pas_cur_domain aag s"
apply (auto simp: invs_def valid_state_def valid_idle_def pred_tcb_at_def obj_at_def
ct_in_state_def guarded_pas_domain_def)
done
lemma handle_invocation_pas_refined:
shows "\<lbrace>pas_refined aag and guarded_pas_domain aag and domain_sep_inv (pasMaySendIrqs aag) st'
and einvs and ct_active and schact_is_rct
and is_subject aag \<circ> cur_thread\<rbrace>
handle_invocation calling blocking
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: handle_invocation_def split_def)
apply (cases blocking, simp)
apply (rule hoare_pre)
apply (((wp syscall_valid without_preemption_wp
handle_fault_pas_refined set_thread_state_pas_refined
set_thread_state_runnable_valid_sched
perform_invocation_pas_refined
hoare_vcg_conj_lift hoare_vcg_all_lift
| wpc
| rule hoare_drop_imps
| simp add: if_apply_def2 conj_comms split del: split_if
del: hoare_True_E_R)+),
((wp lookup_extra_caps_auth lookup_extra_caps_authorised
decode_invocation_authorised
lookup_cap_and_slot_authorised
lookup_cap_and_slot_cur_auth
as_user_pas_refined
lookup_cap_and_slot_valid_fault3
| simp add: split comp_def runnable_eq_active del: split_if)+),
(auto intro: guarded_to_cur_domain simp: ct_in_state_def st_tcb_at_def intro: if_live_then_nonz_capD)[1])+
done
lemma handle_invocation_respects:
"\<lbrace>integrity aag X st and pas_refined aag and guarded_pas_domain aag and domain_sep_inv (pasMaySendIrqs aag) st'
and einvs and ct_active and schact_is_rct
and is_subject aag \<circ> cur_thread
and (op = st)\<rbrace>
handle_invocation calling blocking
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: handle_invocation_def split_def)
apply (wp syscall_valid without_preemption_wp handle_fault_integrity_autarch
reply_from_kernel_integrity_autarch
set_thread_state_integrity_autarch
hoare_vcg_conj_lift
hoare_vcg_all_lift_R hoare_vcg_all_lift
| rule hoare_drop_imps
| wpc | simp add: if_apply_def2
del: hoare_post_taut hoare_True_E_R
split del: split_if)+
apply (simp add: conj_comms pred_conj_def comp_def if_apply_def2 split del: split_if
| wp perform_invocation_respects set_thread_state_pas_refined
set_thread_state_authorised
set_thread_state_runnable_valid_sched
set_thread_state_integrity_autarch
sts_first_restart
decode_invocation_authorised
lookup_extra_caps_auth lookup_extra_caps_authorised
set_thread_state_integrity_autarch
lookup_cap_and_slot_cur_auth lookup_cap_and_slot_authorised
hoare_vcg_const_imp_lift perform_invocation_pas_refined
set_thread_state_ct_st hoare_vcg_const_imp_lift_R
lookup_cap_and_slot_valid_fault3
| (rule valid_validE, strengthen invs_vobjs_strgs)
)+
apply (fastforce intro: st_tcb_ex_cap' guarded_to_cur_domain simp: ct_in_state_def runnable_eq_active)+
done
crunch pas_refined[wp]: delete_caller_cap "pas_refined aag"
crunch cur_thread[wp]: delete_caller_cap "\<lambda>s. P (cur_thread s)"
lemma invs_sym_refs_strg:
"invs s \<longrightarrow> sym_refs (state_refs_of s)" by clarsimp
lemma lookup_slot_for_thread_cap_fault:
"\<lbrace>invs\<rbrace> lookup_slot_for_thread t s -, \<lbrace>\<lambda>f s. valid_fault (CapFault x y f)\<rbrace>"
apply (simp add: lookup_slot_for_thread_def)
apply (wp resolve_address_bits_valid_fault2)
apply clarsimp
apply (erule (1) invs_valid_tcb_ctable)
done
lemma handle_recv_pas_refined:
"\<lbrace>pas_refined aag and invs and is_subject aag \<circ> cur_thread\<rbrace> handle_recv is_blocking \<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: handle_recv_def Let_def lookup_cap_def lookup_cap_def split_def)
apply (wp handle_fault_pas_refined receive_ipc_pas_refined receive_signal_pas_refined
get_cap_auth_wp [where aag=aag] lookup_slot_for_cnode_op_authorised
lookup_slot_for_thread_authorised lookup_slot_for_thread_cap_fault
hoare_vcg_all_lift_R get_ntfn_wp
| wpc | simp
| rename_tac word1 word2 word3, rule_tac Q="\<lambda>rv s. invs s \<and> is_subject aag thread
\<and> (pasSubject aag, Receive, pasObjectAbs aag word1) \<in> pasPolicy aag"
in hoare_strengthen_post, wp, clarsimp simp: invs_valid_objs invs_sym_refs)+
apply (rule_tac Q' = "\<lambda>rv s. pas_refined aag s \<and> invs s \<and> tcb_at thread s
\<and> cur_thread s = thread \<and> is_subject aag (cur_thread s)
\<and> is_subject aag thread" in hoare_post_imp_R [rotated])
apply (fastforce simp: aag_cap_auth_def cap_auth_conferred_def cap_rights_to_auth_def valid_fault_def)
apply (wp user_getreg_inv | strengthen invs_vobjs_strgs invs_sym_refs_strg | simp)+
apply clarsimp
done
crunch respects[wp]: delete_caller_cap "integrity aag X st"
lemma invs_mdb_strgs: "invs s \<longrightarrow> valid_mdb s"
by(auto)
lemma handle_recv_integrity:
"\<lbrace>integrity aag X st and pas_refined aag and einvs and is_subject aag \<circ> cur_thread\<rbrace>
handle_recv is_blocking
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: handle_recv_def Let_def lookup_cap_def lookup_cap_def split_def)
apply (wp handle_fault_integrity_autarch receive_ipc_integrity_autarch receive_signal_integrity_autarch lookup_slot_for_thread_authorised lookup_slot_for_thread_cap_fault
get_cap_auth_wp [where aag=aag] get_ntfn_wp
| wpc | simp
| rule_tac Q="\<lambda>rv s. invs s \<and> is_subject aag thread
\<and> (pasSubject aag, Receive, pasObjectAbs aag x31) \<in> pasPolicy aag"
in hoare_strengthen_post, wp, clarsimp simp: invs_valid_objs invs_sym_refs)+
apply (rule_tac Q' = "\<lambda>rv s. pas_refined aag s \<and> einvs s \<and> is_subject aag (cur_thread s)
\<and> tcb_at thread s \<and> cur_thread s = thread
\<and> is_subject aag thread \<and> integrity aag X st s" in hoare_post_imp_R [rotated])
apply (fastforce simp: aag_cap_auth_def cap_auth_conferred_def
cap_rights_to_auth_def valid_fault_def)
apply (wp user_getreg_inv | strengthen invs_vobjs_strgs invs_sym_refs_strg invs_mdb_strgs | simp)+
apply clarsimp
done
lemma handle_reply_pas_refined[wp]:
"\<lbrace> pas_refined aag and invs and is_subject aag \<circ> cur_thread\<rbrace>
handle_reply
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
unfolding handle_reply_def
apply (rule hoare_pre)
apply (wp do_reply_transfer_pas_refined get_cap_auth_wp [where aag = aag]| wpc)+
apply (clarsimp simp: aag_cap_auth_Reply)
done
lemma handle_reply_respects:
"\<lbrace>integrity aag X st and pas_refined aag
and einvs
and is_subject aag \<circ> cur_thread\<rbrace>
handle_reply
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
unfolding handle_reply_def
apply (rule hoare_pre)
apply (wp do_reply_transfer_respects get_cap_auth_wp [where aag = aag]| wpc)+
apply (clarsimp simp: aag_cap_auth_Reply)
done
lemma ethread_set_time_slice_pas_refined[wp]:
"\<lbrace>pas_refined aag\<rbrace>
ethread_set (tcb_time_slice_update f) thread
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: ethread_set_def set_eobject_def | wp)+
apply (clarsimp simp: pas_refined_def tcb_domain_map_wellformed_aux_def)
apply (erule_tac x="(a, b)" in ballE)
apply force
apply (erule notE)
apply (erule domains_of_state_aux.cases, simp add: get_etcb_def split: split_if_asm)
apply (force intro: domtcbs)+
done
lemma thread_set_time_slice_pas_refined[wp]:
"\<lbrace>pas_refined aag\<rbrace>
thread_set_time_slice tptr time
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: thread_set_time_slice_def | wp)+
done
lemma dec_domain_time_pas_refined[wp]:
"\<lbrace>pas_refined aag\<rbrace>
dec_domain_time
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: dec_domain_time_def | wp)+
apply (clarsimp simp: pas_refined_def tcb_domain_map_wellformed_aux_def)
done
crunch pas_refined[wp]: timer_tick "pas_refined aag"
lemma handle_interrupt_pas_refined:
"\<lbrace>pas_refined aag\<rbrace>
handle_interrupt irq
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: handle_interrupt_def)
apply (rule hoare_pre)
apply (wp send_signal_pas_refined get_cap_wp
| wpc
| simp add: get_irq_slot_def get_irq_state_def)+
done
lemma dec_domain_time_integrity[wp]:
"\<lbrace>integrity aag X st\<rbrace>
dec_domain_time
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (simp add: dec_domain_time_def | wp)+
apply (clarsimp simp: integrity_subjects_def)
done
lemma timer_tick_integrity[wp]:
"\<lbrace>integrity aag X st and pas_refined aag and (\<lambda>s. ct_active s \<longrightarrow> is_subject aag (cur_thread s))\<rbrace>
timer_tick
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (simp add: timer_tick_def)
apply (wp ethread_set_integrity_autarch gts_wp
| wpc | simp add: thread_set_time_slice_def split del: split_if)+
apply (clarsimp simp: ct_in_state_def st_tcb_at_def obj_at_def)
done
lemma handle_interrupt_integrity_autarch:
"\<lbrace>integrity aag X st and pas_refined aag
and invs and (\<lambda>s. ct_active s \<longrightarrow> is_subject aag (cur_thread s))
and K (is_subject_irq aag irq)\<rbrace>
handle_interrupt irq
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: handle_interrupt_def cong: irq_state.case_cong maskInterrupt_def ackInterrupt_def resetTimer_def )
apply (rule hoare_pre)
apply (wp_once send_signal_respects get_cap_auth_wp [where aag = aag] dmo_mol_respects
| simp add: get_irq_slot_def get_irq_state_def ackInterrupt_def resetTimer_def
| wp dmo_no_mem_respects
| wpc)+
apply (fastforce simp: is_cap_simps aag_cap_auth_def cap_auth_conferred_def cap_rights_to_auth_def)
done
lemma hacky_ipc_Send:
"\<lbrakk> (pasObjectAbs aag (interrupt_irq_node s irq), Notify, pasObjectAbs aag p) \<in> pasPolicy aag; pas_refined aag s; pasMaySendIrqs aag \<rbrakk>
\<Longrightarrow> aag_has_auth_to aag Notify p"
unfolding pas_refined_def
apply (clarsimp simp: policy_wellformed_def irq_map_wellformed_aux_def)
apply (drule spec [where x = "pasIRQAbs aag irq"], drule spec [where x = "pasObjectAbs aag p"], erule mp)
apply simp
done
lemma handle_interrupt_integrity:
"\<lbrace>integrity aag X st and pas_refined aag and invs and (\<lambda>s. pasMaySendIrqs aag \<or> interrupt_states s irq \<noteq> IRQSignal)
and (\<lambda>s. ct_active s \<longrightarrow> is_subject aag (cur_thread s))\<rbrace>
handle_interrupt irq
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: handle_interrupt_def maskInterrupt_def ackInterrupt_def resetTimer_def cong: irq_state.case_cong bind_cong)
apply (rule hoare_pre)
apply (wp_once send_signal_respects get_cap_wp dmo_mol_respects dmo_no_mem_respects
| wpc
| simp add: get_irq_slot_def get_irq_state_def ackInterrupt_def resetTimer_def)+
apply clarsimp
apply (rule conjI, fastforce)+ -- "valid_objs etc."
apply (clarsimp simp: cte_wp_at_caps_of_state)
apply (rule_tac s = s in hacky_ipc_Send [where irq = irq])
apply (drule (1) cap_auth_caps_of_state)
apply (clarsimp simp: aag_cap_auth_def is_cap_simps cap_auth_conferred_def cap_rights_to_auth_def split: split_if_asm)
apply assumption+
done
lemma handle_vm_fault_integrity:
"\<lbrace>integrity aag X st and K (is_subject aag thread)\<rbrace>
handle_vm_fault thread vmfault_type
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (cases vmfault_type, simp_all)
apply (rule hoare_pre)
apply (wp as_user_integrity_autarch dmo_wp | simp add: getDFSR_def getFAR_def getIFSR_def)+
done
lemma handle_vm_pas_refined[wp]:
"\<lbrace>pas_refined aag\<rbrace>
handle_vm_fault thread vmfault_type
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (cases vmfault_type, simp_all)
apply (wp | simp)+
done
lemma handle_vm_cur_thread [wp]:
"\<lbrace>\<lambda>s. P (cur_thread s)\<rbrace>
handle_vm_fault thread vmfault_type
\<lbrace>\<lambda>rv s. P (cur_thread s)\<rbrace>"
apply (cases vmfault_type, simp_all)
apply (wp | simp)+
done
lemma handle_vm_state_refs_of [wp]:
"\<lbrace>\<lambda>s. P (state_refs_of s)\<rbrace>
handle_vm_fault thread vmfault_type
\<lbrace>\<lambda>rv s. P (state_refs_of s)\<rbrace>"
apply (cases vmfault_type, simp_all)
apply (wp | simp)+
done
lemma handle_yield_pas_refined[wp]:
"\<lbrace>pas_refined aag\<rbrace>
handle_yield
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
by (simp add: handle_yield_def | wp)+
lemma handle_event_pas_refined:
"\<lbrace>pas_refined aag and guarded_pas_domain aag and domain_sep_inv (pasMaySendIrqs aag) st'
and einvs and schact_is_rct
and (\<lambda>s. ev \<noteq> Interrupt \<longrightarrow> is_subject aag (cur_thread s)) and (\<lambda>s. ev \<noteq> Interrupt \<longrightarrow> ct_active s) \<rbrace>
handle_event ev
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (case_tac ev; simp)
apply (rename_tac syscall)
apply (case_tac syscall; simp add: handle_send_def handle_call_def)
apply ((wp handle_invocation_pas_refined handle_recv_pas_refined
handle_fault_pas_refined
| simp | clarsimp)+)
apply (fastforce simp: valid_fault_def)
apply (wp handle_fault_pas_refined
| simp)+
apply (fastforce simp: valid_fault_def)
apply (wp handle_interrupt_pas_refined handle_fault_pas_refined
hoare_vcg_conj_lift hoare_vcg_all_lift
| wpc
| rule hoare_drop_imps
| strengthen invs_vobjs_strgs
| simp)+
apply auto
done
lemma valid_fault_Unknown [simp]:
"valid_fault (UnknownSyscallException x)"
by (simp add: valid_fault_def)
lemma valid_fault_User [simp]:
"valid_fault (UserException word1 word2)"
by (simp add: valid_fault_def)
declare hy_inv[wp del]
lemma handle_yield_integrity[wp]:
"\<lbrace>integrity aag X st and pas_refined aag and is_subject aag \<circ> cur_thread\<rbrace>
handle_yield
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
by (simp add: handle_yield_def | wp)+
lemma ct_in_state_machine_state_update[simp]: "ct_in_state s (st\<lparr>machine_state := x\<rparr>) = ct_in_state s st"
apply (simp add: ct_in_state_def)
done
crunch integrity[wp]: handle_yield "integrity aag X st"
lemma handle_event_integrity:
"\<lbrace>integrity aag X st and pas_refined aag and guarded_pas_domain aag and domain_sep_inv (pasMaySendIrqs aag) st'
and einvs and schact_is_rct
and (\<lambda>s. ct_active s \<longrightarrow> is_subject aag (cur_thread s)) and (\<lambda>s. ev \<noteq> Interrupt \<longrightarrow> ct_active s) and (op = st)\<rbrace>
handle_event ev
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (case_tac "ev \<noteq> Interrupt")
apply (case_tac ev; simp)
apply (rename_tac syscall)
apply (case_tac syscall, simp_all add: handle_send_def handle_call_def)
apply (wp handle_recv_integrity handle_invocation_respects
handle_reply_respects handle_fault_integrity_autarch
handle_interrupt_integrity handle_vm_fault_integrity
handle_reply_pas_refined handle_vm_fault_valid_fault
handle_reply_valid_sched
hoare_vcg_conj_lift hoare_vcg_all_lift alternative_wp select_wp
| rule dmo_wp
| wpc
| simp add: getActiveIRQ_def domain_sep_inv_def
| clarsimp
| rule conjI hoare_vcg_E_elim
| strengthen invs_vobjs_strgs invs_mdb_strgs
| fastforce)+
done
lemma integrity_restart_context:
"\<lbrakk> integrity aag X st s; pasMayActivate aag;
st_tcb_at (op = Structures_A.Restart) thread s; \<not> is_subject aag thread \<rbrakk>
\<Longrightarrow> \<exists>tcb tcb'. get_tcb thread st = Some tcb \<and> get_tcb thread s = Some tcb' \<and> (tcb_context tcb' = tcb_context tcb \<or>
tcb_context tcb' = (tcb_context tcb)(LR_svc := tcb_context tcb FaultInstruction))"
apply (clarsimp simp: integrity_def)
apply (drule_tac x = thread in spec)
apply (erule integrity_obj.cases, auto simp add: tcb_states_of_state_def get_tcb_def st_tcb_def2)
done
lemma set_thread_state_restart_to_running_respects:
"\<lbrace>integrity aag X st and st_tcb_at (op = Structures_A.Restart) thread
and K (pasMayActivate aag)\<rbrace>
do pc \<leftarrow> as_user thread getRestartPC;
as_user thread $ setNextPC pc;
set_thread_state thread Structures_A.thread_state.Running
od
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (simp add: set_thread_state_def set_object_def as_user_def split_def setNextPC_def
getRestartPC_def setRegister_def bind_assoc getRegister_def)
apply wp
apply (clarsimp simp: in_monad fun_upd_def[symmetric] cong: if_cong)
apply (cases "is_subject aag thread")
apply (cut_tac aag=aag in integrity_update_autarch, simp+)
apply (erule integrity_trans)
apply (clarsimp simp: integrity_def obj_at_def st_tcb_at_def)
apply (clarsimp dest!: get_tcb_SomeD)
apply (rule_tac ntfn'="tcb_bound_notification ya" in tro_tcb_activate [OF refl refl])
apply clarsimp
apply (simp add: tcb_bound_notification_reset_integrity_def)+
done
lemma activate_thread_respects:
"\<lbrace>integrity aag X st and K (pasMayActivate aag)\<rbrace>
activate_thread
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: activate_thread_def arch_activate_idle_thread_def)
apply (rule hoare_pre)
apply (wp set_thread_state_restart_to_running_respects thread_get_wp'
| wpc | simp add: arch_activate_idle_thread_def get_thread_state_def)+
apply (clarsimp simp: st_tcb_at_def obj_at_def)
done
lemma activate_thread_integrity:
"\<lbrace>integrity aag X st and (\<lambda>s. cur_thread s \<noteq> idle_thread s \<longrightarrow> is_subject aag (cur_thread s)) and valid_idle\<rbrace>
activate_thread
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: activate_thread_def arch_activate_idle_thread_def)
apply (rule hoare_pre)
apply (wp gts_wp set_thread_state_integrity_autarch as_user_integrity_autarch | wpc | simp add: arch_activate_idle_thread_def)+
apply(clarsimp simp: valid_idle_def pred_tcb_at_def obj_at_def)
done
lemma activate_thread_pas_refined:
"\<lbrace> pas_refined aag \<rbrace>
activate_thread
\<lbrace>\<lambda>rv. pas_refined aag \<rbrace>"
unfolding activate_thread_def arch_activate_idle_thread_def
get_thread_state_def thread_get_def
apply (rule hoare_pre)
apply (wp set_thread_state_pas_refined hoare_drop_imps
| wpc | simp del: hoare_post_taut)+
done
lemma dmo_storeWord_respects_globals:
"\<lbrace>integrity aag X st and K (ptr_range w 2 \<subseteq> X) \<rbrace>
do_machine_op (storeWord w v)
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (rule hoare_gen_asm)
apply (rule hoare_pre)
apply (simp add: storeWord_def)
apply (wp dmo_wp)
apply clarsimp
apply (simp add: integrity_def split del: split_if)
apply (clarsimp split del: split_if)
apply (case_tac "x \<in> ptr_range w 2")
apply (rule trm_globals)
apply fastforce
apply (auto simp: is_aligned_mask [symmetric] intro!: trm_lrefl ptr_range_memI ptr_range_add_memI)
done
lemma integrity_exclusive_state [iff]:
"integrity aag X st (s\<lparr>machine_state := machine_state s \<lparr>exclusive_state := es \<rparr>\<rparr>)
= integrity aag X st s"
unfolding integrity_def
by simp
lemma dmo_clearExMonitor_respects_globals[wp]:
"\<lbrace>integrity aag X st\<rbrace>
do_machine_op clearExMonitor
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (rule hoare_pre)
apply (simp add: clearExMonitor_def | wp dmo_wp)+
done
lemma integrity_cur_thread [iff]:
"integrity aag X st (s\<lparr>cur_thread := v\<rparr>) = integrity aag X st s"
unfolding integrity_def by simp
crunch arm_globals_frame [wp]: set_vm_root "\<lambda>s. P (arm_globals_frame (arch_state s))"
(simp: crunch_simps)
lemma tcb_sched_action_dequeue_integrity_pasMayEditReadyQueues:
"\<lbrace>integrity aag X st and pas_refined aag and K (pasMayEditReadyQueues aag)\<rbrace>
tcb_sched_action tcb_sched_dequeue thread
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (simp add: tcb_sched_action_def)
apply wp
apply (clarsimp simp: integrity_def integrity_ready_queues_def pas_refined_def tcb_domain_map_wellformed_aux_def etcb_at_def get_etcb_def
split: option.splits)
done
lemma switch_to_thread_respects_pasMayEditReadyQueues:
notes tcb_sched_action_dequeue_integrity[wp del]
shows
"\<lbrace>integrity aag X st and pas_refined aag and K (pasMayEditReadyQueues aag) and
(\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X) \<rbrace>
switch_to_thread t
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
unfolding switch_to_thread_def arch_switch_to_thread_def
apply (simp add: spec_valid_def)
apply (wp dmo_storeWord_respects_globals tcb_sched_action_dequeue_integrity_pasMayEditReadyQueues | simp add: clearExMonitor_def)+
done
lemma switch_to_thread_respects:
"\<lbrace>integrity aag X st and pas_refined aag and K (is_subject aag t) and
(\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X) \<rbrace>
switch_to_thread t
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
unfolding switch_to_thread_def arch_switch_to_thread_def
apply (simp add: spec_valid_def)
apply (wp dmo_storeWord_respects_globals | simp add: clearExMonitor_def)+
done
lemma switch_to_idle_thread_respects:
"\<lbrace>integrity aag X st and (\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X)\<rbrace>
switch_to_idle_thread
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
unfolding switch_to_idle_thread_def arch_switch_to_idle_thread_def
by (wp dmo_storeWord_respects_globals | simp)+
lemma choose_thread_respects_pasMayEditReadyQueues:
"\<lbrace>integrity aag X st and pas_refined aag and einvs and valid_queues and
(\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X) and K (pasMayEditReadyQueues aag) \<rbrace>
choose_thread
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: choose_thread_def guarded_switch_to_def | wp switch_to_thread_respects_pasMayEditReadyQueues switch_to_idle_thread_respects gts_wp)+
done
text {* integrity for @{const choose_thread} without @{const pasMayEditReadyQueues} *}
lemma choose_thread_respects:
"\<lbrace>integrity aag X st and pas_refined aag and pas_cur_domain aag and einvs and valid_queues and
(\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X) \<rbrace>
choose_thread
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: choose_thread_def guarded_switch_to_def | wp switch_to_thread_respects switch_to_idle_thread_respects gts_wp)+
apply (clarsimp simp: pas_refined_def)
apply (clarsimp simp: tcb_domain_map_wellformed_aux_def)
apply (erule_tac x="(hd (max_non_empty_queue (ready_queues s (cur_domain s))), cur_domain s)" in ballE)
apply simp
apply (clarsimp simp: valid_queues_def is_etcb_at_def)
apply (erule_tac x="cur_domain s" in allE)
apply (erule_tac x="Max {prio. ready_queues s (cur_domain s) prio \<noteq> []}" in allE)
apply clarsimp
apply (erule_tac x="hd (max_non_empty_queue (ready_queues s (cur_domain s)))" in ballE)
apply (clarsimp)
apply (erule notE, rule domtcbs)
apply force
apply (simp add: etcb_at_def)
apply (simp add: max_non_empty_queue_def)
apply (erule_tac P="hd A \<in> B" for A B in notE)
apply (rule Max_prop)
apply force+
done
lemma next_domain_integrity [wp]:
"\<lbrace>integrity aag X st\<rbrace>
next_domain
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: next_domain_def thread_set_domain_def ethread_set_def set_eobject_def Let_def | wp)+
apply (clarsimp simp: get_etcb_def integrity_subjects_def integrity_eobj_def lfp_def)
done
lemma next_domain_tcb_domain_map_wellformed [wp]:
"\<lbrace>tcb_domain_map_wellformed aag\<rbrace>
next_domain
\<lbrace>\<lambda>rv. tcb_domain_map_wellformed aag\<rbrace>"
by (simp add: next_domain_def thread_set_domain_def ethread_set_def set_eobject_def Let_def | wp)+
crunch domain_time[wp]: tcb_sched_action "\<lambda>s. P (domain_time s)"
lemma valid_blocked_2_valid_blocked_except[simp]:
"valid_blocked_2 queues kh sa ct \<Longrightarrow> valid_blocked_except_2 t queues kh sa ct"
by (clarsimp simp: valid_blocked_def valid_blocked_except_def)
(* clagged from Schedule_R *)
lemma next_domain_valid_sched:
"\<lbrace> valid_sched and (\<lambda>s. scheduler_action s = choose_new_thread)\<rbrace> next_domain \<lbrace> \<lambda>_. valid_sched \<rbrace>"
apply (simp add: next_domain_def Let_def)
apply (wp, simp add: valid_sched_def valid_sched_action_2_def ct_not_in_q_2_def)
apply (simp add:valid_blocked_2_def)
done
lemma schedule_integrity:
"\<lbrace>einvs and integrity aag X st and pas_refined aag and pas_cur_domain aag
and (\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X \<and> domain_time s \<noteq> 0) \<rbrace>
schedule
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: schedule_def)
apply (rule hoare_pre)
apply (wp alternative_wp switch_to_thread_respects select_wp switch_to_idle_thread_respects
guarded_switch_to_lift choose_thread_respects gts_wp hoare_drop_imps
| wpc
| simp add: allActiveTCBs_def
| rule hoare_pre_cont)+
apply (intro allI conjI impI)
apply (clarsimp simp: valid_sched_def valid_sched_action_def weak_valid_sched_action_2_def switch_in_cur_domain_2_def in_cur_domain_2_def valid_etcbs_def invs_def valid_etcbs_def etcb_at_def st_tcb_at_def obj_at_def is_etcb_at_def split: option.splits)
apply force
apply (clarsimp simp: pas_refined_def tcb_domain_map_wellformed_aux_def)
apply (drule_tac x="(x, cur_domain s)" in bspec)
apply (force intro: domtcbs)
apply force
prefer 10
(* direct clag *)
apply (clarsimp simp: valid_sched_def valid_sched_action_def weak_valid_sched_action_2_def switch_in_cur_domain_2_def in_cur_domain_2_def valid_etcbs_def invs_def valid_etcbs_def etcb_at_def st_tcb_at_def obj_at_def is_etcb_at_def split: option.splits)
apply force
apply (clarsimp simp: pas_refined_def tcb_domain_map_wellformed_aux_def)
apply (drule_tac x="(x, cur_domain s)" in bspec)
apply (force intro: domtcbs)
apply force
apply (auto simp: obj_at_def st_tcb_at_def not_cur_thread_2_def valid_sched_def)
done
lemma schedule_integrity_pasMayEditReadyQueues:
"\<lbrace>einvs and integrity aag X st and pas_refined aag and guarded_pas_domain aag
and (\<lambda>s. ptr_range (arm_globals_frame (arch_state s)) 2 \<subseteq> X)
and K (pasMayEditReadyQueues aag) \<rbrace>
schedule
\<lbrace>\<lambda>rv. integrity aag X st\<rbrace>"
apply (simp add: schedule_def)
apply (rule hoare_pre)
apply (wp guarded_switch_to_lift switch_to_thread_respects_pasMayEditReadyQueues choose_thread_respects_pasMayEditReadyQueues
next_domain_valid_sched next_domain_valid_queues gts_wp hoare_drop_imps
| wpc | simp)+
apply (auto simp: obj_at_def st_tcb_at_def not_cur_thread_2_def valid_sched_def)
done
lemma pas_refined_cur_thread [iff]:
"pas_refined aag (s\<lparr>cur_thread := v\<rparr>) = pas_refined aag s"
unfolding pas_refined_def
by (simp add: state_objs_to_policy_def)
lemma switch_to_thread_pas_refined:
"\<lbrace>pas_refined aag\<rbrace>
switch_to_thread t
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
unfolding switch_to_thread_def arch_switch_to_thread_def
by (wp do_machine_op_pas_refined | simp)+
lemma switch_to_idle_thread_pas_refined:
"\<lbrace>pas_refined aag\<rbrace>
switch_to_idle_thread
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
unfolding switch_to_idle_thread_def arch_switch_to_idle_thread_def
by (wp do_machine_op_pas_refined | simp)+
crunch pas_refined[wp]: choose_thread "pas_refined aag" (wp: switch_to_thread_pas_refined switch_to_idle_thread_pas_refined crunch_wps)
lemma schedule_pas_refined:
"\<lbrace>pas_refined aag\<rbrace>
schedule
\<lbrace>\<lambda>rv. pas_refined aag\<rbrace>"
apply (simp add: schedule_def allActiveTCBs_def)
apply (rule hoare_pre)
apply (wp alternative_wp guarded_switch_to_lift switch_to_thread_pas_refined select_wp switch_to_idle_thread_pas_refined gts_wp| wpc | simp)+
done
lemma handle_interrupt_arch_state [wp]:
"\<lbrace>\<lambda>s :: det_ext state. P (arch_state s)\<rbrace> handle_interrupt irq \<lbrace>\<lambda>_ s. P (arch_state s)\<rbrace>"
unfolding handle_interrupt_def
apply (rule hoare_pre)
apply clarsimp
apply (wp get_cap_inv dxo_wp_weak send_signal_arch_state | wpc | simp add: get_irq_state_def)+
done
lemmas sequence_x_mapM_x = mapM_x_def [symmetric]
crunch arm_globals_frame [wp]: invoke_untyped "\<lambda>s. P (arm_globals_frame (arch_state s))"
(wp: crunch_wps without_preemption_wp syscall_valid do_machine_op_arch
simp: crunch_simps sequence_x_mapM_x
ignore: do_machine_op freeMemory clearMemory)
crunch arm_globals_frame [wp]: cap_delete_one "\<lambda>s. P (arm_globals_frame (arch_state s))"
(wp: crunch_wps without_preemption_wp syscall_valid do_machine_op_arch
hoare_unless_wp dxo_wp_weak
simp: crunch_simps sequence_x_mapM_x
ignore: do_machine_op clearMemory empty_slot_ext tcb_sched_action reschedule_required)
crunch arm_globals_frame [wp]: finalise_cap "\<lambda>s. P (arm_globals_frame (arch_state s))"
(wp: crunch_wps without_preemption_wp syscall_valid do_machine_op_arch
hoare_unless_wp select_wp dxo_wp_weak
simp: crunch_simps sequence_x_mapM_x
ignore: do_machine_op clearMemory empty_slot_ext tcb_sched_action reschedule_required)
abbreviation (input)
"invariant m P \<equiv> \<lbrace>P\<rbrace> m \<lbrace>\<lambda>_. P\<rbrace>"
lemma rec_del_arm_globals_frame [wp]:
"invariant (rec_del call) (\<lambda>s. P (arm_globals_frame (arch_state s)))"
apply (rule rec_del_preservation)
apply (wp preemption_point_inv)
apply simp+
done
crunch arm_globals_frame [wp]: cap_delete "\<lambda>s. P (arm_globals_frame (arch_state s))"
(wp: crunch_wps simp: crunch_simps)
lemma cap_revoke_arm_globals_frame [wp]:
"invariant (cap_revoke slot) (\<lambda>s. P (arm_globals_frame (arch_state s)))"
apply (rule validE_valid)
apply (rule cap_revoke_preservation)
apply (wp preemption_point_inv)
apply simp+
done
crunch_ignore (add:
cap_swap_ext cap_move_ext cap_insert_ext empty_slot_ext create_cap_ext tcb_sched_action attempt_switch_to ethread_set
reschedule_required set_thread_state_ext switch_if_required_to next_domain
set_domain recycle_cap_ext
attempt_switch_to timer_tick set_priority retype_region_ext)
crunch arm_globals_frame [wp]: handle_event "\<lambda>s. P (arm_globals_frame (arch_state s))"
(wp: crunch_wps without_preemption_wp syscall_valid do_machine_op_arch select_wp
check_cap_inv filterM_preserved hoare_unless_wp dxo_wp_weak
simp: crunch_simps ignore: do_machine_op clearMemory
getActiveIRQ getFAR getIFSR getDFSR
)
crunch cur_thread[wp]: cap_swap_for_delete,finalise_cap "\<lambda>s. P (cur_thread s)" (wp: select_wp dxo_wp_weak crunch_wps simp: crunch_simps )
lemma irq_state_indepenedent_cur_thread[simp]: "irq_state_independent_A (\<lambda>s. P (cur_thread s))"
by (simp add: irq_state_independent_def)
lemma rec_del_cur_thread[wp]:"\<lbrace>\<lambda>s. P (cur_thread s)\<rbrace> rec_del a \<lbrace>\<lambda>r s. P (cur_thread s)\<rbrace>"
apply (rule rec_del_preservation)
apply (wp preemption_point_inv|simp)+
done
crunch cur_thread[wp]: cap_delete,cap_move "\<lambda>s. P (cur_thread s)" (wp: CNodeInv_AI.cap_revoke_preservation2 mapM_wp mapM_x_wp crunch_wps dxo_wp_weak simp: filterM_mapM unless_def ignore: without_preemption filterM)
lemma cap_revoke_cur_thread[wp]: "\<lbrace>\<lambda>s. P (cur_thread s)\<rbrace> cap_revoke a \<lbrace>\<lambda>r s. P (cur_thread s)\<rbrace>"
apply (rule CNodeInv_AI.cap_revoke_preservation2)
apply (wp preemption_point_inv|simp)+
done
crunch cur_thread[wp]: cap_recycle "\<lambda>s. P (cur_thread s)" (wp: crunch_wps mapM_wp mapM_x_wp dxo_wp_weak simp: filterM_mapM unless_def ignore: without_preemption filterM)
lemma invoke_cnode_cur_thread[wp]: "\<lbrace>\<lambda>s. P (cur_thread s)\<rbrace> invoke_cnode a \<lbrace>\<lambda>r s. P (cur_thread s)\<rbrace>"
apply (simp add: invoke_cnode_def)
apply (rule hoare_pre)
apply (wp hoare_drop_imps hoare_vcg_all_lift | wpc | simp add: without_preemption_def split del: split_if)+
done
crunch cur_thread[wp]: handle_event "\<lambda>s. P (cur_thread s)" (wp: syscall_valid select_wp crunch_wps check_cap_inv cap_revoke_preservation dxo_wp_weak simp: crunch_simps filterM_mapM unless_def ignore: without_preemption check_cap_at filterM getActiveIRQ resetTimer ackInterrupt)
crunch pas_cur_domain[wp]: attempt_switch_to "pas_cur_domain pas"
crunch pas_cur_domain[wp]: ethread_set "pas_cur_domain pas"
(wp: crunch_wps simp: crunch_simps)
crunch pas_cur_domain[wp]: timer_tick "pas_cur_domain pas"
(wp: crunch_wps simp: crunch_simps)
crunch pas_cur_domain[wp]: switch_if_required_to "pas_cur_domain pas"
crunch pas_cur_domain[wp]: handle_interrupt "pas_cur_domain pas"
crunch idle_thread[wp]: preemption_point "\<lambda>s::det_state. P (idle_thread s)"
(wp: OR_choiceE_weak_wp crunch_wps simp: crunch_simps ignore: do_extended_op OR_choiceE)
(* following idle_thread and cur_domain proofs clagged from infoflow/PasUpdates.thy *)
crunch idle_thread[wp]: cap_swap_for_delete,finalise_cap,cap_move,cap_swap,cap_delete,cap_recycle "\<lambda>s::det_state. P (idle_thread s)" (wp: syscall_valid crunch_wps rec_del_preservation cap_revoke_preservation modify_wp dxo_wp_weak simp: crunch_simps check_cap_at_def filterM_mapM unless_def ignore: without_preemption filterM rec_del check_cap_at cap_revoke)
lemma cap_revoke_idle_thread[wp]:"\<lbrace>\<lambda>s::det_state. P (idle_thread s)\<rbrace> cap_revoke a \<lbrace>\<lambda>r s. P (idle_thread s)\<rbrace>"
apply (rule CNodeInv_AI.cap_revoke_preservation2)
apply wp
done
lemma invoke_cnode_idle_thread[wp]: "\<lbrace>\<lambda>s::det_state. P (idle_thread s)\<rbrace> invoke_cnode a \<lbrace>\<lambda>r s. P (idle_thread s)\<rbrace>"
apply (simp add: invoke_cnode_def)
apply (rule hoare_pre)
apply (wp | wpc | clarsimp simp: without_preemption_def crunch_simps | intro impI conjI | wp_once hoare_drop_imps hoare_vcg_all_lift)+
done
crunch idle_thread[wp]: handle_event "\<lambda>s::det_state. P (idle_thread s)" (wp: syscall_valid crunch_wps rec_del_preservation cap_revoke_preservation dxo_wp_weak simp: crunch_simps check_cap_at_def filterM_mapM unless_def ignore: without_preemption filterM rec_del check_cap_at cap_revoke resetTimer ackInterrupt getFAR getDFSR getIFSR getActiveIRQ)
crunch cur_domain[wp]: transfer_caps_loop, ethread_set, thread_set_priority, set_priority, set_domain, invoke_domain, cap_move_ext, recycle_cap_ext,timer_tick,
cap_move,cap_recycle, attempt_switch_to, switch_if_required_to
"\<lambda>s. P (cur_domain s)" (wp: transfer_caps_loop_pres crunch_wps simp: crunch_simps filterM_mapM unless_def ignore: without_preemption filterM const_on_failure )
lemma invoke_cnode_cur_domain[wp]: "\<lbrace>\<lambda>s. P (cur_domain s)\<rbrace> invoke_cnode a \<lbrace>\<lambda>r s. P (cur_domain s)\<rbrace>"
apply (simp add: invoke_cnode_def)
apply (rule hoare_pre)
apply (wp | wpc | clarsimp | intro impI conjI | wp_once crunch_wps hoare_vcg_all_lift )+
done
crunch cur_domain[wp]: handle_event "\<lambda>s. P (cur_domain s)" (wp: syscall_valid select_wp crunch_wps check_cap_inv cap_revoke_preservation simp: crunch_simps filterM_mapM unless_def ignore: without_preemption check_cap_at filterM getActiveIRQ resetTimer ackInterrupt const_on_failure getFAR getDFSR getIFSR)
lemma handle_event_guarded_pas_domain[wp]:
"\<lbrace>guarded_pas_domain aag\<rbrace> handle_event e \<lbrace>\<lambda>_. guarded_pas_domain aag\<rbrace>"
apply(wp guarded_pas_domain_lift)
done
lemma handle_interrupt_guarded_pas_domain[wp]:
"\<lbrace>guarded_pas_domain aag\<rbrace> handle_interrupt blah \<lbrace>\<lambda>_. guarded_pas_domain aag\<rbrace>"
apply(wp guarded_pas_domain_lift)
done
lemma call_kernel_integrity':
fixes st
defines "X \<equiv> ptr_range (arm_globals_frame (arch_state st)) 2"
shows "st \<turnstile> \<lbrace>einvs and pas_refined aag and is_subject aag \<circ> cur_thread and schact_is_rct and guarded_pas_domain aag
and domain_sep_inv (pasMaySendIrqs aag) st'
and (\<lambda>s. ev \<noteq> Interrupt \<longrightarrow> ct_active s) and K (pasMayActivate aag \<and> pasMayEditReadyQueues aag)\<rbrace>
call_kernel ev
\<lbrace>\<lambda>_. integrity aag X st\<rbrace>"
apply (simp add: call_kernel_def getActiveIRQ_def X_def)
apply (simp add: spec_valid_def)
apply (wp activate_thread_respects schedule_integrity_pasMayEditReadyQueues
handle_interrupt_integrity
dmo_wp alternative_wp select_wp handle_interrupt_pas_refined | simp)+
apply (rule hoare_post_impErr,
rule_tac Q = "integrity aag X st and pas_refined aag and einvs and guarded_pas_domain aag and domain_sep_inv (pasMaySendIrqs aag) st'
and is_subject aag \<circ> cur_thread
and (\<lambda>s. arm_globals_frame (arch_state s)
= arm_globals_frame (arch_state st))
and (\<lambda>_. pasMayActivate aag \<and> pasMayEditReadyQueues aag)" in valid_validE)
apply (rule hoare_pre)
apply (wp handle_event_integrity he_invs handle_event_pas_refined
handle_event_domain_sep_inv handle_event_valid_sched | simp)+
apply (fastforce simp: X_def domain_sep_inv_def)+
apply(fastforce simp: domain_sep_inv_def X_def guarded_pas_domain_def)
done
lemma call_kernel_integrity:
"\<lbrace>pas_refined pas and einvs and (\<lambda>s. ev \<noteq> Interrupt \<longrightarrow> ct_active s) and domain_sep_inv (pasMaySendIrqs pas) st'
and schact_is_rct and guarded_pas_domain pas
and is_subject pas o cur_thread and K (pasMayActivate pas \<and> pasMayEditReadyQueues pas) and (\<lambda>s. s = st)\<rbrace>
call_kernel ev
\<lbrace>\<lambda>_. integrity pas (ptr_range (arm_globals_frame (arch_state st)) 2) st\<rbrace>"
using call_kernel_integrity' [of st pas st' ev]
apply (simp add: spec_valid_def)
apply (erule hoare_chain)
apply clarsimp
apply assumption
done
lemma call_kernel_pas_refined:
"\<lbrace>einvs and pas_refined aag and is_subject aag \<circ> cur_thread and guarded_pas_domain aag and (\<lambda>s. ev \<noteq> Interrupt \<longrightarrow> ct_active s) and schact_is_rct and pas_cur_domain aag and domain_sep_inv (pasMaySendIrqs aag) st'\<rbrace>
call_kernel ev
\<lbrace>\<lambda>_. pas_refined aag\<rbrace>"
apply (simp add: call_kernel_def getActiveIRQ_def)
apply (wp activate_thread_pas_refined schedule_pas_refined handle_interrupt_pas_refined
do_machine_op_pas_refined dmo_wp alternative_wp select_wp)
apply simp
apply (rule hoare_post_impErr [OF valid_validE [where Q = "pas_refined aag and invs"]])
apply (wp he_invs handle_event_pas_refined)
apply auto
done
end