|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * System Control and Management Interface (SCMI) Powercap Protocol | 
|  | * | 
|  | * Copyright (C) 2022 ARM Ltd. | 
|  | */ | 
|  |  | 
|  | #define pr_fmt(fmt) "SCMI Notifications POWERCAP - " fmt | 
|  |  | 
|  | #include <linux/bitfield.h> | 
|  | #include <linux/io.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/scmi_protocol.h> | 
|  |  | 
|  | #include <trace/events/scmi.h> | 
|  |  | 
|  | #include "protocols.h" | 
|  | #include "notify.h" | 
|  |  | 
|  | enum scmi_powercap_protocol_cmd { | 
|  | POWERCAP_DOMAIN_ATTRIBUTES = 0x3, | 
|  | POWERCAP_CAP_GET = 0x4, | 
|  | POWERCAP_CAP_SET = 0x5, | 
|  | POWERCAP_PAI_GET = 0x6, | 
|  | POWERCAP_PAI_SET = 0x7, | 
|  | POWERCAP_DOMAIN_NAME_GET = 0x8, | 
|  | POWERCAP_MEASUREMENTS_GET = 0x9, | 
|  | POWERCAP_CAP_NOTIFY = 0xa, | 
|  | POWERCAP_MEASUREMENTS_NOTIFY = 0xb, | 
|  | POWERCAP_DESCRIBE_FASTCHANNEL = 0xc, | 
|  | }; | 
|  |  | 
|  | enum { | 
|  | POWERCAP_FC_CAP, | 
|  | POWERCAP_FC_PAI, | 
|  | POWERCAP_FC_MAX, | 
|  | }; | 
|  |  | 
|  | struct scmi_msg_resp_powercap_domain_attributes { | 
|  | __le32 attributes; | 
|  | #define SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY(x)		((x) & BIT(31)) | 
|  | #define SUPPORTS_POWERCAP_MEASUREMENTS_CHANGE_NOTIFY(x)	((x) & BIT(30)) | 
|  | #define SUPPORTS_ASYNC_POWERCAP_CAP_SET(x)		((x) & BIT(29)) | 
|  | #define SUPPORTS_EXTENDED_NAMES(x)			((x) & BIT(28)) | 
|  | #define SUPPORTS_POWERCAP_CAP_CONFIGURATION(x)		((x) & BIT(27)) | 
|  | #define SUPPORTS_POWERCAP_MONITORING(x)			((x) & BIT(26)) | 
|  | #define SUPPORTS_POWERCAP_PAI_CONFIGURATION(x)		((x) & BIT(25)) | 
|  | #define SUPPORTS_POWERCAP_FASTCHANNELS(x)		((x) & BIT(22)) | 
|  | #define POWERCAP_POWER_UNIT(x)				\ | 
|  | (FIELD_GET(GENMASK(24, 23), (x))) | 
|  | #define	SUPPORTS_POWER_UNITS_MW(x)			\ | 
|  | (POWERCAP_POWER_UNIT(x) == 0x2) | 
|  | #define	SUPPORTS_POWER_UNITS_UW(x)			\ | 
|  | (POWERCAP_POWER_UNIT(x) == 0x1) | 
|  | u8 name[SCMI_SHORT_NAME_MAX_SIZE]; | 
|  | __le32 min_pai; | 
|  | __le32 max_pai; | 
|  | __le32 pai_step; | 
|  | __le32 min_power_cap; | 
|  | __le32 max_power_cap; | 
|  | __le32 power_cap_step; | 
|  | __le32 sustainable_power; | 
|  | __le32 accuracy; | 
|  | __le32 parent_id; | 
|  | }; | 
|  |  | 
|  | struct scmi_msg_powercap_set_cap_or_pai { | 
|  | __le32 domain; | 
|  | __le32 flags; | 
|  | #define CAP_SET_ASYNC		BIT(1) | 
|  | #define CAP_SET_IGNORE_DRESP	BIT(0) | 
|  | __le32 value; | 
|  | }; | 
|  |  | 
|  | struct scmi_msg_resp_powercap_cap_set_complete { | 
|  | __le32 domain; | 
|  | __le32 power_cap; | 
|  | }; | 
|  |  | 
|  | struct scmi_msg_resp_powercap_meas_get { | 
|  | __le32 power; | 
|  | __le32 pai; | 
|  | }; | 
|  |  | 
|  | struct scmi_msg_powercap_notify_cap { | 
|  | __le32 domain; | 
|  | __le32 notify_enable; | 
|  | }; | 
|  |  | 
|  | struct scmi_msg_powercap_notify_thresh { | 
|  | __le32 domain; | 
|  | __le32 notify_enable; | 
|  | __le32 power_thresh_low; | 
|  | __le32 power_thresh_high; | 
|  | }; | 
|  |  | 
|  | struct scmi_powercap_cap_changed_notify_payld { | 
|  | __le32 agent_id; | 
|  | __le32 domain_id; | 
|  | __le32 power_cap; | 
|  | __le32 pai; | 
|  | }; | 
|  |  | 
|  | struct scmi_powercap_meas_changed_notify_payld { | 
|  | __le32 agent_id; | 
|  | __le32 domain_id; | 
|  | __le32 power; | 
|  | }; | 
|  |  | 
|  | struct scmi_powercap_state { | 
|  | bool enabled; | 
|  | u32 last_pcap; | 
|  | bool meas_notif_enabled; | 
|  | u64 thresholds; | 
|  | #define THRESH_LOW(p, id)				\ | 
|  | (lower_32_bits((p)->states[(id)].thresholds)) | 
|  | #define THRESH_HIGH(p, id)				\ | 
|  | (upper_32_bits((p)->states[(id)].thresholds)) | 
|  | }; | 
|  |  | 
|  | struct powercap_info { | 
|  | u32 version; | 
|  | int num_domains; | 
|  | struct scmi_powercap_state *states; | 
|  | struct scmi_powercap_info *powercaps; | 
|  | }; | 
|  |  | 
|  | static enum scmi_powercap_protocol_cmd evt_2_cmd[] = { | 
|  | POWERCAP_CAP_NOTIFY, | 
|  | POWERCAP_MEASUREMENTS_NOTIFY, | 
|  | }; | 
|  |  | 
|  | static int scmi_powercap_notify(const struct scmi_protocol_handle *ph, | 
|  | u32 domain, int message_id, bool enable); | 
|  |  | 
|  | static int | 
|  | scmi_powercap_attributes_get(const struct scmi_protocol_handle *ph, | 
|  | struct powercap_info *pi) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, PROTOCOL_ATTRIBUTES, 0, | 
|  | sizeof(u32), &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  | if (!ret) { | 
|  | u32 attributes; | 
|  |  | 
|  | attributes = get_unaligned_le32(t->rx.buf); | 
|  | pi->num_domains = FIELD_GET(GENMASK(15, 0), attributes); | 
|  | } | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static inline int | 
|  | scmi_powercap_validate(unsigned int min_val, unsigned int max_val, | 
|  | unsigned int step_val, bool configurable) | 
|  | { | 
|  | if (!min_val || !max_val) | 
|  | return -EPROTO; | 
|  |  | 
|  | if ((configurable && min_val == max_val) || | 
|  | (!configurable && min_val != max_val)) | 
|  | return -EPROTO; | 
|  |  | 
|  | if (min_val != max_val && !step_val) | 
|  | return -EPROTO; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int | 
|  | scmi_powercap_domain_attributes_get(const struct scmi_protocol_handle *ph, | 
|  | struct powercap_info *pinfo, u32 domain) | 
|  | { | 
|  | int ret; | 
|  | u32 flags; | 
|  | struct scmi_xfer *t; | 
|  | struct scmi_powercap_info *dom_info = pinfo->powercaps + domain; | 
|  | struct scmi_msg_resp_powercap_domain_attributes *resp; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, POWERCAP_DOMAIN_ATTRIBUTES, | 
|  | sizeof(domain), sizeof(*resp), &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | put_unaligned_le32(domain, t->tx.buf); | 
|  | resp = t->rx.buf; | 
|  |  | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  | if (!ret) { | 
|  | flags = le32_to_cpu(resp->attributes); | 
|  |  | 
|  | dom_info->id = domain; | 
|  | dom_info->notify_powercap_cap_change = | 
|  | SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY(flags); | 
|  | dom_info->notify_powercap_measurement_change = | 
|  | SUPPORTS_POWERCAP_MEASUREMENTS_CHANGE_NOTIFY(flags); | 
|  | dom_info->async_powercap_cap_set = | 
|  | SUPPORTS_ASYNC_POWERCAP_CAP_SET(flags); | 
|  | dom_info->powercap_cap_config = | 
|  | SUPPORTS_POWERCAP_CAP_CONFIGURATION(flags); | 
|  | dom_info->powercap_monitoring = | 
|  | SUPPORTS_POWERCAP_MONITORING(flags); | 
|  | dom_info->powercap_pai_config = | 
|  | SUPPORTS_POWERCAP_PAI_CONFIGURATION(flags); | 
|  | dom_info->powercap_scale_mw = | 
|  | SUPPORTS_POWER_UNITS_MW(flags); | 
|  | dom_info->powercap_scale_uw = | 
|  | SUPPORTS_POWER_UNITS_UW(flags); | 
|  | dom_info->fastchannels = | 
|  | SUPPORTS_POWERCAP_FASTCHANNELS(flags); | 
|  |  | 
|  | strscpy(dom_info->name, resp->name, SCMI_SHORT_NAME_MAX_SIZE); | 
|  |  | 
|  | dom_info->min_pai = le32_to_cpu(resp->min_pai); | 
|  | dom_info->max_pai = le32_to_cpu(resp->max_pai); | 
|  | dom_info->pai_step = le32_to_cpu(resp->pai_step); | 
|  | ret = scmi_powercap_validate(dom_info->min_pai, | 
|  | dom_info->max_pai, | 
|  | dom_info->pai_step, | 
|  | dom_info->powercap_pai_config); | 
|  | if (ret) { | 
|  | dev_err(ph->dev, | 
|  | "Platform reported inconsistent PAI config for domain %d - %s\n", | 
|  | dom_info->id, dom_info->name); | 
|  | goto clean; | 
|  | } | 
|  |  | 
|  | dom_info->min_power_cap = le32_to_cpu(resp->min_power_cap); | 
|  | dom_info->max_power_cap = le32_to_cpu(resp->max_power_cap); | 
|  | dom_info->power_cap_step = le32_to_cpu(resp->power_cap_step); | 
|  | ret = scmi_powercap_validate(dom_info->min_power_cap, | 
|  | dom_info->max_power_cap, | 
|  | dom_info->power_cap_step, | 
|  | dom_info->powercap_cap_config); | 
|  | if (ret) { | 
|  | dev_err(ph->dev, | 
|  | "Platform reported inconsistent CAP config for domain %d - %s\n", | 
|  | dom_info->id, dom_info->name); | 
|  | goto clean; | 
|  | } | 
|  |  | 
|  | dom_info->sustainable_power = | 
|  | le32_to_cpu(resp->sustainable_power); | 
|  | dom_info->accuracy = le32_to_cpu(resp->accuracy); | 
|  |  | 
|  | dom_info->parent_id = le32_to_cpu(resp->parent_id); | 
|  | if (dom_info->parent_id != SCMI_POWERCAP_ROOT_ZONE_ID && | 
|  | (dom_info->parent_id >= pinfo->num_domains || | 
|  | dom_info->parent_id == dom_info->id)) { | 
|  | dev_err(ph->dev, | 
|  | "Platform reported inconsistent parent ID for domain %d - %s\n", | 
|  | dom_info->id, dom_info->name); | 
|  | ret = -ENODEV; | 
|  | } | 
|  | } | 
|  |  | 
|  | clean: | 
|  | ph->xops->xfer_put(ph, t); | 
|  |  | 
|  | /* | 
|  | * If supported overwrite short name with the extended one; | 
|  | * on error just carry on and use already provided short name. | 
|  | */ | 
|  | if (!ret && SUPPORTS_EXTENDED_NAMES(flags)) | 
|  | ph->hops->extended_name_get(ph, POWERCAP_DOMAIN_NAME_GET, | 
|  | domain, dom_info->name, | 
|  | SCMI_MAX_STR_SIZE); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_num_domains_get(const struct scmi_protocol_handle *ph) | 
|  | { | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | return pi->num_domains; | 
|  | } | 
|  |  | 
|  | static const struct scmi_powercap_info * | 
|  | scmi_powercap_dom_info_get(const struct scmi_protocol_handle *ph, u32 domain_id) | 
|  | { | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (domain_id >= pi->num_domains) | 
|  | return NULL; | 
|  |  | 
|  | return pi->powercaps + domain_id; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_xfer_cap_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 *power_cap) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, POWERCAP_CAP_GET, sizeof(u32), | 
|  | sizeof(u32), &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | put_unaligned_le32(domain_id, t->tx.buf); | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  | if (!ret) | 
|  | *power_cap = get_unaligned_le32(t->rx.buf); | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int __scmi_powercap_cap_get(const struct scmi_protocol_handle *ph, | 
|  | const struct scmi_powercap_info *dom, | 
|  | u32 *power_cap) | 
|  | { | 
|  | if (dom->fc_info && dom->fc_info[POWERCAP_FC_CAP].get_addr) { | 
|  | *power_cap = ioread32(dom->fc_info[POWERCAP_FC_CAP].get_addr); | 
|  | trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_CAP_GET, | 
|  | dom->id, *power_cap, 0); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | return scmi_powercap_xfer_cap_get(ph, dom->id, power_cap); | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_cap_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 *power_cap) | 
|  | { | 
|  | const struct scmi_powercap_info *dom; | 
|  |  | 
|  | if (!power_cap) | 
|  | return -EINVAL; | 
|  |  | 
|  | dom = scmi_powercap_dom_info_get(ph, domain_id); | 
|  | if (!dom) | 
|  | return -EINVAL; | 
|  |  | 
|  | return __scmi_powercap_cap_get(ph, dom, power_cap); | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_xfer_cap_set(const struct scmi_protocol_handle *ph, | 
|  | const struct scmi_powercap_info *pc, | 
|  | u32 power_cap, bool ignore_dresp) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  | struct scmi_msg_powercap_set_cap_or_pai *msg; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, POWERCAP_CAP_SET, | 
|  | sizeof(*msg), 0, &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | msg = t->tx.buf; | 
|  | msg->domain = cpu_to_le32(pc->id); | 
|  | msg->flags = | 
|  | cpu_to_le32(FIELD_PREP(CAP_SET_ASYNC, !!pc->async_powercap_cap_set) | | 
|  | FIELD_PREP(CAP_SET_IGNORE_DRESP, !!ignore_dresp)); | 
|  | msg->value = cpu_to_le32(power_cap); | 
|  |  | 
|  | if (!pc->async_powercap_cap_set || ignore_dresp) { | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  | } else { | 
|  | ret = ph->xops->do_xfer_with_response(ph, t); | 
|  | if (!ret) { | 
|  | struct scmi_msg_resp_powercap_cap_set_complete *resp; | 
|  |  | 
|  | resp = t->rx.buf; | 
|  | if (le32_to_cpu(resp->domain) == pc->id) | 
|  | dev_dbg(ph->dev, | 
|  | "Powercap ID %d CAP set async to %u\n", | 
|  | pc->id, | 
|  | get_unaligned_le32(&resp->power_cap)); | 
|  | else | 
|  | ret = -EPROTO; | 
|  | } | 
|  | } | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int __scmi_powercap_cap_set(const struct scmi_protocol_handle *ph, | 
|  | struct powercap_info *pi, u32 domain_id, | 
|  | u32 power_cap, bool ignore_dresp) | 
|  | { | 
|  | int ret = -EINVAL; | 
|  | const struct scmi_powercap_info *pc; | 
|  |  | 
|  | pc = scmi_powercap_dom_info_get(ph, domain_id); | 
|  | if (!pc || !pc->powercap_cap_config) | 
|  | return ret; | 
|  |  | 
|  | if (power_cap && | 
|  | (power_cap < pc->min_power_cap || power_cap > pc->max_power_cap)) | 
|  | return ret; | 
|  |  | 
|  | if (pc->fc_info && pc->fc_info[POWERCAP_FC_CAP].set_addr) { | 
|  | struct scmi_fc_info *fci = &pc->fc_info[POWERCAP_FC_CAP]; | 
|  |  | 
|  | iowrite32(power_cap, fci->set_addr); | 
|  | ph->hops->fastchannel_db_ring(fci->set_db); | 
|  | trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_CAP_SET, | 
|  | domain_id, power_cap, 0); | 
|  | ret = 0; | 
|  | } else { | 
|  | ret = scmi_powercap_xfer_cap_set(ph, pc, power_cap, | 
|  | ignore_dresp); | 
|  | } | 
|  |  | 
|  | /* Save the last explicitly set non-zero powercap value */ | 
|  | if (PROTOCOL_REV_MAJOR(pi->version) >= 0x2 && !ret && power_cap) | 
|  | pi->states[domain_id].last_pcap = power_cap; | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_cap_set(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 power_cap, | 
|  | bool ignore_dresp) | 
|  | { | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | /* | 
|  | * Disallow zero as a possible explicitly requested powercap: | 
|  | * there are enable/disable operations for this. | 
|  | */ | 
|  | if (!power_cap) | 
|  | return -EINVAL; | 
|  |  | 
|  | /* Just log the last set request if acting on a disabled domain */ | 
|  | if (PROTOCOL_REV_MAJOR(pi->version) >= 0x2 && | 
|  | !pi->states[domain_id].enabled) { | 
|  | pi->states[domain_id].last_pcap = power_cap; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | return __scmi_powercap_cap_set(ph, pi, domain_id, | 
|  | power_cap, ignore_dresp); | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_xfer_pai_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 *pai) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, POWERCAP_PAI_GET, sizeof(u32), | 
|  | sizeof(u32), &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | put_unaligned_le32(domain_id, t->tx.buf); | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  | if (!ret) | 
|  | *pai = get_unaligned_le32(t->rx.buf); | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_pai_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 *pai) | 
|  | { | 
|  | struct scmi_powercap_info *dom; | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (!pai || domain_id >= pi->num_domains) | 
|  | return -EINVAL; | 
|  |  | 
|  | dom = pi->powercaps + domain_id; | 
|  | if (dom->fc_info && dom->fc_info[POWERCAP_FC_PAI].get_addr) { | 
|  | *pai = ioread32(dom->fc_info[POWERCAP_FC_PAI].get_addr); | 
|  | trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_PAI_GET, | 
|  | domain_id, *pai, 0); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | return scmi_powercap_xfer_pai_get(ph, domain_id, pai); | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_xfer_pai_set(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 pai) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  | struct scmi_msg_powercap_set_cap_or_pai *msg; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, POWERCAP_PAI_SET, | 
|  | sizeof(*msg), 0, &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | msg = t->tx.buf; | 
|  | msg->domain = cpu_to_le32(domain_id); | 
|  | msg->flags = cpu_to_le32(0); | 
|  | msg->value = cpu_to_le32(pai); | 
|  |  | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_pai_set(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 pai) | 
|  | { | 
|  | const struct scmi_powercap_info *pc; | 
|  |  | 
|  | pc = scmi_powercap_dom_info_get(ph, domain_id); | 
|  | if (!pc || !pc->powercap_pai_config || !pai || | 
|  | pai < pc->min_pai || pai > pc->max_pai) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (pc->fc_info && pc->fc_info[POWERCAP_FC_PAI].set_addr) { | 
|  | struct scmi_fc_info *fci = &pc->fc_info[POWERCAP_FC_PAI]; | 
|  |  | 
|  | trace_scmi_fc_call(SCMI_PROTOCOL_POWERCAP, POWERCAP_PAI_SET, | 
|  | domain_id, pai, 0); | 
|  | iowrite32(pai, fci->set_addr); | 
|  | ph->hops->fastchannel_db_ring(fci->set_db); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | return scmi_powercap_xfer_pai_set(ph, domain_id, pai); | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_measurements_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 *average_power, | 
|  | u32 *pai) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  | struct scmi_msg_resp_powercap_meas_get *resp; | 
|  | const struct scmi_powercap_info *pc; | 
|  |  | 
|  | pc = scmi_powercap_dom_info_get(ph, domain_id); | 
|  | if (!pc || !pc->powercap_monitoring || !pai || !average_power) | 
|  | return -EINVAL; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, POWERCAP_MEASUREMENTS_GET, | 
|  | sizeof(u32), sizeof(*resp), &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | resp = t->rx.buf; | 
|  | put_unaligned_le32(domain_id, t->tx.buf); | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  | if (!ret) { | 
|  | *average_power = le32_to_cpu(resp->power); | 
|  | *pai = le32_to_cpu(resp->pai); | 
|  | } | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int | 
|  | scmi_powercap_measurements_threshold_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 *power_thresh_low, | 
|  | u32 *power_thresh_high) | 
|  | { | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (!power_thresh_low || !power_thresh_high || | 
|  | domain_id >= pi->num_domains) | 
|  | return -EINVAL; | 
|  |  | 
|  | *power_thresh_low =  THRESH_LOW(pi, domain_id); | 
|  | *power_thresh_high = THRESH_HIGH(pi, domain_id); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int | 
|  | scmi_powercap_measurements_threshold_set(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, u32 power_thresh_low, | 
|  | u32 power_thresh_high) | 
|  | { | 
|  | int ret = 0; | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (domain_id >= pi->num_domains || | 
|  | power_thresh_low > power_thresh_high) | 
|  | return -EINVAL; | 
|  |  | 
|  | /* Anything to do ? */ | 
|  | if (THRESH_LOW(pi, domain_id) == power_thresh_low && | 
|  | THRESH_HIGH(pi, domain_id) == power_thresh_high) | 
|  | return ret; | 
|  |  | 
|  | pi->states[domain_id].thresholds = | 
|  | (FIELD_PREP(GENMASK_ULL(31, 0), power_thresh_low) | | 
|  | FIELD_PREP(GENMASK_ULL(63, 32), power_thresh_high)); | 
|  |  | 
|  | /* Update thresholds if notification already enabled */ | 
|  | if (pi->states[domain_id].meas_notif_enabled) | 
|  | ret = scmi_powercap_notify(ph, domain_id, | 
|  | POWERCAP_MEASUREMENTS_NOTIFY, | 
|  | true); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_cap_enable_set(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, bool enable) | 
|  | { | 
|  | int ret; | 
|  | u32 power_cap; | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (PROTOCOL_REV_MAJOR(pi->version) < 0x2) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (enable == pi->states[domain_id].enabled) | 
|  | return 0; | 
|  |  | 
|  | if (enable) { | 
|  | /* Cannot enable with a zero powercap. */ | 
|  | if (!pi->states[domain_id].last_pcap) | 
|  | return -EINVAL; | 
|  |  | 
|  | ret = __scmi_powercap_cap_set(ph, pi, domain_id, | 
|  | pi->states[domain_id].last_pcap, | 
|  | true); | 
|  | } else { | 
|  | ret = __scmi_powercap_cap_set(ph, pi, domain_id, 0, true); | 
|  | } | 
|  |  | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* | 
|  | * Update our internal state to reflect final platform state: the SCMI | 
|  | * server could have ignored a disable request and kept enforcing some | 
|  | * powercap limit requested by other agents. | 
|  | */ | 
|  | ret = scmi_powercap_cap_get(ph, domain_id, &power_cap); | 
|  | if (!ret) | 
|  | pi->states[domain_id].enabled = !!power_cap; | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_cap_enable_get(const struct scmi_protocol_handle *ph, | 
|  | u32 domain_id, bool *enable) | 
|  | { | 
|  | int ret; | 
|  | u32 power_cap; | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | *enable = true; | 
|  | if (PROTOCOL_REV_MAJOR(pi->version) < 0x2) | 
|  | return 0; | 
|  |  | 
|  | /* | 
|  | * Report always real platform state; platform could have ignored | 
|  | * a previous disable request. Default true on any error. | 
|  | */ | 
|  | ret = scmi_powercap_cap_get(ph, domain_id, &power_cap); | 
|  | if (!ret) | 
|  | *enable = !!power_cap; | 
|  |  | 
|  | /* Update internal state with current real platform state */ | 
|  | pi->states[domain_id].enabled = *enable; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct scmi_powercap_proto_ops powercap_proto_ops = { | 
|  | .num_domains_get = scmi_powercap_num_domains_get, | 
|  | .info_get = scmi_powercap_dom_info_get, | 
|  | .cap_get = scmi_powercap_cap_get, | 
|  | .cap_set = scmi_powercap_cap_set, | 
|  | .cap_enable_set = scmi_powercap_cap_enable_set, | 
|  | .cap_enable_get = scmi_powercap_cap_enable_get, | 
|  | .pai_get = scmi_powercap_pai_get, | 
|  | .pai_set = scmi_powercap_pai_set, | 
|  | .measurements_get = scmi_powercap_measurements_get, | 
|  | .measurements_threshold_set = scmi_powercap_measurements_threshold_set, | 
|  | .measurements_threshold_get = scmi_powercap_measurements_threshold_get, | 
|  | }; | 
|  |  | 
|  | static void scmi_powercap_domain_init_fc(const struct scmi_protocol_handle *ph, | 
|  | u32 domain, struct scmi_fc_info **p_fc) | 
|  | { | 
|  | struct scmi_fc_info *fc; | 
|  |  | 
|  | fc = devm_kcalloc(ph->dev, POWERCAP_FC_MAX, sizeof(*fc), GFP_KERNEL); | 
|  | if (!fc) | 
|  | return; | 
|  |  | 
|  | ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL, | 
|  | POWERCAP_CAP_SET, 4, domain, | 
|  | &fc[POWERCAP_FC_CAP].set_addr, | 
|  | &fc[POWERCAP_FC_CAP].set_db); | 
|  |  | 
|  | ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL, | 
|  | POWERCAP_CAP_GET, 4, domain, | 
|  | &fc[POWERCAP_FC_CAP].get_addr, NULL); | 
|  |  | 
|  | ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL, | 
|  | POWERCAP_PAI_SET, 4, domain, | 
|  | &fc[POWERCAP_FC_PAI].set_addr, | 
|  | &fc[POWERCAP_FC_PAI].set_db); | 
|  |  | 
|  | ph->hops->fastchannel_init(ph, POWERCAP_DESCRIBE_FASTCHANNEL, | 
|  | POWERCAP_PAI_GET, 4, domain, | 
|  | &fc[POWERCAP_FC_PAI].get_addr, NULL); | 
|  |  | 
|  | *p_fc = fc; | 
|  | } | 
|  |  | 
|  | static int scmi_powercap_notify(const struct scmi_protocol_handle *ph, | 
|  | u32 domain, int message_id, bool enable) | 
|  | { | 
|  | int ret; | 
|  | struct scmi_xfer *t; | 
|  |  | 
|  | switch (message_id) { | 
|  | case POWERCAP_CAP_NOTIFY: | 
|  | { | 
|  | struct scmi_msg_powercap_notify_cap *notify; | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, message_id, | 
|  | sizeof(*notify), 0, &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | notify = t->tx.buf; | 
|  | notify->domain = cpu_to_le32(domain); | 
|  | notify->notify_enable = cpu_to_le32(enable ? BIT(0) : 0); | 
|  | break; | 
|  | } | 
|  | case POWERCAP_MEASUREMENTS_NOTIFY: | 
|  | { | 
|  | u32 low, high; | 
|  | struct scmi_msg_powercap_notify_thresh *notify; | 
|  |  | 
|  | /* | 
|  | * Note that we have to pick the most recently configured | 
|  | * thresholds to build a proper POWERCAP_MEASUREMENTS_NOTIFY | 
|  | * enable request and we fail, complaining, if no thresholds | 
|  | * were ever set, since this is an indication the API has been | 
|  | * used wrongly. | 
|  | */ | 
|  | ret = scmi_powercap_measurements_threshold_get(ph, domain, | 
|  | &low, &high); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | if (enable && !low && !high) { | 
|  | dev_err(ph->dev, | 
|  | "Invalid Measurements Notify thresholds: %u/%u\n", | 
|  | low, high); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | ret = ph->xops->xfer_get_init(ph, message_id, | 
|  | sizeof(*notify), 0, &t); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | notify = t->tx.buf; | 
|  | notify->domain = cpu_to_le32(domain); | 
|  | notify->notify_enable = cpu_to_le32(enable ? BIT(0) : 0); | 
|  | notify->power_thresh_low = cpu_to_le32(low); | 
|  | notify->power_thresh_high = cpu_to_le32(high); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | ret = ph->xops->do_xfer(ph, t); | 
|  |  | 
|  | ph->xops->xfer_put(ph, t); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int | 
|  | scmi_powercap_set_notify_enabled(const struct scmi_protocol_handle *ph, | 
|  | u8 evt_id, u32 src_id, bool enable) | 
|  | { | 
|  | int ret, cmd_id; | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (evt_id >= ARRAY_SIZE(evt_2_cmd) || src_id >= pi->num_domains) | 
|  | return -EINVAL; | 
|  |  | 
|  | cmd_id = evt_2_cmd[evt_id]; | 
|  | ret = scmi_powercap_notify(ph, src_id, cmd_id, enable); | 
|  | if (ret) | 
|  | pr_debug("FAIL_ENABLED - evt[%X] dom[%d] - ret:%d\n", | 
|  | evt_id, src_id, ret); | 
|  | else if (cmd_id == POWERCAP_MEASUREMENTS_NOTIFY) | 
|  | /* | 
|  | * On success save the current notification enabled state, so | 
|  | * as to be able to properly update the notification thresholds | 
|  | * when they are modified on a domain for which measurement | 
|  | * notifications were currently enabled. | 
|  | * | 
|  | * This is needed because the SCMI Notification core machinery | 
|  | * and API does not support passing per-notification custom | 
|  | * arguments at callback registration time. | 
|  | * | 
|  | * Note that this can be done here with a simple flag since the | 
|  | * SCMI core Notifications code takes care of keeping proper | 
|  | * per-domain enables refcounting, so that this helper function | 
|  | * will be called only once (for enables) when the first user | 
|  | * registers a callback on this domain and once more (disable) | 
|  | * when the last user de-registers its callback. | 
|  | */ | 
|  | pi->states[src_id].meas_notif_enabled = enable; | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static void * | 
|  | scmi_powercap_fill_custom_report(const struct scmi_protocol_handle *ph, | 
|  | u8 evt_id, ktime_t timestamp, | 
|  | const void *payld, size_t payld_sz, | 
|  | void *report, u32 *src_id) | 
|  | { | 
|  | void *rep = NULL; | 
|  |  | 
|  | switch (evt_id) { | 
|  | case SCMI_EVENT_POWERCAP_CAP_CHANGED: | 
|  | { | 
|  | const struct scmi_powercap_cap_changed_notify_payld *p = payld; | 
|  | struct scmi_powercap_cap_changed_report *r = report; | 
|  |  | 
|  | if (sizeof(*p) != payld_sz) | 
|  | break; | 
|  |  | 
|  | r->timestamp = timestamp; | 
|  | r->agent_id = le32_to_cpu(p->agent_id); | 
|  | r->domain_id = le32_to_cpu(p->domain_id); | 
|  | r->power_cap = le32_to_cpu(p->power_cap); | 
|  | r->pai = le32_to_cpu(p->pai); | 
|  | *src_id = r->domain_id; | 
|  | rep = r; | 
|  | break; | 
|  | } | 
|  | case SCMI_EVENT_POWERCAP_MEASUREMENTS_CHANGED: | 
|  | { | 
|  | const struct scmi_powercap_meas_changed_notify_payld *p = payld; | 
|  | struct scmi_powercap_meas_changed_report *r = report; | 
|  |  | 
|  | if (sizeof(*p) != payld_sz) | 
|  | break; | 
|  |  | 
|  | r->timestamp = timestamp; | 
|  | r->agent_id = le32_to_cpu(p->agent_id); | 
|  | r->domain_id = le32_to_cpu(p->domain_id); | 
|  | r->power = le32_to_cpu(p->power); | 
|  | *src_id = r->domain_id; | 
|  | rep = r; | 
|  | break; | 
|  | } | 
|  | default: | 
|  | break; | 
|  | } | 
|  |  | 
|  | return rep; | 
|  | } | 
|  |  | 
|  | static int | 
|  | scmi_powercap_get_num_sources(const struct scmi_protocol_handle *ph) | 
|  | { | 
|  | struct powercap_info *pi = ph->get_priv(ph); | 
|  |  | 
|  | if (!pi) | 
|  | return -EINVAL; | 
|  |  | 
|  | return pi->num_domains; | 
|  | } | 
|  |  | 
|  | static const struct scmi_event powercap_events[] = { | 
|  | { | 
|  | .id = SCMI_EVENT_POWERCAP_CAP_CHANGED, | 
|  | .max_payld_sz = | 
|  | sizeof(struct scmi_powercap_cap_changed_notify_payld), | 
|  | .max_report_sz = | 
|  | sizeof(struct scmi_powercap_cap_changed_report), | 
|  | }, | 
|  | { | 
|  | .id = SCMI_EVENT_POWERCAP_MEASUREMENTS_CHANGED, | 
|  | .max_payld_sz = | 
|  | sizeof(struct scmi_powercap_meas_changed_notify_payld), | 
|  | .max_report_sz = | 
|  | sizeof(struct scmi_powercap_meas_changed_report), | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static const struct scmi_event_ops powercap_event_ops = { | 
|  | .get_num_sources = scmi_powercap_get_num_sources, | 
|  | .set_notify_enabled = scmi_powercap_set_notify_enabled, | 
|  | .fill_custom_report = scmi_powercap_fill_custom_report, | 
|  | }; | 
|  |  | 
|  | static const struct scmi_protocol_events powercap_protocol_events = { | 
|  | .queue_sz = SCMI_PROTO_QUEUE_SZ, | 
|  | .ops = &powercap_event_ops, | 
|  | .evts = powercap_events, | 
|  | .num_events = ARRAY_SIZE(powercap_events), | 
|  | }; | 
|  |  | 
|  | static int | 
|  | scmi_powercap_protocol_init(const struct scmi_protocol_handle *ph) | 
|  | { | 
|  | int domain, ret; | 
|  | u32 version; | 
|  | struct powercap_info *pinfo; | 
|  |  | 
|  | ret = ph->xops->version_get(ph, &version); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | dev_dbg(ph->dev, "Powercap Version %d.%d\n", | 
|  | PROTOCOL_REV_MAJOR(version), PROTOCOL_REV_MINOR(version)); | 
|  |  | 
|  | pinfo = devm_kzalloc(ph->dev, sizeof(*pinfo), GFP_KERNEL); | 
|  | if (!pinfo) | 
|  | return -ENOMEM; | 
|  |  | 
|  | ret = scmi_powercap_attributes_get(ph, pinfo); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | pinfo->powercaps = devm_kcalloc(ph->dev, pinfo->num_domains, | 
|  | sizeof(*pinfo->powercaps), | 
|  | GFP_KERNEL); | 
|  | if (!pinfo->powercaps) | 
|  | return -ENOMEM; | 
|  |  | 
|  | pinfo->states = devm_kcalloc(ph->dev, pinfo->num_domains, | 
|  | sizeof(*pinfo->states), GFP_KERNEL); | 
|  | if (!pinfo->states) | 
|  | return -ENOMEM; | 
|  |  | 
|  | /* | 
|  | * Note that any failure in retrieving any domain attribute leads to | 
|  | * the whole Powercap protocol initialization failure: this way the | 
|  | * reported Powercap domains are all assured, when accessed, to be well | 
|  | * formed and correlated by sane parent-child relationship (if any). | 
|  | */ | 
|  | for (domain = 0; domain < pinfo->num_domains; domain++) { | 
|  | ret = scmi_powercap_domain_attributes_get(ph, pinfo, domain); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | if (pinfo->powercaps[domain].fastchannels) | 
|  | scmi_powercap_domain_init_fc(ph, domain, | 
|  | &pinfo->powercaps[domain].fc_info); | 
|  |  | 
|  | /* Grab initial state when disable is supported. */ | 
|  | if (PROTOCOL_REV_MAJOR(version) >= 0x2) { | 
|  | ret = __scmi_powercap_cap_get(ph, | 
|  | &pinfo->powercaps[domain], | 
|  | &pinfo->states[domain].last_pcap); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | pinfo->states[domain].enabled = | 
|  | !!pinfo->states[domain].last_pcap; | 
|  | } | 
|  | } | 
|  |  | 
|  | pinfo->version = version; | 
|  | return ph->set_priv(ph, pinfo); | 
|  | } | 
|  |  | 
|  | static const struct scmi_protocol scmi_powercap = { | 
|  | .id = SCMI_PROTOCOL_POWERCAP, | 
|  | .owner = THIS_MODULE, | 
|  | .instance_init = &scmi_powercap_protocol_init, | 
|  | .ops = &powercap_proto_ops, | 
|  | .events = &powercap_protocol_events, | 
|  | }; | 
|  |  | 
|  | DEFINE_SCMI_PROTOCOL_REGISTER_UNREGISTER(powercap, scmi_powercap) |