Commit a0ed4607 authored by Takuya Yoshikawa's avatar Takuya Yoshikawa Committed by Avi Kivity

KVM: MMU: Split the main body of rmap_write_protect() off from others

We will use this in the following patch to implement another function
which needs to write protect pages using the rmap information.

Note that there is a small change in debug printing for large pages:
we do not differentiate them from others to avoid duplicating code.
Signed-off-by: default avatarTakuya Yoshikawa <yoshikawa.takuya@oss.ntt.co.jp>
Signed-off-by: default avatarAvi Kivity <avi@redhat.com>
parent 24899709
...@@ -1010,42 +1010,43 @@ static void drop_spte(struct kvm *kvm, u64 *sptep) ...@@ -1010,42 +1010,43 @@ static void drop_spte(struct kvm *kvm, u64 *sptep)
rmap_remove(kvm, sptep); rmap_remove(kvm, sptep);
} }
int kvm_mmu_rmap_write_protect(struct kvm *kvm, u64 gfn, static int __rmap_write_protect(struct kvm *kvm, unsigned long *rmapp, int level)
struct kvm_memory_slot *slot)
{ {
unsigned long *rmapp; u64 *spte = NULL;
u64 *spte; int write_protected = 0;
int i, write_protected = 0;
rmapp = __gfn_to_rmap(gfn, PT_PAGE_TABLE_LEVEL, slot); while ((spte = rmap_next(rmapp, spte))) {
spte = rmap_next(rmapp, NULL);
while (spte) {
BUG_ON(!(*spte & PT_PRESENT_MASK)); BUG_ON(!(*spte & PT_PRESENT_MASK));
rmap_printk("rmap_write_protect: spte %p %llx\n", spte, *spte); rmap_printk("rmap_write_protect: spte %p %llx\n", spte, *spte);
if (is_writable_pte(*spte)) {
if (!is_writable_pte(*spte))
continue;
if (level == PT_PAGE_TABLE_LEVEL) {
mmu_spte_update(spte, *spte & ~PT_WRITABLE_MASK); mmu_spte_update(spte, *spte & ~PT_WRITABLE_MASK);
write_protected = 1; } else {
BUG_ON(!is_large_pte(*spte));
drop_spte(kvm, spte);
--kvm->stat.lpages;
spte = NULL;
} }
spte = rmap_next(rmapp, spte);
write_protected = 1;
} }
/* check for huge page mappings */ return write_protected;
for (i = PT_DIRECTORY_LEVEL; }
int kvm_mmu_rmap_write_protect(struct kvm *kvm, u64 gfn,
struct kvm_memory_slot *slot)
{
unsigned long *rmapp;
int i, write_protected = 0;
for (i = PT_PAGE_TABLE_LEVEL;
i < PT_PAGE_TABLE_LEVEL + KVM_NR_PAGE_SIZES; ++i) { i < PT_PAGE_TABLE_LEVEL + KVM_NR_PAGE_SIZES; ++i) {
rmapp = __gfn_to_rmap(gfn, i, slot); rmapp = __gfn_to_rmap(gfn, i, slot);
spte = rmap_next(rmapp, NULL); write_protected |= __rmap_write_protect(kvm, rmapp, i);
while (spte) {
BUG_ON(!(*spte & PT_PRESENT_MASK));
BUG_ON(!is_large_pte(*spte));
pgprintk("rmap_write_protect(large): spte %p %llx %lld\n", spte, *spte, gfn);
if (is_writable_pte(*spte)) {
drop_spte(kvm, spte);
--kvm->stat.lpages;
spte = NULL;
write_protected = 1;
}
spte = rmap_next(rmapp, spte);
}
} }
return write_protected; return write_protected;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment