2012-12-10 16:37:02 +00:00
/*
* Copyright ( C ) 2012 , 2013 - ARM Ltd
* Author : Marc Zyngier < marc . zyngier @ arm . com >
*
* Derived from arch / arm / kvm / guest . c :
* Copyright ( C ) 2012 - Virtual Open Systems and Columbia University
* Author : Christoffer Dall < c . dall @ virtualopensystems . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include <linux/errno.h>
# include <linux/err.h>
# include <linux/kvm_host.h>
# include <linux/module.h>
# include <linux/vmalloc.h>
# include <linux/fs.h>
# include <asm/cputype.h>
2016-12-24 11:46:01 -08:00
# include <linux/uaccess.h>
2012-12-10 16:37:02 +00:00
# include <asm/kvm.h>
# include <asm/kvm_emulate.h>
# include <asm/kvm_coproc.h>
2015-07-07 17:30:03 +01:00
# include "trace.h"
2015-11-26 10:09:43 +00:00
# define VM_STAT(x) { #x, offsetof(struct kvm, stat.x), KVM_STAT_VM }
# define VCPU_STAT(x) { #x, offsetof(struct kvm_vcpu, stat.x), KVM_STAT_VCPU }
2012-12-10 16:37:02 +00:00
struct kvm_stats_debugfs_item debugfs_entries [ ] = {
2015-11-26 10:09:43 +00:00
VCPU_STAT ( hvc_exit_stat ) ,
VCPU_STAT ( wfe_exit_stat ) ,
VCPU_STAT ( wfi_exit_stat ) ,
VCPU_STAT ( mmio_exit_user ) ,
VCPU_STAT ( mmio_exit_kernel ) ,
VCPU_STAT ( exits ) ,
2012-12-10 16:37:02 +00:00
{ NULL }
} ;
int kvm_arch_vcpu_setup ( struct kvm_vcpu * vcpu )
{
return 0 ;
}
static u64 core_reg_offset_from_id ( u64 id )
{
return id & ~ ( KVM_REG_ARCH_MASK | KVM_REG_SIZE_MASK | KVM_REG_ARM_CORE ) ;
}
static int get_core_reg ( struct kvm_vcpu * vcpu , const struct kvm_one_reg * reg )
{
/*
* Because the kvm_regs structure is a mix of 32 , 64 and
* 128 bit fields , we index it as if it was a 32 bit
* array . Hence below , nr_regs is the number of entries , and
* off the index in the " array " .
*/
__u32 __user * uaddr = ( __u32 __user * ) ( unsigned long ) reg - > addr ;
struct kvm_regs * regs = vcpu_gp_regs ( vcpu ) ;
int nr_regs = sizeof ( * regs ) / sizeof ( __u32 ) ;
u32 off ;
/* Our ID is an index into the kvm_regs struct. */
off = core_reg_offset_from_id ( reg - > id ) ;
if ( off > = nr_regs | |
( off + ( KVM_REG_SIZE ( reg - > id ) / sizeof ( __u32 ) ) ) > = nr_regs )
return - ENOENT ;
if ( copy_to_user ( uaddr , ( ( u32 * ) regs ) + off , KVM_REG_SIZE ( reg - > id ) ) )
return - EFAULT ;
return 0 ;
}
static int set_core_reg ( struct kvm_vcpu * vcpu , const struct kvm_one_reg * reg )
{
__u32 __user * uaddr = ( __u32 __user * ) ( unsigned long ) reg - > addr ;
struct kvm_regs * regs = vcpu_gp_regs ( vcpu ) ;
int nr_regs = sizeof ( * regs ) / sizeof ( __u32 ) ;
__uint128_t tmp ;
void * valp = & tmp ;
u64 off ;
int err = 0 ;
/* Our ID is an index into the kvm_regs struct. */
off = core_reg_offset_from_id ( reg - > id ) ;
if ( off > = nr_regs | |
( off + ( KVM_REG_SIZE ( reg - > id ) / sizeof ( __u32 ) ) ) > = nr_regs )
return - ENOENT ;
if ( KVM_REG_SIZE ( reg - > id ) > sizeof ( tmp ) )
return - EINVAL ;
if ( copy_from_user ( valp , uaddr , KVM_REG_SIZE ( reg - > id ) ) ) {
err = - EFAULT ;
goto out ;
}
if ( off = = KVM_REG_ARM_CORE_REG ( regs . pstate ) ) {
u32 mode = ( * ( u32 * ) valp ) & COMPAT_PSR_MODE_MASK ;
switch ( mode ) {
2013-02-07 10:46:46 +00:00
case COMPAT_PSR_MODE_USR :
case COMPAT_PSR_MODE_FIQ :
case COMPAT_PSR_MODE_IRQ :
case COMPAT_PSR_MODE_SVC :
case COMPAT_PSR_MODE_ABT :
case COMPAT_PSR_MODE_UND :
2012-12-10 16:37:02 +00:00
case PSR_MODE_EL0t :
case PSR_MODE_EL1t :
case PSR_MODE_EL1h :
break ;
default :
err = - EINVAL ;
goto out ;
}
}
memcpy ( ( u32 * ) regs + off , valp , KVM_REG_SIZE ( reg - > id ) ) ;
out :
return err ;
}
int kvm_arch_vcpu_ioctl_get_regs ( struct kvm_vcpu * vcpu , struct kvm_regs * regs )
{
return - EINVAL ;
}
int kvm_arch_vcpu_ioctl_set_regs ( struct kvm_vcpu * vcpu , struct kvm_regs * regs )
{
return - EINVAL ;
}
static unsigned long num_core_regs ( void )
{
return sizeof ( struct kvm_regs ) / sizeof ( __u32 ) ;
}
2014-07-04 15:54:14 +01:00
/**
* ARM64 versions of the TIMER registers , always available on arm64
*/
# define NUM_TIMER_REGS 3
static bool is_timer_reg ( u64 index )
{
switch ( index ) {
case KVM_REG_ARM_TIMER_CTL :
case KVM_REG_ARM_TIMER_CNT :
case KVM_REG_ARM_TIMER_CVAL :
return true ;
}
return false ;
}
static int copy_timer_indices ( struct kvm_vcpu * vcpu , u64 __user * uindices )
{
if ( put_user ( KVM_REG_ARM_TIMER_CTL , uindices ) )
return - EFAULT ;
uindices + + ;
if ( put_user ( KVM_REG_ARM_TIMER_CNT , uindices ) )
return - EFAULT ;
uindices + + ;
if ( put_user ( KVM_REG_ARM_TIMER_CVAL , uindices ) )
return - EFAULT ;
return 0 ;
}
static int set_timer_reg ( struct kvm_vcpu * vcpu , const struct kvm_one_reg * reg )
{
void __user * uaddr = ( void __user * ) ( long ) reg - > addr ;
u64 val ;
int ret ;
ret = copy_from_user ( & val , uaddr , KVM_REG_SIZE ( reg - > id ) ) ;
if ( ret ! = 0 )
2014-08-26 15:13:23 +01:00
return - EFAULT ;
2014-07-04 15:54:14 +01:00
return kvm_arm_timer_set_reg ( vcpu , reg - > id , val ) ;
}
static int get_timer_reg ( struct kvm_vcpu * vcpu , const struct kvm_one_reg * reg )
{
void __user * uaddr = ( void __user * ) ( long ) reg - > addr ;
u64 val ;
val = kvm_arm_timer_get_reg ( vcpu , reg - > id ) ;
2016-02-28 17:32:07 +02:00
return copy_to_user ( uaddr , & val , KVM_REG_SIZE ( reg - > id ) ) ? - EFAULT : 0 ;
2014-07-04 15:54:14 +01:00
}
2012-12-10 16:37:02 +00:00
/**
* kvm_arm_num_regs - how many registers do we present via KVM_GET_ONE_REG
*
* This is for all registers .
*/
unsigned long kvm_arm_num_regs ( struct kvm_vcpu * vcpu )
{
2014-07-04 15:54:14 +01:00
return num_core_regs ( ) + kvm_arm_num_sys_reg_descs ( vcpu )
+ NUM_TIMER_REGS ;
2012-12-10 16:37:02 +00:00
}
/**
* kvm_arm_copy_reg_indices - get indices of all registers .
*
2016-05-21 13:53:14 +02:00
* We do core registers right here , then we append system regs .
2012-12-10 16:37:02 +00:00
*/
int kvm_arm_copy_reg_indices ( struct kvm_vcpu * vcpu , u64 __user * uindices )
{
unsigned int i ;
const u64 core_reg = KVM_REG_ARM64 | KVM_REG_SIZE_U64 | KVM_REG_ARM_CORE ;
2014-07-04 15:54:14 +01:00
int ret ;
2012-12-10 16:37:02 +00:00
for ( i = 0 ; i < sizeof ( struct kvm_regs ) / sizeof ( __u32 ) ; i + + ) {
if ( put_user ( core_reg | i , uindices ) )
return - EFAULT ;
uindices + + ;
}
2014-07-04 15:54:14 +01:00
ret = copy_timer_indices ( vcpu , uindices ) ;
if ( ret )
return ret ;
uindices + = NUM_TIMER_REGS ;
2012-12-10 16:37:02 +00:00
return kvm_arm_copy_sys_reg_indices ( vcpu , uindices ) ;
}
int kvm_arm_get_reg ( struct kvm_vcpu * vcpu , const struct kvm_one_reg * reg )
{
/* We currently use nothing arch-specific in upper 32 bits */
if ( ( reg - > id & ~ KVM_REG_SIZE_MASK ) > > 32 ! = KVM_REG_ARM64 > > 32 )
return - EINVAL ;
/* Register group 16 means we want a core register. */
if ( ( reg - > id & KVM_REG_ARM_COPROC_MASK ) = = KVM_REG_ARM_CORE )
return get_core_reg ( vcpu , reg ) ;
2014-07-04 15:54:14 +01:00
if ( is_timer_reg ( reg - > id ) )
return get_timer_reg ( vcpu , reg ) ;
2012-12-10 16:37:02 +00:00
return kvm_arm_sys_reg_get_reg ( vcpu , reg ) ;
}
int kvm_arm_set_reg ( struct kvm_vcpu * vcpu , const struct kvm_one_reg * reg )
{
/* We currently use nothing arch-specific in upper 32 bits */
if ( ( reg - > id & ~ KVM_REG_SIZE_MASK ) > > 32 ! = KVM_REG_ARM64 > > 32 )
return - EINVAL ;
/* Register group 16 means we set a core register. */
if ( ( reg - > id & KVM_REG_ARM_COPROC_MASK ) = = KVM_REG_ARM_CORE )
return set_core_reg ( vcpu , reg ) ;
2014-07-04 15:54:14 +01:00
if ( is_timer_reg ( reg - > id ) )
return set_timer_reg ( vcpu , reg ) ;
2012-12-10 16:37:02 +00:00
return kvm_arm_sys_reg_set_reg ( vcpu , reg ) ;
}
int kvm_arch_vcpu_ioctl_get_sregs ( struct kvm_vcpu * vcpu ,
struct kvm_sregs * sregs )
{
return - EINVAL ;
}
int kvm_arch_vcpu_ioctl_set_sregs ( struct kvm_vcpu * vcpu ,
struct kvm_sregs * sregs )
{
return - EINVAL ;
}
int __attribute_const__ kvm_target_cpu ( void )
{
unsigned long implementor = read_cpuid_implementor ( ) ;
unsigned long part_number = read_cpuid_part_number ( ) ;
2013-11-14 15:20:08 +00:00
switch ( implementor ) {
case ARM_CPU_IMP_ARM :
switch ( part_number ) {
case ARM_CPU_PART_AEM_V8 :
return KVM_ARM_TARGET_AEM_V8 ;
case ARM_CPU_PART_FOUNDATION :
return KVM_ARM_TARGET_FOUNDATION_V8 ;
2014-05-20 18:06:03 +01:00
case ARM_CPU_PART_CORTEX_A53 :
return KVM_ARM_TARGET_CORTEX_A53 ;
2013-11-14 15:20:08 +00:00
case ARM_CPU_PART_CORTEX_A57 :
return KVM_ARM_TARGET_CORTEX_A57 ;
} ;
break ;
case ARM_CPU_IMP_APM :
switch ( part_number ) {
case APM_CPU_PART_POTENZA :
return KVM_ARM_TARGET_XGENE_POTENZA ;
} ;
break ;
} ;
2012-12-10 16:37:02 +00:00
2015-06-17 10:00:46 +01:00
/* Return a default generic target */
return KVM_ARM_TARGET_GENERIC_V8 ;
2012-12-10 16:37:02 +00:00
}
2013-09-30 14:20:06 +05:30
int kvm_vcpu_preferred_target ( struct kvm_vcpu_init * init )
{
int target = kvm_target_cpu ( ) ;
if ( target < 0 )
return - ENODEV ;
memset ( init , 0 , sizeof ( * init ) ) ;
/*
* For now , we don ' t return any features .
* In future , we might use features to return target
* specific features available for the preferred
* target type .
*/
init - > target = ( __u32 ) target ;
return 0 ;
}
2012-12-10 16:37:02 +00:00
int kvm_arch_vcpu_ioctl_get_fpu ( struct kvm_vcpu * vcpu , struct kvm_fpu * fpu )
{
return - EINVAL ;
}
int kvm_arch_vcpu_ioctl_set_fpu ( struct kvm_vcpu * vcpu , struct kvm_fpu * fpu )
{
return - EINVAL ;
}
int kvm_arch_vcpu_ioctl_translate ( struct kvm_vcpu * vcpu ,
struct kvm_translation * tr )
{
return - EINVAL ;
}
2015-07-07 17:29:55 +01:00
2015-07-07 17:29:58 +01:00
# define KVM_GUESTDBG_VALID_MASK (KVM_GUESTDBG_ENABLE | \
KVM_GUESTDBG_USE_SW_BP | \
2015-07-07 17:30:02 +01:00
KVM_GUESTDBG_USE_HW | \
2015-07-07 17:29:58 +01:00
KVM_GUESTDBG_SINGLESTEP )
2015-07-07 17:29:55 +01:00
/**
* kvm_arch_vcpu_ioctl_set_guest_debug - set up guest debugging
* @ kvm : pointer to the KVM struct
* @ kvm_guest_debug : the ioctl data buffer
*
* This sets up and enables the VM for guest debugging . Userspace
* passes in a control flag to enable different debug types and
* potentially other architecture specific information in the rest of
* the structure .
*/
int kvm_arch_vcpu_ioctl_set_guest_debug ( struct kvm_vcpu * vcpu ,
struct kvm_guest_debug * dbg )
{
2015-07-07 17:30:03 +01:00
trace_kvm_set_guest_debug ( vcpu , dbg - > control ) ;
2015-07-07 17:29:55 +01:00
if ( dbg - > control & ~ KVM_GUESTDBG_VALID_MASK )
return - EINVAL ;
if ( dbg - > control & KVM_GUESTDBG_ENABLE ) {
vcpu - > guest_debug = dbg - > control ;
2015-07-07 17:30:02 +01:00
/* Hardware assisted Break and Watch points */
if ( vcpu - > guest_debug & KVM_GUESTDBG_USE_HW ) {
vcpu - > arch . external_debug_state = dbg - > arch ;
}
2015-07-07 17:29:55 +01:00
} else {
/* If not enabled clear all flags */
vcpu - > guest_debug = 0 ;
}
return 0 ;
}
2016-01-11 21:35:32 +08:00
int kvm_arm_vcpu_arch_set_attr ( struct kvm_vcpu * vcpu ,
struct kvm_device_attr * attr )
{
int ret ;
switch ( attr - > group ) {
case KVM_ARM_VCPU_PMU_V3_CTRL :
ret = kvm_arm_pmu_v3_set_attr ( vcpu , attr ) ;
break ;
2017-05-02 20:19:15 +02:00
case KVM_ARM_VCPU_TIMER_CTRL :
ret = kvm_arm_timer_set_attr ( vcpu , attr ) ;
break ;
2016-01-11 21:35:32 +08:00
default :
ret = - ENXIO ;
break ;
}
return ret ;
}
int kvm_arm_vcpu_arch_get_attr ( struct kvm_vcpu * vcpu ,
struct kvm_device_attr * attr )
{
int ret ;
switch ( attr - > group ) {
case KVM_ARM_VCPU_PMU_V3_CTRL :
ret = kvm_arm_pmu_v3_get_attr ( vcpu , attr ) ;
break ;
2017-05-02 20:19:15 +02:00
case KVM_ARM_VCPU_TIMER_CTRL :
ret = kvm_arm_timer_get_attr ( vcpu , attr ) ;
break ;
2016-01-11 21:35:32 +08:00
default :
ret = - ENXIO ;
break ;
}
return ret ;
}
int kvm_arm_vcpu_arch_has_attr ( struct kvm_vcpu * vcpu ,
struct kvm_device_attr * attr )
{
int ret ;
switch ( attr - > group ) {
case KVM_ARM_VCPU_PMU_V3_CTRL :
ret = kvm_arm_pmu_v3_has_attr ( vcpu , attr ) ;
break ;
2017-05-02 20:19:15 +02:00
case KVM_ARM_VCPU_TIMER_CTRL :
ret = kvm_arm_timer_has_attr ( vcpu , attr ) ;
break ;
2016-01-11 21:35:32 +08:00
default :
ret = - ENXIO ;
break ;
}
return ret ;
}