KVM: s390: intercepts for diagnose instructions
authorChristian Borntraeger <borntraeger@de.ibm.com>
Tue, 25 Mar 2008 17:47:34 +0000 (18:47 +0100)
committerAvi Kivity <avi@qumranet.com>
Sun, 27 Apr 2008 09:00:46 +0000 (12:00 +0300)
This patch introduces interpretation of some diagnose instruction intercepts.
Diagnose is our classic architected way of doing a hypercall. This patch
features the following diagnose codes:
- vm storage size, that tells the guest about its memory layout
- time slice end, which is used by the guest to indicate that it waits
  for a lock and thus cannot use up its time slice in a useful way
- ipl functions, which a guest can use to reset and reboot itself

In order to implement ipl functions, we also introduce an exit reason that
causes userspace to perform various resets on the virtual machine. All resets
are described in the principles of operation book, except KVM_S390_RESET_IPL
which causes a reboot of the machine.

Acked-by: Martin Schwidefsky <martin.schwidefsky@de.ibm.com>
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Carsten Otte <cotte@de.ibm.com>
Signed-off-by: Avi Kivity <avi@qumranet.com>
arch/s390/kvm/Makefile
arch/s390/kvm/diag.c [new file with mode: 0644]
arch/s390/kvm/intercept.c
arch/s390/kvm/kvm-s390.c
arch/s390/kvm/kvm-s390.h
include/asm-s390/kvm_host.h
include/linux/kvm.h

index f3bf11a88bc7c9131c56c73768c96bcfaa84fa28..e5221ec0b8e3eeb5b008aefb891841e795d98221 100644 (file)
@@ -10,5 +10,5 @@ common-objs = $(addprefix ../../../virt/kvm/, kvm_main.o)
 
 EXTRA_CFLAGS += -Ivirt/kvm -Iarch/s390/kvm
 
-kvm-objs := $(common-objs) kvm-s390.o sie64a.o intercept.o interrupt.o priv.o sigp.o
+kvm-objs := $(common-objs) kvm-s390.o sie64a.o intercept.o interrupt.o priv.o sigp.o diag.o
 obj-$(CONFIG_KVM) += kvm.o
diff --git a/arch/s390/kvm/diag.c b/arch/s390/kvm/diag.c
new file mode 100644 (file)
index 0000000..f639a15
--- /dev/null
@@ -0,0 +1,67 @@
+/*
+ * diag.c - handling diagnose instructions
+ *
+ * Copyright IBM Corp. 2008
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License (version 2 only)
+ * as published by the Free Software Foundation.
+ *
+ *    Author(s): Carsten Otte <cotte@de.ibm.com>
+ *               Christian Borntraeger <borntraeger@de.ibm.com>
+ */
+
+#include <linux/kvm.h>
+#include <linux/kvm_host.h>
+#include "kvm-s390.h"
+
+static int __diag_time_slice_end(struct kvm_vcpu *vcpu)
+{
+       VCPU_EVENT(vcpu, 5, "%s", "diag time slice end");
+       vcpu->stat.diagnose_44++;
+       vcpu_put(vcpu);
+       schedule();
+       vcpu_load(vcpu);
+       return 0;
+}
+
+static int __diag_ipl_functions(struct kvm_vcpu *vcpu)
+{
+       unsigned int reg = vcpu->arch.sie_block->ipa & 0xf;
+       unsigned long subcode = vcpu->arch.guest_gprs[reg] & 0xffff;
+
+       VCPU_EVENT(vcpu, 5, "diag ipl functions, subcode %lx", subcode);
+       switch (subcode) {
+       case 3:
+               vcpu->run->s390_reset_flags = KVM_S390_RESET_CLEAR;
+               break;
+       case 4:
+               vcpu->run->s390_reset_flags = 0;
+               break;
+       default:
+               return -ENOTSUPP;
+       }
+
+       atomic_clear_mask(CPUSTAT_RUNNING, &vcpu->arch.sie_block->cpuflags);
+       vcpu->run->s390_reset_flags |= KVM_S390_RESET_SUBSYSTEM;
+       vcpu->run->s390_reset_flags |= KVM_S390_RESET_IPL;
+       vcpu->run->s390_reset_flags |= KVM_S390_RESET_CPU_INIT;
+       vcpu->run->exit_reason = KVM_EXIT_S390_RESET;
+       VCPU_EVENT(vcpu, 3, "requesting userspace resets %lx",
+         vcpu->run->s390_reset_flags);
+       return -EREMOTE;
+}
+
+int kvm_s390_handle_diag(struct kvm_vcpu *vcpu)
+{
+       int code = (vcpu->arch.sie_block->ipb & 0xfff0000) >> 16;
+
+       switch (code) {
+       case 0x44:
+               return __diag_time_slice_end(vcpu);
+       case 0x308:
+               return __diag_ipl_functions(vcpu);
+       default:
+               return -ENOTSUPP;
+       }
+}
index 9f0d8b2394361aafa4ed66fc032a06e52d727de9..349581a2610344f23c1af6873dc57d6c23761b47 100644 (file)
@@ -95,6 +95,7 @@ static int handle_lctl(struct kvm_vcpu *vcpu)
 }
 
 static intercept_handler_t instruction_handlers[256] = {
+       [0x83] = kvm_s390_handle_diag,
        [0xae] = kvm_s390_handle_sigp,
        [0xb2] = kvm_s390_handle_priv,
        [0xb7] = handle_lctl,
index c632180739ee569c9c894360c6977ddeedf7de81..d3b1de83678b4ad4aa8c3b7be67a6ec4cd90b168 100644 (file)
@@ -63,6 +63,7 @@ struct kvm_stats_debugfs_item debugfs_entries[] = {
        { "instruction_sigp_set_arch", VCPU_STAT(instruction_sigp_arch) },
        { "instruction_sigp_set_prefix", VCPU_STAT(instruction_sigp_prefix) },
        { "instruction_sigp_restart", VCPU_STAT(instruction_sigp_restart) },
+       { "diagnose_44", VCPU_STAT(diagnose_44) },
        { NULL }
 };
 
index e6e5756a0e07d93586b3f0dda236f1b2d2b39680..3893cf12eacf4a4bc595b87ee632f84e6b8c7f89 100644 (file)
@@ -58,4 +58,7 @@ int kvm_s390_handle_sigp(struct kvm_vcpu *vcpu);
 /* implemented in kvm-s390.c */
 int __kvm_s390_vcpu_store_status(struct kvm_vcpu *vcpu,
                                 unsigned long addr);
+/* implemented in diag.c */
+int kvm_s390_handle_diag(struct kvm_vcpu *vcpu);
+
 #endif
index 1c829bdf58893f00aeaca84a42187fd8ad61bcdb..f8204a4f2e0260917655a7877f126e9a53cd2186 100644 (file)
@@ -94,7 +94,9 @@ struct sie_block {
        psw_t   gpsw;                   /* 0x0090 */
        __u64   gg14;                   /* 0x00a0 */
        __u64   gg15;                   /* 0x00a8 */
-       __u8    reservedb0[80];         /* 0x00b0 */
+       __u8    reservedb0[30];         /* 0x00b0 */
+       __u16   iprcc;                  /* 0x00ce */
+       __u8    reservedd0[48];         /* 0x00d0 */
        __u64   gcr[16];                /* 0x0100 */
        __u64   gbea;                   /* 0x0180 */
        __u8    reserved188[120];       /* 0x0188 */
@@ -134,6 +136,7 @@ struct kvm_vcpu_stat {
        u32 instruction_sigp_arch;
        u32 instruction_sigp_prefix;
        u32 instruction_sigp_restart;
+       u32 diagnose_44;
 };
 
 struct io_info {
index 029f0284a2fd5d2e12e1e03bd6fdf8b80daec966..f04bb426618f94c7e3ae02bcc1b47a4edbf579ef 100644 (file)
@@ -75,6 +75,7 @@ struct kvm_irqchip {
 #define KVM_EXIT_SET_TPR          11
 #define KVM_EXIT_TPR_ACCESS       12
 #define KVM_EXIT_S390_SIEIC       13
+#define KVM_EXIT_S390_RESET       14
 
 /* for KVM_RUN, returned by mmap(vcpu_fd, offset=0) */
 struct kvm_run {
@@ -147,6 +148,13 @@ struct kvm_run {
                        __u16 ipa;
                        __u32 ipb;
                } s390_sieic;
+               /* KVM_EXIT_S390_RESET */
+#define KVM_S390_RESET_POR       1
+#define KVM_S390_RESET_CLEAR     2
+#define KVM_S390_RESET_SUBSYSTEM 4
+#define KVM_S390_RESET_CPU_INIT  8
+#define KVM_S390_RESET_IPL       16
+               __u64 s390_reset_flags;
                /* Fix the size of the union. */
                char padding[256];
        };