aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kvm/x86.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/kvm/x86.c')
-rw-r--r--arch/x86/kvm/x86.c59
1 files changed, 59 insertions, 0 deletions
diff --git a/arch/x86/kvm/x86.c b/arch/x86/kvm/x86.c
index 3df0b7a140b0..ca74c1dadf3a 100644
--- a/arch/x86/kvm/x86.c
+++ b/arch/x86/kvm/x86.c
@@ -6239,6 +6239,65 @@ void kvm_arch_destroy_vm(struct kvm *kvm)
6239 put_page(kvm->arch.ept_identity_pagetable); 6239 put_page(kvm->arch.ept_identity_pagetable);
6240} 6240}
6241 6241
6242void kvm_arch_free_memslot(struct kvm_memory_slot *free,
6243 struct kvm_memory_slot *dont)
6244{
6245 int i;
6246
6247 for (i = 0; i < KVM_NR_PAGE_SIZES - 1; ++i) {
6248 if (!dont || free->arch.lpage_info[i] != dont->arch.lpage_info[i]) {
6249 vfree(free->arch.lpage_info[i]);
6250 free->arch.lpage_info[i] = NULL;
6251 }
6252 }
6253}
6254
6255int kvm_arch_create_memslot(struct kvm_memory_slot *slot, unsigned long npages)
6256{
6257 int i;
6258
6259 for (i = 0; i < KVM_NR_PAGE_SIZES - 1; ++i) {
6260 unsigned long ugfn;
6261 int lpages;
6262 int level = i + 2;
6263
6264 lpages = gfn_to_index(slot->base_gfn + npages - 1,
6265 slot->base_gfn, level) + 1;
6266
6267 slot->arch.lpage_info[i] =
6268 vzalloc(lpages * sizeof(*slot->arch.lpage_info[i]));
6269 if (!slot->arch.lpage_info[i])
6270 goto out_free;
6271
6272 if (slot->base_gfn & (KVM_PAGES_PER_HPAGE(level) - 1))
6273 slot->arch.lpage_info[i][0].write_count = 1;
6274 if ((slot->base_gfn + npages) & (KVM_PAGES_PER_HPAGE(level) - 1))
6275 slot->arch.lpage_info[i][lpages - 1].write_count = 1;
6276 ugfn = slot->userspace_addr >> PAGE_SHIFT;
6277 /*
6278 * If the gfn and userspace address are not aligned wrt each
6279 * other, or if explicitly asked to, disable large page
6280 * support for this slot
6281 */
6282 if ((slot->base_gfn ^ ugfn) & (KVM_PAGES_PER_HPAGE(level) - 1) ||
6283 !kvm_largepages_enabled()) {
6284 unsigned long j;
6285
6286 for (j = 0; j < lpages; ++j)
6287 slot->arch.lpage_info[i][j].write_count = 1;
6288 }
6289 }
6290
6291 return 0;
6292
6293out_free:
6294 for (i = 0; i < KVM_NR_PAGE_SIZES - 1; ++i) {
6295 vfree(slot->arch.lpage_info[i]);
6296 slot->arch.lpage_info[i] = NULL;
6297 }
6298 return -ENOMEM;
6299}
6300
6242int kvm_arch_prepare_memory_region(struct kvm *kvm, 6301int kvm_arch_prepare_memory_region(struct kvm *kvm,
6243 struct kvm_memory_slot *memslot, 6302 struct kvm_memory_slot *memslot,
6244 struct kvm_memory_slot old, 6303 struct kvm_memory_slot old,