| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2012-07-20 11:15:04 +02:00
										 |  |  |  * handling diagnose instructions | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-07-20 11:15:04 +02:00
										 |  |  |  * Copyright IBM Corp. 2008, 2011 | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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>
 | 
					
						
							| 
									
										
										
										
											2013-02-28 12:33:21 +01:00
										 |  |  | #include <asm/virtio-ccw.h>
 | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | #include "kvm-s390.h"
 | 
					
						
							| 
									
										
										
										
											2012-07-23 17:20:29 +02:00
										 |  |  | #include "trace.h"
 | 
					
						
							| 
									
										
										
										
											2012-07-23 17:20:30 +02:00
										 |  |  | #include "trace-s390.h"
 | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-30 15:17:03 +01:00
										 |  |  | static int diag_release_pages(struct kvm_vcpu *vcpu) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long start, end; | 
					
						
							|  |  |  | 	unsigned long prefix  = vcpu->arch.sie_block->prefix; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-11 11:20:32 +01:00
										 |  |  | 	start = vcpu->run->s.regs.gprs[(vcpu->arch.sie_block->ipa & 0xf0) >> 4]; | 
					
						
							|  |  |  | 	end = vcpu->run->s.regs.gprs[vcpu->arch.sie_block->ipa & 0xf] + 4096; | 
					
						
							| 
									
										
										
										
											2011-10-30 15:17:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (start & ~PAGE_MASK || end & ~PAGE_MASK || start > end | 
					
						
							|  |  |  | 	    || start < 2 * PAGE_SIZE) | 
					
						
							|  |  |  | 		return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	VCPU_EVENT(vcpu, 5, "diag release pages %lX %lX", start, end); | 
					
						
							|  |  |  | 	vcpu->stat.diagnose_10++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* we checked for start > end above */ | 
					
						
							|  |  |  | 	if (end < prefix || start >= prefix + 2 * PAGE_SIZE) { | 
					
						
							|  |  |  | 		gmap_discard(start, end, vcpu->arch.gmap); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (start < prefix) | 
					
						
							|  |  |  | 			gmap_discard(start, prefix, vcpu->arch.gmap); | 
					
						
							|  |  |  | 		if (end >= prefix) | 
					
						
							|  |  |  | 			gmap_discard(prefix + 2 * PAGE_SIZE, | 
					
						
							|  |  |  | 				     end, vcpu->arch.gmap); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | static int __diag_time_slice_end(struct kvm_vcpu *vcpu) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	VCPU_EVENT(vcpu, 5, "%s", "diag time slice end"); | 
					
						
							|  |  |  | 	vcpu->stat.diagnose_44++; | 
					
						
							| 
									
										
										
										
											2012-04-25 15:30:39 +02:00
										 |  |  | 	kvm_vcpu_on_spin(vcpu); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-25 15:30:38 +02:00
										 |  |  | static int __diag_time_slice_end_directed(struct kvm_vcpu *vcpu) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct kvm *kvm = vcpu->kvm; | 
					
						
							|  |  |  | 	struct kvm_vcpu *tcpu; | 
					
						
							|  |  |  | 	int tid; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tid = vcpu->run->s.regs.gprs[(vcpu->arch.sie_block->ipa & 0xf0) >> 4]; | 
					
						
							|  |  |  | 	vcpu->stat.diagnose_9c++; | 
					
						
							|  |  |  | 	VCPU_EVENT(vcpu, 5, "diag time slice end directed to %d", tid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (tid == vcpu->vcpu_id) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	kvm_for_each_vcpu(i, tcpu, kvm) | 
					
						
							|  |  |  | 		if (tcpu->vcpu_id == tid) { | 
					
						
							|  |  |  | 			kvm_vcpu_yield_to(tcpu); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | static int __diag_ipl_functions(struct kvm_vcpu *vcpu) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int reg = vcpu->arch.sie_block->ipa & 0xf; | 
					
						
							| 
									
										
										
										
											2012-01-11 11:20:32 +01:00
										 |  |  | 	unsigned long subcode = vcpu->run->s.regs.gprs[reg] & 0xffff; | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	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: | 
					
						
							| 
									
										
										
										
											2010-02-26 22:37:41 +01:00
										 |  |  | 		return -EOPNOTSUPP; | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-17 11:00:41 +01:00
										 |  |  | 	atomic_set_mask(CPUSTAT_STOPPED, &vcpu->arch.sie_block->cpuflags); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	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; | 
					
						
							| 
									
										
										
										
											2009-01-09 12:14:56 +01:00
										 |  |  | 	VCPU_EVENT(vcpu, 3, "requesting userspace resets %llx", | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	  vcpu->run->s390_reset_flags); | 
					
						
							| 
									
										
										
										
											2012-07-23 17:20:30 +02:00
										 |  |  | 	trace_kvm_s390_request_resets(vcpu->run->s390_reset_flags); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	return -EREMOTE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-28 12:33:21 +01:00
										 |  |  | static int __diag_virtio_hypercall(struct kvm_vcpu *vcpu) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-12 10:33:45 +02:00
										 |  |  | 	int ret; | 
					
						
							| 
									
										
										
										
											2013-02-28 12:33:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* No virtio-ccw notification? Get out quickly. */ | 
					
						
							|  |  |  | 	if (!vcpu->kvm->arch.css_support || | 
					
						
							|  |  |  | 	    (vcpu->run->s.regs.gprs[1] != KVM_S390_VIRTIO_CCW_NOTIFY)) | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * The layout is as follows: | 
					
						
							|  |  |  | 	 * - gpr 2 contains the subchannel id (passed as addr) | 
					
						
							|  |  |  | 	 * - gpr 3 contains the virtqueue index (passed as datamatch) | 
					
						
							| 
									
										
										
										
											2013-07-03 16:30:54 +02:00
										 |  |  | 	 * - gpr 4 contains the index on the bus (optionally) | 
					
						
							| 
									
										
										
										
											2013-02-28 12:33:21 +01:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2013-07-03 16:30:54 +02:00
										 |  |  | 	ret = kvm_io_bus_write_cookie(vcpu->kvm, KVM_VIRTIO_CCW_NOTIFY_BUS, | 
					
						
							|  |  |  | 				      vcpu->run->s.regs.gprs[2], | 
					
						
							|  |  |  | 				      8, &vcpu->run->s.regs.gprs[3], | 
					
						
							|  |  |  | 				      vcpu->run->s.regs.gprs[4]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Return cookie in gpr 2, but don't overwrite the register if the | 
					
						
							|  |  |  | 	 * diagnose will be handled by userspace. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (ret != -EOPNOTSUPP) | 
					
						
							|  |  |  | 		vcpu->run->s.regs.gprs[2] = ret; | 
					
						
							|  |  |  | 	/* kvm_io_bus_write_cookie returns -EOPNOTSUPP if it found no match. */ | 
					
						
							| 
									
										
										
										
											2013-02-28 12:33:21 +01:00
										 |  |  | 	return ret < 0 ? ret : 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | int kvm_s390_handle_diag(struct kvm_vcpu *vcpu) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int code = (vcpu->arch.sie_block->ipb & 0xfff0000) >> 16; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-20 17:22:02 +02:00
										 |  |  | 	if (vcpu->arch.sie_block->gpsw.mask & PSW_MASK_PSTATE) | 
					
						
							|  |  |  | 		return kvm_s390_inject_program_int(vcpu, PGM_PRIVILEGED_OP); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-23 17:20:29 +02:00
										 |  |  | 	trace_kvm_s390_handle_diag(vcpu, code); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	switch (code) { | 
					
						
							| 
									
										
										
										
											2011-10-30 15:17:03 +01:00
										 |  |  | 	case 0x10: | 
					
						
							|  |  |  | 		return diag_release_pages(vcpu); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	case 0x44: | 
					
						
							|  |  |  | 		return __diag_time_slice_end(vcpu); | 
					
						
							| 
									
										
										
										
											2012-04-25 15:30:38 +02:00
										 |  |  | 	case 0x9c: | 
					
						
							|  |  |  | 		return __diag_time_slice_end_directed(vcpu); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	case 0x308: | 
					
						
							|  |  |  | 		return __diag_ipl_functions(vcpu); | 
					
						
							| 
									
										
										
										
											2013-02-28 12:33:21 +01:00
										 |  |  | 	case 0x500: | 
					
						
							|  |  |  | 		return __diag_virtio_hypercall(vcpu); | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2010-02-26 22:37:41 +01:00
										 |  |  | 		return -EOPNOTSUPP; | 
					
						
							| 
									
										
										
										
											2008-03-25 18:47:34 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } |