2010-02-02 20:25:44 +01:00
# undef DEBUG
/*
* ARM performance counter support .
*
* Copyright ( C ) 2009 picoChip Designs , Ltd . , Jamie Iles
2010-11-13 19:04:32 +00:00
* Copyright ( C ) 2010 ARM Ltd . , Will Deacon < will . deacon @ arm . com >
2010-01-26 18:51:05 +01:00
*
2010-02-02 20:25:44 +01:00
* This code is based on the sparc64 perf event code , which is in turn based
* on the x86 code . Callchain code is based on the ARM OProfile backtrace
* code .
*/
# define pr_fmt(fmt) "hw perfevents: " fmt
2011-08-23 11:59:49 +01:00
# include <linux/bitmap.h>
2010-02-02 20:25:44 +01:00
# include <linux/interrupt.h>
# include <linux/kernel.h>
2011-07-22 10:58:34 -04:00
# include <linux/export.h>
2010-02-02 20:25:44 +01:00
# include <linux/perf_event.h>
2010-04-29 17:13:24 +01:00
# include <linux/platform_device.h>
2010-02-02 20:25:44 +01:00
# include <linux/spinlock.h>
# include <linux/uaccess.h>
# include <asm/cputype.h>
# include <asm/irq.h>
# include <asm/irq_regs.h>
# include <asm/pmu.h>
# include <asm/stacktrace.h>
/*
2011-07-19 22:43:28 +01:00
* ARMv6 supports a maximum of 3 events , starting from index 0. If we add
2010-02-02 20:25:44 +01:00
* another platform that supports more , we need to increase this to be the
* largest of all platforms .
2010-01-26 18:51:05 +01:00
*
* ARMv7 supports up to 32 events :
* cycle counter CCNT + 31 events counters CNT0 . .30 .
* Cortex - A8 has 1 + 4 counters , Cortex - A9 has 1 + 6 counters .
2010-02-02 20:25:44 +01:00
*/
2011-07-19 22:43:28 +01:00
# define ARMPMU_MAX_HWEVENTS 32
2010-02-02 20:25:44 +01:00
2011-06-24 11:30:59 +01:00
static DEFINE_PER_CPU ( struct perf_event * [ ARMPMU_MAX_HWEVENTS ] , hw_events ) ;
static DEFINE_PER_CPU ( unsigned long [ BITS_TO_LONGS ( ARMPMU_MAX_HWEVENTS ) ] , used_mask ) ;
2011-05-17 11:20:11 +01:00
static DEFINE_PER_CPU ( struct pmu_hw_events , cpu_hw_events ) ;
2010-04-30 11:32:44 +01:00
2011-04-28 16:27:54 +01:00
# define to_arm_pmu(p) (container_of(p, struct arm_pmu, pmu))
2010-02-02 20:25:44 +01:00
/* Set at runtime when we know what CPU type we are. */
2011-05-17 11:20:11 +01:00
static struct arm_pmu * cpu_pmu ;
2010-02-02 20:25:44 +01:00
2010-04-30 11:32:44 +01:00
enum arm_perf_pmu_ids
armpmu_get_pmu_id ( void )
{
int id = - ENODEV ;
2011-05-17 11:20:11 +01:00
if ( cpu_pmu ! = NULL )
id = cpu_pmu - > id ;
2010-04-30 11:32:44 +01:00
return id ;
}
EXPORT_SYMBOL_GPL ( armpmu_get_pmu_id ) ;
2010-04-30 11:34:26 +01:00
int
armpmu_get_max_events ( void )
{
int max_events = 0 ;
2011-05-17 11:20:11 +01:00
if ( cpu_pmu ! = NULL )
max_events = cpu_pmu - > num_events ;
2010-04-30 11:34:26 +01:00
return max_events ;
}
EXPORT_SYMBOL_GPL ( armpmu_get_max_events ) ;
2010-09-27 20:22:24 +01:00
int perf_num_counters ( void )
{
return armpmu_get_max_events ( ) ;
}
EXPORT_SYMBOL_GPL ( perf_num_counters ) ;
2010-02-02 20:25:44 +01:00
# define HW_OP_UNSUPPORTED 0xFFFF
# define C(_x) \
PERF_COUNT_HW_CACHE_ # # _x
# define CACHE_OP_UNSUPPORTED 0xFFFF
static int
2011-04-28 15:47:10 +01:00
armpmu_map_cache_event ( const unsigned ( * cache_map )
[ PERF_COUNT_HW_CACHE_MAX ]
[ PERF_COUNT_HW_CACHE_OP_MAX ]
[ PERF_COUNT_HW_CACHE_RESULT_MAX ] ,
u64 config )
2010-02-02 20:25:44 +01:00
{
unsigned int cache_type , cache_op , cache_result , ret ;
cache_type = ( config > > 0 ) & 0xff ;
if ( cache_type > = PERF_COUNT_HW_CACHE_MAX )
return - EINVAL ;
cache_op = ( config > > 8 ) & 0xff ;
if ( cache_op > = PERF_COUNT_HW_CACHE_OP_MAX )
return - EINVAL ;
cache_result = ( config > > 16 ) & 0xff ;
if ( cache_result > = PERF_COUNT_HW_CACHE_RESULT_MAX )
return - EINVAL ;
2011-04-28 15:47:10 +01:00
ret = ( int ) ( * cache_map ) [ cache_type ] [ cache_op ] [ cache_result ] ;
2010-02-02 20:25:44 +01:00
if ( ret = = CACHE_OP_UNSUPPORTED )
return - ENOENT ;
return ret ;
}
2010-11-13 17:13:56 +00:00
static int
2011-04-28 15:47:10 +01:00
armpmu_map_event ( const unsigned ( * event_map ) [ PERF_COUNT_HW_MAX ] , u64 config )
2010-11-13 17:13:56 +00:00
{
2011-04-28 15:47:10 +01:00
int mapping = ( * event_map ) [ config ] ;
return mapping = = HW_OP_UNSUPPORTED ? - ENOENT : mapping ;
2010-11-13 17:13:56 +00:00
}
static int
2011-04-28 15:47:10 +01:00
armpmu_map_raw_event ( u32 raw_event_mask , u64 config )
2010-11-13 17:13:56 +00:00
{
2011-04-28 15:47:10 +01:00
return ( int ) ( config & raw_event_mask ) ;
}
static int map_cpu_event ( struct perf_event * event ,
const unsigned ( * event_map ) [ PERF_COUNT_HW_MAX ] ,
const unsigned ( * cache_map )
[ PERF_COUNT_HW_CACHE_MAX ]
[ PERF_COUNT_HW_CACHE_OP_MAX ]
[ PERF_COUNT_HW_CACHE_RESULT_MAX ] ,
u32 raw_event_mask )
{
u64 config = event - > attr . config ;
switch ( event - > attr . type ) {
case PERF_TYPE_HARDWARE :
return armpmu_map_event ( event_map , config ) ;
case PERF_TYPE_HW_CACHE :
return armpmu_map_cache_event ( cache_map , config ) ;
case PERF_TYPE_RAW :
return armpmu_map_raw_event ( raw_event_mask , config ) ;
}
return - ENOENT ;
2010-11-13 17:13:56 +00:00
}
2011-05-19 10:07:57 +01:00
int
2010-02-02 20:25:44 +01:00
armpmu_event_set_period ( struct perf_event * event ,
struct hw_perf_event * hwc ,
int idx )
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2010-05-21 14:43:08 +02:00
s64 left = local64_read ( & hwc - > period_left ) ;
2010-02-02 20:25:44 +01:00
s64 period = hwc - > sample_period ;
int ret = 0 ;
if ( unlikely ( left < = - period ) ) {
left = period ;
2010-05-21 14:43:08 +02:00
local64_set ( & hwc - > period_left , left ) ;
2010-02-02 20:25:44 +01:00
hwc - > last_period = period ;
ret = 1 ;
}
if ( unlikely ( left < = 0 ) ) {
left + = period ;
2010-05-21 14:43:08 +02:00
local64_set ( & hwc - > period_left , left ) ;
2010-02-02 20:25:44 +01:00
hwc - > last_period = period ;
ret = 1 ;
}
if ( left > ( s64 ) armpmu - > max_period )
left = armpmu - > max_period ;
2010-05-21 14:43:08 +02:00
local64_set ( & hwc - > prev_count , ( u64 ) - left ) ;
2010-02-02 20:25:44 +01:00
armpmu - > write_counter ( idx , ( u64 ) ( - left ) & 0xffffffff ) ;
perf_event_update_userpage ( event ) ;
return ret ;
}
2011-05-19 10:07:57 +01:00
u64
2010-02-02 20:25:44 +01:00
armpmu_event_update ( struct perf_event * event ,
struct hw_perf_event * hwc ,
2011-03-25 17:12:37 +01:00
int idx , int overflow )
2010-02-02 20:25:44 +01:00
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2011-03-25 17:12:37 +01:00
u64 delta , prev_raw_count , new_raw_count ;
2010-02-02 20:25:44 +01:00
again :
2010-05-21 14:43:08 +02:00
prev_raw_count = local64_read ( & hwc - > prev_count ) ;
2010-02-02 20:25:44 +01:00
new_raw_count = armpmu - > read_counter ( idx ) ;
2010-05-21 14:43:08 +02:00
if ( local64_cmpxchg ( & hwc - > prev_count , prev_raw_count ,
2010-02-02 20:25:44 +01:00
new_raw_count ) ! = prev_raw_count )
goto again ;
2011-03-25 17:12:37 +01:00
new_raw_count & = armpmu - > max_period ;
prev_raw_count & = armpmu - > max_period ;
if ( overflow )
2011-04-05 14:01:24 +01:00
delta = armpmu - > max_period - prev_raw_count + new_raw_count + 1 ;
2011-03-25 17:12:37 +01:00
else
delta = new_raw_count - prev_raw_count ;
2010-02-02 20:25:44 +01:00
2010-05-21 14:43:08 +02:00
local64_add ( delta , & event - > count ) ;
local64_sub ( delta , & hwc - > period_left ) ;
2010-02-02 20:25:44 +01:00
return new_raw_count ;
}
static void
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
armpmu_read ( struct perf_event * event )
2010-02-02 20:25:44 +01:00
{
struct hw_perf_event * hwc = & event - > hw ;
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
/* Don't read disabled counters! */
if ( hwc - > idx < 0 )
return ;
2010-02-02 20:25:44 +01:00
2011-03-25 17:12:37 +01:00
armpmu_event_update ( event , hwc , hwc - > idx , 0 ) ;
2010-02-02 20:25:44 +01:00
}
static void
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
armpmu_stop ( struct perf_event * event , int flags )
2010-02-02 20:25:44 +01:00
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2010-02-02 20:25:44 +01:00
struct hw_perf_event * hwc = & event - > hw ;
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
/*
* ARM pmu always has to update the counter , so ignore
* PERF_EF_UPDATE , see comments in armpmu_start ( ) .
*/
if ( ! ( hwc - > state & PERF_HES_STOPPED ) ) {
armpmu - > disable ( hwc , hwc - > idx ) ;
barrier ( ) ; /* why? */
2011-03-25 17:12:37 +01:00
armpmu_event_update ( event , hwc , hwc - > idx , 0 ) ;
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
hwc - > state | = PERF_HES_STOPPED | PERF_HES_UPTODATE ;
}
2010-02-02 20:25:44 +01:00
}
static void
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
armpmu_start ( struct perf_event * event , int flags )
2010-02-02 20:25:44 +01:00
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2010-02-02 20:25:44 +01:00
struct hw_perf_event * hwc = & event - > hw ;
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
/*
* ARM pmu always has to reprogram the period , so ignore
* PERF_EF_RELOAD , see the comment below .
*/
if ( flags & PERF_EF_RELOAD )
WARN_ON_ONCE ( ! ( hwc - > state & PERF_HES_UPTODATE ) ) ;
hwc - > state = 0 ;
2010-02-02 20:25:44 +01:00
/*
* Set the period again . Some counters can ' t be stopped , so when we
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
* were stopped we simply disabled the IRQ source and the counter
2010-02-02 20:25:44 +01:00
* may have been left counting . If we don ' t do this step then we may
* get an interrupt too soon or * way * too late if the overflow has
* happened since disabling .
*/
armpmu_event_set_period ( event , hwc , hwc - > idx ) ;
armpmu - > enable ( hwc , hwc - > idx ) ;
}
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
static void
armpmu_del ( struct perf_event * event , int flags )
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2011-05-17 11:20:11 +01:00
struct pmu_hw_events * hw_events = armpmu - > get_hw_events ( ) ;
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
struct hw_perf_event * hwc = & event - > hw ;
int idx = hwc - > idx ;
WARN_ON ( idx < 0 ) ;
armpmu_stop ( event , PERF_EF_UPDATE ) ;
2011-05-17 11:20:11 +01:00
hw_events - > events [ idx ] = NULL ;
clear_bit ( idx , hw_events - > used_mask ) ;
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
perf_event_update_userpage ( event ) ;
}
2010-02-02 20:25:44 +01:00
static int
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
armpmu_add ( struct perf_event * event , int flags )
2010-02-02 20:25:44 +01:00
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2011-05-17 11:20:11 +01:00
struct pmu_hw_events * hw_events = armpmu - > get_hw_events ( ) ;
2010-02-02 20:25:44 +01:00
struct hw_perf_event * hwc = & event - > hw ;
int idx ;
int err = 0 ;
2010-06-14 08:49:00 +02:00
perf_pmu_disable ( event - > pmu ) ;
2010-06-11 17:32:03 +02:00
2010-02-02 20:25:44 +01:00
/* If we don't have a space for the counter then finish early. */
2011-05-17 11:20:11 +01:00
idx = armpmu - > get_event_idx ( hw_events , hwc ) ;
2010-02-02 20:25:44 +01:00
if ( idx < 0 ) {
err = idx ;
goto out ;
}
/*
* If there is an event in the counter we are going to use then make
* sure it is disabled .
*/
event - > hw . idx = idx ;
armpmu - > disable ( hwc , idx ) ;
2011-05-17 11:20:11 +01:00
hw_events - > events [ idx ] = event ;
2010-02-02 20:25:44 +01:00
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
hwc - > state = PERF_HES_STOPPED | PERF_HES_UPTODATE ;
if ( flags & PERF_EF_START )
armpmu_start ( event , PERF_EF_RELOAD ) ;
2010-02-02 20:25:44 +01:00
/* Propagate our changes to the userspace mapping. */
perf_event_update_userpage ( event ) ;
out :
2010-06-14 08:49:00 +02:00
perf_pmu_enable ( event - > pmu ) ;
2010-02-02 20:25:44 +01:00
return err ;
}
static int
2011-05-17 11:20:11 +01:00
validate_event ( struct pmu_hw_events * hw_events ,
2010-02-02 20:25:44 +01:00
struct perf_event * event )
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2010-02-02 20:25:44 +01:00
struct hw_perf_event fake_event = event - > hw ;
2011-04-27 16:22:21 +01:00
struct pmu * leader_pmu = event - > group_leader - > pmu ;
2010-02-02 20:25:44 +01:00
2011-04-27 16:22:21 +01:00
if ( event - > pmu ! = leader_pmu | | event - > state < = PERF_EVENT_STATE_OFF )
2010-09-02 09:32:08 +01:00
return 1 ;
2010-02-02 20:25:44 +01:00
2011-05-17 11:20:11 +01:00
return armpmu - > get_event_idx ( hw_events , & fake_event ) > = 0 ;
2010-02-02 20:25:44 +01:00
}
static int
validate_group ( struct perf_event * event )
{
struct perf_event * sibling , * leader = event - > group_leader ;
2011-05-17 11:20:11 +01:00
struct pmu_hw_events fake_pmu ;
2011-11-17 15:05:14 +00:00
DECLARE_BITMAP ( fake_used_mask , ARMPMU_MAX_HWEVENTS ) ;
2010-02-02 20:25:44 +01:00
2011-11-17 15:05:14 +00:00
/*
* Initialise the fake PMU . We only need to populate the
* used_mask for the purposes of validation .
*/
memset ( fake_used_mask , 0 , sizeof ( fake_used_mask ) ) ;
fake_pmu . used_mask = fake_used_mask ;
2010-02-02 20:25:44 +01:00
if ( ! validate_event ( & fake_pmu , leader ) )
return - ENOSPC ;
list_for_each_entry ( sibling , & leader - > sibling_list , group_entry ) {
if ( ! validate_event ( & fake_pmu , sibling ) )
return - ENOSPC ;
}
if ( ! validate_event ( & fake_pmu , event ) )
return - ENOSPC ;
return 0 ;
}
2011-02-08 09:24:36 +05:30
static irqreturn_t armpmu_platform_irq ( int irq , void * dev )
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = ( struct arm_pmu * ) dev ;
2011-05-04 09:23:15 +01:00
struct platform_device * plat_device = armpmu - > plat_device ;
struct arm_pmu_platdata * plat = dev_get_platdata ( & plat_device - > dev ) ;
2011-02-08 09:24:36 +05:30
return plat - > handle_irq ( irq , dev , armpmu - > handle_irq ) ;
}
2011-07-27 15:18:59 +01:00
static void
2011-04-28 16:27:54 +01:00
armpmu_release_hardware ( struct arm_pmu * armpmu )
2011-07-27 15:18:59 +01:00
{
int i , irq , irqs ;
2011-05-04 09:23:15 +01:00
struct platform_device * pmu_device = armpmu - > plat_device ;
2011-03-02 15:00:08 +08:00
struct arm_pmu_platdata * plat =
dev_get_platdata ( & pmu_device - > dev ) ;
2011-07-27 15:18:59 +01:00
irqs = min ( pmu_device - > num_resources , num_possible_cpus ( ) ) ;
for ( i = 0 ; i < irqs ; + + i ) {
if ( ! cpumask_test_and_clear_cpu ( i , & armpmu - > active_irqs ) )
continue ;
irq = platform_get_irq ( pmu_device , i ) ;
2011-03-02 15:00:08 +08:00
if ( irq > = 0 ) {
if ( plat & & plat - > disable_irq )
plat - > disable_irq ( irq ) ;
2011-04-28 16:27:54 +01:00
free_irq ( irq , armpmu ) ;
2011-03-02 15:00:08 +08:00
}
2011-07-27 15:18:59 +01:00
}
2011-06-06 10:37:50 +01:00
release_pmu ( armpmu - > type ) ;
2011-07-27 15:18:59 +01:00
}
2010-02-02 20:25:44 +01:00
static int
2011-04-28 16:27:54 +01:00
armpmu_reserve_hardware ( struct arm_pmu * armpmu )
2010-02-02 20:25:44 +01:00
{
2011-02-08 09:24:36 +05:30
struct arm_pmu_platdata * plat ;
irq_handler_t handle_irq ;
2011-07-26 22:10:28 +01:00
int i , err , irq , irqs ;
2011-05-04 09:23:15 +01:00
struct platform_device * pmu_device = armpmu - > plat_device ;
2010-02-02 20:25:44 +01:00
2011-11-22 18:01:46 +00:00
if ( ! pmu_device )
return - ENODEV ;
2011-06-06 10:37:50 +01:00
err = reserve_pmu ( armpmu - > type ) ;
2011-07-26 22:10:28 +01:00
if ( err ) {
2010-02-02 20:25:44 +01:00
pr_warning ( " unable to reserve pmu \n " ) ;
2011-07-26 22:10:28 +01:00
return err ;
2010-02-02 20:25:44 +01:00
}
2011-02-08 09:24:36 +05:30
plat = dev_get_platdata ( & pmu_device - > dev ) ;
if ( plat & & plat - > handle_irq )
handle_irq = armpmu_platform_irq ;
else
handle_irq = armpmu - > handle_irq ;
2011-07-27 15:18:59 +01:00
irqs = min ( pmu_device - > num_resources , num_possible_cpus ( ) ) ;
2011-07-26 22:10:28 +01:00
if ( irqs < 1 ) {
2010-02-02 20:25:44 +01:00
pr_err ( " no irqs for PMUs defined \n " ) ;
return - ENODEV ;
}
2011-07-26 22:10:28 +01:00
for ( i = 0 ; i < irqs ; + + i ) {
2011-07-27 15:18:59 +01:00
err = 0 ;
2010-04-29 17:13:24 +01:00
irq = platform_get_irq ( pmu_device , i ) ;
if ( irq < 0 )
continue ;
2011-07-26 22:10:28 +01:00
/*
* If we have a single PMU interrupt that we can ' t shift ,
* assume that we ' re running on a uniprocessor machine and
2011-07-27 15:18:59 +01:00
* continue . Otherwise , continue without this interrupt .
2011-07-26 22:10:28 +01:00
*/
2011-07-27 15:18:59 +01:00
if ( irq_set_affinity ( irq , cpumask_of ( i ) ) & & irqs > 1 ) {
pr_warning ( " unable to set irq affinity (irq=%d, cpu=%u) \n " ,
irq , i ) ;
continue ;
2011-07-26 22:10:28 +01:00
}
2011-02-08 09:24:36 +05:30
err = request_irq ( irq , handle_irq ,
2010-02-25 15:04:14 +01:00
IRQF_DISABLED | IRQF_NOBALANCING ,
2011-04-28 16:27:54 +01:00
" arm-pmu " , armpmu ) ;
2010-02-02 20:25:44 +01:00
if ( err ) {
2011-07-26 22:10:28 +01:00
pr_err ( " unable to request IRQ%d for ARM PMU counters \n " ,
irq ) ;
2011-04-28 16:27:54 +01:00
armpmu_release_hardware ( armpmu ) ;
2011-07-27 15:18:59 +01:00
return err ;
2011-03-02 15:00:08 +08:00
} else if ( plat & & plat - > enable_irq )
plat - > enable_irq ( irq ) ;
2010-02-02 20:25:44 +01:00
2011-07-27 15:18:59 +01:00
cpumask_set_cpu ( i , & armpmu - > active_irqs ) ;
2010-04-29 17:13:24 +01:00
}
2010-02-02 20:25:44 +01:00
2011-07-27 15:18:59 +01:00
return 0 ;
2010-02-02 20:25:44 +01:00
}
static void
hw_perf_event_destroy ( struct perf_event * event )
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2011-04-27 11:20:11 +01:00
atomic_t * active_events = & armpmu - > active_events ;
struct mutex * pmu_reserve_mutex = & armpmu - > reserve_mutex ;
if ( atomic_dec_and_mutex_lock ( active_events , pmu_reserve_mutex ) ) {
2011-04-28 16:27:54 +01:00
armpmu_release_hardware ( armpmu ) ;
2011-04-27 11:20:11 +01:00
mutex_unlock ( pmu_reserve_mutex ) ;
2010-02-02 20:25:44 +01:00
}
}
2011-07-19 11:57:30 +01:00
static int
event_requires_mode_exclusion ( struct perf_event_attr * attr )
{
return attr - > exclude_idle | | attr - > exclude_user | |
attr - > exclude_kernel | | attr - > exclude_hv ;
}
2010-02-02 20:25:44 +01:00
static int
__hw_perf_event_init ( struct perf_event * event )
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2010-02-02 20:25:44 +01:00
struct hw_perf_event * hwc = & event - > hw ;
int mapping , err ;
2011-04-28 15:47:10 +01:00
mapping = armpmu - > map_event ( event ) ;
2010-02-02 20:25:44 +01:00
if ( mapping < 0 ) {
pr_debug ( " event %x:%llx not supported \n " , event - > attr . type ,
event - > attr . config ) ;
return mapping ;
}
2011-07-19 11:57:30 +01:00
/*
* We don ' t assign an index until we actually place the event onto
* hardware . Use - 1 to signify that we haven ' t decided where to put it
* yet . For SMP systems , each core has it ' s own PMU so we can ' t do any
* clever allocation or constraints checking at this point .
*/
hwc - > idx = - 1 ;
hwc - > config_base = 0 ;
hwc - > config = 0 ;
hwc - > event_base = 0 ;
2010-02-02 20:25:44 +01:00
/*
* Check whether we need to exclude the counter from certain modes .
*/
2011-07-19 11:57:30 +01:00
if ( ( ! armpmu - > set_event_filter | |
armpmu - > set_event_filter ( hwc , & event - > attr ) ) & &
event_requires_mode_exclusion ( & event - > attr ) ) {
2010-02-02 20:25:44 +01:00
pr_debug ( " ARM performance counters do not support "
" mode exclusion \n " ) ;
return - EPERM ;
}
/*
2011-07-19 11:57:30 +01:00
* Store the event encoding into the config_base field .
2010-02-02 20:25:44 +01:00
*/
2011-07-19 11:57:30 +01:00
hwc - > config_base | = ( unsigned long ) mapping ;
2010-02-02 20:25:44 +01:00
if ( ! hwc - > sample_period ) {
hwc - > sample_period = armpmu - > max_period ;
hwc - > last_period = hwc - > sample_period ;
2010-05-21 14:43:08 +02:00
local64_set ( & hwc - > period_left , hwc - > sample_period ) ;
2010-02-02 20:25:44 +01:00
}
err = 0 ;
if ( event - > group_leader ! = event ) {
err = validate_group ( event ) ;
if ( err )
return - EINVAL ;
}
return err ;
}
2010-06-11 13:35:08 +02:00
static int armpmu_event_init ( struct perf_event * event )
2010-02-02 20:25:44 +01:00
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( event - > pmu ) ;
2010-02-02 20:25:44 +01:00
int err = 0 ;
2011-04-27 11:20:11 +01:00
atomic_t * active_events = & armpmu - > active_events ;
2010-02-02 20:25:44 +01:00
2011-04-28 15:47:10 +01:00
if ( armpmu - > map_event ( event ) = = - ENOENT )
2010-06-11 13:35:08 +02:00
return - ENOENT ;
2010-02-02 20:25:44 +01:00
event - > destroy = hw_perf_event_destroy ;
2011-04-27 11:20:11 +01:00
if ( ! atomic_inc_not_zero ( active_events ) ) {
mutex_lock ( & armpmu - > reserve_mutex ) ;
if ( atomic_read ( active_events ) = = 0 )
2011-04-28 16:27:54 +01:00
err = armpmu_reserve_hardware ( armpmu ) ;
2010-02-02 20:25:44 +01:00
if ( ! err )
2011-04-27 11:20:11 +01:00
atomic_inc ( active_events ) ;
mutex_unlock ( & armpmu - > reserve_mutex ) ;
2010-02-02 20:25:44 +01:00
}
if ( err )
2010-06-11 13:35:08 +02:00
return err ;
2010-02-02 20:25:44 +01:00
err = __hw_perf_event_init ( event ) ;
if ( err )
hw_perf_event_destroy ( event ) ;
2010-06-11 13:35:08 +02:00
return err ;
2010-02-02 20:25:44 +01:00
}
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
static void armpmu_enable ( struct pmu * pmu )
2010-02-02 20:25:44 +01:00
{
2011-05-17 11:20:11 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( pmu ) ;
struct pmu_hw_events * hw_events = armpmu - > get_hw_events ( ) ;
2011-08-23 11:59:49 +01:00
int enabled = bitmap_weight ( hw_events - > used_mask , armpmu - > num_events ) ;
2010-02-02 20:25:44 +01:00
2011-07-01 14:38:12 +01:00
if ( enabled )
armpmu - > start ( ) ;
2010-02-02 20:25:44 +01:00
}
perf: Rework the PMU methods
Replace pmu::{enable,disable,start,stop,unthrottle} with
pmu::{add,del,start,stop}, all of which take a flags argument.
The new interface extends the capability to stop a counter while
keeping it scheduled on the PMU. We replace the throttled state with
the generic stopped state.
This also allows us to efficiently stop/start counters over certain
code paths (like IRQ handlers).
It also allows scheduling a counter without it starting, allowing for
a generic frozen state (useful for rotating stopped counters).
The stopped state is implemented in two different ways, depending on
how the architecture implemented the throttled state:
1) We disable the counter:
a) the pmu has per-counter enable bits, we flip that
b) we program a NOP event, preserving the counter state
2) We store the counter state and ignore all read/overflow events
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: paulus <paulus@samba.org>
Cc: stephane eranian <eranian@googlemail.com>
Cc: Robert Richter <robert.richter@amd.com>
Cc: Will Deacon <will.deacon@arm.com>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Cyrill Gorcunov <gorcunov@gmail.com>
Cc: Lin Ming <ming.m.lin@intel.com>
Cc: Yanmin <yanmin_zhang@linux.intel.com>
Cc: Deng-Cheng Zhu <dengcheng.zhu@gmail.com>
Cc: David Miller <davem@davemloft.net>
Cc: Michael Cree <mcree@orcon.net.nz>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2010-06-16 14:37:10 +02:00
static void armpmu_disable ( struct pmu * pmu )
2010-02-02 20:25:44 +01:00
{
2011-04-28 16:27:54 +01:00
struct arm_pmu * armpmu = to_arm_pmu ( pmu ) ;
2011-04-27 10:31:51 +01:00
armpmu - > stop ( ) ;
2010-02-02 20:25:44 +01:00
}
2011-04-27 11:20:11 +01:00
static void __init armpmu_init ( struct arm_pmu * armpmu )
{
atomic_set ( & armpmu - > active_events , 0 ) ;
mutex_init ( & armpmu - > reserve_mutex ) ;
2011-04-28 16:27:54 +01:00
armpmu - > pmu = ( struct pmu ) {
. pmu_enable = armpmu_enable ,
. pmu_disable = armpmu_disable ,
. event_init = armpmu_event_init ,
. add = armpmu_add ,
. del = armpmu_del ,
. start = armpmu_start ,
. stop = armpmu_stop ,
. read = armpmu_read ,
} ;
}
2011-05-19 10:07:57 +01:00
int __init armpmu_register ( struct arm_pmu * armpmu , char * name , int type )
2011-04-28 16:27:54 +01:00
{
armpmu_init ( armpmu ) ;
return perf_pmu_register ( & armpmu - > pmu , name , type ) ;
2011-04-27 11:20:11 +01:00
}
2010-11-13 19:04:32 +00:00
/* Include the PMU-specific implementations. */
# include "perf_event_xscale.c"
# include "perf_event_v6.c"
# include "perf_event_v7.c"
2010-04-30 11:33:33 +01:00
2011-03-25 13:13:34 +01:00
/*
* Ensure the PMU has sane values out of reset .
* This requires SMP to be available , so exists as a separate initcall .
*/
static int __init
2011-05-17 11:20:11 +01:00
cpu_pmu_reset ( void )
2011-03-25 13:13:34 +01:00
{
2011-05-17 11:20:11 +01:00
if ( cpu_pmu & & cpu_pmu - > reset )
return on_each_cpu ( cpu_pmu - > reset , NULL , 1 ) ;
2011-03-25 13:13:34 +01:00
return 0 ;
}
2011-05-17 11:20:11 +01:00
arch_initcall ( cpu_pmu_reset ) ;
2011-03-25 13:13:34 +01:00
2011-07-26 22:10:28 +01:00
/*
* PMU platform driver and devicetree bindings .
*/
static struct of_device_id armpmu_of_device_ids [ ] = {
{ . compatible = " arm,cortex-a9-pmu " } ,
{ . compatible = " arm,cortex-a8-pmu " } ,
{ . compatible = " arm,arm1136-pmu " } ,
{ . compatible = " arm,arm1176-pmu " } ,
{ } ,
} ;
static struct platform_device_id armpmu_plat_device_ids [ ] = {
{ . name = " arm-pmu " } ,
{ } ,
} ;
static int __devinit armpmu_device_probe ( struct platform_device * pdev )
{
2011-05-17 11:20:11 +01:00
cpu_pmu - > plat_device = pdev ;
2011-07-26 22:10:28 +01:00
return 0 ;
}
static struct platform_driver armpmu_driver = {
. driver = {
. name = " arm-pmu " ,
. of_match_table = armpmu_of_device_ids ,
} ,
. probe = armpmu_device_probe ,
. id_table = armpmu_plat_device_ids ,
} ;
static int __init register_pmu_driver ( void )
{
return platform_driver_register ( & armpmu_driver ) ;
}
device_initcall ( register_pmu_driver ) ;
2011-05-17 11:20:11 +01:00
static struct pmu_hw_events * armpmu_get_cpu_events ( void )
2011-05-04 09:23:51 +01:00
{
return & __get_cpu_var ( cpu_hw_events ) ;
}
static void __init cpu_pmu_init ( struct arm_pmu * armpmu )
{
2011-04-28 10:17:04 +01:00
int cpu ;
for_each_possible_cpu ( cpu ) {
2011-05-17 11:20:11 +01:00
struct pmu_hw_events * events = & per_cpu ( cpu_hw_events , cpu ) ;
2011-06-24 11:30:59 +01:00
events - > events = per_cpu ( hw_events , cpu ) ;
events - > used_mask = per_cpu ( used_mask , cpu ) ;
2011-04-28 10:17:04 +01:00
raw_spin_lock_init ( & events - > pmu_lock ) ;
}
2011-05-04 09:23:51 +01:00
armpmu - > get_hw_events = armpmu_get_cpu_events ;
2011-06-06 10:37:50 +01:00
armpmu - > type = ARM_PMU_DEVICE_CPU ;
2011-05-04 09:23:51 +01:00
}
2011-07-26 22:10:28 +01:00
/*
* CPU PMU identification and registration .
*/
2010-02-02 20:25:44 +01:00
static int __init
init_hw_perf_events ( void )
{
unsigned long cpuid = read_cpuid_id ( ) ;
unsigned long implementor = ( cpuid & 0xFF000000 ) > > 24 ;
unsigned long part_number = ( cpuid & 0xFFF0 ) ;
2010-04-30 11:33:33 +01:00
/* ARM Ltd CPUs. */
2010-02-02 20:25:44 +01:00
if ( 0x41 = = implementor ) {
switch ( part_number ) {
case 0xB360 : /* ARM1136 */
case 0xB560 : /* ARM1156 */
case 0xB760 : /* ARM1176 */
2011-05-17 11:20:11 +01:00
cpu_pmu = armv6pmu_init ( ) ;
2010-02-02 20:25:44 +01:00
break ;
case 0xB020 : /* ARM11mpcore */
2011-05-17 11:20:11 +01:00
cpu_pmu = armv6mpcore_pmu_init ( ) ;
2010-02-02 20:25:44 +01:00
break ;
2010-01-26 18:51:05 +01:00
case 0xC080 : /* Cortex-A8 */
2011-05-17 11:20:11 +01:00
cpu_pmu = armv7_a8_pmu_init ( ) ;
2010-01-26 18:51:05 +01:00
break ;
case 0xC090 : /* Cortex-A9 */
2011-05-17 11:20:11 +01:00
cpu_pmu = armv7_a9_pmu_init ( ) ;
2010-01-26 18:51:05 +01:00
break ;
2011-06-03 17:40:15 +01:00
case 0xC050 : /* Cortex-A5 */
2011-05-17 11:20:11 +01:00
cpu_pmu = armv7_a5_pmu_init ( ) ;
2011-06-03 17:40:15 +01:00
break ;
2011-01-19 14:24:38 +00:00
case 0xC0F0 : /* Cortex-A15 */
2011-05-17 11:20:11 +01:00
cpu_pmu = armv7_a15_pmu_init ( ) ;
2011-01-19 14:24:38 +00:00
break ;
2010-04-30 11:33:33 +01:00
}
/* Intel CPUs [xscale]. */
} else if ( 0x69 = = implementor ) {
part_number = ( cpuid > > 13 ) & 0x7 ;
switch ( part_number ) {
case 1 :
2011-05-17 11:20:11 +01:00
cpu_pmu = xscale1pmu_init ( ) ;
2010-04-30 11:33:33 +01:00
break ;
case 2 :
2011-05-17 11:20:11 +01:00
cpu_pmu = xscale2pmu_init ( ) ;
2010-04-30 11:33:33 +01:00
break ;
2010-02-02 20:25:44 +01:00
}
}
2011-05-17 11:20:11 +01:00
if ( cpu_pmu ) {
2010-01-26 18:51:05 +01:00
pr_info ( " enabled with %s PMU driver, %d counters available \n " ,
2011-05-17 11:20:11 +01:00
cpu_pmu - > name , cpu_pmu - > num_events ) ;
cpu_pmu_init ( cpu_pmu ) ;
armpmu_register ( cpu_pmu , " cpu " , PERF_TYPE_RAW ) ;
2010-04-30 11:33:33 +01:00
} else {
pr_info ( " no hardware support available \n " ) ;
}
2010-02-02 20:25:44 +01:00
return 0 ;
}
2010-11-25 18:38:29 +01:00
early_initcall ( init_hw_perf_events ) ;
2010-02-02 20:25:44 +01:00
/*
* Callchain handling code .
*/
/*
* The registers we ' re interested in are at the end of the variable
* length saved register structure . The fp points at the end of this
* structure so the address of this struct is :
* ( struct frame_tail * ) ( xxx - > fp ) - 1
*
* This code has been adapted from the ARM OProfile support .
*/
struct frame_tail {
2010-11-30 18:15:53 +01:00
struct frame_tail __user * fp ;
unsigned long sp ;
unsigned long lr ;
2010-02-02 20:25:44 +01:00
} __attribute__ ( ( packed ) ) ;
/*
* Get the return address for a single stackframe and return a pointer to the
* next frame tail .
*/
2010-11-30 18:15:53 +01:00
static struct frame_tail __user *
user_backtrace ( struct frame_tail __user * tail ,
2010-02-02 20:25:44 +01:00
struct perf_callchain_entry * entry )
{
struct frame_tail buftail ;
/* Also check accessibility of one struct frame_tail beyond */
if ( ! access_ok ( VERIFY_READ , tail , sizeof ( buftail ) ) )
return NULL ;
if ( __copy_from_user_inatomic ( & buftail , tail , sizeof ( buftail ) ) )
return NULL ;
2010-06-29 19:34:05 +02:00
perf_callchain_store ( entry , buftail . lr ) ;
2010-02-02 20:25:44 +01:00
/*
* Frame pointers should strictly progress back up the stack
* ( towards higher addresses ) .
*/
2011-02-09 11:35:12 +01:00
if ( tail + 1 > = buftail . fp )
2010-02-02 20:25:44 +01:00
return NULL ;
return buftail . fp - 1 ;
}
2010-06-30 23:03:51 +02:00
void
perf_callchain_user ( struct perf_callchain_entry * entry , struct pt_regs * regs )
2010-02-02 20:25:44 +01:00
{
2010-11-30 18:15:53 +01:00
struct frame_tail __user * tail ;
2010-02-02 20:25:44 +01:00
2010-11-30 18:15:53 +01:00
tail = ( struct frame_tail __user * ) regs - > ARM_fp - 1 ;
2010-02-02 20:25:44 +01:00
2011-04-18 22:12:59 +01:00
while ( ( entry - > nr < PERF_MAX_STACK_DEPTH ) & &
tail & & ! ( ( unsigned long ) tail & 0x3 ) )
2010-02-02 20:25:44 +01:00
tail = user_backtrace ( tail , entry ) ;
}
/*
* Gets called by walk_stackframe ( ) for every stackframe . This will be called
* whist unwinding the stackframe and is like a subroutine return so we use
* the PC .
*/
static int
callchain_trace ( struct stackframe * fr ,
void * data )
{
struct perf_callchain_entry * entry = data ;
2010-06-29 19:34:05 +02:00
perf_callchain_store ( entry , fr - > pc ) ;
2010-02-02 20:25:44 +01:00
return 0 ;
}
2010-06-30 23:03:51 +02:00
void
perf_callchain_kernel ( struct perf_callchain_entry * entry , struct pt_regs * regs )
2010-02-02 20:25:44 +01:00
{
struct stackframe fr ;
fr . fp = regs - > ARM_fp ;
fr . sp = regs - > ARM_sp ;
fr . lr = regs - > ARM_lr ;
fr . pc = regs - > ARM_pc ;
walk_stackframe ( & fr , callchain_trace , entry ) ;
}