2011-08-30 20:41:05 -03:00
# include <linux/perf_event.h>
# include <linux/types.h>
# include <linux/init.h>
# include <linux/slab.h>
2011-09-27 10:00:40 -07:00
# include <asm/apicdef.h>
2011-08-30 20:41:05 -03:00
# include "perf_event.h"
2010-02-26 12:05:05 +01:00
2010-03-29 13:09:53 +02:00
static __initconst const u64 amd_hw_cache_event_ids
2010-02-26 12:05:05 +01:00
[ PERF_COUNT_HW_CACHE_MAX ]
[ PERF_COUNT_HW_CACHE_OP_MAX ]
[ PERF_COUNT_HW_CACHE_RESULT_MAX ] =
{
[ C ( L1D ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x0040 , /* Data Cache Accesses */
2011-04-16 02:27:53 +02:00
[ C ( RESULT_MISS ) ] = 0x0141 , /* Data Cache Misses */
2010-02-26 12:05:05 +01:00
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x0142 , /* Data Cache Refills :system */
[ C ( RESULT_MISS ) ] = 0 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x0267 , /* Data Prefetcher :attempts */
[ C ( RESULT_MISS ) ] = 0x0167 , /* Data Prefetcher :cancelled */
} ,
} ,
[ C ( L1I ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x0080 , /* Instruction cache fetches */
[ C ( RESULT_MISS ) ] = 0x0081 , /* Instruction cache misses */
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x014B , /* Prefetch Instructions :Load */
[ C ( RESULT_MISS ) ] = 0 ,
} ,
} ,
[ C ( LL ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x037D , /* Requests to L2 Cache :IC+DC */
[ C ( RESULT_MISS ) ] = 0x037E , /* L2 Cache Misses : IC+DC */
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x017F , /* L2 Fill/Writeback */
[ C ( RESULT_MISS ) ] = 0 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = 0 ,
[ C ( RESULT_MISS ) ] = 0 ,
} ,
} ,
[ C ( DTLB ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x0040 , /* Data Cache Accesses */
2010-10-15 15:15:01 +02:00
[ C ( RESULT_MISS ) ] = 0x0746 , /* L1_DTLB_AND_L2_DLTB_MISS.ALL */
2010-02-26 12:05:05 +01:00
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = 0 ,
[ C ( RESULT_MISS ) ] = 0 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = 0 ,
[ C ( RESULT_MISS ) ] = 0 ,
} ,
} ,
[ C ( ITLB ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x0080 , /* Instruction fecthes */
2010-10-15 15:15:01 +02:00
[ C ( RESULT_MISS ) ] = 0x0385 , /* L1_ITLB_AND_L2_ITLB_MISS.ALL */
2010-02-26 12:05:05 +01:00
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
} ,
[ C ( BPU ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0x00c2 , /* Retired Branch Instr. */
[ C ( RESULT_MISS ) ] = 0x00c3 , /* Retired Mispredicted BI */
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
} ,
2011-04-22 23:37:06 +02:00
[ C ( NODE ) ] = {
[ C ( OP_READ ) ] = {
[ C ( RESULT_ACCESS ) ] = 0xb8e9 , /* CPU Request to Memory, l+r */
[ C ( RESULT_MISS ) ] = 0x98e9 , /* CPU Request to Memory, r */
} ,
[ C ( OP_WRITE ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
[ C ( OP_PREFETCH ) ] = {
[ C ( RESULT_ACCESS ) ] = - 1 ,
[ C ( RESULT_MISS ) ] = - 1 ,
} ,
} ,
2010-02-26 12:05:05 +01:00
} ;
/*
* AMD Performance Monitor K7 and later .
*/
static const u64 amd_perfmon_event_map [ ] =
{
2011-04-29 14:17:19 +02:00
[ PERF_COUNT_HW_CPU_CYCLES ] = 0x0076 ,
[ PERF_COUNT_HW_INSTRUCTIONS ] = 0x00c0 ,
[ PERF_COUNT_HW_CACHE_REFERENCES ] = 0x0080 ,
[ PERF_COUNT_HW_CACHE_MISSES ] = 0x0081 ,
[ PERF_COUNT_HW_BRANCH_INSTRUCTIONS ] = 0x00c2 ,
[ PERF_COUNT_HW_BRANCH_MISSES ] = 0x00c3 ,
[ PERF_COUNT_HW_STALLED_CYCLES_FRONTEND ] = 0x00d0 , /* "Decoder empty" event */
[ PERF_COUNT_HW_STALLED_CYCLES_BACKEND ] = 0x00d1 , /* "Dispatch stalls" event */
2010-02-26 12:05:05 +01:00
} ;
static u64 amd_pmu_event_map ( int hw_event )
{
return amd_perfmon_event_map [ hw_event ] ;
}
2010-03-30 17:00:06 +02:00
static int amd_pmu_hw_config ( struct perf_event * event )
2010-02-26 12:05:05 +01:00
{
2010-03-30 17:00:06 +02:00
int ret = x86_pmu_hw_config ( event ) ;
if ( ret )
return ret ;
if ( event - > attr . type ! = PERF_TYPE_RAW )
return 0 ;
event - > hw . config | = event - > attr . config & AMD64_RAW_EVENT_MASK ;
return 0 ;
2010-02-26 12:05:05 +01:00
}
/*
* AMD64 events are detected based on their event codes .
*/
2011-02-02 17:36:12 +01:00
static inline unsigned int amd_get_event_code ( struct hw_perf_event * hwc )
{
return ( ( hwc - > config > > 24 ) & 0x0f00 ) | ( hwc - > config & 0x00ff ) ;
}
2010-02-26 12:05:05 +01:00
static inline int amd_is_nb_event ( struct hw_perf_event * hwc )
{
return ( hwc - > config & 0xe0 ) = = 0xe0 ;
}
2010-03-23 19:31:15 +01:00
static inline int amd_has_nb ( struct cpu_hw_events * cpuc )
{
struct amd_nb * nb = cpuc - > amd_nb ;
return nb & & nb - > nb_id ! = - 1 ;
}
2010-02-26 12:05:05 +01:00
static void amd_put_event_constraints ( struct cpu_hw_events * cpuc ,
struct perf_event * event )
{
struct hw_perf_event * hwc = & event - > hw ;
struct amd_nb * nb = cpuc - > amd_nb ;
int i ;
/*
* only care about NB events
*/
2010-03-23 19:31:15 +01:00
if ( ! ( amd_has_nb ( cpuc ) & & amd_is_nb_event ( hwc ) ) )
2010-02-26 12:05:05 +01:00
return ;
/*
* need to scan whole list because event may not have
* been assigned during scheduling
*
* no race condition possible because event can only
* be removed on one CPU at a time AND PMU is disabled
* when we come here
*/
2010-03-29 18:36:50 +02:00
for ( i = 0 ; i < x86_pmu . num_counters ; i + + ) {
2010-02-26 12:05:05 +01:00
if ( nb - > owners [ i ] = = event ) {
cmpxchg ( nb - > owners + i , event , NULL ) ;
break ;
}
}
}
/*
* AMD64 NorthBridge events need special treatment because
* counter access needs to be synchronized across all cores
* of a package . Refer to BKDG section 3.12
*
* NB events are events measuring L3 cache , Hypertransport
* traffic . They are identified by an event code > = 0xe00 .
* They measure events on the NorthBride which is shared
* by all cores on a package . NB events are counted on a
* shared set of counters . When a NB event is programmed
* in a counter , the data actually comes from a shared
* counter . Thus , access to those counters needs to be
* synchronized .
*
* We implement the synchronization such that no two cores
* can be measuring NB events using the same counters . Thus ,
* we maintain a per - NB allocation table . The available slot
* is propagated using the event_constraint structure .
*
* We provide only one choice for each NB event based on
* the fact that only NB events have restrictions . Consequently ,
* if a counter is available , there is a guarantee the NB event
* will be assigned to it . If no slot is available , an empty
* constraint is returned and scheduling will eventually fail
* for this event .
*
* Note that all cores attached the same NB compete for the same
* counters to host NB events , this is why we use atomic ops . Some
* multi - chip CPUs may have more than one NB .
*
* Given that resources are allocated ( cmpxchg ) , they must be
* eventually freed for others to use . This is accomplished by
* calling amd_put_event_constraints ( ) .
*
* Non NB events are not impacted by this restriction .
*/
static struct event_constraint *
amd_get_event_constraints ( struct cpu_hw_events * cpuc , struct perf_event * event )
{
struct hw_perf_event * hwc = & event - > hw ;
struct amd_nb * nb = cpuc - > amd_nb ;
struct perf_event * old = NULL ;
2010-03-29 18:36:50 +02:00
int max = x86_pmu . num_counters ;
2010-02-26 12:05:05 +01:00
int i , j , k = - 1 ;
/*
* if not NB event or no NB , then no constraints
*/
2010-03-23 19:31:15 +01:00
if ( ! ( amd_has_nb ( cpuc ) & & amd_is_nb_event ( hwc ) ) )
2010-02-26 12:05:05 +01:00
return & unconstrained ;
/*
* detect if already present , if so reuse
*
* cannot merge with actual allocation
* because of possible holes
*
* event can already be present yet not assigned ( in hwc - > idx )
* because of successive calls to x86_schedule_events ( ) from
* hw_perf_group_sched_in ( ) without hw_perf_enable ( )
*/
for ( i = 0 ; i < max ; i + + ) {
/*
* keep track of first free slot
*/
if ( k = = - 1 & & ! nb - > owners [ i ] )
k = i ;
/* already present, reuse */
if ( nb - > owners [ i ] = = event )
goto done ;
}
/*
* not present , so grab a new slot
* starting either at :
*/
if ( hwc - > idx ! = - 1 ) {
/* previous assignment */
i = hwc - > idx ;
} else if ( k ! = - 1 ) {
/* start from free slot found */
i = k ;
} else {
/*
* event not found , no slot found in
* first pass , try again from the
* beginning
*/
i = 0 ;
}
j = i ;
do {
old = cmpxchg ( nb - > owners + i , NULL , event ) ;
if ( ! old )
break ;
if ( + + i = = max )
i = 0 ;
} while ( i ! = j ) ;
done :
if ( ! old )
return & nb - > event_constraints [ i ] ;
return & emptyconstraint ;
}
2010-11-25 08:56:17 +01:00
static struct amd_nb * amd_alloc_nb ( int cpu )
2010-02-26 12:05:05 +01:00
{
struct amd_nb * nb ;
int i ;
2010-11-01 18:52:05 +01:00
nb = kmalloc_node ( sizeof ( struct amd_nb ) , GFP_KERNEL | __GFP_ZERO ,
cpu_to_node ( cpu ) ) ;
2010-02-26 12:05:05 +01:00
if ( ! nb )
return NULL ;
2010-11-25 08:56:17 +01:00
nb - > nb_id = - 1 ;
2010-02-26 12:05:05 +01:00
/*
* initialize all possible NB constraints
*/
2010-03-29 18:36:50 +02:00
for ( i = 0 ; i < x86_pmu . num_counters ; i + + ) {
2010-03-02 21:16:55 +01:00
__set_bit ( i , nb - > event_constraints [ i ] . idxmsk ) ;
2010-02-26 12:05:05 +01:00
nb - > event_constraints [ i ] . weight = 1 ;
}
return nb ;
}
2010-03-23 19:31:15 +01:00
static int amd_pmu_cpu_prepare ( int cpu )
{
struct cpu_hw_events * cpuc = & per_cpu ( cpu_hw_events , cpu ) ;
WARN_ON_ONCE ( cpuc - > amd_nb ) ;
if ( boot_cpu_data . x86_max_cores < 2 )
return NOTIFY_OK ;
2010-11-25 08:56:17 +01:00
cpuc - > amd_nb = amd_alloc_nb ( cpu ) ;
2010-03-23 19:31:15 +01:00
if ( ! cpuc - > amd_nb )
return NOTIFY_BAD ;
return NOTIFY_OK ;
}
static void amd_pmu_cpu_starting ( int cpu )
2010-02-26 12:05:05 +01:00
{
2010-03-23 19:31:15 +01:00
struct cpu_hw_events * cpuc = & per_cpu ( cpu_hw_events , cpu ) ;
struct amd_nb * nb ;
2010-02-26 12:05:05 +01:00
int i , nb_id ;
if ( boot_cpu_data . x86_max_cores < 2 )
return ;
nb_id = amd_get_nb_id ( cpu ) ;
2010-03-23 19:31:15 +01:00
WARN_ON_ONCE ( nb_id = = BAD_APICID ) ;
2010-02-26 12:05:05 +01:00
for_each_online_cpu ( i ) {
2010-03-23 19:31:15 +01:00
nb = per_cpu ( cpu_hw_events , i ) . amd_nb ;
if ( WARN_ON_ONCE ( ! nb ) )
2010-02-26 12:05:05 +01:00
continue ;
2010-03-23 19:31:15 +01:00
if ( nb - > nb_id = = nb_id ) {
2011-07-22 13:41:54 +02:00
cpuc - > kfree_on_online = cpuc - > amd_nb ;
2010-03-23 19:31:15 +01:00
cpuc - > amd_nb = nb ;
break ;
}
2010-02-26 12:05:05 +01:00
}
2010-03-23 19:31:15 +01:00
cpuc - > amd_nb - > nb_id = nb_id ;
cpuc - > amd_nb - > refcnt + + ;
2010-02-26 12:05:05 +01:00
}
2010-03-23 19:31:15 +01:00
static void amd_pmu_cpu_dead ( int cpu )
2010-02-26 12:05:05 +01:00
{
struct cpu_hw_events * cpuhw ;
if ( boot_cpu_data . x86_max_cores < 2 )
return ;
cpuhw = & per_cpu ( cpu_hw_events , cpu ) ;
2010-03-21 21:51:51 +01:00
if ( cpuhw - > amd_nb ) {
2010-03-23 19:31:15 +01:00
struct amd_nb * nb = cpuhw - > amd_nb ;
if ( nb - > nb_id = = - 1 | | - - nb - > refcnt = = 0 )
kfree ( nb ) ;
2010-02-26 12:05:05 +01:00
2010-03-21 21:51:51 +01:00
cpuhw - > amd_nb = NULL ;
}
2010-02-26 12:05:05 +01:00
}
2010-03-29 13:09:53 +02:00
static __initconst const struct x86_pmu amd_pmu = {
2010-03-05 13:01:18 +01:00
. name = " AMD " ,
. handle_irq = x86_pmu_handle_irq ,
. disable_all = x86_pmu_disable_all ,
. enable_all = x86_pmu_enable_all ,
. enable = x86_pmu_enable_event ,
. disable = x86_pmu_disable_event ,
2010-03-30 17:00:06 +02:00
. hw_config = amd_pmu_hw_config ,
2010-03-11 19:54:39 +03:00
. schedule_events = x86_schedule_events ,
2010-03-05 13:01:18 +01:00
. eventsel = MSR_K7_EVNTSEL0 ,
. perfctr = MSR_K7_PERFCTR0 ,
. event_map = amd_pmu_event_map ,
. max_events = ARRAY_SIZE ( amd_perfmon_event_map ) ,
2010-03-29 18:36:50 +02:00
. num_counters = 4 ,
. cntval_bits = 48 ,
. cntval_mask = ( 1ULL < < 48 ) - 1 ,
2010-03-05 13:01:18 +01:00
. apic = 1 ,
/* use highest bit to detect overflow */
. max_period = ( 1ULL < < 47 ) - 1 ,
. get_event_constraints = amd_get_event_constraints ,
. put_event_constraints = amd_put_event_constraints ,
2010-03-23 19:31:15 +01:00
. cpu_prepare = amd_pmu_cpu_prepare ,
. cpu_starting = amd_pmu_cpu_starting ,
. cpu_dead = amd_pmu_cpu_dead ,
2010-03-05 13:01:18 +01:00
} ;
2011-02-02 17:36:12 +01:00
/* AMD Family 15h */
# define AMD_EVENT_TYPE_MASK 0x000000F0ULL
# define AMD_EVENT_FP 0x00000000ULL ... 0x00000010ULL
# define AMD_EVENT_LS 0x00000020ULL ... 0x00000030ULL
# define AMD_EVENT_DC 0x00000040ULL ... 0x00000050ULL
# define AMD_EVENT_CU 0x00000060ULL ... 0x00000070ULL
# define AMD_EVENT_IC_DE 0x00000080ULL ... 0x00000090ULL
# define AMD_EVENT_EX_LS 0x000000C0ULL
# define AMD_EVENT_DE 0x000000D0ULL
# define AMD_EVENT_NB 0x000000E0ULL ... 0x000000F0ULL
/*
* AMD family 15 h event code / PMC mappings :
*
* type = event_code & 0x0F0 :
*
* 0x000 FP PERF_CTL [ 5 : 3 ]
* 0x010 FP PERF_CTL [ 5 : 3 ]
* 0x020 LS PERF_CTL [ 5 : 0 ]
* 0x030 LS PERF_CTL [ 5 : 0 ]
* 0x040 DC PERF_CTL [ 5 : 0 ]
* 0x050 DC PERF_CTL [ 5 : 0 ]
* 0x060 CU PERF_CTL [ 2 : 0 ]
* 0x070 CU PERF_CTL [ 2 : 0 ]
* 0x080 IC / DE PERF_CTL [ 2 : 0 ]
* 0x090 IC / DE PERF_CTL [ 2 : 0 ]
* 0x0A0 - - -
* 0x0B0 - - -
* 0x0C0 EX / LS PERF_CTL [ 5 : 0 ]
* 0x0D0 DE PERF_CTL [ 2 : 0 ]
* 0x0E0 NB NB_PERF_CTL [ 3 : 0 ]
* 0x0F0 NB NB_PERF_CTL [ 3 : 0 ]
*
* Exceptions :
*
2011-04-16 02:27:54 +02:00
* 0x000 FP PERF_CTL [ 3 ] , PERF_CTL [ 5 : 3 ] ( * )
2011-02-02 17:36:12 +01:00
* 0x003 FP PERF_CTL [ 3 ]
2011-04-16 02:27:54 +02:00
* 0x004 FP PERF_CTL [ 3 ] , PERF_CTL [ 5 : 3 ] ( * )
2011-02-02 17:36:12 +01:00
* 0x00B FP PERF_CTL [ 3 ]
* 0x00D FP PERF_CTL [ 3 ]
* 0x023 DE PERF_CTL [ 2 : 0 ]
* 0x02D LS PERF_CTL [ 3 ]
* 0x02E LS PERF_CTL [ 3 , 0 ]
* 0x043 CU PERF_CTL [ 2 : 0 ]
* 0x045 CU PERF_CTL [ 2 : 0 ]
* 0x046 CU PERF_CTL [ 2 : 0 ]
* 0x054 CU PERF_CTL [ 2 : 0 ]
* 0x055 CU PERF_CTL [ 2 : 0 ]
* 0x08F IC PERF_CTL [ 0 ]
* 0x187 DE PERF_CTL [ 0 ]
* 0x188 DE PERF_CTL [ 0 ]
* 0x0DB EX PERF_CTL [ 5 : 0 ]
* 0x0DC LS PERF_CTL [ 5 : 0 ]
* 0x0DD LS PERF_CTL [ 5 : 0 ]
* 0x0DE LS PERF_CTL [ 5 : 0 ]
* 0x0DF LS PERF_CTL [ 5 : 0 ]
* 0x1D6 EX PERF_CTL [ 5 : 0 ]
* 0x1D8 EX PERF_CTL [ 5 : 0 ]
2011-04-16 02:27:54 +02:00
*
* ( * ) depending on the umask all FPU counters may be used
2011-02-02 17:36:12 +01:00
*/
static struct event_constraint amd_f15_PMC0 = EVENT_CONSTRAINT ( 0 , 0x01 , 0 ) ;
static struct event_constraint amd_f15_PMC20 = EVENT_CONSTRAINT ( 0 , 0x07 , 0 ) ;
static struct event_constraint amd_f15_PMC3 = EVENT_CONSTRAINT ( 0 , 0x08 , 0 ) ;
static struct event_constraint amd_f15_PMC30 = EVENT_CONSTRAINT ( 0 , 0x09 , 0 ) ;
static struct event_constraint amd_f15_PMC50 = EVENT_CONSTRAINT ( 0 , 0x3F , 0 ) ;
static struct event_constraint amd_f15_PMC53 = EVENT_CONSTRAINT ( 0 , 0x38 , 0 ) ;
static struct event_constraint *
amd_get_event_constraints_f15h ( struct cpu_hw_events * cpuc , struct perf_event * event )
{
2011-04-16 02:27:54 +02:00
struct hw_perf_event * hwc = & event - > hw ;
unsigned int event_code = amd_get_event_code ( hwc ) ;
2011-02-02 17:36:12 +01:00
switch ( event_code & AMD_EVENT_TYPE_MASK ) {
case AMD_EVENT_FP :
switch ( event_code ) {
2011-04-16 02:27:54 +02:00
case 0x000 :
if ( ! ( hwc - > config & 0x0000F000ULL ) )
break ;
if ( ! ( hwc - > config & 0x00000F00ULL ) )
break ;
return & amd_f15_PMC3 ;
case 0x004 :
if ( hweight_long ( hwc - > config & ARCH_PERFMON_EVENTSEL_UMASK ) < = 1 )
break ;
return & amd_f15_PMC3 ;
2011-02-02 17:36:12 +01:00
case 0x003 :
case 0x00B :
case 0x00D :
return & amd_f15_PMC3 ;
}
2011-04-16 02:27:54 +02:00
return & amd_f15_PMC53 ;
2011-02-02 17:36:12 +01:00
case AMD_EVENT_LS :
case AMD_EVENT_DC :
case AMD_EVENT_EX_LS :
switch ( event_code ) {
case 0x023 :
case 0x043 :
case 0x045 :
case 0x046 :
case 0x054 :
case 0x055 :
return & amd_f15_PMC20 ;
case 0x02D :
return & amd_f15_PMC3 ;
case 0x02E :
return & amd_f15_PMC30 ;
default :
return & amd_f15_PMC50 ;
}
case AMD_EVENT_CU :
case AMD_EVENT_IC_DE :
case AMD_EVENT_DE :
switch ( event_code ) {
case 0x08F :
case 0x187 :
case 0x188 :
return & amd_f15_PMC0 ;
case 0x0DB . . . 0x0DF :
case 0x1D6 :
case 0x1D8 :
return & amd_f15_PMC50 ;
default :
return & amd_f15_PMC20 ;
}
case AMD_EVENT_NB :
/* not yet implemented */
return & emptyconstraint ;
default :
return & emptyconstraint ;
}
}
static __initconst const struct x86_pmu amd_pmu_f15h = {
. name = " AMD Family 15h " ,
. handle_irq = x86_pmu_handle_irq ,
. disable_all = x86_pmu_disable_all ,
. enable_all = x86_pmu_enable_all ,
. enable = x86_pmu_enable_event ,
. disable = x86_pmu_disable_event ,
. hw_config = amd_pmu_hw_config ,
. schedule_events = x86_schedule_events ,
. eventsel = MSR_F15H_PERF_CTL ,
. perfctr = MSR_F15H_PERF_CTR ,
. event_map = amd_pmu_event_map ,
. max_events = ARRAY_SIZE ( amd_perfmon_event_map ) ,
. num_counters = 6 ,
. cntval_bits = 48 ,
. cntval_mask = ( 1ULL < < 48 ) - 1 ,
. apic = 1 ,
/* use highest bit to detect overflow */
. max_period = ( 1ULL < < 47 ) - 1 ,
. get_event_constraints = amd_get_event_constraints_f15h ,
/* nortbridge counters not yet implemented: */
#if 0
. put_event_constraints = amd_put_event_constraints ,
. cpu_prepare = amd_pmu_cpu_prepare ,
. cpu_starting = amd_pmu_cpu_starting ,
. cpu_dead = amd_pmu_cpu_dead ,
# endif
} ;
2011-08-30 20:41:05 -03:00
__init int amd_pmu_init ( void )
2010-02-26 12:05:05 +01:00
{
/* Performance-monitoring supported from K7 and later: */
if ( boot_cpu_data . x86 < 6 )
return - ENODEV ;
2011-02-02 17:36:12 +01:00
/*
* If core performance counter extensions exists , it must be
* family 15 h , otherwise fail . See x86_pmu_addr_offset ( ) .
*/
switch ( boot_cpu_data . x86 ) {
case 0x15 :
if ( ! cpu_has_perfctr_core )
return - ENODEV ;
x86_pmu = amd_pmu_f15h ;
break ;
default :
if ( cpu_has_perfctr_core )
return - ENODEV ;
x86_pmu = amd_pmu ;
break ;
}
2010-02-26 12:05:05 +01:00
/* Events are common for all AMDs */
memcpy ( hw_cache_event_ids , amd_hw_cache_event_ids ,
sizeof ( hw_cache_event_ids ) ) ;
return 0 ;
}