xref: /aosp_15_r20/external/igt-gpu-tools/overlay/rc6.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker  * Copyright © 2013 Intel Corporation
3*d83cc019SAndroid Build Coastguard Worker  *
4*d83cc019SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
5*d83cc019SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
6*d83cc019SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
7*d83cc019SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*d83cc019SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
9*d83cc019SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
10*d83cc019SAndroid Build Coastguard Worker  *
11*d83cc019SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
12*d83cc019SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
13*d83cc019SAndroid Build Coastguard Worker  * Software.
14*d83cc019SAndroid Build Coastguard Worker  *
15*d83cc019SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*d83cc019SAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*d83cc019SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18*d83cc019SAndroid Build Coastguard Worker  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*d83cc019SAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*d83cc019SAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21*d83cc019SAndroid Build Coastguard Worker  * IN THE SOFTWARE.
22*d83cc019SAndroid Build Coastguard Worker  *
23*d83cc019SAndroid Build Coastguard Worker  */
24*d83cc019SAndroid Build Coastguard Worker 
25*d83cc019SAndroid Build Coastguard Worker #include <sys/stat.h>
26*d83cc019SAndroid Build Coastguard Worker #include <stdlib.h>
27*d83cc019SAndroid Build Coastguard Worker #include <stdio.h>
28*d83cc019SAndroid Build Coastguard Worker #include <string.h>
29*d83cc019SAndroid Build Coastguard Worker #include <unistd.h>
30*d83cc019SAndroid Build Coastguard Worker #include <fcntl.h>
31*d83cc019SAndroid Build Coastguard Worker #include <time.h>
32*d83cc019SAndroid Build Coastguard Worker #include <errno.h>
33*d83cc019SAndroid Build Coastguard Worker 
34*d83cc019SAndroid Build Coastguard Worker #include "igt_perf.h"
35*d83cc019SAndroid Build Coastguard Worker 
36*d83cc019SAndroid Build Coastguard Worker #include "rc6.h"
37*d83cc019SAndroid Build Coastguard Worker 
rc6_init(struct rc6 * rc6)38*d83cc019SAndroid Build Coastguard Worker int rc6_init(struct rc6 *rc6)
39*d83cc019SAndroid Build Coastguard Worker {
40*d83cc019SAndroid Build Coastguard Worker 	memset(rc6, 0, sizeof(*rc6));
41*d83cc019SAndroid Build Coastguard Worker 
42*d83cc019SAndroid Build Coastguard Worker 	rc6->fd = perf_i915_open(I915_PMU_RC6_RESIDENCY);
43*d83cc019SAndroid Build Coastguard Worker 	if (rc6->fd < 0) {
44*d83cc019SAndroid Build Coastguard Worker 		struct stat st;
45*d83cc019SAndroid Build Coastguard Worker 		if (stat("/sys/class/drm/card0/power", &st) < 0)
46*d83cc019SAndroid Build Coastguard Worker 			return rc6->error = errno;
47*d83cc019SAndroid Build Coastguard Worker 	}
48*d83cc019SAndroid Build Coastguard Worker 
49*d83cc019SAndroid Build Coastguard Worker 	return 0;
50*d83cc019SAndroid Build Coastguard Worker }
51*d83cc019SAndroid Build Coastguard Worker 
file_to_u64(const char * path)52*d83cc019SAndroid Build Coastguard Worker static uint64_t file_to_u64(const char *path)
53*d83cc019SAndroid Build Coastguard Worker {
54*d83cc019SAndroid Build Coastguard Worker 	char buf[4096];
55*d83cc019SAndroid Build Coastguard Worker 	int fd, len;
56*d83cc019SAndroid Build Coastguard Worker 
57*d83cc019SAndroid Build Coastguard Worker 	fd = open(path, 0);
58*d83cc019SAndroid Build Coastguard Worker 	if (fd < 0)
59*d83cc019SAndroid Build Coastguard Worker 		return -1;
60*d83cc019SAndroid Build Coastguard Worker 
61*d83cc019SAndroid Build Coastguard Worker 	len = read(fd, buf, sizeof(buf)-1);
62*d83cc019SAndroid Build Coastguard Worker 	close(fd);
63*d83cc019SAndroid Build Coastguard Worker 
64*d83cc019SAndroid Build Coastguard Worker 	if (len < 0)
65*d83cc019SAndroid Build Coastguard Worker 		return -1;
66*d83cc019SAndroid Build Coastguard Worker 
67*d83cc019SAndroid Build Coastguard Worker 	buf[len] = '\0';
68*d83cc019SAndroid Build Coastguard Worker 
69*d83cc019SAndroid Build Coastguard Worker 	return strtoull(buf, 0, 0);
70*d83cc019SAndroid Build Coastguard Worker }
71*d83cc019SAndroid Build Coastguard Worker 
clock_ms_to_u64(void)72*d83cc019SAndroid Build Coastguard Worker static uint64_t clock_ms_to_u64(void)
73*d83cc019SAndroid Build Coastguard Worker {
74*d83cc019SAndroid Build Coastguard Worker 	struct timespec tv;
75*d83cc019SAndroid Build Coastguard Worker 
76*d83cc019SAndroid Build Coastguard Worker 	if (clock_gettime(CLOCK_MONOTONIC, &tv) < 0)
77*d83cc019SAndroid Build Coastguard Worker 		return 0;
78*d83cc019SAndroid Build Coastguard Worker 
79*d83cc019SAndroid Build Coastguard Worker 	return (uint64_t)tv.tv_sec * 1000 + tv.tv_nsec / 1000000;
80*d83cc019SAndroid Build Coastguard Worker }
81*d83cc019SAndroid Build Coastguard Worker 
rc6_update(struct rc6 * rc6)82*d83cc019SAndroid Build Coastguard Worker int rc6_update(struct rc6 *rc6)
83*d83cc019SAndroid Build Coastguard Worker {
84*d83cc019SAndroid Build Coastguard Worker 	struct rc6_stat *s = &rc6->stat[rc6->count++&1];
85*d83cc019SAndroid Build Coastguard Worker 	struct rc6_stat *d = &rc6->stat[rc6->count&1];
86*d83cc019SAndroid Build Coastguard Worker 	uint64_t d_time, d_rc6, d_rc6p, d_rc6pp;
87*d83cc019SAndroid Build Coastguard Worker 
88*d83cc019SAndroid Build Coastguard Worker 	if (rc6->error)
89*d83cc019SAndroid Build Coastguard Worker 		return rc6->error;
90*d83cc019SAndroid Build Coastguard Worker 
91*d83cc019SAndroid Build Coastguard Worker 	if (rc6->fd < 0) {
92*d83cc019SAndroid Build Coastguard Worker 		struct stat st;
93*d83cc019SAndroid Build Coastguard Worker 
94*d83cc019SAndroid Build Coastguard Worker 		if (stat("/sys/class/drm/card0/power/rc6_residency_ms", &st) < 0)
95*d83cc019SAndroid Build Coastguard Worker 			return rc6->error = ENOENT;
96*d83cc019SAndroid Build Coastguard Worker 
97*d83cc019SAndroid Build Coastguard Worker 		s->rc6_residency = file_to_u64("/sys/class/drm/card0/power/rc6_residency_ms");
98*d83cc019SAndroid Build Coastguard Worker 		s->rc6p_residency = file_to_u64("/sys/class/drm/card0/power/rc6p_residency_ms");
99*d83cc019SAndroid Build Coastguard Worker 		s->rc6pp_residency = file_to_u64("/sys/class/drm/card0/power/rc6pp_residency_ms");
100*d83cc019SAndroid Build Coastguard Worker 		s->timestamp = clock_ms_to_u64();
101*d83cc019SAndroid Build Coastguard Worker 	} else {
102*d83cc019SAndroid Build Coastguard Worker 		uint64_t data[2];
103*d83cc019SAndroid Build Coastguard Worker 
104*d83cc019SAndroid Build Coastguard Worker 		if (read(rc6->fd, data, sizeof(data)) < sizeof(data))
105*d83cc019SAndroid Build Coastguard Worker 			return rc6->error = errno;
106*d83cc019SAndroid Build Coastguard Worker 
107*d83cc019SAndroid Build Coastguard Worker 		s->timestamp = data[1] / 1e6;
108*d83cc019SAndroid Build Coastguard Worker 		s->rc6_residency = data[0] / 1e6;
109*d83cc019SAndroid Build Coastguard Worker 	}
110*d83cc019SAndroid Build Coastguard Worker 
111*d83cc019SAndroid Build Coastguard Worker 	if (rc6->count == 1)
112*d83cc019SAndroid Build Coastguard Worker 		return EAGAIN;
113*d83cc019SAndroid Build Coastguard Worker 
114*d83cc019SAndroid Build Coastguard Worker 	d_time = s->timestamp - d->timestamp;
115*d83cc019SAndroid Build Coastguard Worker 	if (d_time == 0) {
116*d83cc019SAndroid Build Coastguard Worker 		rc6->count--;
117*d83cc019SAndroid Build Coastguard Worker 		return EAGAIN;
118*d83cc019SAndroid Build Coastguard Worker 	}
119*d83cc019SAndroid Build Coastguard Worker 
120*d83cc019SAndroid Build Coastguard Worker 	d_rc6 = s->rc6_residency - d->rc6_residency;
121*d83cc019SAndroid Build Coastguard Worker 	rc6->rc6 = 100 * d_rc6 / d_time;
122*d83cc019SAndroid Build Coastguard Worker 
123*d83cc019SAndroid Build Coastguard Worker 	d_rc6p = s->rc6p_residency - d->rc6p_residency;
124*d83cc019SAndroid Build Coastguard Worker 	rc6->rc6p = 100 * d_rc6p / d_time;
125*d83cc019SAndroid Build Coastguard Worker 
126*d83cc019SAndroid Build Coastguard Worker 	d_rc6pp = s->rc6pp_residency - d->rc6pp_residency;
127*d83cc019SAndroid Build Coastguard Worker 	rc6->rc6pp = 100 * d_rc6pp / d_time;
128*d83cc019SAndroid Build Coastguard Worker 
129*d83cc019SAndroid Build Coastguard Worker 	rc6->rc6_combined = 100 * (d_rc6 + d_rc6p + d_rc6pp) / d_time;
130*d83cc019SAndroid Build Coastguard Worker 	return 0;
131*d83cc019SAndroid Build Coastguard Worker }
132