aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kvm/mmu.c
diff options
context:
space:
mode:
authorTakuya Yoshikawa <yoshikawa_takuya_b1@lab.ntt.co.jp>2013-01-08 05:47:33 -0500
committerGleb Natapov <gleb@redhat.com>2013-01-14 04:14:28 -0500
commit6b81b05e449e15abb60eaa4f62cdc7954f4d74f0 (patch)
treeaad7979876b3d887e6b5891fdc6148d909a5a516 /arch/x86/kvm/mmu.c
parent9d1beefb71146bbf5f820ab17c450808b0d0b2df (diff)
KVM: MMU: Conditionally reschedule when kvm_mmu_slot_remove_write_access() takes a long time
If the userspace starts dirty logging for a large slot, say 64GB of memory, kvm_mmu_slot_remove_write_access() needs to hold mmu_lock for a long time such as tens of milliseconds. This patch controls the lock hold time by asking the scheduler if we need to reschedule for others. One penalty for this is that we need to flush TLBs before releasing mmu_lock. But since holding mmu_lock for a long time does affect not only the guest, vCPU threads in other words, but also the host as a whole, we should pay for that. In practice, the cost will not be so high because we can protect a fair amount of memory before being rescheduled: on my test environment, cond_resched_lock() was called only once for protecting 12GB of memory even without THP. We can also revisit Avi's "unlocked TLB flush" work later for completely suppressing extra TLB flushes if needed. Reviewed-by: Marcelo Tosatti <mtosatti@redhat.com> Signed-off-by: Takuya Yoshikawa <yoshikawa_takuya_b1@lab.ntt.co.jp> Signed-off-by: Gleb Natapov <gleb@redhat.com>
Diffstat (limited to 'arch/x86/kvm/mmu.c')
-rw-r--r--arch/x86/kvm/mmu.c5
1 files changed, 5 insertions, 0 deletions
diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c
index e5dcae31cebc..9f628f7a40b2 100644
--- a/arch/x86/kvm/mmu.c
+++ b/arch/x86/kvm/mmu.c
@@ -4186,6 +4186,11 @@ void kvm_mmu_slot_remove_write_access(struct kvm *kvm, int slot)
4186 for (index = 0; index <= last_index; ++index, ++rmapp) { 4186 for (index = 0; index <= last_index; ++index, ++rmapp) {
4187 if (*rmapp) 4187 if (*rmapp)
4188 __rmap_write_protect(kvm, rmapp, false); 4188 __rmap_write_protect(kvm, rmapp, false);
4189
4190 if (need_resched() || spin_needbreak(&kvm->mmu_lock)) {
4191 kvm_flush_remote_tlbs(kvm);
4192 cond_resched_lock(&kvm->mmu_lock);
4193 }
4189 } 4194 }
4190 } 4195 }
4191 4196