2017-12-04 10:39:38 +01:00
// SPDX-License-Identifier: GPL-2.0
2015-06-06 12:44:25 +02:00
# include <linux/perf_event.h>
# include <linux/perf_regs.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/bug.h>
# include <asm/ptrace.h>
2017-11-08 07:30:15 +01:00
# include <asm/fpu/api.h>
# include <asm/fpu/types.h>
2015-06-06 12:44:25 +02:00
u64 perf_reg_value ( struct pt_regs * regs , int idx )
{
2017-11-08 07:30:15 +01:00
freg_t fp ;
2015-06-06 12:44:25 +02:00
if ( WARN_ON_ONCE ( ( u32 ) idx > = PERF_REG_S390_MAX ) )
return 0 ;
2017-11-08 07:30:15 +01:00
if ( idx > = PERF_REG_S390_R0 & & idx < = PERF_REG_S390_R15 )
return regs - > gprs [ idx ] ;
if ( idx > = PERF_REG_S390_FP0 & & idx < = PERF_REG_S390_FP15 ) {
if ( ! user_mode ( regs ) )
return 0 ;
idx - = PERF_REG_S390_FP0 ;
fp = MACHINE_HAS_VX ? * ( freg_t * ) ( current - > thread . fpu . vxrs + idx )
: current - > thread . fpu . fprs [ idx ] ;
return fp . ui ;
}
2015-06-06 12:44:25 +02:00
if ( idx = = PERF_REG_S390_MASK )
return regs - > psw . mask ;
if ( idx = = PERF_REG_S390_PC )
return regs - > psw . addr ;
return regs - > gprs [ idx ] ;
}
# define REG_RESERVED (~((1UL << PERF_REG_S390_MAX) - 1))
int perf_reg_validate ( u64 mask )
{
if ( ! mask | | mask & REG_RESERVED )
return - EINVAL ;
return 0 ;
}
u64 perf_reg_abi ( struct task_struct * task )
{
if ( test_tsk_thread_flag ( task , TIF_31BIT ) )
return PERF_SAMPLE_REGS_ABI_32 ;
return PERF_SAMPLE_REGS_ABI_64 ;
}
void perf_get_regs_user ( struct perf_regs * regs_user ,
struct pt_regs * regs ,
struct pt_regs * regs_user_copy )
{
/*
* Use the regs from the first interruption and let
* perf_sample_regs_intr ( ) handle interrupts ( regs = = get_irq_regs ( ) ) .
2017-11-08 07:30:15 +01:00
*
* Also save FPU registers for user - space tasks only .
2015-06-06 12:44:25 +02:00
*/
regs_user - > regs = task_pt_regs ( current ) ;
2017-11-08 07:30:15 +01:00
if ( user_mode ( regs_user - > regs ) )
save_fpu_regs ( ) ;
2015-06-06 12:44:25 +02:00
regs_user - > abi = perf_reg_abi ( current ) ;
}