2009-04-20 06:37:32 -07:00
|
|
|
/*
|
2009-06-02 14:37:05 -07:00
|
|
|
* builtin-stat.c
|
|
|
|
*
|
|
|
|
* Builtin stat command: Give a precise performance counters summary
|
|
|
|
* overview about any workload, CPU or specific PID.
|
|
|
|
*
|
|
|
|
* Sample output:
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-02 14:37:05 -07:00
|
|
|
$ perf stat ~/hackbench 10
|
|
|
|
Time: 0.104
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-02 14:37:05 -07:00
|
|
|
Performance counter stats for '/home/mingo/hackbench':
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-02 14:37:05 -07:00
|
|
|
1255.538611 task clock ticks # 10.143 CPU utilization factor
|
|
|
|
54011 context switches # 0.043 M/sec
|
|
|
|
385 CPU migrations # 0.000 M/sec
|
|
|
|
17755 pagefaults # 0.014 M/sec
|
|
|
|
3808323185 CPU cycles # 3033.219 M/sec
|
|
|
|
1575111190 instructions # 1254.530 M/sec
|
|
|
|
17367895 cache references # 13.833 M/sec
|
|
|
|
7674421 cache misses # 6.112 M/sec
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-02 14:37:05 -07:00
|
|
|
Wall-clock time elapsed: 123.786620 msecs
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-05-26 00:17:18 -07:00
|
|
|
*
|
|
|
|
* Copyright (C) 2008, Red Hat Inc, Ingo Molnar <mingo@redhat.com>
|
|
|
|
*
|
|
|
|
* Improvements and fixes by:
|
|
|
|
*
|
|
|
|
* Arjan van de Ven <arjan@linux.intel.com>
|
|
|
|
* Yanmin Zhang <yanmin.zhang@intel.com>
|
|
|
|
* Wu Fengguang <fengguang.wu@intel.com>
|
|
|
|
* Mike Galbraith <efault@gmx.de>
|
|
|
|
* Paul Mackerras <paulus@samba.org>
|
2009-06-26 14:32:07 -07:00
|
|
|
* Jaswinder Singh Rajput <jaswinder@kernel.org>
|
2009-05-26 00:17:18 -07:00
|
|
|
*
|
|
|
|
* Released under the GPL v2. (and only v2, not any later version)
|
2009-04-20 06:37:32 -07:00
|
|
|
*/
|
|
|
|
|
2009-05-23 09:28:58 -07:00
|
|
|
#include "perf.h"
|
2009-05-27 00:10:38 -07:00
|
|
|
#include "builtin.h"
|
2009-04-26 23:02:14 -07:00
|
|
|
#include "util/util.h"
|
2009-05-26 00:17:18 -07:00
|
|
|
#include "util/parse-options.h"
|
|
|
|
#include "util/parse-events.h"
|
2009-08-16 13:05:48 -07:00
|
|
|
#include "util/event.h"
|
|
|
|
#include "util/debug.h"
|
2009-04-20 06:37:32 -07:00
|
|
|
|
|
|
|
#include <sys/prctl.h>
|
2009-06-13 05:57:28 -07:00
|
|
|
#include <math.h>
|
2009-05-05 08:50:27 -07:00
|
|
|
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 03:02:48 -07:00
|
|
|
static struct perf_event_attr default_attrs[] = {
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-11 05:06:28 -07:00
|
|
|
{ .type = PERF_TYPE_SOFTWARE, .config = PERF_COUNT_SW_TASK_CLOCK },
|
|
|
|
{ .type = PERF_TYPE_SOFTWARE, .config = PERF_COUNT_SW_CONTEXT_SWITCHES},
|
|
|
|
{ .type = PERF_TYPE_SOFTWARE, .config = PERF_COUNT_SW_CPU_MIGRATIONS },
|
|
|
|
{ .type = PERF_TYPE_SOFTWARE, .config = PERF_COUNT_SW_PAGE_FAULTS },
|
|
|
|
|
|
|
|
{ .type = PERF_TYPE_HARDWARE, .config = PERF_COUNT_HW_CPU_CYCLES },
|
|
|
|
{ .type = PERF_TYPE_HARDWARE, .config = PERF_COUNT_HW_INSTRUCTIONS },
|
|
|
|
{ .type = PERF_TYPE_HARDWARE, .config = PERF_COUNT_HW_CACHE_REFERENCES},
|
|
|
|
{ .type = PERF_TYPE_HARDWARE, .config = PERF_COUNT_HW_CACHE_MISSES },
|
|
|
|
|
2009-04-20 06:37:32 -07:00
|
|
|
};
|
2009-05-26 00:17:18 -07:00
|
|
|
|
2009-06-06 00:58:57 -07:00
|
|
|
static int system_wide = 0;
|
2009-07-01 03:37:06 -07:00
|
|
|
static unsigned int nr_cpus = 0;
|
2009-06-24 05:49:34 -07:00
|
|
|
static int run_idx = 0;
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-24 05:49:34 -07:00
|
|
|
static int run_count = 1;
|
|
|
|
static int inherit = 1;
|
2009-04-30 04:53:33 -07:00
|
|
|
static int scale = 1;
|
2009-10-03 17:35:01 -07:00
|
|
|
static pid_t target_pid = -1;
|
|
|
|
static pid_t child_pid = -1;
|
2009-06-26 21:10:30 -07:00
|
|
|
static int null_run = 0;
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-24 05:49:34 -07:00
|
|
|
static int fd[MAX_NR_CPUS][MAX_COUNTERS];
|
2009-06-13 05:57:28 -07:00
|
|
|
|
2009-09-04 08:03:13 -07:00
|
|
|
static int event_scaled[MAX_COUNTERS];
|
2009-06-24 05:49:34 -07:00
|
|
|
|
2009-09-04 06:36:12 -07:00
|
|
|
struct stats
|
|
|
|
{
|
2009-09-04 08:26:26 -07:00
|
|
|
double n, mean, M2;
|
2009-09-04 06:36:12 -07:00
|
|
|
};
|
2009-06-13 05:57:28 -07:00
|
|
|
|
2009-09-04 06:36:08 -07:00
|
|
|
static void update_stats(struct stats *stats, u64 val)
|
|
|
|
{
|
2009-09-04 08:26:26 -07:00
|
|
|
double delta;
|
2009-09-04 06:36:08 -07:00
|
|
|
|
2009-09-04 08:26:26 -07:00
|
|
|
stats->n++;
|
|
|
|
delta = val - stats->mean;
|
|
|
|
stats->mean += delta / stats->n;
|
|
|
|
stats->M2 += delta*(val - stats->mean);
|
2009-09-04 06:36:08 -07:00
|
|
|
}
|
|
|
|
|
2009-09-04 06:36:12 -07:00
|
|
|
static double avg_stats(struct stats *stats)
|
|
|
|
{
|
2009-09-04 08:26:26 -07:00
|
|
|
return stats->mean;
|
2009-09-04 06:36:12 -07:00
|
|
|
}
|
2009-06-13 05:57:28 -07:00
|
|
|
|
2009-09-04 06:36:12 -07:00
|
|
|
/*
|
2009-09-04 08:03:13 -07:00
|
|
|
* http://en.wikipedia.org/wiki/Algorithms_for_calculating_variance
|
|
|
|
*
|
2009-09-04 08:26:26 -07:00
|
|
|
* (\Sum n_i^2) - ((\Sum n_i)^2)/n
|
|
|
|
* s^2 = -------------------------------
|
|
|
|
* n - 1
|
2009-09-04 08:03:13 -07:00
|
|
|
*
|
|
|
|
* http://en.wikipedia.org/wiki/Stddev
|
|
|
|
*
|
|
|
|
* The std dev of the mean is related to the std dev by:
|
|
|
|
*
|
|
|
|
* s
|
|
|
|
* s_mean = -------
|
|
|
|
* sqrt(n)
|
|
|
|
*
|
2009-09-04 06:36:12 -07:00
|
|
|
*/
|
|
|
|
static double stddev_stats(struct stats *stats)
|
|
|
|
{
|
2009-09-04 08:26:26 -07:00
|
|
|
double variance = stats->M2 / (stats->n - 1);
|
|
|
|
double variance_mean = variance / stats->n;
|
2009-06-13 05:57:28 -07:00
|
|
|
|
2009-09-04 08:03:13 -07:00
|
|
|
return sqrt(variance_mean);
|
2009-09-04 06:36:12 -07:00
|
|
|
}
|
2009-06-13 05:57:28 -07:00
|
|
|
|
2009-09-04 06:36:12 -07:00
|
|
|
struct stats event_res_stats[MAX_COUNTERS][3];
|
|
|
|
struct stats runtime_nsecs_stats;
|
|
|
|
struct stats walltime_nsecs_stats;
|
|
|
|
struct stats runtime_cycles_stats;
|
2009-05-29 00:10:54 -07:00
|
|
|
|
2009-07-01 02:35:09 -07:00
|
|
|
#define MATCH_EVENT(t, c, counter) \
|
|
|
|
(attrs[counter].type == PERF_TYPE_##t && \
|
|
|
|
attrs[counter].config == PERF_COUNT_##c)
|
|
|
|
|
2009-06-23 04:42:49 -07:00
|
|
|
#define ERR_PERF_OPEN \
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 03:02:48 -07:00
|
|
|
"Error: counter %d, sys_perf_event_open() syscall returned with %d (%s)\n"
|
2009-06-23 04:42:49 -07:00
|
|
|
|
2009-06-29 04:13:21 -07:00
|
|
|
static void create_perf_stat_counter(int counter, int pid)
|
2009-04-20 06:37:32 -07:00
|
|
|
{
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 03:02:48 -07:00
|
|
|
struct perf_event_attr *attr = attrs + counter;
|
2009-05-05 08:50:27 -07:00
|
|
|
|
2009-04-20 06:37:32 -07:00
|
|
|
if (scale)
|
2009-06-06 00:58:57 -07:00
|
|
|
attr->read_format = PERF_FORMAT_TOTAL_TIME_ENABLED |
|
|
|
|
PERF_FORMAT_TOTAL_TIME_RUNNING;
|
2009-04-20 06:37:32 -07:00
|
|
|
|
|
|
|
if (system_wide) {
|
2009-07-01 03:37:06 -07:00
|
|
|
unsigned int cpu;
|
|
|
|
|
2009-06-23 04:42:49 -07:00
|
|
|
for (cpu = 0; cpu < nr_cpus; cpu++) {
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 03:02:48 -07:00
|
|
|
fd[cpu][counter] = sys_perf_event_open(attr, -1, cpu, -1, 0);
|
2009-06-23 04:42:49 -07:00
|
|
|
if (fd[cpu][counter] < 0 && verbose)
|
|
|
|
fprintf(stderr, ERR_PERF_OPEN, counter,
|
|
|
|
fd[cpu][counter], strerror(errno));
|
2009-04-20 06:37:32 -07:00
|
|
|
}
|
|
|
|
} else {
|
2009-06-29 23:07:19 -07:00
|
|
|
attr->inherit = inherit;
|
|
|
|
attr->disabled = 1;
|
|
|
|
attr->enable_on_exec = 1;
|
2009-04-20 06:37:32 -07:00
|
|
|
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 03:02:48 -07:00
|
|
|
fd[0][counter] = sys_perf_event_open(attr, pid, -1, -1, 0);
|
2009-06-23 04:42:49 -07:00
|
|
|
if (fd[0][counter] < 0 && verbose)
|
|
|
|
fprintf(stderr, ERR_PERF_OPEN, counter,
|
|
|
|
fd[0][counter], strerror(errno));
|
2009-04-20 06:37:32 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
/*
|
|
|
|
* Does the counter have nsecs as a unit?
|
|
|
|
*/
|
|
|
|
static inline int nsec_counter(int counter)
|
|
|
|
{
|
2009-07-01 02:35:09 -07:00
|
|
|
if (MATCH_EVENT(SOFTWARE, SW_CPU_CLOCK, counter) ||
|
|
|
|
MATCH_EVENT(SOFTWARE, SW_TASK_CLOCK, counter))
|
2009-05-29 00:10:54 -07:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-05-29 00:10:54 -07:00
|
|
|
* Read out the results of a single counter:
|
2009-05-29 00:10:54 -07:00
|
|
|
*/
|
2009-05-29 00:10:54 -07:00
|
|
|
static void read_counter(int counter)
|
2009-05-29 00:10:54 -07:00
|
|
|
{
|
2009-09-04 09:23:38 -07:00
|
|
|
u64 count[3], single_count[3];
|
2009-07-01 03:37:06 -07:00
|
|
|
unsigned int cpu;
|
|
|
|
size_t res, nv;
|
2009-05-29 00:10:54 -07:00
|
|
|
int scaled;
|
2009-09-04 06:36:08 -07:00
|
|
|
int i;
|
2009-05-29 00:10:54 -07:00
|
|
|
|
|
|
|
count[0] = count[1] = count[2] = 0;
|
2009-05-29 00:10:54 -07:00
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
nv = scale ? 3 : 1;
|
2009-06-23 04:42:49 -07:00
|
|
|
for (cpu = 0; cpu < nr_cpus; cpu++) {
|
2009-06-07 08:06:46 -07:00
|
|
|
if (fd[cpu][counter] < 0)
|
|
|
|
continue;
|
|
|
|
|
perf_counter tools: Define and use our own u64, s64 etc. definitions
On 64-bit powerpc, __u64 is defined to be unsigned long rather than
unsigned long long. This causes compiler warnings every time we
print a __u64 value with %Lx.
Rather than changing __u64, we define our own u64 to be unsigned long
long on all architectures, and similarly s64 as signed long long.
For consistency we also define u32, s32, u16, s16, u8 and s8. These
definitions are put in a new header, types.h, because these definitions
are needed in util/string.h and util/symbol.h.
The main change here is the mechanical change of __[us]{64,32,16,8}
to remove the "__". The other changes are:
* Create types.h
* Include types.h in perf.h, util/string.h and util/symbol.h
* Add types.h to the LIB_H definition in Makefile
* Added (u64) casts in process_overflow_event() and print_sym_table()
to kill two remaining warnings.
Signed-off-by: Paul Mackerras <paulus@samba.org>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: benh@kernel.crashing.org
LKML-Reference: <19003.33494.495844.956580@cargo.ozlabs.ibm.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-06-19 05:21:42 -07:00
|
|
|
res = read(fd[cpu][counter], single_count, nv * sizeof(u64));
|
|
|
|
assert(res == nv * sizeof(u64));
|
2009-07-01 03:37:06 -07:00
|
|
|
|
2009-06-13 05:57:28 -07:00
|
|
|
close(fd[cpu][counter]);
|
|
|
|
fd[cpu][counter] = -1;
|
2009-05-29 00:10:54 -07:00
|
|
|
|
|
|
|
count[0] += single_count[0];
|
|
|
|
if (scale) {
|
|
|
|
count[1] += single_count[1];
|
|
|
|
count[2] += single_count[2];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
scaled = 0;
|
|
|
|
if (scale) {
|
|
|
|
if (count[2] == 0) {
|
2009-09-04 06:36:08 -07:00
|
|
|
event_scaled[counter] = -1;
|
2009-05-29 00:10:54 -07:00
|
|
|
count[0] = 0;
|
2009-05-29 00:10:54 -07:00
|
|
|
return;
|
|
|
|
}
|
2009-05-29 00:10:54 -07:00
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
if (count[2] < count[1]) {
|
2009-09-04 06:36:08 -07:00
|
|
|
event_scaled[counter] = 1;
|
2009-05-29 00:10:54 -07:00
|
|
|
count[0] = (unsigned long long)
|
|
|
|
((double)count[0] * count[1] / count[2] + 0.5);
|
|
|
|
}
|
|
|
|
}
|
2009-09-04 06:36:08 -07:00
|
|
|
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
update_stats(&event_res_stats[counter][i], count[i]);
|
|
|
|
|
|
|
|
if (verbose) {
|
|
|
|
fprintf(stderr, "%s: %Ld %Ld %Ld\n", event_name(counter),
|
|
|
|
count[0], count[1], count[2]);
|
|
|
|
}
|
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
/*
|
|
|
|
* Save the full runtime - to allow normalization during printout:
|
|
|
|
*/
|
2009-07-01 02:35:09 -07:00
|
|
|
if (MATCH_EVENT(SOFTWARE, SW_TASK_CLOCK, counter))
|
2009-09-04 06:36:08 -07:00
|
|
|
update_stats(&runtime_nsecs_stats, count[0]);
|
2009-07-01 02:35:09 -07:00
|
|
|
if (MATCH_EVENT(HARDWARE, HW_CPU_CYCLES, counter))
|
2009-09-04 06:36:08 -07:00
|
|
|
update_stats(&runtime_cycles_stats, count[0]);
|
2009-05-29 00:10:54 -07:00
|
|
|
}
|
|
|
|
|
2009-07-01 03:37:06 -07:00
|
|
|
static int run_perf_stat(int argc __used, const char **argv)
|
2009-06-13 05:57:28 -07:00
|
|
|
{
|
|
|
|
unsigned long long t0, t1;
|
|
|
|
int status = 0;
|
|
|
|
int counter;
|
|
|
|
int pid;
|
2009-06-29 04:13:21 -07:00
|
|
|
int child_ready_pipe[2], go_pipe[2];
|
|
|
|
char buf;
|
2009-06-13 05:57:28 -07:00
|
|
|
|
|
|
|
if (!system_wide)
|
|
|
|
nr_cpus = 1;
|
|
|
|
|
2009-06-29 04:13:21 -07:00
|
|
|
if (pipe(child_ready_pipe) < 0 || pipe(go_pipe) < 0) {
|
|
|
|
perror("failed to create pipes");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((pid = fork()) < 0)
|
|
|
|
perror("failed to fork");
|
|
|
|
|
|
|
|
if (!pid) {
|
|
|
|
close(child_ready_pipe[0]);
|
|
|
|
close(go_pipe[1]);
|
|
|
|
fcntl(go_pipe[0], F_SETFD, FD_CLOEXEC);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do a dummy execvp to get the PLT entry resolved,
|
|
|
|
* so we avoid the resolver overhead on the real
|
|
|
|
* execvp call.
|
|
|
|
*/
|
|
|
|
execvp("", (char **)argv);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Tell the parent we're ready to go
|
|
|
|
*/
|
|
|
|
close(child_ready_pipe[1]);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait until the parent tells us to go.
|
|
|
|
*/
|
2009-07-01 12:02:10 -07:00
|
|
|
if (read(go_pipe[0], &buf, 1) == -1)
|
|
|
|
perror("unable to read pipe");
|
2009-06-29 04:13:21 -07:00
|
|
|
|
|
|
|
execvp(argv[0], (char **)argv);
|
|
|
|
|
|
|
|
perror(argv[0]);
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
2009-10-03 17:35:01 -07:00
|
|
|
child_pid = pid;
|
|
|
|
|
2009-06-29 04:13:21 -07:00
|
|
|
/*
|
|
|
|
* Wait for the child to be ready to exec.
|
|
|
|
*/
|
|
|
|
close(child_ready_pipe[1]);
|
|
|
|
close(go_pipe[0]);
|
2009-07-01 12:02:10 -07:00
|
|
|
if (read(child_ready_pipe[0], &buf, 1) == -1)
|
|
|
|
perror("unable to read pipe");
|
2009-06-29 04:13:21 -07:00
|
|
|
close(child_ready_pipe[0]);
|
|
|
|
|
2009-06-13 05:57:28 -07:00
|
|
|
for (counter = 0; counter < nr_counters; counter++)
|
2009-06-29 04:13:21 -07:00
|
|
|
create_perf_stat_counter(counter, pid);
|
2009-06-13 05:57:28 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable counters and exec the command:
|
|
|
|
*/
|
|
|
|
t0 = rdclock();
|
|
|
|
|
2009-06-29 04:13:21 -07:00
|
|
|
close(go_pipe[1]);
|
2009-06-13 05:57:28 -07:00
|
|
|
wait(&status);
|
|
|
|
|
|
|
|
t1 = rdclock();
|
|
|
|
|
2009-09-04 06:36:08 -07:00
|
|
|
update_stats(&walltime_nsecs_stats, t1 - t0);
|
2009-06-13 05:57:28 -07:00
|
|
|
|
|
|
|
for (counter = 0; counter < nr_counters; counter++)
|
|
|
|
read_counter(counter);
|
|
|
|
|
|
|
|
return WEXITSTATUS(status);
|
|
|
|
}
|
|
|
|
|
2009-09-04 09:23:38 -07:00
|
|
|
static void print_noise(int counter, double avg)
|
2009-06-13 05:57:28 -07:00
|
|
|
{
|
2009-09-04 09:23:38 -07:00
|
|
|
if (run_count == 1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fprintf(stderr, " ( +- %7.3f%% )",
|
|
|
|
100 * stddev_stats(&event_res_stats[counter][0]) / avg);
|
2009-06-13 05:57:28 -07:00
|
|
|
}
|
|
|
|
|
2009-09-04 09:23:38 -07:00
|
|
|
static void nsec_printout(int counter, double avg)
|
2009-06-13 04:35:00 -07:00
|
|
|
{
|
2009-09-04 06:36:12 -07:00
|
|
|
double msecs = avg / 1e6;
|
2009-06-13 04:35:00 -07:00
|
|
|
|
2009-06-26 14:32:07 -07:00
|
|
|
fprintf(stderr, " %14.6f %-24s", msecs, event_name(counter));
|
2009-06-13 04:35:00 -07:00
|
|
|
|
2009-07-01 02:35:09 -07:00
|
|
|
if (MATCH_EVENT(SOFTWARE, SW_TASK_CLOCK, counter)) {
|
2009-09-04 06:36:12 -07:00
|
|
|
fprintf(stderr, " # %10.3f CPUs ",
|
|
|
|
avg / avg_stats(&walltime_nsecs_stats));
|
2009-06-13 04:35:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-04 09:23:38 -07:00
|
|
|
static void abs_printout(int counter, double avg)
|
2009-06-13 04:35:00 -07:00
|
|
|
{
|
2009-09-22 05:53:51 -07:00
|
|
|
double total, ratio = 0.0;
|
|
|
|
|
2009-09-04 06:36:12 -07:00
|
|
|
fprintf(stderr, " %14.0f %-24s", avg, event_name(counter));
|
2009-06-13 04:35:00 -07:00
|
|
|
|
2009-09-04 06:36:12 -07:00
|
|
|
if (MATCH_EVENT(HARDWARE, HW_INSTRUCTIONS, counter)) {
|
2009-09-22 05:53:51 -07:00
|
|
|
total = avg_stats(&runtime_cycles_stats);
|
|
|
|
|
|
|
|
if (total)
|
|
|
|
ratio = avg / total;
|
|
|
|
|
|
|
|
fprintf(stderr, " # %10.3f IPC ", ratio);
|
2009-06-13 05:57:28 -07:00
|
|
|
} else {
|
2009-09-22 05:53:51 -07:00
|
|
|
total = avg_stats(&runtime_nsecs_stats);
|
|
|
|
|
|
|
|
if (total)
|
|
|
|
ratio = 1000.0 * avg / total;
|
|
|
|
|
|
|
|
fprintf(stderr, " # %10.3f M/sec", ratio);
|
2009-06-13 04:35:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
/*
|
|
|
|
* Print out the results of a single counter:
|
|
|
|
*/
|
|
|
|
static void print_counter(int counter)
|
|
|
|
{
|
2009-09-04 09:23:38 -07:00
|
|
|
double avg = avg_stats(&event_res_stats[counter][0]);
|
2009-09-04 08:03:13 -07:00
|
|
|
int scaled = event_scaled[counter];
|
2009-05-29 00:10:54 -07:00
|
|
|
|
|
|
|
if (scaled == -1) {
|
2009-06-26 14:32:07 -07:00
|
|
|
fprintf(stderr, " %14s %-24s\n",
|
2009-05-29 00:10:54 -07:00
|
|
|
"<not counted>", event_name(counter));
|
|
|
|
return;
|
|
|
|
}
|
2009-05-29 00:10:54 -07:00
|
|
|
|
2009-06-13 04:35:00 -07:00
|
|
|
if (nsec_counter(counter))
|
2009-09-04 09:23:38 -07:00
|
|
|
nsec_printout(counter, avg);
|
2009-06-13 04:35:00 -07:00
|
|
|
else
|
2009-09-04 09:23:38 -07:00
|
|
|
abs_printout(counter, avg);
|
|
|
|
|
|
|
|
print_noise(counter, avg);
|
2009-09-04 06:36:12 -07:00
|
|
|
|
|
|
|
if (scaled) {
|
|
|
|
double avg_enabled, avg_running;
|
|
|
|
|
|
|
|
avg_enabled = avg_stats(&event_res_stats[counter][1]);
|
|
|
|
avg_running = avg_stats(&event_res_stats[counter][2]);
|
2009-05-30 03:38:51 -07:00
|
|
|
|
2009-06-29 12:50:54 -07:00
|
|
|
fprintf(stderr, " (scaled from %.2f%%)",
|
2009-09-04 06:36:12 -07:00
|
|
|
100 * avg_running / avg_enabled);
|
|
|
|
}
|
2009-06-13 04:35:00 -07:00
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
|
2009-06-13 05:57:28 -07:00
|
|
|
static void print_stat(int argc, const char **argv)
|
|
|
|
{
|
|
|
|
int i, counter;
|
|
|
|
|
2009-04-20 06:37:32 -07:00
|
|
|
fflush(stdout);
|
|
|
|
|
|
|
|
fprintf(stderr, "\n");
|
2009-06-03 10:36:07 -07:00
|
|
|
fprintf(stderr, " Performance counter stats for \'%s", argv[0]);
|
|
|
|
|
|
|
|
for (i = 1; i < argc; i++)
|
|
|
|
fprintf(stderr, " %s", argv[i]);
|
|
|
|
|
2009-06-13 05:57:28 -07:00
|
|
|
fprintf(stderr, "\'");
|
|
|
|
if (run_count > 1)
|
|
|
|
fprintf(stderr, " (%d runs)", run_count);
|
|
|
|
fprintf(stderr, ":\n\n");
|
2009-05-29 00:10:54 -07:00
|
|
|
|
2009-05-29 00:10:54 -07:00
|
|
|
for (counter = 0; counter < nr_counters; counter++)
|
|
|
|
print_counter(counter);
|
2009-04-20 06:37:32 -07:00
|
|
|
|
|
|
|
fprintf(stderr, "\n");
|
2009-06-26 21:24:32 -07:00
|
|
|
fprintf(stderr, " %14.9f seconds time elapsed",
|
2009-09-04 06:36:12 -07:00
|
|
|
avg_stats(&walltime_nsecs_stats)/1e9);
|
2009-06-26 21:24:32 -07:00
|
|
|
if (run_count > 1) {
|
|
|
|
fprintf(stderr, " ( +- %7.3f%% )",
|
2009-09-04 06:36:12 -07:00
|
|
|
100*stddev_stats(&walltime_nsecs_stats) /
|
|
|
|
avg_stats(&walltime_nsecs_stats));
|
2009-06-26 21:24:32 -07:00
|
|
|
}
|
|
|
|
fprintf(stderr, "\n\n");
|
2009-04-20 06:37:32 -07:00
|
|
|
}
|
|
|
|
|
2009-06-10 06:55:59 -07:00
|
|
|
static volatile int signr = -1;
|
|
|
|
|
2009-05-26 00:17:18 -07:00
|
|
|
static void skip_signal(int signo)
|
2009-04-20 06:37:32 -07:00
|
|
|
{
|
2009-06-10 06:55:59 -07:00
|
|
|
signr = signo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sig_atexit(void)
|
|
|
|
{
|
2009-10-03 17:35:01 -07:00
|
|
|
if (child_pid != -1)
|
|
|
|
kill(child_pid, SIGTERM);
|
|
|
|
|
2009-06-10 06:55:59 -07:00
|
|
|
if (signr == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
signal(signr, SIG_DFL);
|
|
|
|
kill(getpid(), signr);
|
2009-05-26 00:17:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static const char * const stat_usage[] = {
|
|
|
|
"perf stat [<options>] <command>",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct option options[] = {
|
|
|
|
OPT_CALLBACK('e', "event", NULL, "event",
|
2009-06-06 03:24:17 -07:00
|
|
|
"event selector. use 'perf list' to list available events",
|
|
|
|
parse_events),
|
2009-05-26 00:17:18 -07:00
|
|
|
OPT_BOOLEAN('i', "inherit", &inherit,
|
|
|
|
"child tasks inherit counters"),
|
|
|
|
OPT_INTEGER('p', "pid", &target_pid,
|
|
|
|
"stat events on existing pid"),
|
|
|
|
OPT_BOOLEAN('a', "all-cpus", &system_wide,
|
2009-06-24 05:49:34 -07:00
|
|
|
"system-wide collection from all CPUs"),
|
2009-08-07 01:18:39 -07:00
|
|
|
OPT_BOOLEAN('c', "scale", &scale,
|
2009-06-24 05:49:34 -07:00
|
|
|
"scale/normalize counters"),
|
2009-06-07 08:06:46 -07:00
|
|
|
OPT_BOOLEAN('v', "verbose", &verbose,
|
|
|
|
"be more verbose (show counter open errors, etc)"),
|
2009-06-13 05:57:28 -07:00
|
|
|
OPT_INTEGER('r', "repeat", &run_count,
|
|
|
|
"repeat command and print average + stddev (max: 100)"),
|
2009-06-26 21:10:30 -07:00
|
|
|
OPT_BOOLEAN('n', "null", &null_run,
|
|
|
|
"null run - dont start any counters"),
|
2009-05-26 00:17:18 -07:00
|
|
|
OPT_END()
|
|
|
|
};
|
|
|
|
|
2009-07-01 03:37:06 -07:00
|
|
|
int cmd_stat(int argc, const char **argv, const char *prefix __used)
|
2009-05-26 00:17:18 -07:00
|
|
|
{
|
2009-06-13 05:57:28 -07:00
|
|
|
int status;
|
|
|
|
|
2009-07-22 06:04:12 -07:00
|
|
|
argc = parse_options(argc, argv, options, stat_usage,
|
|
|
|
PARSE_OPT_STOP_AT_NON_OPTION);
|
2009-05-26 00:17:18 -07:00
|
|
|
if (!argc)
|
|
|
|
usage_with_options(stat_usage, options);
|
2009-09-04 06:36:08 -07:00
|
|
|
if (run_count <= 0)
|
2009-06-13 05:57:28 -07:00
|
|
|
usage_with_options(stat_usage, options);
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-06-27 11:19:09 -07:00
|
|
|
/* Set attrs and nr_counters if no event is selected and !null_run */
|
|
|
|
if (!null_run && !nr_counters) {
|
|
|
|
memcpy(attrs, default_attrs, sizeof(default_attrs));
|
|
|
|
nr_counters = ARRAY_SIZE(default_attrs);
|
|
|
|
}
|
2009-04-20 06:37:32 -07:00
|
|
|
|
|
|
|
nr_cpus = sysconf(_SC_NPROCESSORS_ONLN);
|
|
|
|
assert(nr_cpus <= MAX_NR_CPUS);
|
2009-07-01 03:37:06 -07:00
|
|
|
assert((int)nr_cpus >= 0);
|
2009-04-20 06:37:32 -07:00
|
|
|
|
2009-05-15 02:03:23 -07:00
|
|
|
/*
|
|
|
|
* We dont want to block the signals - that would cause
|
|
|
|
* child tasks to inherit that and Ctrl-C would not work.
|
|
|
|
* What we want is for Ctrl-C to work in the exec()-ed
|
|
|
|
* task, but being ignored by perf stat itself:
|
|
|
|
*/
|
2009-06-10 06:55:59 -07:00
|
|
|
atexit(sig_atexit);
|
2009-05-15 02:03:23 -07:00
|
|
|
signal(SIGINT, skip_signal);
|
|
|
|
signal(SIGALRM, skip_signal);
|
|
|
|
signal(SIGABRT, skip_signal);
|
|
|
|
|
2009-06-13 05:57:28 -07:00
|
|
|
status = 0;
|
|
|
|
for (run_idx = 0; run_idx < run_count; run_idx++) {
|
|
|
|
if (run_count != 1 && verbose)
|
2009-06-24 05:49:34 -07:00
|
|
|
fprintf(stderr, "[ perf stat: executing run #%d ... ]\n", run_idx + 1);
|
2009-06-13 05:57:28 -07:00
|
|
|
status = run_perf_stat(argc, argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
print_stat(argc, argv);
|
|
|
|
|
|
|
|
return status;
|
2009-04-20 06:37:32 -07:00
|
|
|
}
|