1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * Copyright (C) 2017 Etnaviv Project
3*61046927SAndroid Build Coastguard Worker * Copyright (C) 2017 Zodiac Inflight Innovations
4*61046927SAndroid Build Coastguard Worker *
5*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
6*61046927SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
7*61046927SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
8*61046927SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9*61046927SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
10*61046927SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
11*61046927SAndroid Build Coastguard Worker *
12*61046927SAndroid Build Coastguard Worker * The above copyright notice and this permission notice (including the next
13*61046927SAndroid Build Coastguard Worker * paragraph) shall be included in all copies or substantial portions of the
14*61046927SAndroid Build Coastguard Worker * Software.
15*61046927SAndroid Build Coastguard Worker *
16*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*61046927SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19*61046927SAndroid Build Coastguard Worker * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*61046927SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21*61046927SAndroid Build Coastguard Worker * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22*61046927SAndroid Build Coastguard Worker * SOFTWARE.
23*61046927SAndroid Build Coastguard Worker *
24*61046927SAndroid Build Coastguard Worker * Authors:
25*61046927SAndroid Build Coastguard Worker * Christian Gmeiner <[email protected]>
26*61046927SAndroid Build Coastguard Worker */
27*61046927SAndroid Build Coastguard Worker
28*61046927SAndroid Build Coastguard Worker #include "etnaviv_priv.h"
29*61046927SAndroid Build Coastguard Worker
etna_perfmon_query_signals(struct etna_perfmon * pm,struct etna_perfmon_domain * dom)30*61046927SAndroid Build Coastguard Worker static int etna_perfmon_query_signals(struct etna_perfmon *pm, struct etna_perfmon_domain *dom)
31*61046927SAndroid Build Coastguard Worker {
32*61046927SAndroid Build Coastguard Worker struct etna_device *dev = pm->pipe->gpu->dev;
33*61046927SAndroid Build Coastguard Worker struct drm_etnaviv_pm_signal req = {
34*61046927SAndroid Build Coastguard Worker .pipe = pm->pipe->id,
35*61046927SAndroid Build Coastguard Worker .domain = dom->id
36*61046927SAndroid Build Coastguard Worker };
37*61046927SAndroid Build Coastguard Worker
38*61046927SAndroid Build Coastguard Worker do {
39*61046927SAndroid Build Coastguard Worker struct etna_perfmon_signal *sig;
40*61046927SAndroid Build Coastguard Worker int ret;
41*61046927SAndroid Build Coastguard Worker
42*61046927SAndroid Build Coastguard Worker ret = drmCommandWriteRead(dev->fd, DRM_ETNAVIV_PM_QUERY_SIG, &req, sizeof(req));
43*61046927SAndroid Build Coastguard Worker if (ret)
44*61046927SAndroid Build Coastguard Worker break;
45*61046927SAndroid Build Coastguard Worker
46*61046927SAndroid Build Coastguard Worker sig = calloc(1, sizeof(*sig));
47*61046927SAndroid Build Coastguard Worker if (!sig)
48*61046927SAndroid Build Coastguard Worker return -ENOMEM;
49*61046927SAndroid Build Coastguard Worker
50*61046927SAndroid Build Coastguard Worker DEBUG_MSG("perfmon signal:");
51*61046927SAndroid Build Coastguard Worker DEBUG_MSG("id = %d", req.id);
52*61046927SAndroid Build Coastguard Worker DEBUG_MSG("name = %s", req.name);
53*61046927SAndroid Build Coastguard Worker
54*61046927SAndroid Build Coastguard Worker sig->domain = dom;
55*61046927SAndroid Build Coastguard Worker sig->signal = req.id;
56*61046927SAndroid Build Coastguard Worker strncpy(sig->name, req.name, sizeof(sig->name));
57*61046927SAndroid Build Coastguard Worker list_addtail(&sig->head, &dom->signals);
58*61046927SAndroid Build Coastguard Worker } while (req.iter != 0xffff);
59*61046927SAndroid Build Coastguard Worker
60*61046927SAndroid Build Coastguard Worker return 0;
61*61046927SAndroid Build Coastguard Worker }
62*61046927SAndroid Build Coastguard Worker
etna_perfmon_query_domains(struct etna_perfmon * pm)63*61046927SAndroid Build Coastguard Worker static int etna_perfmon_query_domains(struct etna_perfmon *pm)
64*61046927SAndroid Build Coastguard Worker {
65*61046927SAndroid Build Coastguard Worker struct etna_device *dev = pm->pipe->gpu->dev;
66*61046927SAndroid Build Coastguard Worker struct drm_etnaviv_pm_domain req = {
67*61046927SAndroid Build Coastguard Worker .pipe = pm->pipe->id
68*61046927SAndroid Build Coastguard Worker };
69*61046927SAndroid Build Coastguard Worker
70*61046927SAndroid Build Coastguard Worker do {
71*61046927SAndroid Build Coastguard Worker struct etna_perfmon_domain *dom;
72*61046927SAndroid Build Coastguard Worker int ret;
73*61046927SAndroid Build Coastguard Worker
74*61046927SAndroid Build Coastguard Worker ret = drmCommandWriteRead(dev->fd, DRM_ETNAVIV_PM_QUERY_DOM, &req, sizeof(req));
75*61046927SAndroid Build Coastguard Worker if (ret)
76*61046927SAndroid Build Coastguard Worker break;
77*61046927SAndroid Build Coastguard Worker
78*61046927SAndroid Build Coastguard Worker dom = calloc(1, sizeof(*dom));
79*61046927SAndroid Build Coastguard Worker if (!dom)
80*61046927SAndroid Build Coastguard Worker return -ENOMEM;
81*61046927SAndroid Build Coastguard Worker
82*61046927SAndroid Build Coastguard Worker list_inithead(&dom->signals);
83*61046927SAndroid Build Coastguard Worker dom->id = req.id;
84*61046927SAndroid Build Coastguard Worker strncpy(dom->name, req.name, sizeof(dom->name));
85*61046927SAndroid Build Coastguard Worker list_addtail(&dom->head, &pm->domains);
86*61046927SAndroid Build Coastguard Worker
87*61046927SAndroid Build Coastguard Worker DEBUG_MSG("perfmon domain:");
88*61046927SAndroid Build Coastguard Worker DEBUG_MSG("id = %d", req.id);
89*61046927SAndroid Build Coastguard Worker DEBUG_MSG("name = %s", req.name);
90*61046927SAndroid Build Coastguard Worker DEBUG_MSG("nr_signals = %d", req.nr_signals);
91*61046927SAndroid Build Coastguard Worker
92*61046927SAndroid Build Coastguard Worker /* Query all available signals for this domain. */
93*61046927SAndroid Build Coastguard Worker if (req.nr_signals > 0) {
94*61046927SAndroid Build Coastguard Worker ret = etna_perfmon_query_signals(pm, dom);
95*61046927SAndroid Build Coastguard Worker if (ret)
96*61046927SAndroid Build Coastguard Worker return ret;
97*61046927SAndroid Build Coastguard Worker }
98*61046927SAndroid Build Coastguard Worker } while (req.iter != 0xff);
99*61046927SAndroid Build Coastguard Worker
100*61046927SAndroid Build Coastguard Worker return 0;
101*61046927SAndroid Build Coastguard Worker }
102*61046927SAndroid Build Coastguard Worker
etna_perfmon_free_signals(struct etna_perfmon_domain * dom)103*61046927SAndroid Build Coastguard Worker static void etna_perfmon_free_signals(struct etna_perfmon_domain *dom)
104*61046927SAndroid Build Coastguard Worker {
105*61046927SAndroid Build Coastguard Worker struct etna_perfmon_signal *sig, *next;
106*61046927SAndroid Build Coastguard Worker
107*61046927SAndroid Build Coastguard Worker LIST_FOR_EACH_ENTRY_SAFE(sig, next, &dom->signals, head) {
108*61046927SAndroid Build Coastguard Worker list_del(&sig->head);
109*61046927SAndroid Build Coastguard Worker free(sig);
110*61046927SAndroid Build Coastguard Worker }
111*61046927SAndroid Build Coastguard Worker }
112*61046927SAndroid Build Coastguard Worker
etna_perfmon_free_domains(struct etna_perfmon * pm)113*61046927SAndroid Build Coastguard Worker static void etna_perfmon_free_domains(struct etna_perfmon *pm)
114*61046927SAndroid Build Coastguard Worker {
115*61046927SAndroid Build Coastguard Worker struct etna_perfmon_domain *dom, *next;
116*61046927SAndroid Build Coastguard Worker
117*61046927SAndroid Build Coastguard Worker LIST_FOR_EACH_ENTRY_SAFE(dom, next, &pm->domains, head) {
118*61046927SAndroid Build Coastguard Worker etna_perfmon_free_signals(dom);
119*61046927SAndroid Build Coastguard Worker list_del(&dom->head);
120*61046927SAndroid Build Coastguard Worker free(dom);
121*61046927SAndroid Build Coastguard Worker }
122*61046927SAndroid Build Coastguard Worker }
123*61046927SAndroid Build Coastguard Worker
etna_perfmon_create(struct etna_pipe * pipe)124*61046927SAndroid Build Coastguard Worker struct etna_perfmon *etna_perfmon_create(struct etna_pipe *pipe)
125*61046927SAndroid Build Coastguard Worker {
126*61046927SAndroid Build Coastguard Worker struct etna_perfmon *pm;
127*61046927SAndroid Build Coastguard Worker int ret;
128*61046927SAndroid Build Coastguard Worker
129*61046927SAndroid Build Coastguard Worker pm = calloc(1, sizeof(*pm));
130*61046927SAndroid Build Coastguard Worker if (!pm) {
131*61046927SAndroid Build Coastguard Worker ERROR_MSG("allocation failed");
132*61046927SAndroid Build Coastguard Worker return NULL;
133*61046927SAndroid Build Coastguard Worker }
134*61046927SAndroid Build Coastguard Worker
135*61046927SAndroid Build Coastguard Worker list_inithead(&pm->domains);
136*61046927SAndroid Build Coastguard Worker pm->pipe = pipe;
137*61046927SAndroid Build Coastguard Worker
138*61046927SAndroid Build Coastguard Worker /* query all available domains and sources for this device */
139*61046927SAndroid Build Coastguard Worker ret = etna_perfmon_query_domains(pm);
140*61046927SAndroid Build Coastguard Worker if (ret)
141*61046927SAndroid Build Coastguard Worker goto fail;
142*61046927SAndroid Build Coastguard Worker
143*61046927SAndroid Build Coastguard Worker return pm;
144*61046927SAndroid Build Coastguard Worker
145*61046927SAndroid Build Coastguard Worker fail:
146*61046927SAndroid Build Coastguard Worker etna_perfmon_del(pm);
147*61046927SAndroid Build Coastguard Worker return NULL;
148*61046927SAndroid Build Coastguard Worker }
149*61046927SAndroid Build Coastguard Worker
etna_perfmon_del(struct etna_perfmon * pm)150*61046927SAndroid Build Coastguard Worker void etna_perfmon_del(struct etna_perfmon *pm)
151*61046927SAndroid Build Coastguard Worker {
152*61046927SAndroid Build Coastguard Worker if (!pm)
153*61046927SAndroid Build Coastguard Worker return;
154*61046927SAndroid Build Coastguard Worker
155*61046927SAndroid Build Coastguard Worker etna_perfmon_free_domains(pm);
156*61046927SAndroid Build Coastguard Worker free(pm);
157*61046927SAndroid Build Coastguard Worker }
158*61046927SAndroid Build Coastguard Worker
etna_perfmon_get_dom_by_name(struct etna_perfmon * pm,const char * name)159*61046927SAndroid Build Coastguard Worker struct etna_perfmon_domain *etna_perfmon_get_dom_by_name(struct etna_perfmon *pm, const char *name)
160*61046927SAndroid Build Coastguard Worker {
161*61046927SAndroid Build Coastguard Worker struct etna_perfmon_domain *dom;
162*61046927SAndroid Build Coastguard Worker
163*61046927SAndroid Build Coastguard Worker if (pm) {
164*61046927SAndroid Build Coastguard Worker LIST_FOR_EACH_ENTRY(dom, &pm->domains, head) {
165*61046927SAndroid Build Coastguard Worker if (!strcmp(dom->name, name))
166*61046927SAndroid Build Coastguard Worker return dom;
167*61046927SAndroid Build Coastguard Worker }
168*61046927SAndroid Build Coastguard Worker }
169*61046927SAndroid Build Coastguard Worker
170*61046927SAndroid Build Coastguard Worker return NULL;
171*61046927SAndroid Build Coastguard Worker }
172*61046927SAndroid Build Coastguard Worker
etna_perfmon_get_sig_by_name(struct etna_perfmon_domain * dom,const char * name)173*61046927SAndroid Build Coastguard Worker struct etna_perfmon_signal *etna_perfmon_get_sig_by_name(struct etna_perfmon_domain *dom, const char *name)
174*61046927SAndroid Build Coastguard Worker {
175*61046927SAndroid Build Coastguard Worker struct etna_perfmon_signal *signal;
176*61046927SAndroid Build Coastguard Worker
177*61046927SAndroid Build Coastguard Worker if (dom) {
178*61046927SAndroid Build Coastguard Worker LIST_FOR_EACH_ENTRY(signal, &dom->signals, head) {
179*61046927SAndroid Build Coastguard Worker if (!strcmp(signal->name, name))
180*61046927SAndroid Build Coastguard Worker return signal;
181*61046927SAndroid Build Coastguard Worker }
182*61046927SAndroid Build Coastguard Worker }
183*61046927SAndroid Build Coastguard Worker
184*61046927SAndroid Build Coastguard Worker return NULL;
185*61046927SAndroid Build Coastguard Worker }
186