提交 bb3b394d 编写于 作者: L Lai Jiangshan 提交者: Paolo Bonzini

KVM: X86: Rename gpte_is_8_bytes to has_4_byte_gpte and invert the direction

This bit is very close to mean "role.quadrant is not in use", except that
it is false also when the MMU is mapping guest physical addresses
directly.  In that case, role.quadrant is indeed not in use, but there
are no guest PTEs at all.

Changing the name and direction of the bit removes the special case,
since a guest with paging disabled, or not considering guest paging
structures as is the case for two-dimensional paging, does not have
to deal with 4-byte guest PTEs.
Suggested-by: NPaolo Bonzini <pbonzini@redhat.com>
Signed-off-by: NLai Jiangshan <laijs@linux.alibaba.com>
Message-Id: <20211124122055.64424-10-jiangshanlai@gmail.com>
Signed-off-by: NPaolo Bonzini <pbonzini@redhat.com>
上级 f8cd457f
...@@ -161,7 +161,7 @@ Shadow pages contain the following information: ...@@ -161,7 +161,7 @@ Shadow pages contain the following information:
If clear, this page corresponds to a guest page table denoted by the gfn If clear, this page corresponds to a guest page table denoted by the gfn
field. field.
role.quadrant: role.quadrant:
When role.gpte_is_8_bytes=0, the guest uses 32-bit gptes while the host uses 64-bit When role.has_4_byte_gpte=1, the guest uses 32-bit gptes while the host uses 64-bit
sptes. That means a guest page table contains more ptes than the host, sptes. That means a guest page table contains more ptes than the host,
so multiple shadow pages are needed to shadow one guest page. so multiple shadow pages are needed to shadow one guest page.
For first-level shadow pages, role.quadrant can be 0 or 1 and denotes the For first-level shadow pages, role.quadrant can be 0 or 1 and denotes the
...@@ -177,9 +177,9 @@ Shadow pages contain the following information: ...@@ -177,9 +177,9 @@ Shadow pages contain the following information:
The page is invalid and should not be used. It is a root page that is The page is invalid and should not be used. It is a root page that is
currently pinned (by a cpu hardware register pointing to it); once it is currently pinned (by a cpu hardware register pointing to it); once it is
unpinned it will be destroyed. unpinned it will be destroyed.
role.gpte_is_8_bytes: role.has_4_byte_gpte:
Reflects the size of the guest PTE for which the page is valid, i.e. '1' Reflects the size of the guest PTE for which the page is valid, i.e. '0'
if 64-bit gptes are in use, '0' if 32-bit gptes are in use. if direct map or 64-bit gptes are in use, '1' if 32-bit gptes are in use.
role.efer_nx: role.efer_nx:
Contains the value of efer.nx for which the page is valid. Contains the value of efer.nx for which the page is valid.
role.cr0_wp: role.cr0_wp:
......
...@@ -296,14 +296,14 @@ struct kvm_kernel_irq_routing_entry; ...@@ -296,14 +296,14 @@ struct kvm_kernel_irq_routing_entry;
* *
* - invalid shadow pages are not accounted, so the bits are effectively 18 * - invalid shadow pages are not accounted, so the bits are effectively 18
* *
* - quadrant will only be used if gpte_is_8_bytes=0 (non-PAE paging); * - quadrant will only be used if has_4_byte_gpte=1 (non-PAE paging);
* execonly and ad_disabled are only used for nested EPT which has * execonly and ad_disabled are only used for nested EPT which has
* gpte_is_8_bytes=1. Therefore, 2 bits are always unused. * has_4_byte_gpte=0. Therefore, 2 bits are always unused.
* *
* - the 4 bits of level are effectively limited to the values 2/3/4/5, * - the 4 bits of level are effectively limited to the values 2/3/4/5,
* as 4k SPs are not tracked (allowed to go unsync). In addition non-PAE * as 4k SPs are not tracked (allowed to go unsync). In addition non-PAE
* paging has exactly one upper level, making level completely redundant * paging has exactly one upper level, making level completely redundant
* when gpte_is_8_bytes=0. * when has_4_byte_gpte=1.
* *
* - on top of this, smep_andnot_wp and smap_andnot_wp are only set if * - on top of this, smep_andnot_wp and smap_andnot_wp are only set if
* cr0_wp=0, therefore these three bits only give rise to 5 possibilities. * cr0_wp=0, therefore these three bits only give rise to 5 possibilities.
...@@ -315,7 +315,7 @@ union kvm_mmu_page_role { ...@@ -315,7 +315,7 @@ union kvm_mmu_page_role {
u32 word; u32 word;
struct { struct {
unsigned level:4; unsigned level:4;
unsigned gpte_is_8_bytes:1; unsigned has_4_byte_gpte:1;
unsigned quadrant:2; unsigned quadrant:2;
unsigned direct:1; unsigned direct:1;
unsigned access:3; unsigned access:3;
......
...@@ -2081,7 +2081,7 @@ static struct kvm_mmu_page *kvm_mmu_get_page(struct kvm_vcpu *vcpu, ...@@ -2081,7 +2081,7 @@ static struct kvm_mmu_page *kvm_mmu_get_page(struct kvm_vcpu *vcpu,
role.level = level; role.level = level;
role.direct = direct; role.direct = direct;
role.access = access; role.access = access;
if (!direct_mmu && !role.gpte_is_8_bytes) { if (role.has_4_byte_gpte) {
quadrant = gaddr >> (PAGE_SHIFT + (PT64_PT_BITS * level)); quadrant = gaddr >> (PAGE_SHIFT + (PT64_PT_BITS * level));
quadrant &= (1 << ((PT32_PT_BITS - PT64_PT_BITS) * level)) - 1; quadrant &= (1 << ((PT32_PT_BITS - PT64_PT_BITS) * level)) - 1;
role.quadrant = quadrant; role.quadrant = quadrant;
...@@ -4746,7 +4746,7 @@ kvm_calc_tdp_mmu_root_page_role(struct kvm_vcpu *vcpu, ...@@ -4746,7 +4746,7 @@ kvm_calc_tdp_mmu_root_page_role(struct kvm_vcpu *vcpu,
role.base.ad_disabled = (shadow_accessed_mask == 0); role.base.ad_disabled = (shadow_accessed_mask == 0);
role.base.level = kvm_mmu_get_tdp_level(vcpu); role.base.level = kvm_mmu_get_tdp_level(vcpu);
role.base.direct = true; role.base.direct = true;
role.base.gpte_is_8_bytes = true; role.base.has_4_byte_gpte = false;
return role; return role;
} }
...@@ -4791,7 +4791,7 @@ kvm_calc_shadow_root_page_role_common(struct kvm_vcpu *vcpu, ...@@ -4791,7 +4791,7 @@ kvm_calc_shadow_root_page_role_common(struct kvm_vcpu *vcpu,
role.base.smep_andnot_wp = role.ext.cr4_smep && !____is_cr0_wp(regs); role.base.smep_andnot_wp = role.ext.cr4_smep && !____is_cr0_wp(regs);
role.base.smap_andnot_wp = role.ext.cr4_smap && !____is_cr0_wp(regs); role.base.smap_andnot_wp = role.ext.cr4_smap && !____is_cr0_wp(regs);
role.base.gpte_is_8_bytes = ____is_cr0_pg(regs) && ____is_cr4_pae(regs); role.base.has_4_byte_gpte = ____is_cr0_pg(regs) && !____is_cr4_pae(regs);
return role; return role;
} }
...@@ -4890,7 +4890,7 @@ kvm_calc_shadow_ept_root_page_role(struct kvm_vcpu *vcpu, bool accessed_dirty, ...@@ -4890,7 +4890,7 @@ kvm_calc_shadow_ept_root_page_role(struct kvm_vcpu *vcpu, bool accessed_dirty,
role.base.smm = vcpu->arch.root_mmu.mmu_role.base.smm; role.base.smm = vcpu->arch.root_mmu.mmu_role.base.smm;
role.base.level = level; role.base.level = level;
role.base.gpte_is_8_bytes = true; role.base.has_4_byte_gpte = false;
role.base.direct = false; role.base.direct = false;
role.base.ad_disabled = !accessed_dirty; role.base.ad_disabled = !accessed_dirty;
role.base.guest_mode = true; role.base.guest_mode = true;
...@@ -5168,7 +5168,7 @@ static bool detect_write_misaligned(struct kvm_mmu_page *sp, gpa_t gpa, ...@@ -5168,7 +5168,7 @@ static bool detect_write_misaligned(struct kvm_mmu_page *sp, gpa_t gpa,
gpa, bytes, sp->role.word); gpa, bytes, sp->role.word);
offset = offset_in_page(gpa); offset = offset_in_page(gpa);
pte_size = sp->role.gpte_is_8_bytes ? 8 : 4; pte_size = sp->role.has_4_byte_gpte ? 4 : 8;
/* /*
* Sometimes, the OS only writes the last one bytes to update status * Sometimes, the OS only writes the last one bytes to update status
...@@ -5192,7 +5192,7 @@ static u64 *get_written_sptes(struct kvm_mmu_page *sp, gpa_t gpa, int *nspte) ...@@ -5192,7 +5192,7 @@ static u64 *get_written_sptes(struct kvm_mmu_page *sp, gpa_t gpa, int *nspte)
page_offset = offset_in_page(gpa); page_offset = offset_in_page(gpa);
level = sp->role.level; level = sp->role.level;
*nspte = 1; *nspte = 1;
if (!sp->role.gpte_is_8_bytes) { if (sp->role.has_4_byte_gpte) {
page_offset <<= 1; /* 32->64 */ page_offset <<= 1; /* 32->64 */
/* /*
* A 32-bit pde maps 4MB while the shadow pdes map * A 32-bit pde maps 4MB while the shadow pdes map
......
...@@ -35,7 +35,7 @@ ...@@ -35,7 +35,7 @@
" %snxe %sad root %u %s%c", \ " %snxe %sad root %u %s%c", \
__entry->mmu_valid_gen, \ __entry->mmu_valid_gen, \
__entry->gfn, role.level, \ __entry->gfn, role.level, \
role.gpte_is_8_bytes ? 8 : 4, \ role.has_4_byte_gpte ? 4 : 8, \
role.quadrant, \ role.quadrant, \
role.direct ? " direct" : "", \ role.direct ? " direct" : "", \
access_str[role.access], \ access_str[role.access], \
......
...@@ -165,7 +165,7 @@ static union kvm_mmu_page_role page_role_for_level(struct kvm_vcpu *vcpu, ...@@ -165,7 +165,7 @@ static union kvm_mmu_page_role page_role_for_level(struct kvm_vcpu *vcpu,
role = vcpu->arch.mmu->mmu_role.base; role = vcpu->arch.mmu->mmu_role.base;
role.level = level; role.level = level;
role.direct = true; role.direct = true;
role.gpte_is_8_bytes = true; role.has_4_byte_gpte = false;
role.access = ACC_ALL; role.access = ACC_ALL;
role.ad_disabled = !shadow_accessed_mask; role.ad_disabled = !shadow_accessed_mask;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册