From: Tvrtko Ursulin <tvrtko.ursulin@xxxxxxxxx> Wire up to the RAPL PMU for GPU energy readings. The only complication is that we have to add code to parse: # cat /sys/devices/power/events/energy-gpu.scale 2.3283064365386962890625e-10 v2: Link with -lm. v3: strtod can handle scientific notation, even though my initial reading of the man page did not spot that. (Chris Wilson) Signed-off-by: Tvrtko Ursulin <tvrtko.ursulin@xxxxxxxxx> --- lib/igt_perf.c | 16 +++++-- lib/igt_perf.h | 1 + overlay/Makefile.am | 2 +- overlay/power.c | 127 ++++++++++++++++++++++++++++++++++++---------------- overlay/power.h | 2 + 5 files changed, 104 insertions(+), 44 deletions(-) diff --git a/lib/igt_perf.c b/lib/igt_perf.c index 208474302fcc..0221461e918f 100644 --- a/lib/igt_perf.c +++ b/lib/igt_perf.c @@ -27,11 +27,12 @@ uint64_t i915_type_id(void) return strtoull(buf, NULL, 0); } -static int _perf_open(uint64_t config, int group, uint64_t format) +static int +_perf_open(uint64_t type, uint64_t config, int group, uint64_t format) { struct perf_event_attr attr = { }; - attr.type = i915_type_id(); + attr.type = type; if (attr.type == 0) return -ENOENT; @@ -46,11 +47,18 @@ static int _perf_open(uint64_t config, int group, uint64_t format) int perf_i915_open(uint64_t config) { - return _perf_open(config, -1, PERF_FORMAT_TOTAL_TIME_ENABLED); + return _perf_open(i915_type_id(), config, -1, + PERF_FORMAT_TOTAL_TIME_ENABLED); } int perf_i915_open_group(uint64_t config, int group) { - return _perf_open(config, group, + return _perf_open(i915_type_id(), config, group, PERF_FORMAT_TOTAL_TIME_ENABLED | PERF_FORMAT_GROUP); } + +int igt_perf_open(uint64_t type, uint64_t config) +{ + return _perf_open(type, config, -1, + PERF_FORMAT_TOTAL_TIME_ENABLED); +} diff --git a/lib/igt_perf.h b/lib/igt_perf.h index 285823786324..b1f525739c69 100644 --- a/lib/igt_perf.h +++ b/lib/igt_perf.h @@ -99,5 +99,6 @@ perf_event_open(struct perf_event_attr *attr, uint64_t i915_type_id(void); int perf_i915_open(uint64_t config); int perf_i915_open_group(uint64_t config, int group); +int igt_perf_open(uint64_t type, uint64_t config); #endif /* I915_PERF_H */ diff --git a/overlay/Makefile.am b/overlay/Makefile.am index cefde2d040f8..f49f54ac3590 100644 --- a/overlay/Makefile.am +++ b/overlay/Makefile.am @@ -63,7 +63,7 @@ intel_gpu_overlay_SOURCES += \ intel_gpu_overlay_SOURCES += $(both_x11_sources) -intel_gpu_overlay_LDADD = $(LDADD) -lrt +intel_gpu_overlay_LDADD = $(LDADD) -lrt -lm EXTRA_DIST= \ README \ diff --git a/overlay/power.c b/overlay/power.c index 805f4ca7805c..9ac90fde8786 100644 --- a/overlay/power.c +++ b/overlay/power.c @@ -30,60 +30,107 @@ #include <fcntl.h> #include <time.h> #include <errno.h> +#include <ctype.h> +#include <math.h> #include "igt_perf.h" #include "power.h" #include "debugfs.h" -/* XXX Is this exposed through RAPL? */ - -int power_init(struct power *power) +static int +filename_to_buf(const char *filename, char *buf, unsigned int bufsize) { - char buf[4096]; - int fd, len; - - memset(power, 0, sizeof(*power)); - - power->fd = -1; + int fd; + ssize_t ret; - sprintf(buf, "%s/i915_energy_uJ", debugfs_dri_path); - fd = open(buf, 0); + fd = open(filename, O_RDONLY); if (fd < 0) - return power->error = errno; + return -1; - len = read(fd, buf, sizeof(buf)); + ret = read(fd, buf, bufsize - 1); close(fd); + if (ret < 1) + return -1; - if (len < 0) - return power->error = errno; - - buf[len] = '\0'; - if (strtoull(buf, 0, 0) == 0) - return power->error = EINVAL; + buf[ret] = '\0'; return 0; } -static uint64_t file_to_u64(const char *name) +static uint64_t filename_to_u64(const char *filename, int base) { - char buf[4096]; - int fd, len; + char buf[64], *b; - sprintf(buf, "%s/%s", debugfs_dri_path, name); - fd = open(buf, 0); - if (fd < 0) + if (filename_to_buf(filename, buf, sizeof(buf))) return 0; - len = read(fd, buf, sizeof(buf)-1); - close(fd); + /* + * Handle both single integer and key=value formats by skipping + * leading non-digits. + */ + b = buf; + while (*b && !isdigit(*b)) + b++; + + return strtoull(b, NULL, base); +} + +static uint64_t debugfs_file_to_u64(const char *name) +{ + char buf[1024]; + + snprintf(buf, sizeof(buf), "%s/%s", debugfs_dri_path, name); + + return filename_to_u64(buf, 0); +} + +static uint64_t rapl_type_id(void) +{ + return filename_to_u64("/sys/devices/power/type", 10); +} + +static uint64_t rapl_gpu_power(void) +{ + return filename_to_u64("/sys/devices/power/events/energy-gpu", 0); +} - if (len < 0) +static double filename_to_double(const char *filename) +{ + char buf[64]; + + if (filename_to_buf(filename, buf, sizeof(buf))) return 0; - buf[len] = '\0'; + return strtod(buf, NULL); +} + +static double rapl_gpu_power_scale(void) +{ + return filename_to_double("/sys/devices/power/events/energy-gpu.scale"); +} + +int power_init(struct power *power) +{ + uint64_t val; + + memset(power, 0, sizeof(*power)); + + power->fd = igt_perf_open(rapl_type_id(), rapl_gpu_power()); + if (power->fd >= 0) { + power->rapl_scale = rapl_gpu_power_scale(); + + if (power->rapl_scale != NAN) { + power->rapl_scale *= 1e3; /* from nano to micro */ + return 0; + } + } + + val = debugfs_file_to_u64("i915_energy_uJ"); + if (val == 0) + return power->error = EINVAL; - return strtoull(buf, 0, 0); + return 0; } static uint64_t clock_ms_to_u64(void) @@ -93,30 +140,30 @@ static uint64_t clock_ms_to_u64(void) if (clock_gettime(CLOCK_MONOTONIC, &tv) < 0) return 0; - return (uint64_t)tv.tv_sec * 1000 + tv.tv_nsec / 1000000; + return (uint64_t)tv.tv_sec * 1e3 + tv.tv_nsec / 1e6; } int power_update(struct power *power) { - struct power_stat *s = &power->stat[power->count++&1]; - struct power_stat *d = &power->stat[power->count&1]; + struct power_stat *s = &power->stat[power->count++ & 1]; + struct power_stat *d = &power->stat[power->count & 1]; uint64_t d_time; if (power->error) return power->error; - if (power->fd != -1) { + if (power->fd >= 0) { uint64_t data[2]; int len; len = read(power->fd, data, sizeof(data)); - if (len < 0) + if (len != sizeof(data)) return power->error = errno; - s->energy = data[0]; - s->timestamp = data[1] / (1000*1000); + s->energy = llround((double)data[0] * power->rapl_scale); + s->timestamp = data[1] / 1e6; } else { - s->energy = file_to_u64("i915_energy_uJ"); + s->energy = debugfs_file_to_u64("i915_energy_uJ") / 1e3; s->timestamp = clock_ms_to_u64(); } @@ -124,7 +171,9 @@ int power_update(struct power *power) return EAGAIN; d_time = s->timestamp - d->timestamp; - power->power_mW = (s->energy - d->energy) / d_time; + power->power_mW = round((double)(s->energy - d->energy) * + (1e3f / d_time)); power->new_sample = 1; + return 0; } diff --git a/overlay/power.h b/overlay/power.h index bf8346ce46b4..28abfc32234b 100644 --- a/overlay/power.h +++ b/overlay/power.h @@ -39,6 +39,8 @@ struct power { int new_sample; uint64_t power_mW; + + double rapl_scale; }; int power_init(struct power *power); -- 2.9.5 _______________________________________________ Intel-gfx mailing list Intel-gfx@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/intel-gfx