2 * in-kernel handling for sie intercepts
4 * Copyright IBM Corp. 2008, 2009
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License (version 2 only)
8 * as published by the Free Software Foundation.
14 #include <linux/kvm_host.h>
15 #include <linux/errno.h>
16 #include <linux/pagemap.h>
18 #include <asm/kvm_host.h>
23 #include "trace-s390.h"
26 static const intercept_handler_t instruction_handlers[256] = {
27 [0x01] = kvm_s390_handle_01,
28 [0x82] = kvm_s390_handle_lpsw,
29 [0x83] = kvm_s390_handle_diag,
30 [0xae] = kvm_s390_handle_sigp,
31 [0xb2] = kvm_s390_handle_b2,
32 [0xb7] = kvm_s390_handle_lctl,
33 [0xb9] = kvm_s390_handle_b9,
34 [0xe5] = kvm_s390_handle_e5,
35 [0xeb] = kvm_s390_handle_eb,
38 static int handle_noop(struct kvm_vcpu *vcpu)
40 switch (vcpu->arch.sie_block->icptcode) {
42 vcpu->stat.exit_null++;
45 vcpu->stat.exit_external_request++;
48 vcpu->stat.exit_external_interrupt++;
56 static int handle_stop(struct kvm_vcpu *vcpu)
60 vcpu->stat.exit_stop_request++;
61 spin_lock_bh(&vcpu->arch.local_int.lock);
63 trace_kvm_s390_stop_request(vcpu->arch.local_int.action_bits);
65 if (vcpu->arch.local_int.action_bits & ACTION_STOP_ON_STOP) {
66 atomic_set_mask(CPUSTAT_STOPPED,
67 &vcpu->arch.sie_block->cpuflags);
68 vcpu->arch.local_int.action_bits &= ~ACTION_STOP_ON_STOP;
69 VCPU_EVENT(vcpu, 3, "%s", "cpu stopped");
73 if (vcpu->arch.local_int.action_bits & ACTION_STORE_ON_STOP) {
74 vcpu->arch.local_int.action_bits &= ~ACTION_STORE_ON_STOP;
75 /* store status must be called unlocked. Since local_int.lock
76 * only protects local_int.* and not guest memory we can give
78 spin_unlock_bh(&vcpu->arch.local_int.lock);
79 rc = kvm_s390_vcpu_store_status(vcpu,
80 KVM_S390_STORE_STATUS_NOADDR);
84 spin_unlock_bh(&vcpu->arch.local_int.lock);
88 static int handle_validity(struct kvm_vcpu *vcpu)
90 int viwhy = vcpu->arch.sie_block->ipb >> 16;
92 vcpu->stat.exit_validity++;
93 trace_kvm_s390_intercept_validity(vcpu, viwhy);
94 WARN_ONCE(true, "kvm: unhandled validity intercept 0x%x\n", viwhy);
98 static int handle_instruction(struct kvm_vcpu *vcpu)
100 intercept_handler_t handler;
102 vcpu->stat.exit_instruction++;
103 trace_kvm_s390_intercept_instruction(vcpu,
104 vcpu->arch.sie_block->ipa,
105 vcpu->arch.sie_block->ipb);
106 handler = instruction_handlers[vcpu->arch.sie_block->ipa >> 8];
108 return handler(vcpu);
112 static int handle_prog(struct kvm_vcpu *vcpu)
114 vcpu->stat.exit_program_interruption++;
116 /* Restore ITDB to Program-Interruption TDB in guest memory */
117 if (IS_TE_ENABLED(vcpu) &&
118 !(current->thread.per_flags & PER_FLAG_NO_TE) &&
119 IS_ITDB_VALID(vcpu)) {
120 copy_to_guest(vcpu, TDB_ADDR, vcpu->arch.sie_block->itdba,
121 sizeof(struct kvm_s390_itdb));
122 memset((void *) vcpu->arch.sie_block->itdba, 0,
123 sizeof(struct kvm_s390_itdb));
126 trace_kvm_s390_intercept_prog(vcpu, vcpu->arch.sie_block->iprcc);
127 return kvm_s390_inject_program_int(vcpu, vcpu->arch.sie_block->iprcc);
130 static int handle_instruction_and_prog(struct kvm_vcpu *vcpu)
134 vcpu->stat.exit_instr_and_program++;
135 rc = handle_instruction(vcpu);
136 rc2 = handle_prog(vcpu);
138 if (rc == -EOPNOTSUPP)
139 vcpu->arch.sie_block->icptcode = 0x04;
145 static const intercept_handler_t intercept_funcs[] = {
146 [0x00 >> 2] = handle_noop,
147 [0x04 >> 2] = handle_instruction,
148 [0x08 >> 2] = handle_prog,
149 [0x0C >> 2] = handle_instruction_and_prog,
150 [0x10 >> 2] = handle_noop,
151 [0x14 >> 2] = handle_noop,
152 [0x18 >> 2] = handle_noop,
153 [0x1C >> 2] = kvm_s390_handle_wait,
154 [0x20 >> 2] = handle_validity,
155 [0x28 >> 2] = handle_stop,
158 int kvm_handle_sie_intercept(struct kvm_vcpu *vcpu)
160 intercept_handler_t func;
161 u8 code = vcpu->arch.sie_block->icptcode;
163 if (code & 3 || (code >> 2) >= ARRAY_SIZE(intercept_funcs))
165 func = intercept_funcs[code >> 2];