Commit f2d3155e authored by Nico Boehr's avatar Nico Boehr Committed by Janosch Frank

KVM: s390: disable migration mode when dirty tracking is disabled

Migration mode is a VM attribute which enables tracking of changes in
storage attributes (PGSTE). It assumes dirty tracking is enabled on all
memslots to keep a dirty bitmap of pages with changed storage attributes.

When enabling migration mode, we currently check that dirty tracking is
enabled for all memslots. However, userspace can disable dirty tracking
without disabling migration mode.

Since migration mode is pointless with dirty tracking disabled, disable
migration mode whenever userspace disables dirty tracking on any slot.

Also update the documentation to clarify that dirty tracking must be
enabled when enabling migration mode, which is already enforced by the
code in kvm_s390_vm_start_migration().

Also highlight in the documentation for KVM_S390_GET_CMMA_BITS that it
can now fail with -EINVAL when dirty tracking is disabled while
migration mode is on. Move all the error codes to a table so this stays
readable.

To disable migration mode, slots_lock should be held, which is taken
in kvm_set_memory_region() and thus held in
kvm_arch_prepare_memory_region().

Restructure the prepare code a bit so all the sanity checking is done
before disabling migration mode. This ensures migration mode isn't
disabled when some sanity check fails.

Cc: stable@vger.kernel.org
Fixes: 190df4a2 ("KVM: s390: CMMA tracking, ESSA emulation, migration mode")
Signed-off-by: default avatarNico Boehr <nrb@linux.ibm.com>
Reviewed-by: default avatarJanosch Frank <frankja@linux.ibm.com>
Reviewed-by: default avatarClaudio Imbrenda <imbrenda@linux.ibm.com>
Link: https://lore.kernel.org/r/20230127140532.230651-2-nrb@linux.ibm.com
Message-Id: <20230127140532.230651-2-nrb@linux.ibm.com>
[frankja@linux.ibm.com: fixed commit message typo, moved api.rst error table upwards]
Signed-off-by: default avatarJanosch Frank <frankja@linux.ibm.com>
parent a2ce98d6
...@@ -4449,6 +4449,18 @@ not holding a previously reported uncorrected error). ...@@ -4449,6 +4449,18 @@ not holding a previously reported uncorrected error).
:Parameters: struct kvm_s390_cmma_log (in, out) :Parameters: struct kvm_s390_cmma_log (in, out)
:Returns: 0 on success, a negative value on error :Returns: 0 on success, a negative value on error
Errors:
====== =============================================================
ENOMEM not enough memory can be allocated to complete the task
ENXIO if CMMA is not enabled
EINVAL if KVM_S390_CMMA_PEEK is not set but migration mode was not enabled
EINVAL if KVM_S390_CMMA_PEEK is not set but dirty tracking has been
disabled (and thus migration mode was automatically disabled)
EFAULT if the userspace address is invalid or if no page table is
present for the addresses (e.g. when using hugepages).
====== =============================================================
This ioctl is used to get the values of the CMMA bits on the s390 This ioctl is used to get the values of the CMMA bits on the s390
architecture. It is meant to be used in two scenarios: architecture. It is meant to be used in two scenarios:
...@@ -4529,12 +4541,6 @@ mask is unused. ...@@ -4529,12 +4541,6 @@ mask is unused.
values points to the userspace buffer where the result will be stored. values points to the userspace buffer where the result will be stored.
This ioctl can fail with -ENOMEM if not enough memory can be allocated to
complete the task, with -ENXIO if CMMA is not enabled, with -EINVAL if
KVM_S390_CMMA_PEEK is not set but migration mode was not enabled, with
-EFAULT if the userspace address is invalid or if no page table is
present for the addresses (e.g. when using hugepages).
4.108 KVM_S390_SET_CMMA_BITS 4.108 KVM_S390_SET_CMMA_BITS
---------------------------- ----------------------------
......
...@@ -302,6 +302,10 @@ Allows userspace to start migration mode, needed for PGSTE migration. ...@@ -302,6 +302,10 @@ Allows userspace to start migration mode, needed for PGSTE migration.
Setting this attribute when migration mode is already active will have Setting this attribute when migration mode is already active will have
no effects. no effects.
Dirty tracking must be enabled on all memslots, else -EINVAL is returned. When
dirty tracking is disabled on any memslot, migration mode is automatically
stopped.
:Parameters: none :Parameters: none
:Returns: -ENOMEM if there is not enough free memory to start migration mode; :Returns: -ENOMEM if there is not enough free memory to start migration mode;
-EINVAL if the state of the VM is invalid (e.g. no memory defined); -EINVAL if the state of the VM is invalid (e.g. no memory defined);
......
...@@ -5633,23 +5633,40 @@ int kvm_arch_prepare_memory_region(struct kvm *kvm, ...@@ -5633,23 +5633,40 @@ int kvm_arch_prepare_memory_region(struct kvm *kvm,
if (kvm_s390_pv_get_handle(kvm)) if (kvm_s390_pv_get_handle(kvm))
return -EINVAL; return -EINVAL;
if (change == KVM_MR_DELETE || change == KVM_MR_FLAGS_ONLY) if (change != KVM_MR_DELETE && change != KVM_MR_FLAGS_ONLY) {
return 0; /*
* A few sanity checks. We can have memory slots which have to be
* located/ended at a segment boundary (1MB). The memory in userland is
* ok to be fragmented into various different vmas. It is okay to mmap()
* and munmap() stuff in this slot after doing this call at any time
*/
/* A few sanity checks. We can have memory slots which have to be if (new->userspace_addr & 0xffffful)
located/ended at a segment boundary (1MB). The memory in userland is return -EINVAL;
ok to be fragmented into various different vmas. It is okay to mmap()
and munmap() stuff in this slot after doing this call at any time */
if (new->userspace_addr & 0xffffful) size = new->npages * PAGE_SIZE;
return -EINVAL; if (size & 0xffffful)
return -EINVAL;
size = new->npages * PAGE_SIZE; if ((new->base_gfn * PAGE_SIZE) + size > kvm->arch.mem_limit)
if (size & 0xffffful) return -EINVAL;
return -EINVAL; }
if ((new->base_gfn * PAGE_SIZE) + size > kvm->arch.mem_limit) if (!kvm->arch.migration_mode)
return -EINVAL; return 0;
/*
* Turn off migration mode when:
* - userspace creates a new memslot with dirty logging off,
* - userspace modifies an existing memslot (MOVE or FLAGS_ONLY) and
* dirty logging is turned off.
* Migration mode expects dirty page logging being enabled to store
* its dirty bitmap.
*/
if (change != KVM_MR_DELETE &&
!(new->flags & KVM_MEM_LOG_DIRTY_PAGES))
WARN(kvm_s390_vm_stop_migration(kvm),
"Failed to stop migration mode");
return 0; return 0;
} }
......
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