From nobody Sat Oct 4 21:03:12 2025 Received: from foss.arm.com (foss.arm.com [217.140.110.172]) by smtp.subspace.kernel.org (Postfix) with ESMTP id 791682F747A; Wed, 13 Aug 2025 11:47:03 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=217.140.110.172 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1755085625; cv=none; b=PNZAVoHyiFPMTHb5qmrJqhNi3loRs60M7ow1bBKh6LHhp2mWWvgnlb7dxfB5pOYE61TCRYqbwp8SxfAyztG6639qzY66YD+EMYghsGBtQAyKgRlaeEnNGbooWPFPiBE+1yBO5owXm1Uwxv7Hau6njvLLEYOcPK0d9FnS63LIXRM= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1755085625; c=relaxed/simple; bh=U8LgDVBMfLKig+DW61cy5I4wo/5s4exWi1AiZsrGiaA=; h=From:To:Cc:Subject:Date:Message-ID:In-Reply-To:References: MIME-Version; b=gjw4pk6mlVvqJry8i68YcuaEuF0RQNNxGQNJ3OhJyAjw0YtXbnlcJ54x4DuUq29XGBn8Tnf2FCUQi0LHECGAfRBVCTq48MNEYZm/4pmwy8T2xl2aW/HVI2ap4lKA5igHcnWrmjm5q9O+KdbH+MLaSOXo6m+IoVoY8DtjKI2U8nY= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=arm.com; spf=pass smtp.mailfrom=arm.com; arc=none smtp.client-ip=217.140.110.172 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=arm.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=arm.com Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id BBFFE12FC; Wed, 13 Aug 2025 04:46:54 -0700 (PDT) Received: from pluto.guest.local (usa-sjc-mx-foss1.foss.arm.com [172.31.20.19]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPSA id 066A73F738; Wed, 13 Aug 2025 04:46:59 -0700 (PDT) From: Cristian Marussi To: linux-kernel@vger.kernel.org, linux-arm-kernel@lists.infradead.org, arm-scmi@vger.kernel.org, linux-pm@vger.kernel.org Cc: sudeep.holla@arm.com, james.quinlan@broadcom.com, f.fainelli@gmail.com, vincent.guittot@linaro.org, etienne.carriere@st.com, peng.fan@oss.nxp.com, michal.simek@amd.com, quic_sibis@quicinc.com, dan.carpenter@linaro.org, d-gole@ti.com, souvik.chakravarty@arm.com, Cristian Marussi Subject: [PATCH 4/8] firmware: arm_scmi: Add SCMIv4.0 Powercap basic support Date: Wed, 13 Aug 2025 12:46:05 +0100 Message-ID: <20250813114609.1305571-5-cristian.marussi@arm.com> X-Mailer: git-send-email 2.47.0 In-Reply-To: <20250813114609.1305571-1-cristian.marussi@arm.com> References: <20250813114609.1305571-1-cristian.marussi@arm.com> Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable Content-Type: text/plain; charset="utf-8" Add SCMIv4.0 Powercap support for enumerating multiple CPLs of a domain when available. Signed-off-by: Cristian Marussi --- drivers/firmware/arm_scmi/powercap.c | 477 +++++++++++++++++++++------ include/linux/scmi_protocol.h | 1 + 2 files changed, 379 insertions(+), 99 deletions(-) diff --git a/drivers/firmware/arm_scmi/powercap.c b/drivers/firmware/arm_sc= mi/powercap.c index 3432f55ace14..37fe3bd0c099 100644 --- a/drivers/firmware/arm_scmi/powercap.c +++ b/drivers/firmware/arm_scmi/powercap.c @@ -33,6 +33,7 @@ enum scmi_powercap_protocol_cmd { POWERCAP_CAP_NOTIFY =3D 0xa, POWERCAP_MEASUREMENTS_NOTIFY =3D 0xb, POWERCAP_DESCRIBE_FASTCHANNEL =3D 0xc, + POWERCAP_CPC_ATTRIBUTES =3D 0xd, }; =20 enum { @@ -69,17 +70,56 @@ struct scmi_msg_resp_powercap_domain_attributes { __le32 parent_id; }; =20 +struct scmi_msg_resp_powercap_domain_attributes_v3 { + __le32 attributes; +#define SUPPORTS_POWERCAP_MAI_CONFIGURATION(x) ((x) & BIT(25)) +#define SUPPORTS_POWERCAP_FASTCHANNELS(x) ((x) & BIT(22)) +#define SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY_V3(x) ((x) & BIT(21)) +#define SUPPORTS_POWERCAP_CAI_CONFIGURATION(x) ((x) & BIT(20)) + u8 name[SCMI_SHORT_NAME_MAX_SIZE]; + __le32 min_mai; + __le32 max_mai; + __le32 mai_step; + __le32 min_power_cap; + __le32 max_power_cap; + __le32 power_cap_step; + __le32 sustainable_power; + __le32 accuracy; + __le32 parent_id; + __le32 min_cai; + __le32 max_cai; + __le32 cai_step; +}; + +struct scmi_msg_powercap_get_v3 { + __le32 domain_id; + __le32 cpli; +}; + struct scmi_msg_powercap_set_cap_or_pai { - __le32 domain; + __le32 domain_id; __le32 flags; #define CAP_SET_ASYNC BIT(1) #define CAP_SET_IGNORE_DRESP BIT(0) __le32 value; }; =20 +struct scmi_msg_powercap_set_cap_v3 { + __le32 domain_id; + __le32 cpli; + __le32 flags; + __le32 power_cap; +}; + struct scmi_msg_resp_powercap_cap_set_complete { - __le32 domain; + __le32 domain_id; + __le32 power_cap; +}; + +struct scmi_msg_resp_powercap_cap_set_complete_v3 { + __le32 domain_id; __le32 power_cap; + __le32 cpli; }; =20 struct scmi_msg_resp_powercap_meas_get { @@ -112,6 +152,33 @@ struct scmi_powercap_meas_changed_notify_payld { __le32 power; }; =20 +struct scmi_msg_powercap_cpc { + __le32 domain_id; + __le32 desc_index; +}; + +struct scmi_msg_resp_powercap_cpc { + __le32 num_cpl; +#define NUM_RETURNED(n) (le32_get_bits((n), GENMASK(15, 0))) +#define NUM_REMAINING(n) (le32_get_bits((n), GENMASK(31, 16))) + struct { + __le32 cpli; + __le32 flags; + __le32 min_power_cap; + __le32 max_power_cap; + __le32 power_cap_step; + __le32 min_cai; + __le32 max_cai; + __le32 cai_step; + u8 name[SCMI_SHORT_NAME_MAX_SIZE]; + } desc[]; +}; + +struct scmi_cpls_priv { + u32 domain_id; + struct scmi_powercap_cpl_info *cpli; +}; + struct scmi_powercap_state { bool enabled; u32 last_pcap; @@ -130,6 +197,11 @@ struct powercap_info { bool notify_measurements_cmd; struct scmi_powercap_state *states; struct scmi_powercap_info *powercaps; + int (*xfer_cap_get)(const struct scmi_protocol_handle *ph, + u32 domain_id, u32 cpl_id, u32 *power_cap); + int (*xfer_cap_set)(const struct scmi_protocol_handle *ph, + const struct scmi_powercap_info *pc, + u32 cpl_id, u32 power_cap, bool ignore_dresp); }; =20 static enum scmi_powercap_protocol_cmd evt_2_cmd[] =3D { @@ -193,111 +265,241 @@ scmi_powercap_validate(unsigned int min_val, unsign= ed int max_val, return 0; } =20 +static void iter_powercap_cpls_prepare_message(void *message, + unsigned int desc_index, + const void *priv) +{ + struct scmi_msg_powercap_cpc *msg =3D message; + const struct scmi_cpls_priv *p =3D priv; + + msg->domain_id =3D cpu_to_le32(p->domain_id); + msg->desc_index =3D cpu_to_le32(desc_index); +} + +static int iter_powercap_cpls_update_state(struct scmi_iterator_state *st, + const void *response, void *priv) +{ + const struct scmi_msg_resp_powercap_cpc *r =3D response; + + st->num_returned =3D NUM_RETURNED(r->num_cpl); + st->num_remaining =3D NUM_REMAINING(r->num_cpl); + + return 0; +} + static int -scmi_powercap_domain_attributes_get(const struct scmi_protocol_handle *ph, - struct powercap_info *pinfo, - struct scmi_powercap_info *dom_info) +iter_powercap_cpls_process_response(const struct scmi_protocol_handle *ph, + const void *response, + struct scmi_iterator_state *st, void *priv) { - int ret; - u32 flags; - struct scmi_xfer *t; - struct scmi_msg_resp_powercap_domain_attributes *resp; + const struct scmi_msg_resp_powercap_cpc *r =3D response; + struct scmi_cpls_priv *p =3D priv; + struct scmi_powercap_cpl_info *cpl; =20 - ret =3D ph->xops->xfer_get_init(ph, POWERCAP_DOMAIN_ATTRIBUTES, - sizeof(dom_info->id), sizeof(*resp), &t); - if (ret) - return ret; + cpl =3D &p->cpli[st->desc_index + st->loop_idx]; =20 - put_unaligned_le32(dom_info->id, t->tx.buf); - resp =3D t->rx.buf; + cpl->id =3D le32_to_cpu(r->desc[st->loop_idx].cpli); + cpl->cap_config =3D r->desc[st->loop_idx].flags & BIT(0); =20 - ret =3D ph->xops->do_xfer(ph, t); - if (!ret) { - flags =3D le32_to_cpu(resp->attributes); + cpl->min_power_cap =3D le32_to_cpu(r->desc[st->loop_idx].min_power_cap); + cpl->max_power_cap =3D le32_to_cpu(r->desc[st->loop_idx].max_power_cap); + cpl->power_cap_step =3D le32_to_cpu(r->desc[st->loop_idx].power_cap_step); + if (!cpl->power_cap_step && cpl->min_power_cap !=3D cpl->max_power_cap) + return -EINVAL; + + cpl->min_avg_ivl =3D le32_to_cpu(r->desc[st->loop_idx].min_cai); + cpl->max_avg_ivl =3D le32_to_cpu(r->desc[st->loop_idx].max_cai); + cpl->avg_ivl_step =3D le32_to_cpu(r->desc[st->loop_idx].cai_step); + if (!cpl->avg_ivl_step && cpl->min_avg_ivl !=3D cpl->max_avg_ivl) + return -EINVAL; =20 - if (pinfo->notify_cap_cmd) + cpl->avg_ivl_config =3D cpl->min_avg_ivl !=3D cpl->max_avg_ivl; + + strscpy(cpl->name, r->desc[st->loop_idx].name, SCMI_SHORT_NAME_MAX_SIZE); + + return 0; +} + +static int scmi_powercap_cpls_enumerate(const struct scmi_protocol_handle = *ph, + struct scmi_powercap_info *dom_info) +{ + void *iter; + struct scmi_iterator_ops ops =3D { + .prepare_message =3D iter_powercap_cpls_prepare_message, + .update_state =3D iter_powercap_cpls_update_state, + .process_response =3D iter_powercap_cpls_process_response, + }; + struct scmi_cpls_priv cpriv =3D { + .domain_id =3D dom_info->id, + .cpli =3D dom_info->cpli, + }; + + iter =3D ph->hops->iter_response_init(ph, &ops, dom_info->num_cpli, + POWERCAP_CPC_ATTRIBUTES, + sizeof(struct scmi_msg_powercap_cpc), + &cpriv); + if (IS_ERR(iter)) + return PTR_ERR(iter); + + return ph->hops->iter_response_run(iter); +} + +static int +scmi_powercap_domain_attrs_process(const struct scmi_protocol_handle *ph, + struct powercap_info *pinfo, + struct scmi_powercap_info *dom_info, void *r) +{ + struct scmi_msg_resp_powercap_domain_attributes *resp =3D r; + u32 flags =3D resp->attributes; + bool cap_config; + int ret; + + cap_config =3D SUPPORTS_POWERCAP_CAP_CONFIGURATION(flags); + if (PROTOCOL_REV_MAJOR(pinfo->version) < 0x3) { + dom_info->num_cpli =3D 1; + } else { + dom_info->num_cpli =3D le32_get_bits(resp->attributes, + GENMASK(18, 15)); + if (cap_config && !dom_info->num_cpli) + return -EINVAL; + } + + dom_info->cpli =3D devm_kcalloc(ph->dev, dom_info->num_cpli, + sizeof(*dom_info->cpli), GFP_KERNEL); + if (!dom_info->cpli) + return -ENOMEM; + + if (pinfo->notify_cap_cmd) { + if (PROTOCOL_REV_MAJOR(pinfo->version) < 0x3) dom_info->notify_powercap_cap_change =3D SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY(flags); - if (pinfo->notify_measurements_cmd) - dom_info->notify_powercap_measurement_change =3D - SUPPORTS_POWERCAP_MEASUREMENTS_CHANGE_NOTIFY(flags); - dom_info->async_powercap_cap_set =3D - SUPPORTS_ASYNC_POWERCAP_CAP_SET(flags); - - dom_info->powercap_monitoring =3D - SUPPORTS_POWERCAP_MONITORING(flags); - dom_info->powercap_scale_mw =3D - SUPPORTS_POWER_UNITS_MW(flags); - dom_info->powercap_scale_uw =3D - SUPPORTS_POWER_UNITS_UW(flags); - dom_info->fastchannels =3D - SUPPORTS_POWERCAP_FASTCHANNELS(flags); - - strscpy(dom_info->name, resp->name, SCMI_SHORT_NAME_MAX_SIZE); - - dom_info->sustainable_power =3D - le32_to_cpu(resp->sustainable_power); - dom_info->accuracy =3D le32_to_cpu(resp->accuracy); - - dom_info->parent_id =3D le32_to_cpu(resp->parent_id); - if (dom_info->parent_id !=3D SCMI_POWERCAP_ROOT_ZONE_ID && - (dom_info->parent_id >=3D pinfo->num_domains || - dom_info->parent_id =3D=3D dom_info->id)) { - dev_err(ph->dev, - "Platform reported inconsistent parent ID for domain %d - %s\n", - dom_info->id, dom_info->name); - ret =3D -ENODEV; - } + else + dom_info->notify_powercap_cap_change =3D + SUPPORTS_POWERCAP_CAP_CHANGE_NOTIFY_V3(flags); + } + + if (pinfo->notify_measurements_cmd) + dom_info->notify_powercap_measurement_change =3D + SUPPORTS_POWERCAP_MEASUREMENTS_CHANGE_NOTIFY(flags); + =20 + dom_info->extended_names =3D SUPPORTS_EXTENDED_NAMES(flags); + + dom_info->async_powercap_cap_set =3D + SUPPORTS_ASYNC_POWERCAP_CAP_SET(flags); + + dom_info->powercap_monitoring =3D + SUPPORTS_POWERCAP_MONITORING(flags); + dom_info->powercap_scale_mw =3D + SUPPORTS_POWER_UNITS_MW(flags); + dom_info->powercap_scale_uw =3D + SUPPORTS_POWER_UNITS_UW(flags); + dom_info->fastchannels =3D + SUPPORTS_POWERCAP_FASTCHANNELS(flags); + + strscpy(dom_info->name, resp->name, SCMI_SHORT_NAME_MAX_SIZE); + + dom_info->sustainable_power =3D + le32_to_cpu(resp->sustainable_power); + dom_info->accuracy =3D le32_to_cpu(resp->accuracy); + + dom_info->parent_id =3D le32_to_cpu(resp->parent_id); + if (dom_info->parent_id !=3D SCMI_POWERCAP_ROOT_ZONE_ID && + (dom_info->parent_id >=3D pinfo->num_domains || + dom_info->parent_id =3D=3D dom_info->id)) { + dev_err(ph->dev, + "Platform reported inconsistent parent ID for domain %d - %s\n", + dom_info->id, dom_info->name); + return -ENODEV; + } + + dom_info->cpli[0].id =3D CPL0; + if (PROTOCOL_REV_MAJOR(pinfo->version) < 0x3) dom_info->cpli[0].avg_ivl_config =3D SUPPORTS_POWERCAP_PAI_CONFIGURATION(flags); + else + dom_info->cpli[0].avg_ivl_config =3D + SUPPORTS_POWERCAP_CAI_CONFIGURATION(flags); + + if (PROTOCOL_REV_MAJOR(pinfo->version) < 0x3) { dom_info->cpli[0].min_avg_ivl =3D le32_to_cpu(resp->min_pai); dom_info->cpli[0].max_avg_ivl =3D le32_to_cpu(resp->max_pai); dom_info->cpli[0].avg_ivl_step =3D le32_to_cpu(resp->pai_step); - ret =3D scmi_powercap_validate(dom_info->cpli[0].min_avg_ivl, - dom_info->cpli[0].max_avg_ivl, - dom_info->cpli[0].avg_ivl_step, - dom_info->cpli[0].avg_ivl_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; - } + } else { + struct scmi_msg_resp_powercap_domain_attributes_v3 *resp =3D r; =20 - dom_info->cpli[0].cap_config =3D - SUPPORTS_POWERCAP_CAP_CONFIGURATION(flags); - dom_info->cpli[0].min_power_cap =3D le32_to_cpu(resp->min_power_cap); - dom_info->cpli[0].max_power_cap =3D le32_to_cpu(resp->max_power_cap); - dom_info->cpli[0].power_cap_step =3D le32_to_cpu(resp->power_cap_step); - ret =3D scmi_powercap_validate(dom_info->cpli[0].min_power_cap, - dom_info->cpli[0].max_power_cap, - dom_info->cpli[0].power_cap_step, - dom_info->cpli[0].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->cpli[0].min_avg_ivl =3D le32_to_cpu(resp->min_cai); + dom_info->cpli[0].max_avg_ivl =3D le32_to_cpu(resp->max_cai); + dom_info->cpli[0].avg_ivl_step =3D le32_to_cpu(resp->cai_step); + } =20 - /* Just using same short name */ - strscpy(dom_info->cpli[0].name, dom_info->name, - SCMI_SHORT_NAME_MAX_SIZE); + ret =3D scmi_powercap_validate(dom_info->cpli[0].min_avg_ivl, + dom_info->cpli[0].max_avg_ivl, + dom_info->cpli[0].avg_ivl_step, + dom_info->cpli[0].avg_ivl_config); + if (ret) { + dev_err(ph->dev, + "Platform reported inconsistent PAI config for domain %d - %s\n", + dom_info->id, dom_info->name); + return ret; } =20 -clean: + dom_info->cpli[0].cap_config =3D cap_config; + dom_info->cpli[0].min_power_cap =3D le32_to_cpu(resp->min_power_cap); + dom_info->cpli[0].max_power_cap =3D le32_to_cpu(resp->max_power_cap); + dom_info->cpli[0].power_cap_step =3D le32_to_cpu(resp->power_cap_step); + ret =3D scmi_powercap_validate(dom_info->cpli[0].min_power_cap, + dom_info->cpli[0].max_power_cap, + dom_info->cpli[0].power_cap_step, + dom_info->cpli[0].cap_config); + if (ret) { + dev_err(ph->dev, + "Platform reported inconsistent CAP config for domain %d - %s\n", + dom_info->id, dom_info->name); + return ret; + } + /* Just using same short name */ + strscpy(dom_info->cpli[0].name, dom_info->name, SCMI_SHORT_NAME_MAX_SIZE); + + return 0; +} + +static int +scmi_powercap_domain_attributes_get(const struct scmi_protocol_handle *ph, + struct powercap_info *pinfo, + struct scmi_powercap_info *dom_info) +{ + int ret; + struct scmi_xfer *t; + struct scmi_msg_resp_powercap_domain_attributes *resp; + + ret =3D ph->xops->xfer_get_init(ph, POWERCAP_DOMAIN_ATTRIBUTES, + sizeof(dom_info->id), 0, &t); + if (ret) + return ret; + + put_unaligned_le32(dom_info->id, t->tx.buf); + resp =3D t->rx.buf; + + ret =3D ph->xops->do_xfer(ph, t); + if (!ret) + ret =3D scmi_powercap_domain_attrs_process(ph, pinfo, dom_info, resp); + ph->xops->xfer_put(ph, t); =20 /* * 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)) + if (!ret && dom_info->extended_names) ph->hops->extended_name_get(ph, POWERCAP_DOMAIN_NAME_GET, dom_info->id, NULL, dom_info->name, SCMI_MAX_STR_SIZE); =20 + /* When protocol version > 0x3 there can possibly be more than 1 CPLs */ + if (!ret && dom_info->num_cpli > 1) + ret =3D scmi_powercap_cpls_enumerate(ph, dom_info); + return ret; } =20 @@ -307,14 +509,7 @@ scmi_powercap_domain_initialize(const struct scmi_prot= ocol_handle *ph, { struct scmi_powercap_info *dom_info =3D pinfo->powercaps + domain; =20 - dom_info->num_cpli =3D 1; - dom_info->cpli =3D devm_kcalloc(ph->dev, dom_info->num_cpli, - sizeof(*dom_info->cpli), GFP_KERNEL); - if (!dom_info->cpli) - return -ENOMEM; - dom_info->id =3D domain; - dom_info->cpli[0].id =3D CPL0; =20 return scmi_powercap_domain_attributes_get(ph, pinfo, dom_info); } @@ -338,7 +533,7 @@ scmi_powercap_dom_info_get(const struct scmi_protocol_h= andle *ph, u32 domain_id) } =20 static int scmi_powercap_xfer_cap_get(const struct scmi_protocol_handle *p= h, - u32 domain_id, u32 *power_cap) + u32 domain_id, u32 cpl_id, u32 *power_cap) { int ret; struct scmi_xfer *t; @@ -349,6 +544,33 @@ static int scmi_powercap_xfer_cap_get(const struct scm= i_protocol_handle *ph, return ret; =20 put_unaligned_le32(domain_id, t->tx.buf); + + ret =3D ph->xops->do_xfer(ph, t); + if (!ret) + *power_cap =3D get_unaligned_le32(t->rx.buf); + + ph->xops->xfer_put(ph, t); + + return ret; +} + +static int scmi_powercap_xfer_cap_get_v3(const struct scmi_protocol_handle= *ph, + u32 domain_id, u32 cpl_id, + u32 *power_cap) +{ + int ret; + struct scmi_xfer *t; + struct scmi_msg_powercap_get_v3 *msg; + + ret =3D ph->xops->xfer_get_init(ph, POWERCAP_CAP_GET, sizeof(*msg), + sizeof(u32), &t); + if (ret) + return ret; + + msg =3D t->tx.buf; + msg->domain_id =3D cpu_to_le32(domain_id); + msg->cpli =3D cpu_to_le32(cpl_id); + ret =3D ph->xops->do_xfer(ph, t); if (!ret) *power_cap =3D get_unaligned_le32(t->rx.buf); @@ -362,6 +584,8 @@ static int __scmi_powercap_cap_get(const struct scmi_pr= otocol_handle *ph, const struct scmi_powercap_info *dom, u32 cpl_id, u32 *power_cap) { + struct powercap_info *pi =3D ph->get_priv(ph); + if (dom->cpli[cpl_id].fc_info && dom->cpli[cpl_id].fc_info[POWERCAP_FC_CAP].get_addr) { *power_cap =3D ioread32(dom->cpli[cpl_id].fc_info[POWERCAP_FC_CAP].get_a= ddr); @@ -370,7 +594,7 @@ static int __scmi_powercap_cap_get(const struct scmi_pr= otocol_handle *ph, return 0; } =20 - return scmi_powercap_xfer_cap_get(ph, dom->id, power_cap); + return pi->xfer_cap_get(ph, dom->id, cpl_id, power_cap); } =20 static int scmi_powercap_cap_get(const struct scmi_protocol_handle *ph, @@ -403,7 +627,7 @@ static int scmi_powercap_xfer_cap_set(const struct scmi= _protocol_handle *ph, return ret; =20 msg =3D t->tx.buf; - msg->domain =3D cpu_to_le32(pc->id); + msg->domain_id =3D cpu_to_le32(pc->id); msg->flags =3D cpu_to_le32(FIELD_PREP(CAP_SET_ASYNC, pc->async_powercap_cap_set) | FIELD_PREP(CAP_SET_IGNORE_DRESP, ignore_dresp)); @@ -417,7 +641,7 @@ static int scmi_powercap_xfer_cap_set(const struct scmi= _protocol_handle *ph, struct scmi_msg_resp_powercap_cap_set_complete *resp; =20 resp =3D t->rx.buf; - if (le32_to_cpu(resp->domain) =3D=3D pc->id) + if (le32_to_cpu(resp->domain_id) =3D=3D pc->id) dev_dbg(ph->dev, "Powercap ID %d CAP set async to %u\n", pc->id, @@ -431,6 +655,51 @@ static int scmi_powercap_xfer_cap_set(const struct scm= i_protocol_handle *ph, return ret; } =20 +static int scmi_powercap_xfer_cap_set_v3(const struct scmi_protocol_handle= *ph, + const struct scmi_powercap_info *pc, + u32 cpl_id, u32 power_cap, + bool ignore_dresp) +{ + int ret; + struct scmi_xfer *t; + struct scmi_msg_powercap_set_cap_v3 *msg; + + ret =3D ph->xops->xfer_get_init(ph, POWERCAP_CAP_SET, + sizeof(*msg), 0, &t); + if (ret) + return ret; + + msg =3D t->tx.buf; + msg->domain_id =3D cpu_to_le32(pc->id); + msg->cpli =3D cpu_to_le32(cpl_id); + msg->flags =3D + cpu_to_le32(FIELD_PREP(CAP_SET_ASYNC, pc->async_powercap_cap_set) | + FIELD_PREP(CAP_SET_IGNORE_DRESP, ignore_dresp)); + msg->power_cap =3D cpu_to_le32(power_cap); + + if (!pc->async_powercap_cap_set || ignore_dresp) { + ret =3D ph->xops->do_xfer(ph, t); + } else { + ret =3D ph->xops->do_xfer_with_response(ph, t); + if (!ret) { + struct scmi_msg_resp_powercap_cap_set_complete_v3 *resp; + + resp =3D t->rx.buf; + if (le32_to_cpu(resp->domain_id) =3D=3D pc->id && + le32_to_cpu(resp->cpli) =3D=3D pc->cpli[cpl_id].id) + dev_dbg(ph->dev, + "Powercap ID:%d/CPLI:%d CAP set async to %u\n", + pc->id, cpl_id, + get_unaligned_le32(&resp->power_cap)); + else + ret =3D -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 cpl_id, u32 power_cap, bool ignore_dresp) @@ -457,12 +726,12 @@ static int __scmi_powercap_cap_set(const struct scmi_= protocol_handle *ph, domain_id, power_cap, 0); ret =3D 0; } else { - ret =3D scmi_powercap_xfer_cap_set(ph, pc, cpl_id, power_cap, - ignore_dresp); + ret =3D pi->xfer_cap_set(ph, pc, cpl_id, power_cap, ignore_dresp); } =20 - /* Save the last explicitly set non-zero powercap value */ - if (PROTOCOL_REV_MAJOR(pi->version) >=3D 0x2 && !ret && power_cap) + /* Save the last explicitly set non-zero powercap value for CPL0 */ + if (PROTOCOL_REV_MAJOR(pi->version) >=3D 0x2 && !ret && + cpl_id =3D=3D CPL0 && power_cap) pi->states[domain_id].last_pcap =3D power_cap; =20 return ret; @@ -481,8 +750,8 @@ static int scmi_powercap_cap_set(const struct scmi_prot= ocol_handle *ph, if (!power_cap) return -EINVAL; =20 - /* Just log the last set request if acting on a disabled domain */ - if (PROTOCOL_REV_MAJOR(pi->version) >=3D 0x2 && + /* Just log the last set request on CPL0 on a disabled domain */ + if (PROTOCOL_REV_MAJOR(pi->version) >=3D 0x2 && cpl_id =3D=3D CPL0 && !pi->states[domain_id].enabled) { pi->states[domain_id].last_pcap =3D power_cap; return 0; @@ -555,7 +824,7 @@ static int scmi_powercap_xfer_pai_set(const struct scmi= _protocol_handle *ph, return ret; =20 msg =3D t->tx.buf; - msg->domain =3D cpu_to_le32(domain_id); + msg->domain_id =3D cpu_to_le32(domain_id); msg->flags =3D cpu_to_le32(0); msg->value =3D cpu_to_le32(pai); =20 @@ -1019,6 +1288,17 @@ scmi_powercap_protocol_init(const struct scmi_protoc= ol_handle *ph) if (!pinfo) return -ENOMEM; =20 + pinfo->version =3D version; + ph->set_priv(ph, pinfo, version); + + if (PROTOCOL_REV_MAJOR(version) < 0x3) { + pinfo->xfer_cap_get =3D scmi_powercap_xfer_cap_get; + pinfo->xfer_cap_set =3D scmi_powercap_xfer_cap_set; + } else { + pinfo->xfer_cap_get =3D scmi_powercap_xfer_cap_get_v3; + pinfo->xfer_cap_set =3D scmi_powercap_xfer_cap_set_v3; + } + ret =3D scmi_powercap_attributes_get(ph, pinfo); if (ret) return ret; @@ -1062,8 +1342,7 @@ scmi_powercap_protocol_init(const struct scmi_protoco= l_handle *ph) } } =20 - pinfo->version =3D version; - return ph->set_priv(ph, pinfo, version); + return 0; } =20 static const struct scmi_protocol scmi_powercap =3D { diff --git a/include/linux/scmi_protocol.h b/include/linux/scmi_protocol.h index 0f48d8dcf561..a98213bff60a 100644 --- a/include/linux/scmi_protocol.h +++ b/include/linux/scmi_protocol.h @@ -673,6 +673,7 @@ struct scmi_powercap_info { bool powercap_monitoring; bool powercap_scale_mw; bool powercap_scale_uw; + bool extended_names; bool fastchannels; char name[SCMI_MAX_STR_SIZE]; unsigned int sustainable_power; --=20 2.47.0