Loading...
Note: File does not exist in v3.1.
1// SPDX-License-Identifier: GPL-2.0
2/*
3 * System Control and Management Interface (SCMI) Powercap Protocol
4 *
5 * Copyright (C) 2022 ARM Ltd.
6 */
7
8#define pr_fmt(fmt) "SCMI Notifications POWERCAP - " fmt
9
10#include <linux/bitfield.h>
11#include <linux/io.h>
12#include <linux/module.h>
13#include <linux/scmi_protocol.h>
14
15#include <trace/events/scmi.h>
16
17#include "protocols.h"
18#include "notify.h"
19
20enum scmi_powercap_protocol_cmd {
21 POWERCAP_DOMAIN_ATTRIBUTES = 0x3,
22 POWERCAP_CAP_GET = 0x4,
23 POWERCAP_CAP_SET = 0x5,
24 POWERCAP_PAI_GET = 0x6,
25 POWERCAP_PAI_SET = 0x7,
26 POWERCAP_DOMAIN_NAME_GET = 0x8,
27 POWERCAP_MEASUREMENTS_GET = 0x9,
28 POWERCAP_CAP_NOTIFY = 0xa,
29 POWERCAP_MEASUREMENTS_NOTIFY = 0xb,
30 POWERCAP_DESCRIBE_FASTCHANNEL = 0xc,
31};
32
33enum {
34 POWERCAP_FC_CAP,
35 POWERCAP_FC_PAI,
36 POWERCAP_FC_MAX,
37};
38
39struct scmi_msg_resp_powercap_domain_attributes {
40 __le32 attributes;
41#define SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY(x) ((x) & BIT(31))
42#define SUPPORTS_POWERCAP_MEASUREMENTS_CHANGE_NOTIFY(x) ((x) & BIT(30))
43#define SUPPORTS_ASYNC_POWERCAP_CAP_SET(x) ((x) & BIT(29))
44#define SUPPORTS_EXTENDED_NAMES(x) ((x) & BIT(28))
45#define SUPPORTS_POWERCAP_CAP_CONFIGURATION(x) ((x) & BIT(27))
46#define SUPPORTS_POWERCAP_MONITORING(x) ((x) & BIT(26))
47#define SUPPORTS_POWERCAP_PAI_CONFIGURATION(x) ((x) & BIT(25))
48#define SUPPORTS_POWERCAP_FASTCHANNELS(x) ((x) & BIT(22))
49#define POWERCAP_POWER_UNIT(x) \
50 (FIELD_GET(GENMASK(24, 23), (x)))
51#define SUPPORTS_POWER_UNITS_MW(x) \
52 (POWERCAP_POWER_UNIT(x) == 0x2)
53#define SUPPORTS_POWER_UNITS_UW(x) \
54 (POWERCAP_POWER_UNIT(x) == 0x1)
55 u8 name[SCMI_SHORT_NAME_MAX_SIZE];
56 __le32 min_pai;
57 __le32 max_pai;
58 __le32 pai_step;
59 __le32 min_power_cap;
60 __le32 max_power_cap;
61 __le32 power_cap_step;
62 __le32 sustainable_power;
63 __le32 accuracy;
64 __le32 parent_id;
65};
66
67struct scmi_msg_powercap_set_cap_or_pai {
68 __le32 domain;
69 __le32 flags;
70#define CAP_SET_ASYNC BIT(1)
71#define CAP_SET_IGNORE_DRESP BIT(0)
72 __le32 value;
73};
74
75struct scmi_msg_resp_powercap_cap_set_complete {
76 __le32 domain;
77 __le32 power_cap;
78};
79
80struct scmi_msg_resp_powercap_meas_get {
81 __le32 power;
82 __le32 pai;
83};
84
85struct scmi_msg_powercap_notify_cap {
86 __le32 domain;
87 __le32 notify_enable;
88};
89
90struct scmi_msg_powercap_notify_thresh {
91 __le32 domain;
92 __le32 notify_enable;
93 __le32 power_thresh_low;
94 __le32 power_thresh_high;
95};
96
97struct scmi_powercap_cap_changed_notify_payld {
98 __le32 agent_id;
99 __le32 domain_id;
100 __le32 power_cap;
101 __le32 pai;
102};
103
104struct scmi_powercap_meas_changed_notify_payld {
105 __le32 agent_id;
106 __le32 domain_id;
107 __le32 power;
108};
109
110struct scmi_powercap_state {
111 bool meas_notif_enabled;
112 u64 thresholds;
113#define THRESH_LOW(p, id) \
114 (lower_32_bits((p)->states[(id)].thresholds))
115#define THRESH_HIGH(p, id) \
116 (upper_32_bits((p)->states[(id)].thresholds))
117};
118
119struct powercap_info {
120 u32 version;
121 int num_domains;
122 struct scmi_powercap_state *states;
123 struct scmi_powercap_info *powercaps;
124};
125
126static enum scmi_powercap_protocol_cmd evt_2_cmd[] = {
127 POWERCAP_CAP_NOTIFY,
128 POWERCAP_MEASUREMENTS_NOTIFY,
129};
130
131static int scmi_powercap_notify(const struct scmi_protocol_handle *ph,
132 u32 domain, int message_id, bool enable);
133
134static int
135scmi_powercap_attributes_get(const struct scmi_protocol_handle *ph,
136 struct powercap_info *pi)
137{
138 int ret;
139 struct scmi_xfer *t;
140
141 ret = ph->xops->xfer_get_init(ph, PROTOCOL_ATTRIBUTES, 0,
142 sizeof(u32), &t);
143 if (ret)
144 return ret;
145
146 ret = ph->xops->do_xfer(ph, t);
147 if (!ret) {
148 u32 attributes;
149
150 attributes = get_unaligned_le32(t->rx.buf);
151 pi->num_domains = FIELD_GET(GENMASK(15, 0), attributes);
152 }
153
154 ph->xops->xfer_put(ph, t);
155 return ret;
156}
157
158static inline int
159scmi_powercap_validate(unsigned int min_val, unsigned int max_val,
160 unsigned int step_val, bool configurable)
161{
162 if (!min_val || !max_val)
163 return -EPROTO;
164
165 if ((configurable && min_val == max_val) ||
166 (!configurable && min_val != max_val))
167 return -EPROTO;
168
169 if (min_val != max_val && !step_val)
170 return -EPROTO;
171
172 return 0;
173}
174
175static int
176scmi_powercap_domain_attributes_get(const struct scmi_protocol_handle *ph,
177 struct powercap_info *pinfo, u32 domain)
178{
179 int ret;
180 u32 flags;
181 struct scmi_xfer *t;
182 struct scmi_powercap_info *dom_info = pinfo->powercaps + domain;
183 struct scmi_msg_resp_powercap_domain_attributes *resp;
184
185 ret = ph->xops->xfer_get_init(ph, POWERCAP_DOMAIN_ATTRIBUTES,
186 sizeof(domain), sizeof(*resp), &t);
187 if (ret)
188 return ret;
189
190 put_unaligned_le32(domain, t->tx.buf);
191 resp = t->rx.buf;
192
193 ret = ph->xops->do_xfer(ph, t);
194 if (!ret) {
195 flags = le32_to_cpu(resp->attributes);
196
197 dom_info->id = domain;
198 dom_info->notify_powercap_cap_change =
199 SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY(flags);
200 dom_info->notify_powercap_measurement_change =
201 SUPPORTS_POWERCAP_MEASUREMENTS_CHANGE_NOTIFY(flags);
202 dom_info->async_powercap_cap_set =
203 SUPPORTS_ASYNC_POWERCAP_CAP_SET(flags);
204 dom_info->powercap_cap_config =
205 SUPPORTS_POWERCAP_CAP_CONFIGURATION(flags);
206 dom_info->powercap_monitoring =
207 SUPPORTS_POWERCAP_MONITORING(flags);
208 dom_info->powercap_pai_config =
209 SUPPORTS_POWERCAP_PAI_CONFIGURATION(flags);
210 dom_info->powercap_scale_mw =
211 SUPPORTS_POWER_UNITS_MW(flags);
212 dom_info->powercap_scale_uw =
213 SUPPORTS_POWER_UNITS_UW(flags);
214 dom_info->fastchannels =
215 SUPPORTS_POWERCAP_FASTCHANNELS(flags);
216
217 strscpy(dom_info->name, resp->name, SCMI_SHORT_NAME_MAX_SIZE);
218
219 dom_info->min_pai = le32_to_cpu(resp->min_pai);
220 dom_info->max_pai = le32_to_cpu(resp->max_pai);
221 dom_info->pai_step = le32_to_cpu(resp->pai_step);
222 ret = scmi_powercap_validate(dom_info->min_pai,
223 dom_info->max_pai,
224 dom_info->pai_step,
225 dom_info->powercap_pai_config);
226 if (ret) {
227 dev_err(ph->dev,
228 "Platform reported inconsistent PAI config for domain %d - %s\n",
229 dom_info->id, dom_info->name);
230 goto clean;
231 }
232
233 dom_info->min_power_cap = le32_to_cpu(resp->min_power_cap);
234 dom_info->max_power_cap = le32_to_cpu(resp->max_power_cap);
235 dom_info->power_cap_step = le32_to_cpu(resp->power_cap_step);
236 ret = scmi_powercap_validate(dom_info->min_power_cap,
237 dom_info->max_power_cap,
238 dom_info->power_cap_step,
239 dom_info->powercap_cap_config);
240 if (ret) {
241 dev_err(ph->dev,
242 "Platform reported inconsistent CAP config for domain %d - %s\n",
243 dom_info->id, dom_info->name);
244 goto clean;
245 }
246
247 dom_info->sustainable_power =
248 le32_to_cpu(resp->sustainable_power);
249 dom_info->accuracy = le32_to_cpu(resp->accuracy);
250
251 dom_info->parent_id = le32_to_cpu(resp->parent_id);
252 if (dom_info->parent_id != SCMI_POWERCAP_ROOT_ZONE_ID &&
253 (dom_info->parent_id >= pinfo->num_domains ||
254 dom_info->parent_id == dom_info->id)) {
255 dev_err(ph->dev,
256 "Platform reported inconsistent parent ID for domain %d - %s\n",
257 dom_info->id, dom_info->name);
258 ret = -ENODEV;
259 }
260 }
261
262clean:
263 ph->xops->xfer_put(ph, t);
264
265 /*
266 * If supported overwrite short name with the extended one;
267 * on error just carry on and use already provided short name.
268 */
269 if (!ret && SUPPORTS_EXTENDED_NAMES(flags))
270 ph->hops->extended_name_get(ph, POWERCAP_DOMAIN_NAME_GET,
271 domain, dom_info->name,
272 SCMI_MAX_STR_SIZE);
273
274 return ret;
275}
276
277static int scmi_powercap_num_domains_get(const struct scmi_protocol_handle *ph)
278{
279 struct powercap_info *pi = ph->get_priv(ph);
280
281 return pi->num_domains;
282}
283
284static const struct scmi_powercap_info *
285scmi_powercap_dom_info_get(const struct scmi_protocol_handle *ph, u32 domain_id)
286{
287 struct powercap_info *pi = ph->get_priv(ph);
288
289 if (domain_id >= pi->num_domains)
290 return NULL;
291
292 return pi->powercaps + domain_id;
293}
294
295static int scmi_powercap_xfer_cap_get(const struct scmi_protocol_handle *ph,
296 u32 domain_id, u32 *power_cap)
297{
298 int ret;
299 struct scmi_xfer *t;
300
301 ret = ph->xops->xfer_get_init(ph, POWERCAP_CAP_GET, sizeof(u32),
302 sizeof(u32), &t);
303 if (ret)
304 return ret;
305
306 put_unaligned_le32(domain_id, t->tx.buf);
307 ret = ph->xops->do_xfer(ph, t);
308 if (!ret)
309 *power_cap = get_unaligned_le32(t->rx.buf);
310
311 ph->xops->xfer_put(ph, t);
312
313 return ret;
314}
315
316static int scmi_powercap_cap_get(const struct scmi_protocol_handle *ph,
317 u32 domain_id, u32 *power_cap)
318{
319 struct scmi_powercap_info *dom;
320 struct powercap_info *pi = ph->get_priv(ph);
321
322 if (!power_cap || domain_id >= pi->num_domains)
323 return -EINVAL;
324
325 dom = pi->powercaps + domain_id;
326 if (dom->fc_info && dom->fc_info[POWERCAP_FC_CAP].get_addr) {
327 *power_cap = ioread32(dom->fc_info[POWERCAP_FC_CAP].get_addr);
328 trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_CAP_GET,
329 domain_id, *power_cap, 0);
330 return 0;
331 }
332
333 return scmi_powercap_xfer_cap_get(ph, domain_id, power_cap);
334}
335
336static int scmi_powercap_xfer_cap_set(const struct scmi_protocol_handle *ph,
337 const struct scmi_powercap_info *pc,
338 u32 power_cap, bool ignore_dresp)
339{
340 int ret;
341 struct scmi_xfer *t;
342 struct scmi_msg_powercap_set_cap_or_pai *msg;
343
344 ret = ph->xops->xfer_get_init(ph, POWERCAP_CAP_SET,
345 sizeof(*msg), 0, &t);
346 if (ret)
347 return ret;
348
349 msg = t->tx.buf;
350 msg->domain = cpu_to_le32(pc->id);
351 msg->flags =
352 cpu_to_le32(FIELD_PREP(CAP_SET_ASYNC, !!pc->async_powercap_cap_set) |
353 FIELD_PREP(CAP_SET_IGNORE_DRESP, !!ignore_dresp));
354 msg->value = cpu_to_le32(power_cap);
355
356 if (!pc->async_powercap_cap_set || ignore_dresp) {
357 ret = ph->xops->do_xfer(ph, t);
358 } else {
359 ret = ph->xops->do_xfer_with_response(ph, t);
360 if (!ret) {
361 struct scmi_msg_resp_powercap_cap_set_complete *resp;
362
363 resp = t->rx.buf;
364 if (le32_to_cpu(resp->domain) == pc->id)
365 dev_dbg(ph->dev,
366 "Powercap ID %d CAP set async to %u\n",
367 pc->id,
368 get_unaligned_le32(&resp->power_cap));
369 else
370 ret = -EPROTO;
371 }
372 }
373
374 ph->xops->xfer_put(ph, t);
375 return ret;
376}
377
378static int scmi_powercap_cap_set(const struct scmi_protocol_handle *ph,
379 u32 domain_id, u32 power_cap,
380 bool ignore_dresp)
381{
382 const struct scmi_powercap_info *pc;
383
384 pc = scmi_powercap_dom_info_get(ph, domain_id);
385 if (!pc || !pc->powercap_cap_config || !power_cap ||
386 power_cap < pc->min_power_cap ||
387 power_cap > pc->max_power_cap)
388 return -EINVAL;
389
390 if (pc->fc_info && pc->fc_info[POWERCAP_FC_CAP].set_addr) {
391 struct scmi_fc_info *fci = &pc->fc_info[POWERCAP_FC_CAP];
392
393 iowrite32(power_cap, fci->set_addr);
394 ph->hops->fastchannel_db_ring(fci->set_db);
395 trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_CAP_SET,
396 domain_id, power_cap, 0);
397 return 0;
398 }
399
400 return scmi_powercap_xfer_cap_set(ph, pc, power_cap, ignore_dresp);
401}
402
403static int scmi_powercap_xfer_pai_get(const struct scmi_protocol_handle *ph,
404 u32 domain_id, u32 *pai)
405{
406 int ret;
407 struct scmi_xfer *t;
408
409 ret = ph->xops->xfer_get_init(ph, POWERCAP_PAI_GET, sizeof(u32),
410 sizeof(u32), &t);
411 if (ret)
412 return ret;
413
414 put_unaligned_le32(domain_id, t->tx.buf);
415 ret = ph->xops->do_xfer(ph, t);
416 if (!ret)
417 *pai = get_unaligned_le32(t->rx.buf);
418
419 ph->xops->xfer_put(ph, t);
420
421 return ret;
422}
423
424static int scmi_powercap_pai_get(const struct scmi_protocol_handle *ph,
425 u32 domain_id, u32 *pai)
426{
427 struct scmi_powercap_info *dom;
428 struct powercap_info *pi = ph->get_priv(ph);
429
430 if (!pai || domain_id >= pi->num_domains)
431 return -EINVAL;
432
433 dom = pi->powercaps + domain_id;
434 if (dom->fc_info && dom->fc_info[POWERCAP_FC_PAI].get_addr) {
435 *pai = ioread32(dom->fc_info[POWERCAP_FC_PAI].get_addr);
436 trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_PAI_GET,
437 domain_id, *pai, 0);
438 return 0;
439 }
440
441 return scmi_powercap_xfer_pai_get(ph, domain_id, pai);
442}
443
444static int scmi_powercap_xfer_pai_set(const struct scmi_protocol_handle *ph,
445 u32 domain_id, u32 pai)
446{
447 int ret;
448 struct scmi_xfer *t;
449 struct scmi_msg_powercap_set_cap_or_pai *msg;
450
451 ret = ph->xops->xfer_get_init(ph, POWERCAP_PAI_SET,
452 sizeof(*msg), 0, &t);
453 if (ret)
454 return ret;
455
456 msg = t->tx.buf;
457 msg->domain = cpu_to_le32(domain_id);
458 msg->flags = cpu_to_le32(0);
459 msg->value = cpu_to_le32(pai);
460
461 ret = ph->xops->do_xfer(ph, t);
462
463 ph->xops->xfer_put(ph, t);
464 return ret;
465}
466
467static int scmi_powercap_pai_set(const struct scmi_protocol_handle *ph,
468 u32 domain_id, u32 pai)
469{
470 const struct scmi_powercap_info *pc;
471
472 pc = scmi_powercap_dom_info_get(ph, domain_id);
473 if (!pc || !pc->powercap_pai_config || !pai ||
474 pai < pc->min_pai || pai > pc->max_pai)
475 return -EINVAL;
476
477 if (pc->fc_info && pc->fc_info[POWERCAP_FC_PAI].set_addr) {
478 struct scmi_fc_info *fci = &pc->fc_info[POWERCAP_FC_PAI];
479
480 trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_PAI_SET,
481 domain_id, pai, 0);
482 iowrite32(pai, fci->set_addr);
483 ph->hops->fastchannel_db_ring(fci->set_db);
484 return 0;
485 }
486
487 return scmi_powercap_xfer_pai_set(ph, domain_id, pai);
488}
489
490static int scmi_powercap_measurements_get(const struct scmi_protocol_handle *ph,
491 u32 domain_id, u32 *average_power,
492 u32 *pai)
493{
494 int ret;
495 struct scmi_xfer *t;
496 struct scmi_msg_resp_powercap_meas_get *resp;
497 const struct scmi_powercap_info *pc;
498
499 pc = scmi_powercap_dom_info_get(ph, domain_id);
500 if (!pc || !pc->powercap_monitoring || !pai || !average_power)
501 return -EINVAL;
502
503 ret = ph->xops->xfer_get_init(ph, POWERCAP_MEASUREMENTS_GET,
504 sizeof(u32), sizeof(*resp), &t);
505 if (ret)
506 return ret;
507
508 resp = t->rx.buf;
509 put_unaligned_le32(domain_id, t->tx.buf);
510 ret = ph->xops->do_xfer(ph, t);
511 if (!ret) {
512 *average_power = le32_to_cpu(resp->power);
513 *pai = le32_to_cpu(resp->pai);
514 }
515
516 ph->xops->xfer_put(ph, t);
517 return ret;
518}
519
520static int
521scmi_powercap_measurements_threshold_get(const struct scmi_protocol_handle *ph,
522 u32 domain_id, u32 *power_thresh_low,
523 u32 *power_thresh_high)
524{
525 struct powercap_info *pi = ph->get_priv(ph);
526
527 if (!power_thresh_low || !power_thresh_high ||
528 domain_id >= pi->num_domains)
529 return -EINVAL;
530
531 *power_thresh_low = THRESH_LOW(pi, domain_id);
532 *power_thresh_high = THRESH_HIGH(pi, domain_id);
533
534 return 0;
535}
536
537static int
538scmi_powercap_measurements_threshold_set(const struct scmi_protocol_handle *ph,
539 u32 domain_id, u32 power_thresh_low,
540 u32 power_thresh_high)
541{
542 int ret = 0;
543 struct powercap_info *pi = ph->get_priv(ph);
544
545 if (domain_id >= pi->num_domains ||
546 power_thresh_low > power_thresh_high)
547 return -EINVAL;
548
549 /* Anything to do ? */
550 if (THRESH_LOW(pi, domain_id) == power_thresh_low &&
551 THRESH_HIGH(pi, domain_id) == power_thresh_high)
552 return ret;
553
554 pi->states[domain_id].thresholds =
555 (FIELD_PREP(GENMASK_ULL(31, 0), power_thresh_low) |
556 FIELD_PREP(GENMASK_ULL(63, 32), power_thresh_high));
557
558 /* Update thresholds if notification already enabled */
559 if (pi->states[domain_id].meas_notif_enabled)
560 ret = scmi_powercap_notify(ph, domain_id,
561 POWERCAP_MEASUREMENTS_NOTIFY,
562 true);
563
564 return ret;
565}
566
567static const struct scmi_powercap_proto_ops powercap_proto_ops = {
568 .num_domains_get = scmi_powercap_num_domains_get,
569 .info_get = scmi_powercap_dom_info_get,
570 .cap_get = scmi_powercap_cap_get,
571 .cap_set = scmi_powercap_cap_set,
572 .pai_get = scmi_powercap_pai_get,
573 .pai_set = scmi_powercap_pai_set,
574 .measurements_get = scmi_powercap_measurements_get,
575 .measurements_threshold_set = scmi_powercap_measurements_threshold_set,
576 .measurements_threshold_get = scmi_powercap_measurements_threshold_get,
577};
578
579static void scmi_powercap_domain_init_fc(const struct scmi_protocol_handle *ph,
580 u32 domain, struct scmi_fc_info **p_fc)
581{
582 struct scmi_fc_info *fc;
583
584 fc = devm_kcalloc(ph->dev, POWERCAP_FC_MAX, sizeof(*fc), GFP_KERNEL);
585 if (!fc)
586 return;
587
588 ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL,
589 POWERCAP_CAP_SET, 4, domain,
590 &fc[POWERCAP_FC_CAP].set_addr,
591 &fc[POWERCAP_FC_CAP].set_db);
592
593 ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL,
594 POWERCAP_CAP_GET, 4, domain,
595 &fc[POWERCAP_FC_CAP].get_addr, NULL);
596
597 ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL,
598 POWERCAP_PAI_SET, 4, domain,
599 &fc[POWERCAP_FC_PAI].set_addr,
600 &fc[POWERCAP_FC_PAI].set_db);
601
602 ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL,
603 POWERCAP_PAI_GET, 4, domain,
604 &fc[POWERCAP_FC_PAI].get_addr, NULL);
605
606 *p_fc = fc;
607}
608
609static int scmi_powercap_notify(const struct scmi_protocol_handle *ph,
610 u32 domain, int message_id, bool enable)
611{
612 int ret;
613 struct scmi_xfer *t;
614
615 switch (message_id) {
616 case POWERCAP_CAP_NOTIFY:
617 {
618 struct scmi_msg_powercap_notify_cap *notify;
619
620 ret = ph->xops->xfer_get_init(ph, message_id,
621 sizeof(*notify), 0, &t);
622 if (ret)
623 return ret;
624
625 notify = t->tx.buf;
626 notify->domain = cpu_to_le32(domain);
627 notify->notify_enable = cpu_to_le32(enable ? BIT(0) : 0);
628 break;
629 }
630 case POWERCAP_MEASUREMENTS_NOTIFY:
631 {
632 u32 low, high;
633 struct scmi_msg_powercap_notify_thresh *notify;
634
635 /*
636 * Note that we have to pick the most recently configured
637 * thresholds to build a proper POWERCAP_MEASUREMENTS_NOTIFY
638 * enable request and we fail, complaining, if no thresholds
639 * were ever set, since this is an indication the API has been
640 * used wrongly.
641 */
642 ret = scmi_powercap_measurements_threshold_get(ph, domain,
643 &low, &high);
644 if (ret)
645 return ret;
646
647 if (enable && !low && !high) {
648 dev_err(ph->dev,
649 "Invalid Measurements Notify thresholds: %u/%u\n",
650 low, high);
651 return -EINVAL;
652 }
653
654 ret = ph->xops->xfer_get_init(ph, message_id,
655 sizeof(*notify), 0, &t);
656 if (ret)
657 return ret;
658
659 notify = t->tx.buf;
660 notify->domain = cpu_to_le32(domain);
661 notify->notify_enable = cpu_to_le32(enable ? BIT(0) : 0);
662 notify->power_thresh_low = cpu_to_le32(low);
663 notify->power_thresh_high = cpu_to_le32(high);
664 break;
665 }
666 default:
667 return -EINVAL;
668 }
669
670 ret = ph->xops->do_xfer(ph, t);
671
672 ph->xops->xfer_put(ph, t);
673 return ret;
674}
675
676static int
677scmi_powercap_set_notify_enabled(const struct scmi_protocol_handle *ph,
678 u8 evt_id, u32 src_id, bool enable)
679{
680 int ret, cmd_id;
681 struct powercap_info *pi = ph->get_priv(ph);
682
683 if (evt_id >= ARRAY_SIZE(evt_2_cmd) || src_id >= pi->num_domains)
684 return -EINVAL;
685
686 cmd_id = evt_2_cmd[evt_id];
687 ret = scmi_powercap_notify(ph, src_id, cmd_id, enable);
688 if (ret)
689 pr_debug("FAIL_ENABLED - evt[%X] dom[%d] - ret:%d\n",
690 evt_id, src_id, ret);
691 else if (cmd_id == POWERCAP_MEASUREMENTS_NOTIFY)
692 /*
693 * On success save the current notification enabled state, so
694 * as to be able to properly update the notification thresholds
695 * when they are modified on a domain for which measurement
696 * notifications were currently enabled.
697 *
698 * This is needed because the SCMI Notification core machinery
699 * and API does not support passing per-notification custom
700 * arguments at callback registration time.
701 *
702 * Note that this can be done here with a simple flag since the
703 * SCMI core Notifications code takes care of keeping proper
704 * per-domain enables refcounting, so that this helper function
705 * will be called only once (for enables) when the first user
706 * registers a callback on this domain and once more (disable)
707 * when the last user de-registers its callback.
708 */
709 pi->states[src_id].meas_notif_enabled = enable;
710
711 return ret;
712}
713
714static void *
715scmi_powercap_fill_custom_report(const struct scmi_protocol_handle *ph,
716 u8 evt_id, ktime_t timestamp,
717 const void *payld, size_t payld_sz,
718 void *report, u32 *src_id)
719{
720 void *rep = NULL;
721
722 switch (evt_id) {
723 case SCMI_EVENT_POWERCAP_CAP_CHANGED:
724 {
725 const struct scmi_powercap_cap_changed_notify_payld *p = payld;
726 struct scmi_powercap_cap_changed_report *r = report;
727
728 if (sizeof(*p) != payld_sz)
729 break;
730
731 r->timestamp = timestamp;
732 r->agent_id = le32_to_cpu(p->agent_id);
733 r->domain_id = le32_to_cpu(p->domain_id);
734 r->power_cap = le32_to_cpu(p->power_cap);
735 r->pai = le32_to_cpu(p->pai);
736 *src_id = r->domain_id;
737 rep = r;
738 break;
739 }
740 case SCMI_EVENT_POWERCAP_MEASUREMENTS_CHANGED:
741 {
742 const struct scmi_powercap_meas_changed_notify_payld *p = payld;
743 struct scmi_powercap_meas_changed_report *r = report;
744
745 if (sizeof(*p) != payld_sz)
746 break;
747
748 r->timestamp = timestamp;
749 r->agent_id = le32_to_cpu(p->agent_id);
750 r->domain_id = le32_to_cpu(p->domain_id);
751 r->power = le32_to_cpu(p->power);
752 *src_id = r->domain_id;
753 rep = r;
754 break;
755 }
756 default:
757 break;
758 }
759
760 return rep;
761}
762
763static int
764scmi_powercap_get_num_sources(const struct scmi_protocol_handle *ph)
765{
766 struct powercap_info *pi = ph->get_priv(ph);
767
768 if (!pi)
769 return -EINVAL;
770
771 return pi->num_domains;
772}
773
774static const struct scmi_event powercap_events[] = {
775 {
776 .id = SCMI_EVENT_POWERCAP_CAP_CHANGED,
777 .max_payld_sz =
778 sizeof(struct scmi_powercap_cap_changed_notify_payld),
779 .max_report_sz =
780 sizeof(struct scmi_powercap_cap_changed_report),
781 },
782 {
783 .id = SCMI_EVENT_POWERCAP_MEASUREMENTS_CHANGED,
784 .max_payld_sz =
785 sizeof(struct scmi_powercap_meas_changed_notify_payld),
786 .max_report_sz =
787 sizeof(struct scmi_powercap_meas_changed_report),
788 },
789};
790
791static const struct scmi_event_ops powercap_event_ops = {
792 .get_num_sources = scmi_powercap_get_num_sources,
793 .set_notify_enabled = scmi_powercap_set_notify_enabled,
794 .fill_custom_report = scmi_powercap_fill_custom_report,
795};
796
797static const struct scmi_protocol_events powercap_protocol_events = {
798 .queue_sz = SCMI_PROTO_QUEUE_SZ,
799 .ops = &powercap_event_ops,
800 .evts = powercap_events,
801 .num_events = ARRAY_SIZE(powercap_events),
802};
803
804static int
805scmi_powercap_protocol_init(const struct scmi_protocol_handle *ph)
806{
807 int domain, ret;
808 u32 version;
809 struct powercap_info *pinfo;
810
811 ret = ph->xops->version_get(ph, &version);
812 if (ret)
813 return ret;
814
815 dev_dbg(ph->dev, "Powercap Version %d.%d\n",
816 PROTOCOL_REV_MAJOR(version), PROTOCOL_REV_MINOR(version));
817
818 pinfo = devm_kzalloc(ph->dev, sizeof(*pinfo), GFP_KERNEL);
819 if (!pinfo)
820 return -ENOMEM;
821
822 ret = scmi_powercap_attributes_get(ph, pinfo);
823 if (ret)
824 return ret;
825
826 pinfo->powercaps = devm_kcalloc(ph->dev, pinfo->num_domains,
827 sizeof(*pinfo->powercaps),
828 GFP_KERNEL);
829 if (!pinfo->powercaps)
830 return -ENOMEM;
831
832 /*
833 * Note that any failure in retrieving any domain attribute leads to
834 * the whole Powercap protocol initialization failure: this way the
835 * reported Powercap domains are all assured, when accessed, to be well
836 * formed and correlated by sane parent-child relationship (if any).
837 */
838 for (domain = 0; domain < pinfo->num_domains; domain++) {
839 ret = scmi_powercap_domain_attributes_get(ph, pinfo, domain);
840 if (ret)
841 return ret;
842
843 if (pinfo->powercaps[domain].fastchannels)
844 scmi_powercap_domain_init_fc(ph, domain,
845 &pinfo->powercaps[domain].fc_info);
846 }
847
848 pinfo->states = devm_kcalloc(ph->dev, pinfo->num_domains,
849 sizeof(*pinfo->states), GFP_KERNEL);
850 if (!pinfo->states)
851 return -ENOMEM;
852
853 pinfo->version = version;
854
855 return ph->set_priv(ph, pinfo);
856}
857
858static const struct scmi_protocol scmi_powercap = {
859 .id = SCMI_PROTOCOL_POWERCAP,
860 .owner = THIS_MODULE,
861 .instance_init = &scmi_powercap_protocol_init,
862 .ops = &powercap_proto_ops,
863 .events = &powercap_protocol_events,
864};
865
866DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(powercap, scmi_powercap)