lh-l4v/proof/invariant-abstract/X64/ArchRetype_AI.thy

1287 lines
51 KiB
Plaintext

(*
* Copyright 2022, Proofcraft Pty Ltd
* Copyright 2014, General Dynamics C4 Systems
*
* SPDX-License-Identifier: GPL-2.0-only
*)
(*
Arch-specific retype invariants
*)
theory ArchRetype_AI
imports Retype_AI
begin
context Arch begin global_naming X64
named_theorems Retype_AI_assms
lemma arch_kobj_size_cong[Retype_AI_assms]:
assumes "ty = ty'" "n = n'"
shows "arch_kobj_size (default_arch_object ty dev n ) =
arch_kobj_size (default_arch_object ty' dev' n')"
by (simp add: assms default_arch_object_def split: aobject_type.splits)
lemma clearMemoryVM_return[simp, Retype_AI_assms]:
"clearMemoryVM a b = return ()"
by (simp add: clearMemoryVM_def)
lemma slot_bits_def2 [Retype_AI_assms]: "slot_bits = cte_level_bits"
by (simp add: slot_bits_def cte_level_bits_def)
definition
"no_gs_types \<equiv> UNIV - {Structures_A.CapTableObject,
ArchObject SmallPageObj, ArchObject LargePageObj,
ArchObject HugePageObj}"
lemma no_gs_types_simps [simp, Retype_AI_assms]:
"Untyped \<in> no_gs_types"
"TCBObject \<in> no_gs_types"
"EndpointObject \<in> no_gs_types"
"NotificationObject \<in> no_gs_types"
"ArchObject PageTableObj \<in> no_gs_types"
"ArchObject PageDirectoryObj \<in> no_gs_types"
"ArchObject ASIDPoolObj \<in> no_gs_types"
by (simp_all add: no_gs_types_def)
lemma retype_region_ret_folded [Retype_AI_assms]:
"\<lbrace>\<top>\<rbrace> retype_region y n bits ty d
\<lbrace>\<lambda>r s. r = retype_addrs y ty n bits\<rbrace>"
unfolding retype_region_def
apply (simp add: pageBits_def)
apply wp
apply (simp add:retype_addrs_def)
done
lemmas [wp] = unless_wp
(* These also prove facts about copy_global_mappings *)
crunch pspace_aligned[wp]: init_arch_objects "pspace_aligned"
(ignore: clearMemory wp: crunch_wps simp: crunch_simps unless_def)
crunch pspace_distinct[wp]: init_arch_objects "pspace_distinct"
(ignore: clearMemory set_object set_pml4
wp: crunch_wps set_object_distinct
simp: crunch_simps unless_def set_arch_obj_simps)
crunch mdb_inv[wp]: init_arch_objects "\<lambda>s. P (cdt s)"
(ignore: clearMemory set_pml4 wp: crunch_wps simp: crunch_simps unless_def set_arch_obj_simps)
crunch valid_mdb[wp]: init_arch_objects "valid_mdb"
(ignore: clearMemory set_pml4 wp: crunch_wps simp: crunch_simps unless_def set_arch_obj_simps)
crunch cte_wp_at[wp]: init_arch_objects "\<lambda>s. P (cte_wp_at P' p s)"
(ignore: clearMemory set_pml4
wp: crunch_wps set_aobject_cte_wp_at
simp: crunch_simps unless_def set_arch_obj_simps)
crunch typ_at[wp]: init_arch_objects "\<lambda>s. P (typ_at T p s)"
(ignore: clearMemory wp: crunch_wps simp: crunch_simps unless_def)
lemma mdb_cte_at_store_pml4e[wp]:
"\<lbrace>\<lambda>s. mdb_cte_at (swp (cte_wp_at ((\<noteq>) cap.NullCap)) s) (cdt s)\<rbrace>
store_pml4e y pml4e
\<lbrace>\<lambda>r s. mdb_cte_at (swp (cte_wp_at ((\<noteq>) cap.NullCap)) s) (cdt s)\<rbrace>"
apply (clarsimp simp:mdb_cte_at_def)
apply (simp only: imp_conv_disj)
apply (wp hoare_vcg_disj_lift hoare_vcg_all_lift)
done
lemma get_pml4e_valid[wp]:
"\<lbrace>valid_vspace_objs
and \<exists>\<rhd> (x && ~~mask pml4_bits)
and K (ucast (x && mask pml4_bits >> word_size_bits) \<notin> kernel_mapping_slots)\<rbrace>
get_pml4e x
\<lbrace>valid_pml4e\<rbrace>"
apply (simp add: get_pml4e_def)
apply wp
apply clarsimp
apply (drule (2) valid_vspace_objsD)
apply simp
done
lemma get_pml4e_wellformed[wp]:
"\<lbrace>valid_objs\<rbrace> get_pml4e x \<lbrace>\<lambda>rv _. wellformed_pml4e rv\<rbrace>"
apply (simp add: get_pml4e_def)
apply wp
apply (fastforce simp: obj_at_def valid_objs_def dom_def valid_obj_def)
done
lemma store_pml4e_wellformed[wp]:
"\<lbrace>\<lambda>s. wellformed_pml4e a\<rbrace> store_pml4e x p \<lbrace>\<lambda>rv s. wellformed_pml4e a\<rbrace>"
by (wpsimp simp: store_pml4e_def)
lemma store_pml4e_valid_objs[wp]:
"\<lbrace>valid_objs and K (wellformed_pml4e p)\<rbrace> store_pml4e x p \<lbrace>\<lambda>rv. valid_objs\<rbrace>"
apply (wpsimp simp: store_pml4e_def set_arch_obj_simps)
by (fastforce simp: valid_objs_def obj_at_def dom_def valid_obj_def)
crunch valid_objs[wp]: init_arch_objects "valid_objs"
(ignore: clearMemory wp: crunch_wps simp: unless_def)
crunch valid_arch_state[wp]: init_arch_objects "valid_arch_state"
(ignore: clearMemory set_pml4 set_object wp: crunch_wps simp: unless_def crunch_simps set_arch_obj_simps)
lemmas init_arch_objects_valid_cap[wp] = valid_cap_typ [OF init_arch_objects_typ_at]
lemmas init_arch_objects_cap_table[wp] = cap_table_at_lift_valid [OF init_arch_objects_typ_at]
crunch device_state_inv[wp]: clearMemory "\<lambda>ms. P (device_state ms)"
(wp: mapM_x_wp ignore_del: clearMemory)
crunch pspace_respects_device_region[wp]: reserve_region pspace_respects_device_region
crunch cap_refs_respects_device_region[wp]: reserve_region cap_refs_respects_device_region
crunch invs [wp]: reserve_region "invs"
crunch iflive[wp]: copy_global_mappings "if_live_then_nonz_cap"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch zombies[wp]: copy_global_mappings "zombies_final"
(wp: crunch_wps simp: set_arch_obj_simps)
crunch state_refs_of[wp]: copy_global_mappings "\<lambda>s. P (state_refs_of s)"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch valid_idle[wp]: copy_global_mappings "valid_idle"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps is_tcb_def)
crunch only_idle[wp]: copy_global_mappings "only_idle"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch ifunsafe[wp]: copy_global_mappings "if_unsafe_then_cap"
(wp: crunch_wps simp: set_arch_obj_simps)
crunch reply_caps[wp]: copy_global_mappings "valid_reply_caps"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch reply_masters[wp]: copy_global_mappings "valid_reply_masters"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch valid_global[wp]: copy_global_mappings "valid_global_refs"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch irq_node[wp]: copy_global_mappings "\<lambda>s. P (interrupt_irq_node s)"
(wp: crunch_wps)
crunch irq_states[wp]: copy_global_mappings "\<lambda>s. P (interrupt_states s)"
(wp: crunch_wps)
crunch caps_of_state[wp]: copy_global_mappings "\<lambda>s. P (caps_of_state s)"
(wp: crunch_wps)
crunch pspace_in_kernel_window[wp]: copy_global_mappings "pspace_in_kernel_window"
(wp: crunch_wps)
crunch cap_refs_in_kernel_window[wp]: copy_global_mappings "cap_refs_in_kernel_window"
(wp: crunch_wps simp: set_arch_obj_simps)
crunch pspace_respects_device_region[wp]: copy_global_mappings "pspace_respects_device_region"
(wp: crunch_wps ignore: set_pml4 simp: set_arch_obj_simps)
crunch cap_refs_respects_device_region[wp]: copy_global_mappings "cap_refs_respects_device_region"
(wp: crunch_wps simp: set_arch_obj_simps)
lemma glob_vs_refs_add_one':
"glob_vs_refs (ArchObj (PageMapL4 (pm(p := pml4e)))) =
glob_vs_refs (ArchObj (PageMapL4 pm))
- Pair (VSRef (ucast p) (Some APageMapL4)) ` set_option (pml4e_ref (pm p))
\<union> Pair (VSRef (ucast p) (Some APageMapL4)) ` set_option (pml4e_ref pml4e)"
apply (simp add: glob_vs_refs_def)
apply (rule set_eqI)
apply clarsimp
apply (rule iffI)
apply (clarsimp del: disjCI dest!: graph_ofD split: if_split_asm)
apply (rule disjI1)
apply (rule conjI)
apply (rule_tac x="(aa, ba)" in image_eqI)
apply simp
apply (simp add: graph_of_def)
apply clarsimp
apply (erule disjE)
apply (clarsimp dest!: graph_ofD)
apply (rule_tac x="(aa,ba)" in image_eqI)
apply simp
apply (clarsimp simp: graph_of_def)
apply clarsimp
apply (rule_tac x="(p,x)" in image_eqI)
apply simp
apply (clarsimp simp: graph_of_def)
done
lemma mapM_x_store_pml4e_eq_kernel_mappings_restr:
"pm \<in> S \<and> is_aligned pm pml4_bits \<and> is_aligned pm' pml4_bits
\<and> set xs \<subseteq> {..< 2 ^ (pml4_bits - word_size_bits)}
\<Longrightarrow>
\<lbrace>\<lambda>s. equal_kernel_mappings (s \<lparr> kheap := restrict_map (kheap s) (- S) \<rparr>)\<rbrace>
mapM_x (\<lambda>idx. get_pml4e (pm' + (idx << word_size_bits)) >>=
store_pml4e (pm + (idx << word_size_bits))) xs
\<lbrace>\<lambda>rv s. equal_kernel_mappings (s \<lparr> kheap := restrict_map (kheap s) (- S) \<rparr>)
\<and> (\<forall>x \<in> set xs.
(\<exists>pmv pmv'. ko_at (ArchObj (PageMapL4 pmv)) pm s
\<and> ko_at (ArchObj (PageMapL4 pmv')) pm' s
\<and> pmv (ucast x) = pmv' (ucast x)))\<rbrace>"
apply (induct xs rule: rev_induct, simp_all add: mapM_x_Nil mapM_x_append mapM_x_singleton)
apply (erule hoare_seq_ext[rotated])
apply (simp add: store_pml4e_def set_object_def set_arch_obj_simps cong: bind_cong)
apply (wp get_object_wp get_pml4e_wp)
apply (clarsimp simp: obj_at_def split del: if_split)
apply (frule shiftl_less_t2n)
apply (simp add: pml4_bits_def simple_bit_simps)
apply (simp add: is_aligned_add_helper split del: if_split)
apply (cut_tac x=x and n=word_size_bits in shiftl_shiftr_id)
apply (simp add: word_size_bits_def)
apply (erule less_le_trans)
apply (simp add: pml4_bits_def simple_bit_simps)
apply (clarsimp simp: fun_upd_def[symmetric] is_aligned_add_helper)
done
lemma equal_kernel_mappings_specific_def:
"ko_at (ArchObj (PageMapL4 pm)) p s
\<Longrightarrow> equal_kernel_mappings s
= (\<forall>p' pm'. ko_at (ArchObj (PageMapL4 pm')) p' s
\<longrightarrow> (\<forall>w \<in> kernel_mapping_slots. pm' w = pm w))"
apply (rule iffI; clarsimp simp: equal_kernel_mappings_def)
apply (rename_tac p' p'' pm' pm'' w)
apply (subgoal_tac "pm' w = pm w \<and> pm'' w = pm w")
apply (erule conjE, erule(1) trans[OF _ sym])
by blast
lemma invs_aligned_pml4D:
"\<lbrakk> pspace_aligned s; valid_arch_state s \<rbrakk> \<Longrightarrow> is_aligned (x64_global_pml4 (arch_state s)) pml4_bits"
apply (clarsimp simp: valid_arch_state_def)
apply (drule (1) is_aligned_pml4)
apply (simp add: pml4_bits_def pageBits_def)
done
(* FIXME: MOVE ? *)
lemma unat_ucast_below_64:
fixes x :: "'a :: len word"
shows "LENGTH ('a) < 64 \<Longrightarrow> unat (ucast x :: word64) = unat x"
unfolding ucast_def unat_def
apply (subst int_word_uint)
apply (subst mod_pos_pos_trivial)
apply simp
apply (rule lt2p_lem)
apply simp
apply simp
done
lemma copy_global_equal_kernel_mappings_restricted:
"is_aligned pm pml4_bits \<Longrightarrow>
\<lbrace>\<lambda>s. equal_kernel_mappings (s \<lparr> kheap := restrict_map (kheap s) (- (insert pm S)) \<rparr>)
\<and> x64_global_pml4 (arch_state s) \<notin> (insert pm S)
\<and> pspace_aligned s \<and> valid_arch_state s\<rbrace>
copy_global_mappings pm
\<lbrace>\<lambda>rv s. equal_kernel_mappings (s \<lparr> kheap := restrict_map (kheap s) (- S) \<rparr>)\<rbrace>"
apply (simp add: copy_global_mappings_def)
apply (rule hoare_seq_ext [OF _ gets_sp])
apply (rule hoare_chain)
apply (rule hoare_vcg_conj_lift)
apply (rule_tac P="global_pm \<notin> (insert pm S)" in hoare_vcg_prop)
apply (rule_tac P="is_aligned global_pm pml4_bits" in hoare_gen_asm(1))
apply (rule_tac S="insert pm S" in mapM_x_store_pml4e_eq_kernel_mappings_restr)
apply clarsimp
apply (erule order_le_less_trans)
apply (simp add: pml4_bits_def simple_bit_simps)
apply (clarsimp simp: invs_aligned_pml4D)
apply clarsimp
apply (frule_tac x="get_pml4_index pptr_base" in spec)
apply (drule mp)
apply (simp add: pptr_base_def pptrBase_def get_pml4_index_def bit_simps mask_def)
apply (clarsimp simp: obj_at_def)
apply (subst equal_kernel_mappings_specific_def)
apply (fastforce simp add: obj_at_def restrict_map_def)
apply (subst(asm) equal_kernel_mappings_specific_def)
apply (fastforce simp add: obj_at_def restrict_map_def)
apply (clarsimp simp: restrict_map_def obj_at_def)
apply (drule_tac x="ucast w" in spec, drule mp)
apply (clarsimp simp: kernel_mapping_slots_def get_pml4_index_def)
apply (rule conjI)
apply (simp add: pptr_base_shift_cast_le)
apply (rule word_le_minus_one_leq)
apply (rule order_less_le_trans, rule ucast_less)
apply simp
apply (simp add: ptTranslationBits_def)
apply (simp add: ucast_down_ucast_id word_size source_size_def
target_size_def is_down_def)
apply (drule_tac x=p' in spec)
apply (simp split: if_split_asm)
done
lemma store_pde_valid_global_pd_mappings[wp]:
"\<lbrace>valid_global_objs and valid_global_vspace_mappings
and (\<lambda>s. p && ~~ mask pml4_bits \<notin> global_refs s)\<rbrace>
store_pml4e p pml4e
\<lbrace>\<lambda>rv. valid_global_vspace_mappings\<rbrace>"
apply (simp add: store_pml4e_def set_pml4_def)
apply (wp set_aobj_valid_global_vspace_mappings)
apply simp
done
lemma store_pde_valid_ioc[wp]:
"\<lbrace>valid_ioc\<rbrace> store_pml4e ptr pml4e \<lbrace>\<lambda>_. valid_ioc\<rbrace>"
by (wpsimp simp: store_pml4e_def set_arch_obj_simps)
lemma store_pde_vms[wp]:
"\<lbrace>valid_machine_state\<rbrace> store_pml4e ptr pml4e \<lbrace>\<lambda>_. valid_machine_state\<rbrace>"
by (wpsimp simp: store_pml4e_def set_arch_obj_simps)
lemma valid_arch_caps_table_caps:
"valid_arch_caps s \<Longrightarrow> valid_table_caps s"
by (simp add: valid_arch_caps_def)
lemma valid_table_caps_aobj_upd_invalid_pml4e2:
"\<lbrakk>valid_table_caps s; kheap s p = Some (ArchObj (PageMapL4 pml4)); valid_objs s;
pml4e_ref_pages pml4e = None \<or> (\<forall>slot cap. caps_of_state s slot = Some cap \<longrightarrow> is_pml4_cap cap \<longrightarrow> p \<in> obj_refs cap \<longrightarrow>
(entry \<in> kernel_mapping_slots \<and> the (pml4e_ref_pages pml4e) \<in> set (second_level_tables (arch_state s))))\<rbrakk>
\<Longrightarrow> valid_table_caps_aobj (caps_of_state s) (arch_state s) (ArchObj (PageMapL4 (pml4(entry := pml4e)))) p"
apply (clarsimp simp: valid_table_caps_def valid_table_caps_aobj_def all_comm
empty_table_def
split: option.splits
simp del: split_paired_All )
apply (drule_tac x = cap in spec)
apply (erule impE)
apply fastforce
apply (drule_tac x = p in spec)
apply (intro impI allI conjI)
apply ((clarsimp simp: obj_at_def dest!: invs_valid_objs caps_of_state_valid | drule(2) valid_cap_typ_at)+)[12]
apply (clarsimp simp: obj_at_def dest!: ref_pages_Some pml4e_ref_pages_SomeD ref_pages_NoneD)
apply (fastforce simp: pml4e_ref_pages_def)
apply (clarsimp simp: pml4e_ref_pages_def pml4e_ref_def[split_simps pml4e.split])
apply (fastforce simp: obj_at_def dest!: ref_pages_Some pml4e_ref_pages_SomeD ref_pages_NoneD split: pml4e.split_asm)
apply (fastforce simp: obj_at_def)
apply (clarsimp simp: obj_at_def dest!: invs_valid_objs caps_of_state_valid | drule(2) valid_cap_typ_at)+
done
lemmas pml4e_ref_simps[simp] = pml4e_ref_def[split_simps pml4e.split]
lemmas pml4e_ref_pages_simps[simp] = pml4e_ref_pages_def[split_simps pml4e.split]
lemma pml4e_ref_pages_eq_refs:
"pml4e_ref_pages a = pml4e_ref a"
by (clarsimp simp: pml4e_ref_pages_def split: pml4e.splits)
lemma in_kernel_mapping_slotsI:
"\<lbrakk>get_pml4_index pptr_base \<le> x; (x::word64) << word_size_bits < 2 ^ pml4_bits; x << word_size_bits >> word_size_bits = x\<rbrakk> \<Longrightarrow> ucast x \<in> kernel_mapping_slots"
apply (clarsimp simp: kernel_mapping_slots_def pptr_base_def bit_simps pptrBase_def get_pml4_index_def mask_def)
apply word_bitwise
apply auto
done
lemma set_object_ioports[wp]:
"\<lbrace>valid_ioports and obj_at (same_caps obj) ptr\<rbrace> set_object ptr obj \<lbrace>\<lambda>rv. valid_ioports\<rbrace>"
by (wpsimp simp: set_object_def get_object_def valid_ioports_def caps_of_state_after_update)
lemma update_aobj_ioports[wp]:
"\<lbrace>valid_ioports\<rbrace> set_object ptr (ArchObj obj) \<lbrace>\<lambda>rv. valid_ioports\<rbrace>"
apply (subst set_object_def)
apply (wpsimp wp: get_object_wp)
apply (clarsimp simp: obj_at_def a_type_def valid_ioports_def caps_of_state_after_update
split: kernel_object.split_asm if_splits arch_kernel_obj.split_asm)
done
lemma copy_global_invs_mappings_restricted:
"\<lbrace>(\<lambda>s. all_invs_but_equal_kernel_mappings_restricted (insert pm S) s)
and (\<lambda>s. insert pm S \<inter> global_refs s = {})
and K (is_aligned pm pml4_bits)\<rbrace>
copy_global_mappings pm
\<lbrace>\<lambda>rv. all_invs_but_equal_kernel_mappings_restricted S\<rbrace>"
supply set_arch_obj_simps[simp]
apply (rule hoare_gen_asm)
apply (simp add: valid_pspace_def pred_conj_def)
apply (rule hoare_conjI, wp copy_global_equal_kernel_mappings_restricted)
apply (clarsimp simp: global_refs_def)
apply (rule hoare_post_add, rule hoare_vcg_conj_lift, rule hoare_TrueI)
apply (simp add: copy_global_mappings_def valid_pspace_def)
apply (rule hoare_seq_ext [OF _ gets_sp])
apply (rule hoare_strengthen_post)
apply (rule mapM_x_wp[where S="{x. get_pml4_index pptr_base \<le> x
\<and> x < 2 ^ (pml4_bits - word_size_bits)}"])
apply simp_all
apply (wpsimp wp: valid_irq_node_typ valid_irq_handlers_lift get_pml4e_wp
simp: store_pml4e_def valid_asid_map_def)
apply (clarsimp simp: valid_global_objs_def)
apply (frule(1) invs_aligned_pml4D)
apply (frule shiftl_less_t2n)
apply (simp add: pml4_bits_def simple_bit_simps)
apply (clarsimp simp: is_aligned_add_helper)
apply (cut_tac x=x and n=word_size_bits in shiftl_shiftr_id)
apply (simp add: word_size_bits_def)
apply (erule order_less_le_trans)
apply (simp add: pml4_bits_def simple_bit_simps)
apply (rule conjI)
apply (auto simp: valid_objs_def valid_obj_def dom_def obj_at_def)[1]
apply (clarsimp simp: obj_at_def empty_table_def kernel_vsrefs_def get_pml4_index_def aa_type_simps)
apply (intro conjI)
apply (clarsimp split: option.split_asm if_split_asm)+
apply (drule valid_vspace_objsD, (fastforce simp: obj_at_def)+)[1]
apply (clarsimp split: option.split_asm if_split_asm)
apply (clarsimp simp: valid_global_objs_upd_def empty_table_def)
apply (erule(2) valid_table_caps_aobj_upd_invalid_pml4e2[OF valid_arch_caps_table_caps])
apply (clarsimp simp: pml4e_ref_pages_eq_refs in_kernel_mapping_slotsI[unfolded get_pml4_index_def])
apply (clarsimp simp: valid_arch_state_asid_table_strg)
apply (clarsimp split: option.split_asm if_split_asm)+
apply (clarsimp simp: valid_global_objs_upd_def global_refs_def)
apply (erule(1) valid_kernel_mappings_if_pm_pml4e)
apply clarsimp
apply (rule ccontr)
apply (drule_tac x = "(ucast x)" in spec)
apply (clarsimp split: option.split_asm if_split_asm)+
apply (drule word_leq_minus_one_le[rotated])
apply (auto simp: pml4_bits_def simple_bit_simps)
done
lemma copy_global_mappings_valid_ioc[wp]:
"\<lbrace>valid_ioc\<rbrace> copy_global_mappings pm \<lbrace>\<lambda>_. valid_ioc\<rbrace>"
by (wpsimp wp: mapM_x_wp[of UNIV] simp: copy_global_mappings_def)
lemma copy_global_mappings_vms[wp]:
"\<lbrace>valid_machine_state\<rbrace> copy_global_mappings pd \<lbrace>\<lambda>_. valid_machine_state\<rbrace>"
by (wpsimp wp: mapM_x_wp[of UNIV] simp: copy_global_mappings_def)
lemma copy_global_mappings_invs:
"\<lbrace>invs and (\<lambda>s. pm \<notin> global_refs s)
and K (is_aligned pm pml4_bits)\<rbrace>
copy_global_mappings pm
\<lbrace>\<lambda>rv. invs\<rbrace>"
apply (fold all_invs_but_equal_kernel_mappings_restricted_eq)
apply (rule hoare_pre, rule copy_global_invs_mappings_restricted)
apply (clarsimp simp: equal_kernel_mappings_def obj_at_def
restrict_map_def)
done
crunch global_refs_inv[wp]: copy_global_mappings "\<lambda>s. P (global_refs s)"
(wp: crunch_wps)
lemma mapM_copy_global_invs_mappings_restricted:
"\<lbrace>\<lambda>s. all_invs_but_equal_kernel_mappings_restricted (set pms) s
\<and> (set pms \<inter> global_refs s = {})
\<and> (\<forall>pm \<in> set pms. is_aligned pm pml4_bits)\<rbrace>
mapM_x copy_global_mappings pms
\<lbrace>\<lambda>rv. invs\<rbrace>"
apply (fold all_invs_but_equal_kernel_mappings_restricted_eq)
apply (induct pms, simp_all only: mapM_x_Nil mapM_x_Cons K_bind_def)
apply (wp, simp)
apply (rule hoare_seq_ext, assumption, thin_tac "P" for P)
apply (rule hoare_conjI)
apply (rule hoare_pre, rule copy_global_invs_mappings_restricted)
apply clarsimp
apply (rule hoare_pre, wp)
apply clarsimp
done
lemma dmo_eq_kernel_restricted [wp, Retype_AI_assms]:
"\<lbrace>\<lambda>s. equal_kernel_mappings (kheap_update (f (kheap s)) s)\<rbrace>
do_machine_op m
\<lbrace>\<lambda>rv s. equal_kernel_mappings (kheap_update (f (kheap s)) s)\<rbrace>"
apply (simp add: do_machine_op_def split_def)
apply wp
apply (simp add: equal_kernel_mappings_def obj_at_def)
done
definition
"post_retype_invs_check tp \<equiv> tp = ArchObject PML4Obj"
declare post_retype_invs_check_def[simp]
end
context begin interpretation Arch .
requalify_consts post_retype_invs_check
end
definition
post_retype_invs :: "apiobject_type \<Rightarrow> machine_word list \<Rightarrow> 'z::state_ext state \<Rightarrow> bool"
where
"post_retype_invs tp refs \<equiv>
if post_retype_invs_check tp
then all_invs_but_equal_kernel_mappings_restricted (set refs)
else invs"
global_interpretation Retype_AI_post_retype_invs?: Retype_AI_post_retype_invs
where post_retype_invs_check = post_retype_invs_check
and post_retype_invs = post_retype_invs
by (unfold_locales; fact post_retype_invs_def)
context Arch begin global_naming X64
lemma init_arch_objects_invs_from_restricted:
"\<lbrace>post_retype_invs new_type refs
and (\<lambda>s. global_refs s \<inter> set refs = {})
and K (\<forall>ref \<in> set refs. is_aligned ref (obj_bits_api new_type obj_sz))\<rbrace>
init_arch_objects new_type ptr bits obj_sz refs
\<lbrace>\<lambda>_. invs\<rbrace>"
apply (simp add: init_arch_objects_def)
apply (rule hoare_pre)
apply (wpsimp wp: mapM_copy_global_invs_mappings_restricted
hoare_vcg_const_Ball_lift
valid_irq_node_typ)
apply (auto simp: post_retype_invs_def default_arch_object_def
pml4_bits_def pageBits_def obj_bits_api_def
global_refs_def)
done
lemma obj_bits_api_neq_0 [Retype_AI_assms]:
"ty \<noteq> Untyped \<Longrightarrow> 0 < obj_bits_api ty us"
unfolding obj_bits_api_def
by (auto simp: slot_bits_def default_arch_object_def simple_bit_simps
split: Structures_A.apiobject_type.splits aobject_type.splits)
lemma vs_lookup_sub2:
assumes ko: "\<And>ko p. \<lbrakk> ko_at ko p s; vs_refs ko \<noteq> {} \<rbrakk> \<Longrightarrow> obj_at (\<lambda>ko'. vs_refs ko \<subseteq> vs_refs ko') p s'"
assumes table: "graph_of (x64_asid_table (arch_state s)) \<subseteq> graph_of (x64_asid_table (arch_state s'))"
shows "vs_lookup s \<subseteq> vs_lookup s'"
unfolding vs_lookup_def
apply (rule Image_mono)
apply (rule vs_lookup_trans_sub2)
apply (erule (1) ko)
apply (unfold vs_asid_refs_def)
apply (rule image_mono)
apply (rule table)
done
lemma vs_lookup_pages_sub2:
assumes ko: "\<And>ko p. \<lbrakk> ko_at ko p s; vs_refs_pages ko \<noteq> {} \<rbrakk> \<Longrightarrow> obj_at (\<lambda>ko'. vs_refs_pages ko \<subseteq> vs_refs_pages ko') p s'"
assumes table: "graph_of (x64_asid_table (arch_state s)) \<subseteq> graph_of (x64_asid_table (arch_state s'))"
shows "vs_lookup_pages s \<subseteq> vs_lookup_pages s'"
unfolding vs_lookup_pages_def
apply (rule Image_mono)
apply (rule vs_lookup_pages_trans_sub2)
apply (erule (1) ko)
apply (unfold vs_asid_refs_def)
apply (rule image_mono)
apply (rule table)
done
end
global_interpretation Retype_AI_slot_bits?: Retype_AI_slot_bits
proof goal_cases
interpret Arch .
case 1 show ?case by (unfold_locales; fact Retype_AI_assms)
qed
context Arch begin global_naming X64
lemma valid_untyped_helper [Retype_AI_assms]:
assumes valid_c : "s \<turnstile> c"
and cte_at : "cte_wp_at ((=) c) q s"
and tyunt: "ty \<noteq> Untyped"
and cover : "range_cover ptr sz (obj_bits_api ty us) n"
and range : "is_untyped_cap c \<Longrightarrow> usable_untyped_range c \<inter> {ptr..ptr + of_nat (n * 2 ^ (obj_bits_api ty us)) - 1} = {}"
and pn : "pspace_no_overlap_range_cover ptr sz s"
and cn : "caps_no_overlap ptr sz s"
and vp : "valid_pspace s"
shows "valid_cap c
(s\<lparr>kheap := \<lambda>x. if x \<in> set (retype_addrs ptr ty n us) then Some (default_object ty dev us) else kheap s x\<rparr>)"
(is "valid_cap c ?ns")
proof -
have obj_at_pres: "\<And>P x. obj_at P x s \<Longrightarrow> obj_at P x ?ns"
by (clarsimp simp: obj_at_def dest: domI)
(erule pspace_no_overlapC [OF pn _ _ cover vp])
note blah[simp del] = atLeastAtMost_iff atLeastatMost_subset_iff atLeastLessThan_iff
Int_atLeastAtMost atLeastatMost_empty_iff
have cover':"range_cover ptr sz (obj_bits (default_object ty dev us)) n"
using cover tyunt
by (clarsimp simp: obj_bits_dev_irr)
show ?thesis
using cover valid_c range usable_range_emptyD[where cap = c] cte_at
apply (clarsimp simp: valid_cap_def elim!: obj_at_pres
split: cap.splits option.splits arch_cap.splits)
defer
apply ((fastforce elim!: obj_at_pres)+)[5]
apply (rename_tac word nat1 nat2)
apply (clarsimp simp: valid_untyped_def is_cap_simps obj_at_def split: if_split_asm)
apply (thin_tac "\<forall>x. Q x" for Q)
apply (frule retype_addrs_obj_range_subset_strong[where dev = dev,OF _ _ tyunt])
apply (simp add: obj_bits_dev_irr tyunt)
apply (frule usable_range_subseteq)
apply (simp add: is_cap_simps)
apply (clarsimp simp: cap_aligned_def split: if_split_asm)
apply (frule aligned_ranges_subset_or_disjoint)
apply (erule retype_addrs_aligned[where sz = sz])
apply (simp add: range_cover_def)
apply (simp add: range_cover_def word_bits_def)
apply (simp add: range_cover_def)
apply (clarsimp simp: default_obj_range Int_ac tyunt
split: if_split_asm)
apply (elim disjE)
apply (drule(2) subset_trans[THEN disjoint_subset2])
apply (drule Int_absorb2)+
apply (simp add: is_cap_simps free_index_of_def)
apply simp
apply (drule(1) disjoint_subset2[rotated])
apply (simp add: Int_ac)
apply (thin_tac "\<forall>x. Q x" for Q)
apply (frule retype_addrs_obj_range_subset[OF _ cover' tyunt])
apply (clarsimp simp: cap_aligned_def)
apply (frule aligned_ranges_subset_or_disjoint)
apply (erule retype_addrs_aligned[where sz = sz])
apply (simp add: range_cover_def)
apply (simp add: range_cover_def word_bits_def)
apply (simp add: range_cover_def)
apply (clarsimp simp: default_obj_range Int_ac tyunt
split: if_split_asm)
apply (erule disjE)
apply (simp add: cte_wp_at_caps_of_state)
apply (drule cn[unfolded caps_no_overlap_def,THEN bspec,OF ranI])
apply (simp add: p_assoc_help[symmetric])
apply (erule impE)
apply blast (* set arith *)
apply blast (* set arith *)
apply blast (* set arith *)
done
qed
lemma valid_default_arch_tcb:
"\<And>s. valid_arch_tcb default_arch_tcb s"
by (simp add: default_arch_tcb_def valid_arch_tcb_def)
end
global_interpretation Retype_AI_valid_untyped_helper?: Retype_AI_valid_untyped_helper
proof goal_cases
interpret Arch .
case 1 show ?case by (unfold_locales; fact Retype_AI_assms)
qed
locale retype_region_proofs_arch
= retype_region_proofs s ty us ptr sz n ps s'
+ Arch
for s :: "'state_ext :: state_ext state"
and ty us ptr sz n ps s'
context retype_region_proofs begin
interpretation Arch .
lemma valid_cap:
assumes cap:
"(s::'state_ext state) \<turnstile> cap \<and> untyped_range cap \<inter> {ptr .. (ptr && ~~ mask sz) + 2 ^ sz - 1} = {}"
shows "s' \<turnstile> cap"
proof -
note blah[simp del] = atLeastAtMost_iff atLeastatMost_subset_iff atLeastLessThan_iff
Int_atLeastAtMost atLeastatMost_empty_iff
have cover':"range_cover ptr sz (obj_bits (default_object ty dev us)) n"
using cover tyunt
by (clarsimp simp: obj_bits_dev_irr)
show ?thesis
using cap
apply (case_tac cap)
unfolding valid_cap_def
apply (simp_all add: valid_cap_def obj_at_pres cte_at_pres
split: option.split_asm arch_cap.split_asm
option.splits)
apply (clarsimp simp add: valid_untyped_def ps_def s'_def)
apply (intro conjI)
apply clarsimp
apply (drule disjoint_subset [OF retype_addrs_obj_range_subset [OF _ cover' tyunt]])
apply (simp add: Int_ac p_assoc_help[symmetric])
apply simp
apply clarsimp
apply (drule disjoint_subset [OF retype_addrs_obj_range_subset [OF _ cover' tyunt]])
apply (simp add: Int_ac p_assoc_help[symmetric])
apply simp
using cover tyunt
apply (simp add: obj_bits_api_def2 split: Structures_A.apiobject_type.splits)
apply clarsimp+
apply (fastforce elim!: obj_at_pres)+
done
qed
lemma valid_global_refs:
"valid_global_refs s \<Longrightarrow> valid_global_refs s'"
apply (simp add: valid_global_refs_def valid_refs_def global_refs_def idle_s')
apply (simp add: cte_retype cap_range_def)
done
lemma valid_arch_state:
"valid_arch_state s \<Longrightarrow> valid_arch_state s'"
by (clarsimp simp: valid_arch_state_def obj_at_pres
valid_asid_table_def valid_global_pts_def valid_global_pds_def valid_global_pdpts_def)
lemma vs_refs_default [simp]:
"vs_refs (default_object ty dev us) = {}"
by (simp add: default_object_def default_arch_object_def tyunt vs_refs_def
o_def pml4e_ref_def pdpte_ref_def pde_ref_def graph_of_def
split: Structures_A.apiobject_type.splits aobject_type.splits)
lemma vs_refs_pages_default [simp]:
"vs_refs_pages (default_object ty dev us) = {}"
by (simp add: default_object_def default_arch_object_def tyunt vs_refs_pages_def
o_def pml4e_ref_pages_def pdpte_ref_pages_def graph_of_def
split: Structures_A.apiobject_type.splits aobject_type.splits)
lemma vs_lookup':
"vs_lookup s' = vs_lookup s"
apply (rule order_antisym)
apply (rule vs_lookup_sub2)
apply (clarsimp simp: obj_at_def s'_def ps_def split: if_split_asm)
apply simp
apply (rule vs_lookup_sub)
apply (clarsimp simp: obj_at_def s'_def ps_def split: if_split_asm dest!: orthr)
apply simp
done
lemma vs_lookup_pages':
"vs_lookup_pages s' = vs_lookup_pages s"
apply (rule order_antisym)
apply (rule vs_lookup_pages_sub2)
apply (clarsimp simp: obj_at_def s'_def ps_def split: if_split_asm)
apply simp
apply (rule vs_lookup_pages_sub)
apply (clarsimp simp: obj_at_def s'_def ps_def split: if_split_asm dest!: orthr)
apply simp
done
lemma hyp_refs_eq: "state_hyp_refs_of s' = state_hyp_refs_of s"
unfolding s'_def ps_def by (auto simp: state_hyp_refs_of_def split: option.splits)
end
context retype_region_proofs_arch begin
lemma valid_vspace_obj_pres: "valid_vspace_obj ao s \<Longrightarrow> valid_vspace_obj ao s'"
apply (cases ao; simp add: obj_at_pres)
apply (erule allEI ballEI; rename_tac t i; case_tac "t i"; fastforce simp: data_at_def obj_at_pres)+
done
end
context retype_region_proofs begin
interpretation retype_region_proofs_arch ..
lemma valid_vspace_objs':
assumes vv: "valid_vspace_objs s"
shows "valid_vspace_objs s'"
proof
fix p ao
assume p: "(\<exists>\<rhd> p) s'"
assume "ko_at (ArchObj ao) p s'"
hence "ko_at (ArchObj ao) p s \<or> ArchObj ao = default_object ty dev us"
by (simp add: ps_def obj_at_def s'_def split: if_split_asm)
moreover
{ assume "ArchObj ao = default_object ty dev us" with tyunt
have "valid_vspace_obj ao s'" by (rule valid_vspace_obj_default)
}
moreover
{ assume "ko_at (ArchObj ao) p s"
with vv p
have "valid_vspace_obj ao s"
by (auto simp: vs_lookup' elim: valid_vspace_objsD)
hence "valid_vspace_obj ao s'"
by (rule valid_vspace_obj_pres)
}
ultimately
show "valid_vspace_obj ao s'" by blast
qed
(* ML \<open>val pre_ctxt_0 = @{context}\<close> *)
sublocale retype_region_proofs_gen?: retype_region_proofs_gen
by (unfold_locales,
auto simp: hyp_refs_eq[simplified s'_def ps_def]
valid_default_arch_tcb)
(* local_setup \<open>note_new_facts pre_ctxt_0\<close> *)
end
context Arch begin global_naming X64 (*FIXME: arch_split*)
definition
valid_vs_lookup2 :: "(vs_ref list \<times> machine_word) set \<Rightarrow> (cslot_ptr \<rightharpoonup> cap) \<Rightarrow> bool"
where
"valid_vs_lookup2 lookup caps \<equiv>
\<forall>p ref. (ref, p) \<in> lookup \<longrightarrow>
ref \<noteq> [VSRef 0 (Some AASIDPool), VSRef 0 None] \<and>
(\<exists>p' cap. caps p' = Some cap \<and> p \<in> obj_refs cap \<and> vs_cap_ref cap = Some ref)"
lemma valid_vs_lookup_def2:
"valid_vs_lookup s = valid_vs_lookup2 (vs_lookup_pages s) (null_filter (caps_of_state s))"
apply (simp add: valid_vs_lookup_def valid_vs_lookup2_def)
apply (intro iff_allI imp_cong[OF refl] disj_cong[OF refl]
iff_exI conj_cong[OF refl])
apply (auto simp: null_filter_def)
done
lemma unique_table_caps_null:
"unique_table_caps (null_filter s)
= unique_table_caps s"
apply (simp add: unique_table_caps_def)
apply (intro iff_allI)
apply (clarsimp simp: null_filter_def)
done
lemma unique_table_refs_null:
"unique_table_refs (null_filter s)
= unique_table_refs s"
apply (simp only: unique_table_refs_def)
apply (intro iff_allI)
apply (clarsimp simp: null_filter_def)
apply (auto dest!: obj_ref_none_no_asid[rule_format]
simp: table_cap_ref_def)
done
lemma all_ioports_issued_null:
"all_ioports_issued (null_filter s) = all_ioports_issued s"
apply (clarsimp simp: all_ioports_issued_def ran_null_filter)
apply (rule ext)
by (auto simp: cap_ioports_def split: cap.splits arch_cap.splits)
lemma ioports_no_overlap_null:
"ioports_no_overlap (null_filter s) = ioports_no_overlap s"
apply (clarsimp simp: ioports_no_overlap_def)
apply (intro iffI; clarsimp)
apply (case_tac cap; clarsimp simp: ran_null_filter)
apply (drule_tac x="ArchObjectCap x12" in bspec, clarsimp)
apply (case_tac cap'; clarsimp)
by (case_tac cap; clarsimp simp: ran_null_filter)
lemma pspace_respects_device_regionI:
assumes uat: "\<And>ptr sz. kheap s ptr = Some (ArchObj (DataPage False sz))
\<Longrightarrow> obj_range ptr (ArchObj $ DataPage False sz) \<subseteq> - device_region s"
and dat: "\<And>ptr sz. kheap s ptr = Some (ArchObj (DataPage True sz))
\<Longrightarrow> obj_range ptr (ArchObj $ DataPage True sz) \<subseteq> device_region s"
and inv: "pspace_aligned s" "valid_objs s"
shows "pspace_respects_device_region s"
apply (simp add: pspace_respects_device_region_def,intro conjI)
apply (rule subsetI)
apply (clarsimp simp: dom_def user_mem_def obj_at_def in_user_frame_def split: if_split_asm)
apply (frule uat)
apply (cut_tac ko = "(ArchObj (DataPage False sz))" in p_in_obj_range_internal[OF _ inv])
prefer 2
apply (fastforce simp: obj_bits_def)
apply simp
apply (rule subsetI)
apply (clarsimp simp: dom_def device_mem_def obj_at_def in_device_frame_def split: if_split_asm)
apply (frule dat)
apply (cut_tac ko = "(ArchObj (DataPage True sz))" in p_in_obj_range_internal[OF _ inv])
prefer 2
apply (fastforce simp: obj_bits_def)
apply simp
done
lemma obj_range_respect_device_range:
"\<lbrakk>kheap s ptr = Some (ArchObj (DataPage dev sz));pspace_aligned s\<rbrakk> \<Longrightarrow>
obj_range ptr (ArchObj $ DataPage dev sz) \<subseteq> (if dev then dom (device_mem s) else dom (user_mem s))"
apply (drule(1) pspace_alignedD[rotated])
apply (clarsimp simp: user_mem_def in_user_frame_def obj_at_def obj_range_def device_mem_def in_device_frame_def)
apply (intro impI conjI)
apply clarsimp
apply (rule exI[where x = sz])
apply (simp add: mask_in_range[symmetric,THEN iffD1] a_type_def)
apply clarsimp
apply (rule exI[where x = sz])
apply (simp add: mask_in_range[symmetric,THEN iffD1] a_type_def)
done
lemma pspace_respects_device_regionD:
assumes inv: "pspace_aligned s" "valid_objs s" "pspace_respects_device_region s"
shows uat: "\<And>ptr sz. kheap s ptr = Some (ArchObj (DataPage False sz))
\<Longrightarrow> obj_range ptr (ArchObj $ DataPage False sz) \<subseteq> - device_region s"
and dat: "\<And>ptr sz. kheap s ptr = Some (ArchObj (DataPage True sz))
\<Longrightarrow> obj_range ptr (ArchObj $ DataPage True sz) \<subseteq> device_region s"
using inv
apply (simp_all add: pspace_respects_device_region_def)
apply (rule subsetI)
apply (drule obj_range_respect_device_range[OF _ inv(1)])
apply (clarsimp split: if_splits)
apply (drule(1) subsetD[rotated])
apply (drule(1) subsetD[rotated])
apply (simp add: dom_def)
apply (rule subsetI)
apply (drule obj_range_respect_device_range[OF _ inv(1)])
apply (clarsimp split: if_splits)
apply (drule(1) subsetD[rotated])
apply (drule(1) subsetD[rotated])
apply (simp add: dom_def)
done
lemma default_obj_dev:
"\<lbrakk>ty \<noteq> Untyped;default_object ty dev us = ArchObj (DataPage dev' sz)\<rbrakk> \<Longrightarrow> dev = dev'"
by (clarsimp simp: default_object_def default_arch_object_def
split: apiobject_type.split_asm aobject_type.split_asm)
definition
region_in_kernel_window :: "machine_word set \<Rightarrow> 'z state \<Rightarrow> bool"
where
"region_in_kernel_window S \<equiv>
\<lambda>s. \<forall>x \<in> S. x64_kernel_vspace (arch_state s) x = X64VSpaceKernelWindow"
end
context begin interpretation Arch .
requalify_consts region_in_kernel_window
end
lemma cap_range_respects_device_region_cong[cong]:
"device_state (machine_state s) = device_state (machine_state s')
\<Longrightarrow> cap_range_respects_device_region cap s = cap_range_respects_device_region cap s'"
by (clarsimp simp: cap_range_respects_device_region_def)
context retype_region_proofs_arch begin
lemmas unique_table_caps_eq
= arg_cong[where f=unique_table_caps, OF null_filter,
simplified unique_table_caps_null]
lemmas unique_table_refs_eq
= arg_cong[where f=unique_table_refs, OF null_filter,
simplified unique_table_refs_null]
lemma valid_table_caps:
"valid_table_caps s \<Longrightarrow> valid_table_caps s'"
apply (simp add: valid_table_caps_def
del: imp_disjL)
apply (elim allEI, intro impI, simp)
apply (frule caps_retype[rotated])
apply clarsimp
apply (rule obj_at_pres)
apply simp
done
lemma valid_arch_caps:
"valid_arch_caps s \<Longrightarrow> valid_arch_caps s'"
by (clarsimp simp add: valid_arch_caps_def null_filter
valid_vs_lookup_def2 vs_lookup_pages'
unique_table_caps_eq
unique_table_refs_eq
valid_table_caps)
lemmas all_ioports_issued_eq
= arg_cong[where f=all_ioports_issued, OF null_filter,
simplified all_ioports_issued_null]
lemmas ioports_no_overlap_eq
= arg_cong[where f=ioports_no_overlap, OF null_filter,
simplified ioports_no_overlap_null]
lemma valid_ioports:
"valid_ioports s \<Longrightarrow> valid_ioports s'"
by (clarsimp simp: valid_ioports_def ioports_no_overlap_eq all_ioports_issued_eq)
lemma valid_global_objs:
"valid_global_objs s \<Longrightarrow> valid_global_objs s'"
apply (simp add: valid_global_objs_def valid_vso_at_def)
apply (elim conjE, intro conjI ballI)
apply (erule exEI)
apply (simp add: obj_at_pres valid_vspace_obj_pres)
apply (simp add: obj_at_pres)
apply (rule exEI, erule(1) bspec, simp add: obj_at_pres)+
done
lemma valid_kernel_mappings:
"valid_kernel_mappings s \<Longrightarrow> valid_kernel_mappings s'"
apply (simp add: valid_kernel_mappings_def s'_def
ball_ran_eq ps_def)
apply (simp add: default_object_def valid_kernel_mappings_if_pm_def
tyunt default_arch_object_def pml4e_ref_def
split: Structures_A.apiobject_type.split
aobject_type.split)
done
lemma equal_kernel_mappings:
"equal_kernel_mappings s \<Longrightarrow>
if ty = ArchObject PML4Obj
then equal_kernel_mappings
(s'\<lparr>kheap := kheap s' |` (- set (retype_addrs ptr ty n us))\<rparr>)
else equal_kernel_mappings s'"
apply (simp add: equal_kernel_mappings_def)
apply (intro conjI impI)
apply (elim allEI)
apply (simp add: obj_at_def restrict_map_def)
apply (simp add: s'_def ps_def)
apply (elim allEI)
apply (simp add: obj_at_def restrict_map_def)
apply (simp add: s'_def ps_def)
apply (simp add: default_object_def default_arch_object_def tyunt
split: Structures_A.apiobject_type.split
aobject_type.split)
done
lemma valid_global_vspace_mappings:
"valid_global_vspace_mappings s
\<Longrightarrow> valid_global_vspace_mappings s'"
apply (erule valid_global_vspace_mappings_pres)
apply (simp | erule obj_at_pres)+
done
lemma pspace_in_kernel_window:
"\<lbrakk> pspace_in_kernel_window (s :: 'state_ext state);
region_in_kernel_window {ptr .. (ptr &&~~ mask sz) + 2 ^ sz - 1} s \<rbrakk>
\<Longrightarrow> pspace_in_kernel_window s'"
apply (simp add: pspace_in_kernel_window_def s'_def ps_def)
apply (clarsimp simp: region_in_kernel_window_def
del: ballI)
apply (drule retype_addrs_mem_subset_ptr_bits[OF cover tyunt])
apply (fastforce simp: field_simps obj_bits_dev_irr tyunt)
done
lemma pspace_respects_device_region:
assumes psp_resp_dev: "pspace_respects_device_region s"
and cap_refs_resp_dev: "cap_refs_respects_device_region s"
shows "pspace_respects_device_region s'"
proof -
note blah[simp del] = untyped_range.simps usable_untyped_range.simps atLeastAtMost_iff
atLeastatMost_subset_iff atLeastLessThan_iff
Int_atLeastAtMost atLeastatMost_empty_iff split_paired_Ex
show ?thesis
apply (cut_tac vp)
apply (rule pspace_respects_device_regionI)
apply (clarsimp simp add: pspace_respects_device_region_def s'_def ps_def
split: if_split_asm )
apply (drule retype_addrs_obj_range_subset[OF _ _ tyunt])
using cover tyunt
apply (simp add: obj_bits_api_def3 split: if_splits)
apply (frule default_obj_dev[OF tyunt],simp)
apply (drule(1) subsetD)
apply (rule exE[OF dev])
apply (drule cap_refs_respects_device_region_cap_range[OF _ cap_refs_resp_dev])
apply (fastforce split: if_splits)
apply (drule pspace_respects_device_regionD[OF _ _ psp_resp_dev, rotated -1])
apply fastforce
apply fastforce
apply fastforce
apply (clarsimp simp add: pspace_respects_device_region_def s'_def ps_def
split: if_split_asm )
apply (drule retype_addrs_obj_range_subset[OF _ _ tyunt])
using cover tyunt
apply (simp add: obj_bits_api_def4 split: if_splits)
apply (frule default_obj_dev[OF tyunt],simp)
apply (drule(1) subsetD)
apply (rule exE[OF dev])
apply (drule cap_refs_respects_device_region_cap_range[OF _ cap_refs_resp_dev])
apply (fastforce split: if_splits)
apply (drule pspace_respects_device_regionD[OF _ _ psp_resp_dev, rotated -1])
apply fastforce
apply fastforce
apply fastforce
using valid_pspace
apply fastforce+
done
qed
lemma cap_refs_respects_device_region:
assumes psp_resp_dev: "pspace_respects_device_region s"
and cap_refs_resp_dev: "cap_refs_respects_device_region s"
shows "cap_refs_respects_device_region s'"
using cap_refs_resp_dev
apply (clarsimp simp: cap_refs_respects_device_region_def
simp del: split_paired_All split_paired_Ex)
apply (drule_tac x = "(a,b)" in spec)
apply (erule notE)
apply (subst(asm) cte_retype)
apply (simp add: cap_range_respects_device_region_def cap_range_def)
apply (clarsimp simp: cte_wp_at_caps_of_state s'_def dom_def)
done
lemma vms:
"valid_machine_state s \<Longrightarrow> valid_machine_state s'"
apply (simp add: s'_def ps_def valid_machine_state_def in_user_frame_def)
apply (rule allI, erule_tac x=p in allE, clarsimp)
apply (rule_tac x=sz in exI, clarsimp simp: obj_at_def orthr)
done
end
context retype_region_proofs begin
interpretation retype_region_proofs_arch ..
lemma post_retype_invs:
"\<lbrakk> invs s; region_in_kernel_window {ptr .. (ptr && ~~ mask sz) + 2 ^ sz - 1} s \<rbrakk>
\<Longrightarrow> post_retype_invs ty (retype_addrs ptr ty n us) s'"
using equal_kernel_mappings
by (clarsimp simp: invs_def post_retype_invs_def valid_state_def
unsafe_rep2 null_filter valid_idle
valid_reply_caps valid_reply_masters
valid_global_refs valid_arch_state
valid_irq_node_def obj_at_pres
valid_arch_caps valid_global_objs
valid_vspace_objs' valid_irq_handlers
valid_mdb_rep2 mdb_and_revokable
valid_pspace cur_tcb only_idle valid_ioports
valid_kernel_mappings valid_asid_map_def
valid_global_vspace_mappings valid_ioc vms
pspace_in_kernel_window pspace_respects_device_region
cap_refs_respects_device_region
cap_refs_in_kernel_window valid_irq_states
split: if_split_asm)
(* ML \<open>val pre_ctxt_1 = @{context}\<close> *)
sublocale retype_region_proofs_invs?: retype_region_proofs_invs
where region_in_kernel_window = region_in_kernel_window
and post_retype_invs_check = post_retype_invs_check
and post_retype_invs = post_retype_invs
using post_retype_invs valid_cap valid_global_refs valid_arch_state valid_vspace_objs'
by unfold_locales (auto simp: s'_def ps_def)
(* local_setup \<open>note_new_facts pre_ctxt_1\<close> *)
lemmas post_retype_invs_axioms = retype_region_proofs_invs_axioms
end
context Arch begin global_naming X64
named_theorems Retype_AI_assms'
lemma invs_post_retype_invs [Retype_AI_assms']:
"invs s \<Longrightarrow> post_retype_invs ty refs s"
apply (clarsimp simp: post_retype_invs_def invs_def valid_state_def)
apply (clarsimp simp: equal_kernel_mappings_def obj_at_def
restrict_map_def)
done
lemmas equal_kernel_mappings_trans_state
= more_update.equal_kernel_mappings_update
lemmas retype_region_proofs_assms [Retype_AI_assms']
= retype_region_proofs.post_retype_invs_axioms
end
global_interpretation Retype_AI?: Retype_AI
where no_gs_types = X64.no_gs_types
and post_retype_invs_check = post_retype_invs_check
and post_retype_invs = post_retype_invs
and region_in_kernel_window = region_in_kernel_window
proof goal_cases
interpret Arch .
case 1 show ?case
by (intro_locales; (unfold_locales; fact Retype_AI_assms)?)
(simp add: Retype_AI_axioms_def Retype_AI_assms')
qed
context Arch begin global_naming X64
lemma retype_region_plain_invs:
"\<lbrace>invs and caps_no_overlap ptr sz and pspace_no_overlap_range_cover ptr sz
and caps_overlap_reserved {ptr..ptr + of_nat n * 2 ^ obj_bits_api ty us - 1}
and region_in_kernel_window {ptr .. (ptr &&~~ mask sz) + 2 ^ sz - 1}
and (\<lambda>s. \<exists>slot. cte_wp_at (\<lambda>c. {ptr..(ptr && ~~ mask sz) + (2 ^ sz - 1)} \<subseteq> cap_range c \<and> cap_is_device c = dev) slot s)
and K (ty = Structures_A.CapTableObject \<longrightarrow> 0 < us)
and K (range_cover ptr sz (obj_bits_api ty us) n)
and K (ty \<noteq> ArchObject PML4Obj)\<rbrace>
retype_region ptr n us ty dev\<lbrace>\<lambda>rv. invs\<rbrace>"
apply (rule hoare_gen_asm)
apply (rule hoare_strengthen_post[OF retype_region_post_retype_invs])
apply (simp add: post_retype_invs_def)
done
lemma storeWord_um_eq_0:
"\<lbrace>\<lambda>m. underlying_memory m p = 0\<rbrace>
storeWord x 0
\<lbrace>\<lambda>_ m. underlying_memory m p = 0\<rbrace>"
by (simp add: storeWord_def word_rsplit_0 upto0_7_def word_bits_def | wp)+
lemma clearMemory_um_eq_0:
"\<lbrace>\<lambda>m. underlying_memory m p = 0\<rbrace>
clearMemory ptr bits
\<lbrace>\<lambda>_ m. underlying_memory m p = 0\<rbrace>"
apply (clarsimp simp: clearMemory_def)
apply (wp mapM_x_wp_inv | simp)+
apply (wp hoare_drop_imps storeWord_um_eq_0)
apply (fastforce simp: ignore_failure_def split: if_split_asm)
done
lemma invs_irq_state_independent:
"invs (s\<lparr>machine_state := machine_state s\<lparr>irq_state := f (irq_state (machine_state s))\<rparr>\<rparr>)
= invs s"
by (clarsimp simp: irq_state_independent_A_def invs_def
valid_state_def valid_pspace_def valid_mdb_def valid_ioc_def valid_idle_def
only_idle_def if_unsafe_then_cap_def valid_reply_caps_def
valid_reply_masters_def valid_global_refs_def valid_arch_state_def
valid_irq_node_def valid_irq_handlers_def valid_machine_state_def
valid_arch_caps_def valid_global_objs_def
valid_kernel_mappings_def equal_kernel_mappings_def
vspace_at_asid_def
pspace_in_kernel_window_def cap_refs_in_kernel_window_def
cur_tcb_def sym_refs_def state_refs_of_def
swp_def valid_irq_states_def)
crunch irq_masks_inv[wp]: storeWord, clearMemory "\<lambda>s. P (irq_masks s)"
(wp: crunch_wps ignore_del: storeWord clearMemory)
crunch underlying_mem_0[wp]: clearMemory
"\<lambda>s. underlying_memory s p = 0"
(wp: crunch_wps storeWord_um_eq_0 ignore_del: clearMemory)
lemma clearMemory_invs[wp]:
"\<lbrace>invs\<rbrace> do_machine_op (clearMemory w sz) \<lbrace>\<lambda>_. invs\<rbrace>"
apply (wp dmo_invs1)
apply clarsimp
apply (intro conjI impI allI)
apply (clarsimp simp: invs_def valid_state_def)
apply (erule_tac p=p in valid_machine_stateE)
apply (clarsimp simp: use_valid[OF _ clearMemory_underlying_mem_0])
apply (clarsimp simp: use_valid[OF _ clearMemory_irq_masks_inv[where P="(=) v" for v], OF _ refl])
done
lemma caps_region_kernel_window_imp:
"caps_of_state s p = Some cap
\<Longrightarrow> cap_refs_in_kernel_window s
\<Longrightarrow> S \<subseteq> cap_range cap
\<Longrightarrow> region_in_kernel_window S s"
apply (simp add: region_in_kernel_window_def)
apply (drule(1) cap_refs_in_kernel_windowD)
apply blast
done
(* make these available in the generic theory? *)
crunch irq_node[wp]: init_arch_objects "\<lambda>s. P (interrupt_irq_node s)"
(wp: crunch_wps)
lemma init_arch_objects_excap[wp]:
"\<lbrace>ex_cte_cap_wp_to P p\<rbrace> init_arch_objects tp ptr bits us refs \<lbrace>\<lambda>rv. ex_cte_cap_wp_to P p\<rbrace>"
by (wp ex_cte_cap_to_pres )
crunch pred_tcb_at[wp]: init_arch_objects "pred_tcb_at proj P t"
(wp: crunch_wps ignore: set_object set_pml4 simp: set_arch_obj_simps)
lemma valid_arch_mdb_detype:
"valid_arch_mdb (is_original_cap s) (caps_of_state s) \<Longrightarrow>
valid_arch_mdb (is_original_cap (detype (untyped_range cap) s))
(\<lambda>p. if fst p \<in> untyped_range cap then None else caps_of_state s p)"
by (simp add: valid_arch_mdb_def ioport_revocable_def detype_def)
lemmas init_arch_objects_wps
= init_arch_objects_cte_wp_at
init_arch_objects_valid_cap
init_arch_objects_cap_table
init_arch_objects_excap
init_arch_objects_pred_tcb_at
end
end