aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
authorPaul Mackerras <paulus@samba.org>2009-02-10 22:35:35 -0500
committerPaul Mackerras <paulus@samba.org>2009-02-10 23:06:59 -0500
commit0475f9ea8e2cc030298908949e0d5da9f2fc2cfe (patch)
treeeb2585d92e00ae4c7fc7e0654ffacde7e8a57e1c /kernel
parentd278c48435625cb6b7edcf6a547620768b175709 (diff)
perf_counters: allow users to count user, kernel and/or hypervisor events
Impact: new perf_counter feature This extends the perf_counter_hw_event struct with bits that specify that events in user, kernel and/or hypervisor mode should not be counted (i.e. should be excluded), and adds code to program the PMU mode selection bits accordingly on x86 and powerpc. For software counters, we don't currently have the infrastructure to distinguish which mode an event occurs in, so we currently fail the counter initialization if the setting of the hw_event.exclude_* bits would require us to distinguish. Context switches and CPU migrations are currently considered to occur in kernel mode. On x86, this changes the previous policy that only root can count kernel events. Now non-root users can count kernel events or exclude them. Non-root users still can't use NMI events, though. On x86 we don't appear to have any way to control whether hypervisor events are counted or not, so hw_event.exclude_hv is ignored. On powerpc, the selection of whether to count events in user, kernel and/or hypervisor mode is PMU-wide, not per-counter, so this adds a check that the hw_event.exclude_* settings are the same as other events on the PMU. Counters being added to a group have to have the same settings as the other hardware counters in the group. Counters and groups can only be enabled in hw_perf_group_sched_in or power_perf_enable if they have the same settings as any other counters already on the PMU. If we are not running on a hypervisor, the exclude_hv setting is ignored (by forcing it to 0) since we can't ever get any hypervisor events. Signed-off-by: Paul Mackerras <paulus@samba.org>
Diffstat (limited to 'kernel')
-rw-r--r--kernel/perf_counter.c26
1 files changed, 22 insertions, 4 deletions
diff --git a/kernel/perf_counter.c b/kernel/perf_counter.c
index 544193cbc478..89d5e3fe9700 100644
--- a/kernel/perf_counter.c
+++ b/kernel/perf_counter.c
@@ -1567,11 +1567,25 @@ sw_perf_counter_init(struct perf_counter *counter)
1567{ 1567{
1568 const struct hw_perf_counter_ops *hw_ops = NULL; 1568 const struct hw_perf_counter_ops *hw_ops = NULL;
1569 1569
1570 /*
1571 * Software counters (currently) can't in general distinguish
1572 * between user, kernel and hypervisor events.
1573 * However, context switches and cpu migrations are considered
1574 * to be kernel events, and page faults are never hypervisor
1575 * events.
1576 */
1570 switch (counter->hw_event.type) { 1577 switch (counter->hw_event.type) {
1571 case PERF_COUNT_CPU_CLOCK: 1578 case PERF_COUNT_CPU_CLOCK:
1572 hw_ops = &perf_ops_cpu_clock; 1579 if (!(counter->hw_event.exclude_user ||
1580 counter->hw_event.exclude_kernel ||
1581 counter->hw_event.exclude_hv))
1582 hw_ops = &perf_ops_cpu_clock;
1573 break; 1583 break;
1574 case PERF_COUNT_TASK_CLOCK: 1584 case PERF_COUNT_TASK_CLOCK:
1585 if (counter->hw_event.exclude_user ||
1586 counter->hw_event.exclude_kernel ||
1587 counter->hw_event.exclude_hv)
1588 break;
1575 /* 1589 /*
1576 * If the user instantiates this as a per-cpu counter, 1590 * If the user instantiates this as a per-cpu counter,
1577 * use the cpu_clock counter instead. 1591 * use the cpu_clock counter instead.
@@ -1582,13 +1596,17 @@ sw_perf_counter_init(struct perf_counter *counter)
1582 hw_ops = &perf_ops_cpu_clock; 1596 hw_ops = &perf_ops_cpu_clock;
1583 break; 1597 break;
1584 case PERF_COUNT_PAGE_FAULTS: 1598 case PERF_COUNT_PAGE_FAULTS:
1585 hw_ops = &perf_ops_page_faults; 1599 if (!(counter->hw_event.exclude_user ||
1600 counter->hw_event.exclude_kernel))
1601 hw_ops = &perf_ops_page_faults;
1586 break; 1602 break;
1587 case PERF_COUNT_CONTEXT_SWITCHES: 1603 case PERF_COUNT_CONTEXT_SWITCHES:
1588 hw_ops = &perf_ops_context_switches; 1604 if (!counter->hw_event.exclude_kernel)
1605 hw_ops = &perf_ops_context_switches;
1589 break; 1606 break;
1590 case PERF_COUNT_CPU_MIGRATIONS: 1607 case PERF_COUNT_CPU_MIGRATIONS:
1591 hw_ops = &perf_ops_cpu_migrations; 1608 if (!counter->hw_event.exclude_kernel)
1609 hw_ops = &perf_ops_cpu_migrations;
1592 break; 1610 break;
1593 default: 1611 default:
1594 break; 1612 break;