提交 261874b0 编写于 作者: A Alex Williamson 提交者: Gleb Natapov

kvm: Force IOMMU remapping on memory slot read-only flag changes

Memory slot flags can be altered without changing other parameters of
the slot.  The read-only attribute is the only one the IOMMU cares
about, so generate an un-map, re-map when this occurs.  This also
avoid unnecessarily re-mapping the slot when no IOMMU visible changes
are made.
Reviewed-by: NXiao Guangrong <xiaoguangrong@linux.vnet.ibm.com>
Signed-off-by: NAlex Williamson <alex.williamson@redhat.com>
Signed-off-by: NGleb Natapov <gleb@redhat.com>
上级 3f0c3d0b
...@@ -731,6 +731,7 @@ int __kvm_set_memory_region(struct kvm *kvm, ...@@ -731,6 +731,7 @@ int __kvm_set_memory_region(struct kvm *kvm,
struct kvm_memory_slot *slot; struct kvm_memory_slot *slot;
struct kvm_memory_slot old, new; struct kvm_memory_slot old, new;
struct kvm_memslots *slots = NULL, *old_memslots; struct kvm_memslots *slots = NULL, *old_memslots;
bool old_iommu_mapped;
r = check_memory_region_flags(mem); r = check_memory_region_flags(mem);
if (r) if (r)
...@@ -772,6 +773,8 @@ int __kvm_set_memory_region(struct kvm *kvm, ...@@ -772,6 +773,8 @@ int __kvm_set_memory_region(struct kvm *kvm,
new.npages = npages; new.npages = npages;
new.flags = mem->flags; new.flags = mem->flags;
old_iommu_mapped = old.npages;
/* /*
* Disallow changing a memory slot's size or changing anything about * Disallow changing a memory slot's size or changing anything about
* zero sized slots that doesn't involve making them non-zero. * zero sized slots that doesn't involve making them non-zero.
...@@ -835,6 +838,7 @@ int __kvm_set_memory_region(struct kvm *kvm, ...@@ -835,6 +838,7 @@ int __kvm_set_memory_region(struct kvm *kvm,
/* slot was deleted or moved, clear iommu mapping */ /* slot was deleted or moved, clear iommu mapping */
kvm_iommu_unmap_pages(kvm, &old); kvm_iommu_unmap_pages(kvm, &old);
old_iommu_mapped = false;
/* From this point no new shadow pages pointing to a deleted, /* From this point no new shadow pages pointing to a deleted,
* or moved, memslot will be created. * or moved, memslot will be created.
* *
...@@ -863,11 +867,27 @@ int __kvm_set_memory_region(struct kvm *kvm, ...@@ -863,11 +867,27 @@ int __kvm_set_memory_region(struct kvm *kvm,
goto out_free; goto out_free;
} }
/* map new memory slot into the iommu */ /*
* IOMMU mapping: New slots need to be mapped. Old slots need to be
* un-mapped and re-mapped if their base changes or if flags that the
* iommu cares about change (read-only). Base change unmapping is
* handled above with slot deletion, so we only unmap incompatible
* flags here. Anything else the iommu might care about for existing
* slots (size changes, userspace addr changes) is disallowed above,
* so any other attribute changes getting here can be skipped.
*/
if (npages) { if (npages) {
r = kvm_iommu_map_pages(kvm, &new); if (old_iommu_mapped &&
if (r) ((new.flags ^ old.flags) & KVM_MEM_READONLY)) {
goto out_slots; kvm_iommu_unmap_pages(kvm, &old);
old_iommu_mapped = false;
}
if (!old_iommu_mapped) {
r = kvm_iommu_map_pages(kvm, &new);
if (r)
goto out_slots;
}
} }
/* actual memory is freed via old in kvm_free_physmem_slot below */ /* actual memory is freed via old in kvm_free_physmem_slot below */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册