[Intel-gfx] [PATCH] drm/i915/pmu: Only enumerate available counters in sysfs
Chris Wilson
chris at chris-wilson.co.uk
Fri Dec 22 14:52:05 UTC 2017
Quoting Tvrtko Ursulin (2017-12-21 17:13:16)
> From: Tvrtko Ursulin <tvrtko.ursulin at intel.com>
>
> Switch over to dynamically creating device attributes, which are in turn
> used by the perf core to expose available counters in sysfs.
>
> This way we do not expose counters which are not avaiable on the current
> platform, and are so more consistent between what we reply to open
> attempts via the perf_event_open(2), and what is discoverable in sysfs.
>
> Signed-off-by: Tvrtko Ursulin <tvrtko.ursulin at intel.com>
> ---
> +#define __event(__config, __name, __unit) \
> +{ \
> + .config = (__config), \
> + .name = (__name), \
> + .unit = (__unit), \
> +}
> +
> +#define __engine_event(__sample, __name) \
> +{ \
> + .sample = (__sample), \
> + .name = (__name), \
> +}
> +
> +#define __i915_attr(__p, __name, __config) \
> +{ \
> + (__p)->attr.attr.name = (__name); \
> + (__p)->attr.attr.mode = 0444; \
> + (__p)->attr.show = i915_pmu_event_show; \
> + (__p)->val = (__config); \
> +}
> +
> +#define __pmu_attr(__p, __name, __str) \
> +{ \
> + (__p)->attr.attr.name = (__name); \
> + (__p)->attr.attr.mode = 0444; \
> + (__p)->attr.show = perf_event_sysfs_show; \
> + (__p)->event_str = (__str); \
> +}
> +
> +static struct attribute **
> +create_event_attributes(struct drm_i915_private *i915)
> +{
> + static const struct {
> + u64 config;
> + const char *name;
> + const char *unit;
> + } events[] = {
> + __event(I915_PMU_ACTUAL_FREQUENCY, "actual-frequency", "MHz"),
> + __event(I915_PMU_REQUESTED_FREQUENCY, "requested-frequency", "MHz"),
> + __event(I915_PMU_INTERRUPTS, "interrupts", NULL),
> + __event(I915_PMU_RC6_RESIDENCY, "rc6-residency", "ns"),
> + };
> + static const struct {
> + enum drm_i915_pmu_engine_sample sample;
> + char *name;
> + } engine_events[] = {
> + __engine_event(I915_SAMPLE_BUSY, "busy"),
> + __engine_event(I915_SAMPLE_SEMA, "sema"),
> + __engine_event(I915_SAMPLE_WAIT, "wait"),
Are these macros that useful? vs { I915_SAMPLE_BUSY, "busy" } and
{ I915_PMU_ACTUAL_FREQUENCY, "actual-frequency", "MHz" },
> + };
> + unsigned int count = 0;
> + struct perf_pmu_events_attr *pmu_attr, *pmu_p;
> + struct i915_ext_attribute *i915_attr, *i915_p;
> + struct intel_engine_cs *engine;
> + struct attribute **attr, **p;
> + enum intel_engine_id id;
> + unsigned int i;
> +
> + /* Count how many counters we will be exposing. */
> + for (i = 0; i < ARRAY_SIZE(events); i++) {
> + if (!config_status(i915, events[i].config))
> + count++;
> + }
> +
> + for_each_engine(engine, i915, id) {
> + for (i = 0; i < ARRAY_SIZE(engine_events); i++) {
> + if (!engine_event_status(engine,
> + engine_events[i].sample))
> + count++;
> + }
> + }
> +
> + /* Allocate attribute objects and table. */
> + i915_attr = kzalloc(count * sizeof(*i915_attr), GFP_KERNEL);
> + if (!i915_attr)
> + return NULL;
> +
> + pmu_attr = kzalloc(count * sizeof(*pmu_attr), GFP_KERNEL);
> + if (!pmu_attr) {
> + kfree(i915_attr);
> + return NULL;
> + }
> +
> + /* Max one pointer of each attribute type plus a termination entry. */
> + attr = kzalloc((count * 2 + 1) * sizeof(attr), GFP_KERNEL);
> + if (!attr) {
> + kfree(pmu_attr);
> + kfree(i915_attr);
> + return NULL;
Joonas wants you to feed through the same error unwind.
> + }
> +
> + i915_p = i915_attr;
> + pmu_p = pmu_attr;
> + p = attr;
i915_attr_iter, pmu_attr_iter and attr_iter?
> + /* Initialize supported non-engine counters. */
> + for (i = 0; i < ARRAY_SIZE(events); i++) {
> + char *str;
> +
> + if (config_status(i915, events[i].config))
> + continue;
> +
> + str = kstrdup(events[i].name, GFP_KERNEL);
> + if (!str)
> + goto err;
> +
> + __i915_attr(i915_p, str, events[i].config);
*attr_iter++ = &i915_attr_iter->attr.attr;
*i915_attr_iter++ = (struct i915_ext_attribute) {
.attr.attr.name = str,
.attr.attr.mode = 0444,
.attr.show = i915_pmu_event_show,
.val = events[i].config,
}
?
Mere suggestions. Doesn't look that bad
-Chris
More information about the Intel-gfx
mailing list