2008-03-25 18:47:34 +01:00
/*
* diag . c - handling diagnose instructions
*
2011-10-30 15:17:03 +01: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>
# include "kvm-s390.h"
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 ) ;
return - EREMOTE ;
}
int kvm_s390_handle_diag ( struct kvm_vcpu * vcpu )
{
int code = ( vcpu - > arch . sie_block - > ipb & 0xfff0000 ) > > 16 ;
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 ) ;
default :
2010-02-26 22:37:41 +01:00
return - EOPNOTSUPP ;
2008-03-25 18:47:34 +01:00
}
}