KVM: s390: Intercept the tprot instruction
authorMatthew Rosato <mjrosato@linux.vnet.ibm.com>
Tue, 29 Jan 2013 16:48:20 +0000 (11:48 -0500)
committerChristian Borntraeger <borntraeger@de.ibm.com>
Fri, 30 May 2014 07:39:40 +0000 (09:39 +0200)
Based on original patch from Jeng-fang (Nick) Wang

When standby memory is specified for a guest Linux, but no virtual memory has
been allocated on the Qemu host backing that guest, the guest memory detection
process encounters a memory access exception which is not thrown from the KVM
handle_tprot() instruction-handler function. The access exception comes from
sie64a returning EFAULT, which then passes an addressing exception to the guest.
Unfortunately this does not the proper PSW fixup (nullifying vs.
suppressing) so the guest will get a fault for the wrong address.

Let's just intercept the tprot instruction all the time to do the right thing
and not go the page fault handler path for standby memory. tprot is only used
by Linux during startup so some exits should be ok.
Without this patch, standby memory cannot be used with KVM.

Signed-off-by: Nick Wang <jfwang@us.ibm.com>
Reviewed-by: Christian Borntraeger <borntraeger@de.ibm.com>
Reviewed-by: Cornelia Huck <cornelia.huck@de.ibm.com>
Tested-by: Matthew Rosato <mjrosato@linux.vnet.ibm.com>
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
arch/s390/include/asm/kvm_host.h
arch/s390/kvm/kvm-s390.c

index a27f5007062ada98c6b616f9dcee6376868c6dda..4181d7baabba99d810cbb53db383d65c12116409 100644 (file)
@@ -110,6 +110,7 @@ struct kvm_s390_sie_block {
 #define ICTL_ISKE      0x00004000
 #define ICTL_SSKE      0x00002000
 #define ICTL_RRBE      0x00001000
+#define ICTL_TPROT     0x00000200
        __u32   ictl;                   /* 0x0048 */
        __u32   eca;                    /* 0x004c */
 #define ICPT_INST      0x04
index 06d1888f91798d1a5861331fc7fc2af1e769d023..43e191b257898c788a27279cb39c3e681be81bbd 100644 (file)
@@ -637,7 +637,9 @@ int kvm_arch_vcpu_setup(struct kvm_vcpu *vcpu)
        if (sclp_has_siif())
                vcpu->arch.sie_block->eca |= 1;
        vcpu->arch.sie_block->fac   = (int) (long) vfacilities;
-       vcpu->arch.sie_block->ictl |= ICTL_ISKE | ICTL_SSKE | ICTL_RRBE;
+       vcpu->arch.sie_block->ictl |= ICTL_ISKE | ICTL_SSKE | ICTL_RRBE |
+                                     ICTL_TPROT;
+
        if (kvm_s390_cmma_enabled(vcpu->kvm)) {
                rc = kvm_s390_vcpu_setup_cmma(vcpu);
                if (rc)