Commit b00233b5 authored by Andi Kleen's avatar Andi Kleen Committed by Ingo Molnar

perf/x86: Export some PMU attributes in caps/ directory

It can be difficult to figure out for user programs what features
the x86 CPU PMU driver actually supports. Currently it requires
grepping in dmesg, but dmesg is not always available.

This adds a caps directory to /sys/bus/event_source/devices/cpu/,
similar to the caps already used on intel_pt, which can be used to
discover the available capabilities cleanly.

Three capabilities are defined:

 - pmu_name:	Underlying CPU name known to the driver
 - max_precise:	Max precise level supported
 - branches:	Known depth of LBR.

Example:

  % grep . /sys/bus/event_source/devices/cpu/caps/*
  /sys/bus/event_source/devices/cpu/caps/branches:32
  /sys/bus/event_source/devices/cpu/caps/max_precise:3
  /sys/bus/event_source/devices/cpu/caps/pmu_name:skylake
Signed-off-by: default avatarAndi Kleen <ak@linux.intel.com>
Signed-off-by: default avatarPeter Zijlstra (Intel) <peterz@infradead.org>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Link: http://lkml.kernel.org/r/20170822185201.9261-3-andi@firstfloor.orgSigned-off-by: default avatarIngo Molnar <mingo@kernel.org>
parent a5df70c3
...@@ -487,22 +487,28 @@ static inline int precise_br_compat(struct perf_event *event) ...@@ -487,22 +487,28 @@ static inline int precise_br_compat(struct perf_event *event)
return m == b; return m == b;
} }
int x86_pmu_hw_config(struct perf_event *event) int x86_pmu_max_precise(void)
{ {
if (event->attr.precise_ip) { int precise = 0;
int precise = 0;
/* Support for constant skid */
if (x86_pmu.pebs_active && !x86_pmu.pebs_broken) {
precise++;
/* Support for constant skid */ /* Support for IP fixup */
if (x86_pmu.pebs_active && !x86_pmu.pebs_broken) { if (x86_pmu.lbr_nr || x86_pmu.intel_cap.pebs_format >= 2)
precise++; precise++;
/* Support for IP fixup */ if (x86_pmu.pebs_prec_dist)
if (x86_pmu.lbr_nr || x86_pmu.intel_cap.pebs_format >= 2) precise++;
precise++; }
return precise;
}
if (x86_pmu.pebs_prec_dist) int x86_pmu_hw_config(struct perf_event *event)
precise++; {
} if (event->attr.precise_ip) {
int precise = x86_pmu_max_precise();
if (event->attr.precise_ip > precise) if (event->attr.precise_ip > precise)
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -1752,6 +1758,10 @@ ssize_t x86_event_sysfs_show(char *page, u64 config, u64 event) ...@@ -1752,6 +1758,10 @@ ssize_t x86_event_sysfs_show(char *page, u64 config, u64 event)
static struct attribute_group x86_pmu_attr_group; static struct attribute_group x86_pmu_attr_group;
static struct attribute_group x86_pmu_caps_group = {
.name = "caps",
};
static int __init init_hw_perf_events(void) static int __init init_hw_perf_events(void)
{ {
struct x86_pmu_quirk *quirk; struct x86_pmu_quirk *quirk;
...@@ -1798,6 +1808,7 @@ static int __init init_hw_perf_events(void) ...@@ -1798,6 +1808,7 @@ static int __init init_hw_perf_events(void)
0, x86_pmu.num_counters, 0, 0); 0, x86_pmu.num_counters, 0, 0);
x86_pmu_format_group.attrs = x86_pmu.format_attrs; x86_pmu_format_group.attrs = x86_pmu.format_attrs;
x86_pmu_caps_group.attrs = x86_pmu.caps_attrs;
if (x86_pmu.event_attrs) if (x86_pmu.event_attrs)
x86_pmu_events_group.attrs = x86_pmu.event_attrs; x86_pmu_events_group.attrs = x86_pmu.event_attrs;
...@@ -2217,6 +2228,7 @@ static const struct attribute_group *x86_pmu_attr_groups[] = { ...@@ -2217,6 +2228,7 @@ static const struct attribute_group *x86_pmu_attr_groups[] = {
&x86_pmu_attr_group, &x86_pmu_attr_group,
&x86_pmu_format_group, &x86_pmu_format_group,
&x86_pmu_events_group, &x86_pmu_events_group,
&x86_pmu_caps_group,
NULL, NULL,
}; };
......
...@@ -3795,6 +3795,46 @@ static ssize_t freeze_on_smi_store(struct device *cdev, ...@@ -3795,6 +3795,46 @@ static ssize_t freeze_on_smi_store(struct device *cdev,
static DEVICE_ATTR_RW(freeze_on_smi); static DEVICE_ATTR_RW(freeze_on_smi);
static ssize_t branches_show(struct device *cdev,
struct device_attribute *attr,
char *buf)
{
return snprintf(buf, PAGE_SIZE, "%d\n", x86_pmu.lbr_nr);
}
static DEVICE_ATTR_RO(branches);
static struct attribute *lbr_attrs[] = {
&dev_attr_branches.attr,
NULL
};
static char pmu_name_str[30];
static ssize_t pmu_name_show(struct device *cdev,
struct device_attribute *attr,
char *buf)
{
return snprintf(buf, PAGE_SIZE, "%s\n", pmu_name_str);
}
static DEVICE_ATTR_RO(pmu_name);
static ssize_t max_precise_show(struct device *cdev,
struct device_attribute *attr,
char *buf)
{
return snprintf(buf, PAGE_SIZE, "%d\n", x86_pmu_max_precise());
}
static DEVICE_ATTR_RO(max_precise);
static struct attribute *intel_pmu_caps_attrs[] = {
&dev_attr_pmu_name.attr,
&dev_attr_max_precise.attr,
NULL
};
static struct attribute *intel_pmu_attrs[] = { static struct attribute *intel_pmu_attrs[] = {
&dev_attr_freeze_on_smi.attr, &dev_attr_freeze_on_smi.attr,
NULL, NULL,
...@@ -3810,6 +3850,7 @@ __init int intel_pmu_init(void) ...@@ -3810,6 +3850,7 @@ __init int intel_pmu_init(void)
struct extra_reg *er; struct extra_reg *er;
int version, i; int version, i;
struct attribute **extra_attr = NULL; struct attribute **extra_attr = NULL;
char *name;
if (!cpu_has(&boot_cpu_data, X86_FEATURE_ARCH_PERFMON)) { if (!cpu_has(&boot_cpu_data, X86_FEATURE_ARCH_PERFMON)) {
switch (boot_cpu_data.x86) { switch (boot_cpu_data.x86) {
...@@ -3877,6 +3918,7 @@ __init int intel_pmu_init(void) ...@@ -3877,6 +3918,7 @@ __init int intel_pmu_init(void)
switch (boot_cpu_data.x86_model) { switch (boot_cpu_data.x86_model) {
case INTEL_FAM6_CORE_YONAH: case INTEL_FAM6_CORE_YONAH:
pr_cont("Core events, "); pr_cont("Core events, ");
name = "core";
break; break;
case INTEL_FAM6_CORE2_MEROM: case INTEL_FAM6_CORE2_MEROM:
...@@ -3892,6 +3934,7 @@ __init int intel_pmu_init(void) ...@@ -3892,6 +3934,7 @@ __init int intel_pmu_init(void)
x86_pmu.event_constraints = intel_core2_event_constraints; x86_pmu.event_constraints = intel_core2_event_constraints;
x86_pmu.pebs_constraints = intel_core2_pebs_event_constraints; x86_pmu.pebs_constraints = intel_core2_pebs_event_constraints;
pr_cont("Core2 events, "); pr_cont("Core2 events, ");
name = "core2";
break; break;
case INTEL_FAM6_NEHALEM: case INTEL_FAM6_NEHALEM:
...@@ -3924,6 +3967,7 @@ __init int intel_pmu_init(void) ...@@ -3924,6 +3967,7 @@ __init int intel_pmu_init(void)
extra_attr = nhm_format_attr; extra_attr = nhm_format_attr;
pr_cont("Nehalem events, "); pr_cont("Nehalem events, ");
name = "nehalem";
break; break;
case INTEL_FAM6_ATOM_PINEVIEW: case INTEL_FAM6_ATOM_PINEVIEW:
...@@ -3940,6 +3984,7 @@ __init int intel_pmu_init(void) ...@@ -3940,6 +3984,7 @@ __init int intel_pmu_init(void)
x86_pmu.pebs_constraints = intel_atom_pebs_event_constraints; x86_pmu.pebs_constraints = intel_atom_pebs_event_constraints;
x86_pmu.pebs_aliases = intel_pebs_aliases_core2; x86_pmu.pebs_aliases = intel_pebs_aliases_core2;
pr_cont("Atom events, "); pr_cont("Atom events, ");
name = "bonnell";
break; break;
case INTEL_FAM6_ATOM_SILVERMONT1: case INTEL_FAM6_ATOM_SILVERMONT1:
...@@ -3959,6 +4004,7 @@ __init int intel_pmu_init(void) ...@@ -3959,6 +4004,7 @@ __init int intel_pmu_init(void)
x86_pmu.cpu_events = slm_events_attrs; x86_pmu.cpu_events = slm_events_attrs;
extra_attr = slm_format_attr; extra_attr = slm_format_attr;
pr_cont("Silvermont events, "); pr_cont("Silvermont events, ");
name = "silvermont";
break; break;
case INTEL_FAM6_ATOM_GOLDMONT: case INTEL_FAM6_ATOM_GOLDMONT:
...@@ -3985,6 +4031,7 @@ __init int intel_pmu_init(void) ...@@ -3985,6 +4031,7 @@ __init int intel_pmu_init(void)
x86_pmu.cpu_events = glm_events_attrs; x86_pmu.cpu_events = glm_events_attrs;
extra_attr = slm_format_attr; extra_attr = slm_format_attr;
pr_cont("Goldmont events, "); pr_cont("Goldmont events, ");
name = "goldmont";
break; break;
case INTEL_FAM6_ATOM_GEMINI_LAKE: case INTEL_FAM6_ATOM_GEMINI_LAKE:
...@@ -4012,6 +4059,7 @@ __init int intel_pmu_init(void) ...@@ -4012,6 +4059,7 @@ __init int intel_pmu_init(void)
event_attr_td_total_slots_scale_glm.event_str = "4"; event_attr_td_total_slots_scale_glm.event_str = "4";
extra_attr = slm_format_attr; extra_attr = slm_format_attr;
pr_cont("Goldmont plus events, "); pr_cont("Goldmont plus events, ");
name = "goldmont_plus";
break; break;
case INTEL_FAM6_WESTMERE: case INTEL_FAM6_WESTMERE:
...@@ -4042,6 +4090,7 @@ __init int intel_pmu_init(void) ...@@ -4042,6 +4090,7 @@ __init int intel_pmu_init(void)
intel_pmu_pebs_data_source_nhm(); intel_pmu_pebs_data_source_nhm();
extra_attr = nhm_format_attr; extra_attr = nhm_format_attr;
pr_cont("Westmere events, "); pr_cont("Westmere events, ");
name = "westmere";
break; break;
case INTEL_FAM6_SANDYBRIDGE: case INTEL_FAM6_SANDYBRIDGE:
...@@ -4080,6 +4129,7 @@ __init int intel_pmu_init(void) ...@@ -4080,6 +4129,7 @@ __init int intel_pmu_init(void)
extra_attr = nhm_format_attr; extra_attr = nhm_format_attr;
pr_cont("SandyBridge events, "); pr_cont("SandyBridge events, ");
name = "sandybridge";
break; break;
case INTEL_FAM6_IVYBRIDGE: case INTEL_FAM6_IVYBRIDGE:
...@@ -4116,6 +4166,7 @@ __init int intel_pmu_init(void) ...@@ -4116,6 +4166,7 @@ __init int intel_pmu_init(void)
extra_attr = nhm_format_attr; extra_attr = nhm_format_attr;
pr_cont("IvyBridge events, "); pr_cont("IvyBridge events, ");
name = "ivybridge";
break; break;
...@@ -4146,6 +4197,7 @@ __init int intel_pmu_init(void) ...@@ -4146,6 +4197,7 @@ __init int intel_pmu_init(void)
extra_attr = boot_cpu_has(X86_FEATURE_RTM) ? extra_attr = boot_cpu_has(X86_FEATURE_RTM) ?
hsw_format_attr : nhm_format_attr; hsw_format_attr : nhm_format_attr;
pr_cont("Haswell events, "); pr_cont("Haswell events, ");
name = "haswell";
break; break;
case INTEL_FAM6_BROADWELL_CORE: case INTEL_FAM6_BROADWELL_CORE:
...@@ -4184,6 +4236,7 @@ __init int intel_pmu_init(void) ...@@ -4184,6 +4236,7 @@ __init int intel_pmu_init(void)
extra_attr = boot_cpu_has(X86_FEATURE_RTM) ? extra_attr = boot_cpu_has(X86_FEATURE_RTM) ?
hsw_format_attr : nhm_format_attr; hsw_format_attr : nhm_format_attr;
pr_cont("Broadwell events, "); pr_cont("Broadwell events, ");
name = "broadwell";
break; break;
case INTEL_FAM6_XEON_PHI_KNL: case INTEL_FAM6_XEON_PHI_KNL:
...@@ -4203,6 +4256,7 @@ __init int intel_pmu_init(void) ...@@ -4203,6 +4256,7 @@ __init int intel_pmu_init(void)
x86_pmu.flags |= PMU_FL_NO_HT_SHARING; x86_pmu.flags |= PMU_FL_NO_HT_SHARING;
extra_attr = slm_format_attr; extra_attr = slm_format_attr;
pr_cont("Knights Landing/Mill events, "); pr_cont("Knights Landing/Mill events, ");
name = "knights-landing";
break; break;
case INTEL_FAM6_SKYLAKE_MOBILE: case INTEL_FAM6_SKYLAKE_MOBILE:
...@@ -4239,6 +4293,7 @@ __init int intel_pmu_init(void) ...@@ -4239,6 +4293,7 @@ __init int intel_pmu_init(void)
intel_pmu_pebs_data_source_skl( intel_pmu_pebs_data_source_skl(
boot_cpu_data.x86_model == INTEL_FAM6_SKYLAKE_X); boot_cpu_data.x86_model == INTEL_FAM6_SKYLAKE_X);
pr_cont("Skylake events, "); pr_cont("Skylake events, ");
name = "skylake";
break; break;
default: default:
...@@ -4246,6 +4301,7 @@ __init int intel_pmu_init(void) ...@@ -4246,6 +4301,7 @@ __init int intel_pmu_init(void)
case 1: case 1:
x86_pmu.event_constraints = intel_v1_event_constraints; x86_pmu.event_constraints = intel_v1_event_constraints;
pr_cont("generic architected perfmon v1, "); pr_cont("generic architected perfmon v1, ");
name = "generic_arch_v1";
break; break;
default: default:
/* /*
...@@ -4253,10 +4309,13 @@ __init int intel_pmu_init(void) ...@@ -4253,10 +4309,13 @@ __init int intel_pmu_init(void)
*/ */
x86_pmu.event_constraints = intel_gen_event_constraints; x86_pmu.event_constraints = intel_gen_event_constraints;
pr_cont("generic architected perfmon, "); pr_cont("generic architected perfmon, ");
name = "generic_arch_v2+";
break; break;
} }
} }
snprintf(pmu_name_str, sizeof pmu_name_str, "%s", name);
if (version >= 2 && extra_attr) { if (version >= 2 && extra_attr) {
x86_pmu.format_attrs = merge_attr(intel_arch3_formats_attr, x86_pmu.format_attrs = merge_attr(intel_arch3_formats_attr,
extra_attr); extra_attr);
...@@ -4309,8 +4368,13 @@ __init int intel_pmu_init(void) ...@@ -4309,8 +4368,13 @@ __init int intel_pmu_init(void)
x86_pmu.lbr_nr = 0; x86_pmu.lbr_nr = 0;
} }
if (x86_pmu.lbr_nr) x86_pmu.caps_attrs = intel_pmu_caps_attrs;
if (x86_pmu.lbr_nr) {
x86_pmu.caps_attrs = merge_attr(x86_pmu.caps_attrs, lbr_attrs);
pr_cont("%d-deep LBR, ", x86_pmu.lbr_nr); pr_cont("%d-deep LBR, ", x86_pmu.lbr_nr);
}
/* /*
* Access extra MSR may cause #GP under certain circumstances. * Access extra MSR may cause #GP under certain circumstances.
* E.g. KVM doesn't support offcore event * E.g. KVM doesn't support offcore event
......
...@@ -558,6 +558,7 @@ struct x86_pmu { ...@@ -558,6 +558,7 @@ struct x86_pmu {
int attr_rdpmc; int attr_rdpmc;
struct attribute **format_attrs; struct attribute **format_attrs;
struct attribute **event_attrs; struct attribute **event_attrs;
struct attribute **caps_attrs;
ssize_t (*events_sysfs_show)(char *page, u64 config); ssize_t (*events_sysfs_show)(char *page, u64 config);
struct attribute **cpu_events; struct attribute **cpu_events;
...@@ -742,6 +743,8 @@ int x86_reserve_hardware(void); ...@@ -742,6 +743,8 @@ int x86_reserve_hardware(void);
void x86_release_hardware(void); void x86_release_hardware(void);
int x86_pmu_max_precise(void);
void hw_perf_lbr_event_destroy(struct perf_event *event); void hw_perf_lbr_event_destroy(struct perf_event *event);
int x86_setup_perfctr(struct perf_event *event); int x86_setup_perfctr(struct perf_event *event);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment