2008-02-19 23:43:25 +01:00
/*
2005-04-16 15:20:36 -07:00
* @ file op_model_ppro . h
2008-08-18 14:50:31 +02:00
* Family 6 perfmon and architectural perfmon MSR operations
2005-04-16 15:20:36 -07:00
*
* @ remark Copyright 2002 OProfile authors
2008-08-18 14:50:31 +02:00
* @ remark Copyright 2008 Intel Corporation
2005-04-16 15:20:36 -07:00
* @ remark Read the file COPYING
*
* @ author John Levon
* @ author Philippe Elie
* @ author Graydon Hoare
2008-08-18 14:50:31 +02:00
* @ author Andi Kleen
2009-05-25 15:10:32 +02:00
* @ author Robert Richter < robert . richter @ amd . com >
2005-04-16 15:20:36 -07:00
*/
# include <linux/oprofile.h>
2008-08-18 14:50:31 +02:00
# include <linux/slab.h>
2005-04-16 15:20:36 -07:00
# include <asm/ptrace.h>
# include <asm/msr.h>
# include <asm/apic.h>
2006-06-26 13:57:01 +02:00
# include <asm/nmi.h>
2008-02-19 23:43:25 +01:00
2005-04-16 15:20:36 -07:00
# include "op_x86_model.h"
# include "op_counter.h"
2008-08-18 14:50:31 +02:00
static int num_counters = 2 ;
static int counter_width = 32 ;
2005-04-16 15:20:36 -07:00
2009-05-25 15:10:32 +02:00
# define MSR_PPRO_EVENTSEL_RESERVED ((0xFFFFFFFFULL<<32)|(1ULL<<21))
2005-04-16 15:20:36 -07:00
2011-08-01 11:08:59 -04:00
static u64 reset_value [ OP_MAX_COUNTER ] ;
2008-02-19 23:43:25 +01:00
2010-03-23 20:01:54 +01:00
static void ppro_shutdown ( struct op_msrs const * const msrs )
{
int i ;
for ( i = 0 ; i < num_counters ; + + i ) {
if ( ! msrs - > counters [ i ] . addr )
continue ;
release_perfctr_nmi ( MSR_P6_PERFCTR0 + i ) ;
release_evntsel_nmi ( MSR_P6_EVNTSEL0 + i ) ;
}
}
2010-02-26 17:20:55 +01:00
static int ppro_fill_in_addresses ( struct op_msrs * const msrs )
2005-04-16 15:20:36 -07:00
{
2006-09-26 10:52:26 +02:00
int i ;
2008-08-18 14:50:31 +02:00
for ( i = 0 ; i < num_counters ; i + + ) {
2010-03-23 19:33:21 +01:00
if ( ! reserve_perfctr_nmi ( MSR_P6_PERFCTR0 + i ) )
2010-02-26 17:20:55 +01:00
goto fail ;
2010-03-23 19:33:21 +01:00
if ( ! reserve_evntsel_nmi ( MSR_P6_EVNTSEL0 + i ) ) {
release_perfctr_nmi ( MSR_P6_PERFCTR0 + i ) ;
2010-02-26 17:20:55 +01:00
goto fail ;
2010-03-23 19:33:21 +01:00
}
/* both registers must be reserved */
msrs - > counters [ i ] . addr = MSR_P6_PERFCTR0 + i ;
msrs - > controls [ i ] . addr = MSR_P6_EVNTSEL0 + i ;
2010-02-26 17:20:55 +01:00
continue ;
fail :
if ( ! counter_config [ i ] . enabled )
continue ;
op_x86_warn_reserved ( i ) ;
ppro_shutdown ( msrs ) ;
return - EBUSY ;
2006-09-26 10:52:26 +02:00
}
2010-02-26 17:20:55 +01:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2009-05-25 19:31:44 +02:00
static void ppro_setup_ctrs ( struct op_x86_model_spec const * model ,
struct op_msrs const * const msrs )
2005-04-16 15:20:36 -07:00
{
2009-05-25 15:10:32 +02:00
u64 val ;
2005-04-16 15:20:36 -07:00
int i ;
2016-03-29 17:41:54 +02:00
if ( boot_cpu_has ( X86_FEATURE_ARCH_PERFMON ) ) {
2008-08-18 14:50:31 +02:00
union cpuid10_eax eax ;
eax . full = cpuid_eax ( 0xa ) ;
2009-02-19 17:34:03 +01:00
/*
* For Core2 ( family 6 , model 15 ) , don ' t reset the
* counter width :
*/
if ( ! ( eax . split . version_id = = 0 & &
2010-12-18 16:30:05 +01:00
__this_cpu_read ( cpu_info . x86 ) = = 6 & &
__this_cpu_read ( cpu_info . x86_model ) = = 15 ) ) {
2009-02-19 17:34:03 +01:00
if ( counter_width < eax . split . bit_width )
counter_width = eax . split . bit_width ;
}
2008-08-18 14:50:31 +02:00
}
2005-04-16 15:20:36 -07:00
/* clear all counters */
2009-07-07 19:25:39 +02:00
for ( i = 0 ; i < num_counters ; + + i ) {
2010-02-26 17:20:55 +01:00
if ( ! msrs - > controls [ i ] . addr )
2006-09-26 10:52:26 +02:00
continue ;
2009-05-25 15:10:32 +02:00
rdmsrl ( msrs - > controls [ i ] . addr , val ) ;
2010-03-01 14:21:23 +01:00
if ( val & ARCH_PERFMON_EVENTSEL_ENABLE )
2010-02-23 18:14:58 +01:00
op_x86_warn_in_use ( i ) ;
2009-05-25 15:10:32 +02:00
val & = model - > reserved ;
wrmsrl ( msrs - > controls [ i ] . addr , val ) ;
2010-03-23 19:33:21 +01:00
/*
* avoid a false detection of ctr overflows in NMI *
* handler
*/
2008-08-18 14:50:31 +02:00
wrmsrl ( msrs - > counters [ i ] . addr , - 1LL ) ;
2005-04-16 15:20:36 -07:00
}
/* enable active counters */
2008-08-18 14:50:31 +02:00
for ( i = 0 ; i < num_counters ; + + i ) {
2009-06-04 02:36:44 +02:00
if ( counter_config [ i ] . enabled & & msrs - > counters [ i ] . addr ) {
2005-04-16 15:20:36 -07:00
reset_value [ i ] = counter_config [ i ] . count ;
2008-08-18 14:50:31 +02:00
wrmsrl ( msrs - > counters [ i ] . addr , - reset_value [ i ] ) ;
2009-05-25 15:10:32 +02:00
rdmsrl ( msrs - > controls [ i ] . addr , val ) ;
val & = model - > reserved ;
val | = op_x86_get_ctrl ( model , & counter_config [ i ] ) ;
wrmsrl ( msrs - > controls [ i ] . addr , val ) ;
2006-09-26 10:52:26 +02:00
} else {
reset_value [ i ] = 0 ;
2005-04-16 15:20:36 -07:00
}
}
}
2008-02-19 23:43:25 +01:00
2005-04-16 15:20:36 -07:00
static int ppro_check_ctrs ( struct pt_regs * const regs ,
struct op_msrs const * const msrs )
{
2008-11-07 14:02:49 +01:00
u64 val ;
2005-04-16 15:20:36 -07:00
int i ;
2008-02-19 23:43:25 +01:00
2009-07-07 19:25:39 +02:00
for ( i = 0 ; i < num_counters ; + + i ) {
2006-09-26 10:52:26 +02:00
if ( ! reset_value [ i ] )
continue ;
2008-11-07 14:02:49 +01:00
rdmsrl ( msrs - > counters [ i ] . addr , val ) ;
2009-05-25 17:59:06 +02:00
if ( val & ( 1ULL < < ( counter_width - 1 ) ) )
continue ;
oprofile_add_sample ( regs , i ) ;
wrmsrl ( msrs - > counters [ i ] . addr , - reset_value [ i ] ) ;
2005-04-16 15:20:36 -07:00
}
/* Only P6 based Pentium M need to re-unmask the apic vector but it
* doesn ' t hurt other P6 variant */
apic_write ( APIC_LVTPC , apic_read ( APIC_LVTPC ) & ~ APIC_LVT_MASKED ) ;
/* We can't work out if we really handled an interrupt. We
* might have caught a * second * counter just after overflowing
* the interrupt for this counter then arrives
* and we don ' t find a counter that ' s overflowed , so we
* would return 0 and get dazed + confused . Instead we always
* assume we found an overflow . This sucks .
*/
return 1 ;
}
2008-02-19 23:43:25 +01:00
2005-04-16 15:20:36 -07:00
static void ppro_start ( struct op_msrs const * const msrs )
{
2009-05-25 18:11:52 +02:00
u64 val ;
2006-09-29 02:00:01 -07:00
int i ;
2006-09-26 10:52:26 +02:00
2008-08-18 14:50:31 +02:00
for ( i = 0 ; i < num_counters ; + + i ) {
2006-09-29 02:00:01 -07:00
if ( reset_value [ i ] ) {
2009-05-25 18:11:52 +02:00
rdmsrl ( msrs - > controls [ i ] . addr , val ) ;
2010-03-01 14:21:23 +01:00
val | = ARCH_PERFMON_EVENTSEL_ENABLE ;
2009-05-25 18:11:52 +02:00
wrmsrl ( msrs - > controls [ i ] . addr , val ) ;
2006-09-29 02:00:01 -07:00
}
2006-09-26 10:52:26 +02:00
}
2005-04-16 15:20:36 -07:00
}
static void ppro_stop ( struct op_msrs const * const msrs )
{
2009-05-25 18:11:52 +02:00
u64 val ;
2006-09-29 02:00:01 -07:00
int i ;
2006-09-26 10:52:26 +02:00
2008-08-18 14:50:31 +02:00
for ( i = 0 ; i < num_counters ; + + i ) {
2006-09-29 02:00:01 -07:00
if ( ! reset_value [ i ] )
continue ;
2009-05-25 18:11:52 +02:00
rdmsrl ( msrs - > controls [ i ] . addr , val ) ;
2010-03-01 14:21:23 +01:00
val & = ~ ARCH_PERFMON_EVENTSEL_ENABLE ;
2009-05-25 18:11:52 +02:00
wrmsrl ( msrs - > controls [ i ] . addr , val ) ;
2006-09-26 10:52:26 +02:00
}
}
2009-07-09 15:12:35 +02:00
struct op_x86_model_spec op_ppro_spec = {
2009-05-14 17:10:52 +02:00
. num_counters = 2 ,
. num_controls = 2 ,
2009-05-25 15:10:32 +02:00
. reserved = MSR_PPRO_EVENTSEL_RESERVED ,
2008-10-15 22:19:41 +02:00
. fill_in_addresses = & ppro_fill_in_addresses ,
. setup_ctrs = & ppro_setup_ctrs ,
. check_ctrs = & ppro_check_ctrs ,
. start = & ppro_start ,
. stop = & ppro_stop ,
. shutdown = & ppro_shutdown
2008-08-18 14:50:31 +02:00
} ;
/*
* Architectural performance monitoring .
*
* Newer Intel CPUs ( Core1 + ) have support for architectural
* events described in CPUID 0xA . See the IA32 SDM Vol3b .18 for details .
* The advantage of this is that it can be done without knowing about
* the specific CPU .
*/
2008-10-12 15:12:34 -04:00
static void arch_perfmon_setup_counters ( void )
2008-08-18 14:50:31 +02:00
{
union cpuid10_eax eax ;
eax . full = cpuid_eax ( 0xa ) ;
/* Workaround for BIOS bugs in 6/15. Taken from perfmon2 */
2017-11-07 18:53:07 +01:00
if ( eax . split . version_id = = 0 & & boot_cpu_data . x86 = = 6 & &
boot_cpu_data . x86_model = = 15 ) {
2008-08-18 14:50:31 +02:00
eax . split . version_id = 2 ;
2010-03-29 18:36:50 +02:00
eax . split . num_counters = 2 ;
2008-08-18 14:50:31 +02:00
eax . split . bit_width = 40 ;
}
2011-08-16 23:39:53 +02:00
num_counters = min ( ( int ) eax . split . num_counters , OP_MAX_COUNTER ) ;
2008-08-18 14:50:31 +02:00
op_arch_perfmon_spec . num_counters = num_counters ;
op_arch_perfmon_spec . num_controls = num_counters ;
}
2008-10-12 15:12:34 -04:00
static int arch_perfmon_init ( struct oprofile_operations * ignore )
{
arch_perfmon_setup_counters ( ) ;
return 0 ;
}
2008-08-18 14:50:31 +02:00
struct op_x86_model_spec op_arch_perfmon_spec = {
2009-05-25 15:10:32 +02:00
. reserved = MSR_PPRO_EVENTSEL_RESERVED ,
2008-10-12 15:12:34 -04:00
. init = & arch_perfmon_init ,
2008-08-18 14:50:31 +02:00
/* num_counters/num_controls filled in at runtime */
2008-09-05 17:12:36 +02:00
. fill_in_addresses = & ppro_fill_in_addresses ,
2008-08-18 14:50:31 +02:00
/* user space does the cpuid check for available events */
2008-09-05 17:12:36 +02:00
. setup_ctrs = & ppro_setup_ctrs ,
. check_ctrs = & ppro_check_ctrs ,
. start = & ppro_start ,
. stop = & ppro_stop ,
. shutdown = & ppro_shutdown
2005-04-16 15:20:36 -07:00
} ;