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-05-21 17:29:52 +02:00
# include <asm/pgalloc.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"
2013-10-07 17:11:48 +02:00
# include "gaccess.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 ;
2014-05-13 16:58:30 +02:00
unsigned long prefix = kvm_s390_get_prefix ( vcpu ) ;
2011-10-30 15:17:03 +01:00
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 ;
2015-07-07 15:19:32 +02:00
vcpu - > stat . diagnose_10 + + ;
2011-10-30 15:17:03 +01:00
2014-09-03 21:23:13 +02:00
if ( start & ~ PAGE_MASK | | end & ~ PAGE_MASK | | start > = end
2011-10-30 15:17:03 +01:00
| | start < 2 * PAGE_SIZE )
return kvm_s390_inject_program_int ( vcpu , PGM_SPECIFICATION ) ;
VCPU_EVENT ( vcpu , 5 , " diag release pages %lX %lX " , start , end ) ;
2014-09-03 21:23:13 +02:00
/*
* We checked for start > = end above , so lets check for the
* fast path ( no prefix swap page involved )
*/
if ( end < = prefix | | start > = prefix + 2 * PAGE_SIZE ) {
2014-04-29 09:34:41 +02:00
gmap_discard ( vcpu - > arch . gmap , start , end ) ;
2011-10-30 15:17:03 +01:00
} else {
2014-09-03 21:23:13 +02:00
/*
* This is slow path . gmap_discard will check for start
* so lets split this into before prefix , prefix , after
* prefix and let gmap_discard make some of these calls
* NOPs .
*/
gmap_discard ( vcpu - > arch . gmap , start , prefix ) ;
if ( start < = prefix )
gmap_discard ( vcpu - > arch . gmap , 0 , 4096 ) ;
if ( end > prefix + 4096 )
gmap_discard ( vcpu - > arch . gmap , 4096 , 8192 ) ;
gmap_discard ( vcpu - > arch . gmap , prefix + 2 * PAGE_SIZE , end ) ;
2011-10-30 15:17:03 +01:00
}
return 0 ;
}
2013-10-07 17:11:48 +02:00
static int __diag_page_ref_service ( struct kvm_vcpu * vcpu )
{
struct prs_parm {
u16 code ;
u16 subcode ;
u16 parm_len ;
u16 parm_version ;
u64 token_addr ;
u64 select_mask ;
u64 compare_mask ;
u64 zarch ;
} ;
struct prs_parm parm ;
int rc ;
u16 rx = ( vcpu - > arch . sie_block - > ipa & 0xf0 ) > > 4 ;
u16 ry = ( vcpu - > arch . sie_block - > ipa & 0x0f ) ;
2015-07-07 15:19:32 +02:00
vcpu - > stat . diagnose_258 + + ;
2013-10-07 17:11:48 +02:00
if ( vcpu - > run - > s . regs . gprs [ rx ] & 7 )
return kvm_s390_inject_program_int ( vcpu , PGM_SPECIFICATION ) ;
2015-01-19 13:24:51 +03:00
rc = read_guest ( vcpu , vcpu - > run - > s . regs . gprs [ rx ] , rx , & parm , sizeof ( parm ) ) ;
2014-01-01 16:36:07 +01:00
if ( rc )
return kvm_s390_inject_prog_cond ( vcpu , rc ) ;
2013-10-07 17:11:48 +02:00
if ( parm . parm_version ! = 2 | | parm . parm_len < 5 | | parm . code ! = 0x258 )
return kvm_s390_inject_program_int ( vcpu , PGM_SPECIFICATION ) ;
switch ( parm . subcode ) {
case 0 : /* TOKEN */
if ( vcpu - > arch . pfault_token ! = KVM_S390_PFAULT_TOKEN_INVALID ) {
/*
* If the pagefault handshake is already activated ,
* the token must not be changed . We have to return
* decimal 8 instead , as mandated in SC24 - 6084.
*/
vcpu - > run - > s . regs . gprs [ ry ] = 8 ;
return 0 ;
}
if ( ( parm . compare_mask & parm . select_mask ) ! = parm . compare_mask | |
parm . token_addr & 7 | | parm . zarch ! = 0x8000000000000000ULL )
return kvm_s390_inject_program_int ( vcpu , PGM_SPECIFICATION ) ;
2014-01-01 16:36:07 +01:00
if ( kvm_is_error_gpa ( vcpu - > kvm , parm . token_addr ) )
2013-10-07 17:11:48 +02:00
return kvm_s390_inject_program_int ( vcpu , PGM_ADDRESSING ) ;
vcpu - > arch . pfault_token = parm . token_addr ;
vcpu - > arch . pfault_select = parm . select_mask ;
vcpu - > arch . pfault_compare = parm . compare_mask ;
vcpu - > run - > s . regs . gprs [ ry ] = 0 ;
rc = 0 ;
break ;
case 1 : /*
* CANCEL
* Specification allows to let already pending tokens survive
* the cancel , therefore to reduce code complexity , we assume
* all outstanding tokens are already pending .
*/
if ( parm . token_addr | | parm . select_mask | |
parm . compare_mask | | parm . zarch )
return kvm_s390_inject_program_int ( vcpu , PGM_SPECIFICATION ) ;
vcpu - > run - > s . regs . gprs [ ry ] = 0 ;
/*
* If the pfault handling was not established or is already
* canceled SC24 - 6084 requests to return decimal 4.
*/
if ( vcpu - > arch . pfault_token = = KVM_S390_PFAULT_TOKEN_INVALID )
vcpu - > run - > s . regs . gprs [ ry ] = 4 ;
else
vcpu - > arch . pfault_token = KVM_S390_PFAULT_TOKEN_INVALID ;
rc = 0 ;
break ;
default :
rc = - EOPNOTSUPP ;
break ;
}
return rc ;
}
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 ) ;
2015-07-07 15:19:32 +02:00
vcpu - > stat . diagnose_308 + + ;
2008-03-25 18:47:34 +01:00
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
}
2014-04-10 17:35:00 +02:00
if ( ! kvm_s390_user_cpu_state_ctrl ( vcpu - > kvm ) )
kvm_s390_vcpu_stop ( vcpu ) ;
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
2015-07-07 15:19:32 +02:00
vcpu - > stat . diagnose_500 + + ;
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
*/
2015-03-26 14:39:28 +00:00
ret = kvm_io_bus_write_cookie ( vcpu , KVM_VIRTIO_CCW_NOTIFY_BUS ,
2013-12-09 18:30:01 +01:00
vcpu - > run - > s . regs . gprs [ 2 ] & 0xffffffff ,
2013-07-03 16:30:54 +02:00
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 )
{
2015-01-19 13:24:51 +03:00
int code = kvm_s390_get_base_disp_rs ( vcpu , NULL ) & 0xffff ;
2008-03-25 18:47:34 +01:00
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 ) ;
2013-10-07 17:11:48 +02:00
case 0x258 :
return __diag_page_ref_service ( 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
}
}