Message ID | 20230726083808.140780-2-takahiro.akashi@linaro.org |
---|---|
State | Superseded |
Headers | show |
Series | firmware: scmi: add SCMI base protocol support | expand |
On Wed, 26 Jul 2023 at 02:38, AKASHI Takahiro <takahiro.akashi@linaro.org> wrote: > > The commit 85dc58289238 ("firmware: scmi: prepare uclass to pass channel > reference") added an explicit parameter, channel, but it seems to make > the code complex. > > Hiding this parameter will allow for adding a generic (protocol-agnostic) > helper function, i.e. for PROTOCOL_VERSION, in a later patch. > > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > --- > v2 > * new patch > --- > drivers/clk/clk_scmi.c | 27 ++------- > drivers/firmware/scmi/scmi_agent-uclass.c | 74 ++++++++++++++++++----- > drivers/power/regulator/scmi_regulator.c | 27 +++------ > drivers/reset/reset-scmi.c | 19 +----- > include/scmi_agent.h | 15 +++-- > 5 files changed, 86 insertions(+), 76 deletions(-) > Reviewed-by: Simon Glass <sjg@chromium.org>
Hello Takahiro-san, > From: U-Boot <u-boot-bounces@lists.denx.de> on behalf of Simon Glass <sjg@chromium.org> > Sent: Thursday, July 27, 2023 2:50 AM > > On Wed, 26 Jul 2023 at 02:38, AKASHI Takahiro > <takahiro.akashi@linaro.org> wrote: > > > > The commit 85dc58289238 ("firmware: scmi: prepare uclass to pass channel > > reference") added an explicit parameter, channel, but it seems to make > > the code complex. > > > > Hiding this parameter will allow for adding a generic (protocol-agnostic) > > helper function, i.e. for PROTOCOL_VERSION, in a later patch. > > > > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > > --- > > v2 > > * new patch > > --- > > drivers/clk/clk_scmi.c | 27 ++------- > > drivers/firmware/scmi/scmi_agent-uclass.c | 74 ++++++++++++++++++----- > > drivers/power/regulator/scmi_regulator.c | 27 +++------ > > drivers/reset/reset-scmi.c | 19 +----- > > include/scmi_agent.h | 15 +++-- > > 5 files changed, 86 insertions(+), 76 deletions(-) > > > > Reviewed-by: Simon Glass <sjg@chromium.org> Sorry for this late feedback. I initially saw no issues with this patch series but testing the series against stm32mp135f-dk board which is using SCMI resources, I saw the board failed to boot. The issue is related to udevice parent tree and common clock framework. I'll post comments on patch 01/12. Best regards, Etienne
Hello Takahiro-san, Sorry again for this late feedback. Testing the series against stm32mp135f-dk board which is using SCMI resources, I see the board fails to boot. > From: AKASHI Takahiro <takahiro.akashi@linaro.org> > To: trini@konsulko.com, sjg@chromium.org > Cc: etienne.carriere@st.com, u-boot@lists.denx.de, > AKASHI Takahiro <takahiro.akashi@linaro.org> > Subject: [PATCH v2 01/12] scmi: refactor the code to hide a channel from devices > Date: Wed, 26 Jul 2023 17:37:57 +0900 [thread overview] > Message-ID: <20230726083808.140780-2-takahiro.akashi@linaro.org> (raw) > In-Reply-To: <20230726083808.140780-1-takahiro.akashi@linaro.org> > > The commit 85dc58289238 ("firmware: scmi: prepare uclass to pass channel > reference") added an explicit parameter, channel, but it seems to make > the code complex. > > Hiding this parameter will allow for adding a generic (protocol-agnostic) > helper function, i.e. for PROTOCOL_VERSION, in a later patch. > > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > > --- > v2 > * new patch > --- > drivers/clk/clk_scmi.c | 27 ++------- > drivers/firmware/scmi/scmi_agent-uclass.c | 74 ++++++++++++++++++----- > drivers/power/regulator/scmi_regulator.c | 27 +++------ > drivers/reset/reset-scmi.c | 19 +----- > include/scmi_agent.h | 15 +++-- > 5 files changed, 86 insertions(+), 76 deletions(-) > > diff --git a/drivers/clk/clk_scmi.c b/drivers/clk/clk_scmi.c > index d172fed24c9d..34a49363a51a 100644 > --- a/drivers/clk/clk_scmi.c > +++ b/drivers/clk/clk_scmi.c > @@ -13,17 +13,8 @@ > #include <asm/types.h> > #include <linux/clk-provider.h> > > -/** > - * struct scmi_clk_priv - Private data for SCMI clocks > - * @channel: Reference to the SCMI channel to use > - */ > -struct scmi_clk_priv { > - struct scmi_channel *channel; > -}; > - > static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > { > - struct scmi_clk_priv *priv = dev_get_priv(dev); > struct scmi_clk_protocol_attr_out out; > struct scmi_msg msg = { > .protocol_id = SCMI_PROTOCOL_ID_CLOCK, > @@ -33,7 +24,7 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > }; > int ret; > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(dev, &msg); > if (ret) > return ret; > > @@ -44,7 +35,6 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > { > - struct scmi_clk_priv *priv = dev_get_priv(dev); > struct scmi_clk_attribute_in in = { > .clock_id = clkid, > }; > @@ -59,7 +49,7 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > }; > int ret; > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(dev, &msg); > if (ret) > return ret; > > @@ -70,7 +60,6 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > static int scmi_clk_gate(struct clk *clk, int enable) > { > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > struct scmi_clk_state_in in = { > .clock_id = clk->id, > .attributes = enable, > @@ -81,7 +70,7 @@ static int scmi_clk_gate(struct clk *clk, int enable) > in, out); > int ret; > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(clk->dev, &msg); > if (ret) > return ret; > > @@ -100,7 +89,6 @@ static int scmi_clk_disable(struct clk *clk) > > static ulong scmi_clk_get_rate(struct clk *clk) > { > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > struct scmi_clk_rate_get_in in = { > .clock_id = clk->id, > }; > @@ -110,7 +98,7 @@ static ulong scmi_clk_get_rate(struct clk *clk) > in, out); > int ret; > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(clk->dev, &msg); > if (ret < 0) > return ret; > > @@ -123,7 +111,6 @@ static ulong scmi_clk_get_rate(struct clk *clk) > > static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > { > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > struct scmi_clk_rate_set_in in = { > .clock_id = clk->id, > .flags = SCMI_CLK_RATE_ROUND_CLOSEST, > @@ -136,7 +123,7 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > in, out); > int ret; > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(clk->dev, &msg); > if (ret < 0) > return ret; > > @@ -149,12 +136,11 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > static int scmi_clk_probe(struct udevice *dev) > { > - struct scmi_clk_priv *priv = dev_get_priv(dev); > struct clk *clk; > size_t num_clocks, i; > int ret; > > - ret = devm_scmi_of_get_channel(dev, &priv->channel); > + ret = devm_scmi_of_get_channel(dev); > if (ret) > return ret; > > @@ -205,5 +191,4 @@ U_BOOT_DRIVER(scmi_clock) = { > .id = UCLASS_CLK, > .ops = &scmi_clk_ops, > .probe = scmi_clk_probe, > - .priv_auto = sizeof(struct scmi_clk_priv *), > }; > diff --git a/drivers/firmware/scmi/scmi_agent-uclass.c b/drivers/firmware/scmi/scmi_agent-uclass.c > index 02de692d66f3..39cf15c88f75 100644 > --- a/drivers/firmware/scmi/scmi_agent-uclass.c > +++ b/drivers/firmware/scmi/scmi_agent-uclass.c > @@ -8,6 +8,7 @@ > #include <common.h> > #include <dm.h> > #include <errno.h> > +#include <scmi_agent.h> > #include <scmi_agent-uclass.h> > #include <scmi_protocols.h> > #include <dm/device_compat.h> > @@ -129,43 +130,88 @@ static const struct scmi_agent_ops *transport_dev_ops(struct udevice *dev) > return (const struct scmi_agent_ops *)dev->driver->ops; > } > > -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > +/** > + * scmi_of_get_channel() - Get SCMI channel handle > + * > + * @dev: SCMI agent device > + * @channel: Output reference to the SCMI channel upon success > + * > + * On return, @channel will be set. > + * Return 0 on success and a negative errno on failure > + */ > +static int scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > +{ > + const struct scmi_agent_ops *ops; > + > + ops = transport_dev_ops(dev); > + if (ops->of_get_channel) > + return ops->of_get_channel(dev, channel); > + else > + return -EPROTONOSUPPORT; > +} > + > +int devm_scmi_of_get_channel(struct udevice *dev) > { > struct udevice *parent; > + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); This parent lookup does not match when CONFIG_CLK_CCF is enabled. When so, the scmi clock device is on the clock itself but a child udevice of some udevice held by the common clock framework (CCF). Platforms from the stm32mp13 family do enable SCMI clock and CCF. It can propose a way to go, using find_scmi_transport_device() to provide the agent's 1st child private data as an output argument: //patch-start -static struct udevice *find_scmi_transport_device(struct udevice *dev) +static struct udevice * +find_scmi_transport_device(struct udevice *dev, + struct scmi_agent_proto_priv **priv) { struct udevice *parent = dev; + struct scmi_agent_proto_priv *parent_priv; do { + parent_priv = dev_get_parent_priv(parent); parent = dev_get_parent(parent); } while (parent && device_get_uclass_id(parent) != UCLASS_SCMI_AGENT); if (!parent) dev_err(dev, "Invalid SCMI device, agent not found\n"); + else if (priv) + *priv = parent_priv; return parent; } //end patch Tested OK on stm32mp13 with the remaining patch series applied (minor conflics). BR, etienne > + int ret; > > parent = find_scmi_transport_device(dev); > if (!parent) > return -ENODEV; > > - if (transport_dev_ops(parent)->of_get_channel) > - return transport_dev_ops(parent)->of_get_channel(parent, channel); > + ret = scmi_of_get_channel(parent, &priv->channel); > + if (ret == -EPROTONOSUPPORT) { > + /* Drivers without a get_channel operator don't need a channel ref */ > + priv->channel = NULL; > > - /* Drivers without a get_channel operator don't need a channel ref */ > - *channel = NULL; > + return 0; > + } > > - return 0; > + return ret; > } > > -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > - struct scmi_msg *msg) > +/** > + * scmi_process_msg() - Send and process an SCMI message > + * > + * Send a message to an SCMI server. > + * Caller sets scmi_msg::out_msg_sz to the output message buffer size. > + * > + * @dev: SCMI agent device > + * @channel: Communication channel for the device > + * @msg: Message structure reference > + * > + * On return, scmi_msg::out_msg_sz stores the response payload size. > + * Return: 0 on success and a negative errno on failure > + */ > +static int scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > + struct scmi_msg *msg) > { > const struct scmi_agent_ops *ops; > + > + ops = transport_dev_ops(dev); > + if (ops->process_msg) > + return ops->process_msg(dev, channel, msg); > + else > + return -EPROTONOSUPPORT; > +} > + > +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg) > +{ > struct udevice *parent; > + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); > > parent = find_scmi_transport_device(dev); > if (!parent) > return -ENODEV; > > - ops = transport_dev_ops(parent); > - > - if (ops->process_msg) > - return ops->process_msg(parent, channel, msg); > - > - return -EPROTONOSUPPORT; > + return scmi_process_msg(parent, priv->channel, msg); > } > > UCLASS_DRIVER(scmi_agent) = { > .id = UCLASS_SCMI_AGENT, > .name = "scmi_agent", > .post_bind = scmi_bind_protocols, > + .per_device_plat_auto = sizeof(struct scmi_agent_priv), Typo: this added line belong to patch 04/12. Regards, Etienne > + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), > }; > diff --git a/drivers/power/regulator/scmi_regulator.c b/drivers/power/regulator/scmi_regulator.c > index 801148036ff6..08918b20872c 100644 > --- a/drivers/power/regulator/scmi_regulator.c > +++ b/drivers/power/regulator/scmi_regulator.c > @@ -25,18 +25,9 @@ struct scmi_regulator_platdata { > u32 domain_id; > }; > > -/** > - * struct scmi_regulator_priv - Private data for SCMI voltage regulator > - * @channel: Reference to the SCMI channel to use > - */ > -struct scmi_regulator_priv { > - struct scmi_channel *channel; > -}; > - > static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > { > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > struct scmi_voltd_config_set_in in = { > .domain_id = pdata->domain_id, > .config = enable ? SCMI_VOLTD_CONFIG_ON : SCMI_VOLTD_CONFIG_OFF, > @@ -47,7 +38,7 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > in, out); > int ret; > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(dev, &msg); > if (ret) > return ret; > > @@ -57,7 +48,6 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > static int scmi_voltd_get_enable(struct udevice *dev) > { > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > struct scmi_voltd_config_get_in in = { > .domain_id = pdata->domain_id, > }; > @@ -67,7 +57,7 @@ static int scmi_voltd_get_enable(struct udevice *dev) > in, out); > int ret; > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(dev, &msg); > if (ret < 0) > return ret; > > @@ -80,7 +70,6 @@ static int scmi_voltd_get_enable(struct udevice *dev) > > static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > { > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > struct scmi_voltd_level_set_in in = { > .domain_id = pdata->domain_id, > @@ -92,7 +81,7 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > in, out); > int ret; > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(dev, &msg); > if (ret < 0) > return ret; > > @@ -101,7 +90,6 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > static int scmi_voltd_get_voltage_level(struct udevice *dev) > { > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > struct scmi_voltd_level_get_in in = { > .domain_id = pdata->domain_id, > @@ -112,7 +100,7 @@ static int scmi_voltd_get_voltage_level(struct udevice *dev) > in, out); > int ret; > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(dev, &msg); > if (ret < 0) > return ret; > > @@ -140,7 +128,6 @@ static int scmi_regulator_of_to_plat(struct udevice *dev) > static int scmi_regulator_probe(struct udevice *dev) > { > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > struct scmi_voltd_attr_in in = { 0 }; > struct scmi_voltd_attr_out out = { 0 }; > struct scmi_msg scmi_msg = { > @@ -153,14 +140,14 @@ static int scmi_regulator_probe(struct udevice *dev) > }; > int ret; > > - ret = devm_scmi_of_get_channel(dev->parent, &priv->channel); > + ret = devm_scmi_of_get_channel(dev); > if (ret) > return ret; > > /* Check voltage domain is known from SCMI server */ > in.domain_id = pdata->domain_id; > > - ret = devm_scmi_process_msg(dev, priv->channel, &scmi_msg); > + ret = devm_scmi_process_msg(dev, &scmi_msg); > if (ret) { > dev_err(dev, "Failed to query voltage domain %u: %d\n", > pdata->domain_id, ret); > @@ -184,7 +171,6 @@ U_BOOT_DRIVER(scmi_regulator) = { > .probe = scmi_regulator_probe, > .of_to_plat = scmi_regulator_of_to_plat, > .plat_auto = sizeof(struct scmi_regulator_platdata), > - .priv_auto = sizeof(struct scmi_regulator_priv *), > }; > > static int scmi_regulator_bind(struct udevice *dev) > @@ -209,4 +195,5 @@ U_BOOT_DRIVER(scmi_voltage_domain) = { > .name = "scmi_voltage_domain", > .id = UCLASS_NOP, > .bind = scmi_regulator_bind, > + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), > }; > diff --git a/drivers/reset/reset-scmi.c b/drivers/reset/reset-scmi.c > index 122556162ec3..b76711f0a8fb 100644 > --- a/drivers/reset/reset-scmi.c > +++ b/drivers/reset/reset-scmi.c > @@ -13,17 +13,8 @@ > #include <scmi_protocols.h> > #include <asm/types.h> > > -/** > - * struct scmi_reset_priv - Private data for SCMI reset controller > - * @channel: Reference to the SCMI channel to use > - */ > -struct scmi_reset_priv { > - struct scmi_channel *channel; > -}; > - > static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) > { > - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); > struct scmi_rd_reset_in in = { > .domain_id = rst->id, > .flags = assert_not_deassert ? SCMI_RD_RESET_FLAG_ASSERT : 0, > @@ -35,7 +26,7 @@ static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) > in, out); > int ret; > > - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(rst->dev, &msg); > if (ret) > return ret; > > @@ -54,7 +45,6 @@ static int scmi_reset_deassert(struct reset_ctl *rst) > > static int scmi_reset_request(struct reset_ctl *rst) > { > - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); > struct scmi_rd_attr_in in = { > .domain_id = rst->id, > }; > @@ -68,7 +58,7 @@ static int scmi_reset_request(struct reset_ctl *rst) > * We don't really care about the attribute, just check > * the reset domain exists. > */ > - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); > + ret = devm_scmi_process_msg(rst->dev, &msg); > if (ret) > return ret; > > @@ -83,9 +73,7 @@ static const struct reset_ops scmi_reset_domain_ops = { > > static int scmi_reset_probe(struct udevice *dev) > { > - struct scmi_reset_priv *priv = dev_get_priv(dev); > - > - return devm_scmi_of_get_channel(dev, &priv->channel); > + return devm_scmi_of_get_channel(dev); > } > > U_BOOT_DRIVER(scmi_reset_domain) = { > @@ -93,5 +81,4 @@ U_BOOT_DRIVER(scmi_reset_domain) = { > .id = UCLASS_RESET, > .ops = &scmi_reset_domain_ops, > .probe = scmi_reset_probe, > - .priv_auto = sizeof(struct scmi_reset_priv *), > }; > diff --git a/include/scmi_agent.h b/include/scmi_agent.h > index ee6286366df7..577892029ff8 100644 > --- a/include/scmi_agent.h > +++ b/include/scmi_agent.h > @@ -15,6 +15,14 @@ > struct udevice; > struct scmi_channel; > > +/** > + * struct scmi_agent_proto_priv - Private data in device for SCMI agent > + * @channel: Reference to the SCMI channel to use > + */ > +struct scmi_agent_proto_priv { > + struct scmi_channel *channel; > +}; > + > /* > * struct scmi_msg - Context of a SCMI message sent and the response received > * > @@ -49,10 +57,9 @@ struct scmi_msg { > * devm_scmi_of_get_channel() - Get SCMI channel handle from SCMI agent DT node > * > * @dev: Device requesting a channel > - * @channel: Output reference to the SCMI channel upon success > * @return 0 on success and a negative errno on failure > */ > -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel); > +int devm_scmi_of_get_channel(struct udevice *dev); > > /** > * devm_scmi_process_msg() - Send and process an SCMI message > @@ -62,12 +69,10 @@ int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > * On return, scmi_msg::out_msg_sz stores the response payload size. > * > * @dev: SCMI device > - * @channel: Communication channel for the device > * @msg: Message structure reference > * Return: 0 on success and a negative errno on failure > */ > -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > - struct scmi_msg *msg); > +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg); > > /** > * scmi_to_linux_errno() - Convert an SCMI error code into a Linux errno code > -- > 2.41.0
Hi Etienne, On Thu, Aug 03, 2023 at 09:24:51AM +0000, Etienne CARRIERE wrote: > Hello Takahiro-san, > > Sorry again for this late feedback. > Testing the series against stm32mp135f-dk board which is using SCMI resources, I see the board fails to boot. > > > From: AKASHI Takahiro <takahiro.akashi@linaro.org> > > To: trini@konsulko.com, sjg@chromium.org > > Cc: etienne.carriere@st.com, u-boot@lists.denx.de, > > AKASHI Takahiro <takahiro.akashi@linaro.org> > > Subject: [PATCH v2 01/12] scmi: refactor the code to hide a channel from devices > > Date: Wed, 26 Jul 2023 17:37:57 +0900 [thread overview] > > Message-ID: <20230726083808.140780-2-takahiro.akashi@linaro.org> (raw) > > In-Reply-To: <20230726083808.140780-1-takahiro.akashi@linaro.org> > > > > The commit 85dc58289238 ("firmware: scmi: prepare uclass to pass channel > > reference") added an explicit parameter, channel, but it seems to make > > the code complex. > > > > Hiding this parameter will allow for adding a generic (protocol-agnostic) > > helper function, i.e. for PROTOCOL_VERSION, in a later patch. > > > > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > > > > --- > > v2 > > * new patch > > --- > > drivers/clk/clk_scmi.c | 27 ++------- > > drivers/firmware/scmi/scmi_agent-uclass.c | 74 ++++++++++++++++++----- > > drivers/power/regulator/scmi_regulator.c | 27 +++------ > > drivers/reset/reset-scmi.c | 19 +----- > > include/scmi_agent.h | 15 +++-- > > 5 files changed, 86 insertions(+), 76 deletions(-) > > > > diff --git a/drivers/clk/clk_scmi.c b/drivers/clk/clk_scmi.c > > index d172fed24c9d..34a49363a51a 100644 > > --- a/drivers/clk/clk_scmi.c > > +++ b/drivers/clk/clk_scmi.c > > @@ -13,17 +13,8 @@ > > #include <asm/types.h> > > #include <linux/clk-provider.h> > > > > -/** > > - * struct scmi_clk_priv - Private data for SCMI clocks > > - * @channel: Reference to the SCMI channel to use > > - */ > > -struct scmi_clk_priv { > > - struct scmi_channel *channel; > > -}; > > - > > static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > { > > - struct scmi_clk_priv *priv = dev_get_priv(dev); > > struct scmi_clk_protocol_attr_out out; > > struct scmi_msg msg = { > > .protocol_id = SCMI_PROTOCOL_ID_CLOCK, > > @@ -33,7 +24,7 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > }; > > int ret; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(dev, &msg); > > if (ret) > > return ret; > > > > @@ -44,7 +35,6 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > > > static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > { > > - struct scmi_clk_priv *priv = dev_get_priv(dev); > > struct scmi_clk_attribute_in in = { > > .clock_id = clkid, > > }; > > @@ -59,7 +49,7 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > }; > > int ret; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(dev, &msg); > > if (ret) > > return ret; > > > > @@ -70,7 +60,6 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > > > static int scmi_clk_gate(struct clk *clk, int enable) > > { > > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > > struct scmi_clk_state_in in = { > > .clock_id = clk->id, > > .attributes = enable, > > @@ -81,7 +70,7 @@ static int scmi_clk_gate(struct clk *clk, int enable) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(clk->dev, &msg); > > if (ret) > > return ret; > > > > @@ -100,7 +89,6 @@ static int scmi_clk_disable(struct clk *clk) > > > > static ulong scmi_clk_get_rate(struct clk *clk) > > { > > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > > struct scmi_clk_rate_get_in in = { > > .clock_id = clk->id, > > }; > > @@ -110,7 +98,7 @@ static ulong scmi_clk_get_rate(struct clk *clk) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(clk->dev, &msg); > > if (ret < 0) > > return ret; > > > > @@ -123,7 +111,6 @@ static ulong scmi_clk_get_rate(struct clk *clk) > > > > static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > { > > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > > struct scmi_clk_rate_set_in in = { > > .clock_id = clk->id, > > .flags = SCMI_CLK_RATE_ROUND_CLOSEST, > > @@ -136,7 +123,7 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(clk->dev, &msg); > > if (ret < 0) > > return ret; > > > > @@ -149,12 +136,11 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > > > static int scmi_clk_probe(struct udevice *dev) > > { > > - struct scmi_clk_priv *priv = dev_get_priv(dev); > > struct clk *clk; > > size_t num_clocks, i; > > int ret; > > > > - ret = devm_scmi_of_get_channel(dev, &priv->channel); > > + ret = devm_scmi_of_get_channel(dev); > > if (ret) > > return ret; > > > > @@ -205,5 +191,4 @@ U_BOOT_DRIVER(scmi_clock) = { > > .id = UCLASS_CLK, > > .ops = &scmi_clk_ops, > > .probe = scmi_clk_probe, > > - .priv_auto = sizeof(struct scmi_clk_priv *), > > }; > > diff --git a/drivers/firmware/scmi/scmi_agent-uclass.c b/drivers/firmware/scmi/scmi_agent-uclass.c > > index 02de692d66f3..39cf15c88f75 100644 > > --- a/drivers/firmware/scmi/scmi_agent-uclass.c > > +++ b/drivers/firmware/scmi/scmi_agent-uclass.c > > @@ -8,6 +8,7 @@ > > #include <common.h> > > #include <dm.h> > > #include <errno.h> > > +#include <scmi_agent.h> > > #include <scmi_agent-uclass.h> > > #include <scmi_protocols.h> > > #include <dm/device_compat.h> > > @@ -129,43 +130,88 @@ static const struct scmi_agent_ops *transport_dev_ops(struct udevice *dev) > > return (const struct scmi_agent_ops *)dev->driver->ops; > > } > > > > -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > > +/** > > + * scmi_of_get_channel() - Get SCMI channel handle > > + * > > + * @dev: SCMI agent device > > + * @channel: Output reference to the SCMI channel upon success > > + * > > + * On return, @channel will be set. > > + * Return 0 on success and a negative errno on failure > > + */ > > +static int scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > > +{ > > + const struct scmi_agent_ops *ops; > > + > > + ops = transport_dev_ops(dev); > > + if (ops->of_get_channel) > > + return ops->of_get_channel(dev, channel); > > + else > > + return -EPROTONOSUPPORT; > > +} > > + > > +int devm_scmi_of_get_channel(struct udevice *dev) > > { > > struct udevice *parent; > > + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); > > This parent lookup does not match when CONFIG_CLK_CCF is enabled. When so, > the scmi clock device is on the clock itself but a child udevice of some > udevice held by the common clock framework (CCF). > Platforms from the stm32mp13 family do enable SCMI clock and CCF. I tested my patch on sandbox, where CCF is also enabled, and "ut dm scmi_clocks" is passed without any error. So I'm not sure if the point you mentioned above is a root cause. -Takahiro Akashi > It can propose a way to go, using find_scmi_transport_device() to provide > the agent's 1st child private data as an output argument: > //patch-start > -static struct udevice *find_scmi_transport_device(struct udevice *dev) > +static struct udevice * > +find_scmi_transport_device(struct udevice *dev, > + struct scmi_agent_proto_priv **priv) > { > struct udevice *parent = dev; > + struct scmi_agent_proto_priv *parent_priv; > > do { > + parent_priv = dev_get_parent_priv(parent); > parent = dev_get_parent(parent); > } while (parent && device_get_uclass_id(parent) != UCLASS_SCMI_AGENT); > > if (!parent) > dev_err(dev, "Invalid SCMI device, agent not found\n"); > + else if (priv) > + *priv = parent_priv; > > return parent; > } > //end patch > > Tested OK on stm32mp13 with the remaining patch series applied (minor conflics). > > BR, > etienne > > > > + int ret; > > > > parent = find_scmi_transport_device(dev); > > if (!parent) > > return -ENODEV; > > > > - if (transport_dev_ops(parent)->of_get_channel) > > - return transport_dev_ops(parent)->of_get_channel(parent, channel); > > + ret = scmi_of_get_channel(parent, &priv->channel); > > + if (ret == -EPROTONOSUPPORT) { > > + /* Drivers without a get_channel operator don't need a channel ref */ > > + priv->channel = NULL; > > > > - /* Drivers without a get_channel operator don't need a channel ref */ > > - *channel = NULL; > > + return 0; > > + } > > > > - return 0; > > + return ret; > > } > > > > -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > > - struct scmi_msg *msg) > > +/** > > + * scmi_process_msg() - Send and process an SCMI message > > + * > > + * Send a message to an SCMI server. > > + * Caller sets scmi_msg::out_msg_sz to the output message buffer size. > > + * > > + * @dev: SCMI agent device > > + * @channel: Communication channel for the device > > + * @msg: Message structure reference > > + * > > + * On return, scmi_msg::out_msg_sz stores the response payload size. > > + * Return: 0 on success and a negative errno on failure > > + */ > > +static int scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > > + struct scmi_msg *msg) > > { > > const struct scmi_agent_ops *ops; > > + > > + ops = transport_dev_ops(dev); > > + if (ops->process_msg) > > + return ops->process_msg(dev, channel, msg); > > + else > > + return -EPROTONOSUPPORT; > > +} > > + > > +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg) > > +{ > > struct udevice *parent; > > + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); > > > > parent = find_scmi_transport_device(dev); > > if (!parent) > > return -ENODEV; > > > > - ops = transport_dev_ops(parent); > > - > > - if (ops->process_msg) > > - return ops->process_msg(parent, channel, msg); > > - > > - return -EPROTONOSUPPORT; > > + return scmi_process_msg(parent, priv->channel, msg); > > } > > > > UCLASS_DRIVER(scmi_agent) = { > > .id = UCLASS_SCMI_AGENT, > > .name = "scmi_agent", > > .post_bind = scmi_bind_protocols, > > + .per_device_plat_auto = sizeof(struct scmi_agent_priv), > > Typo: this added line belong to patch 04/12. > > Regards, > Etienne > > > + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), > > }; > > diff --git a/drivers/power/regulator/scmi_regulator.c b/drivers/power/regulator/scmi_regulator.c > > index 801148036ff6..08918b20872c 100644 > > --- a/drivers/power/regulator/scmi_regulator.c > > +++ b/drivers/power/regulator/scmi_regulator.c > > @@ -25,18 +25,9 @@ struct scmi_regulator_platdata { > > u32 domain_id; > > }; > > > > -/** > > - * struct scmi_regulator_priv - Private data for SCMI voltage regulator > > - * @channel: Reference to the SCMI channel to use > > - */ > > -struct scmi_regulator_priv { > > - struct scmi_channel *channel; > > -}; > > - > > static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > > { > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > struct scmi_voltd_config_set_in in = { > > .domain_id = pdata->domain_id, > > .config = enable ? SCMI_VOLTD_CONFIG_ON : SCMI_VOLTD_CONFIG_OFF, > > @@ -47,7 +38,7 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(dev, &msg); > > if (ret) > > return ret; > > > > @@ -57,7 +48,6 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > > static int scmi_voltd_get_enable(struct udevice *dev) > > { > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > struct scmi_voltd_config_get_in in = { > > .domain_id = pdata->domain_id, > > }; > > @@ -67,7 +57,7 @@ static int scmi_voltd_get_enable(struct udevice *dev) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(dev, &msg); > > if (ret < 0) > > return ret; > > > > @@ -80,7 +70,6 @@ static int scmi_voltd_get_enable(struct udevice *dev) > > > > static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > { > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > struct scmi_voltd_level_set_in in = { > > .domain_id = pdata->domain_id, > > @@ -92,7 +81,7 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(dev, &msg); > > if (ret < 0) > > return ret; > > > > @@ -101,7 +90,6 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > > > static int scmi_voltd_get_voltage_level(struct udevice *dev) > > { > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > struct scmi_voltd_level_get_in in = { > > .domain_id = pdata->domain_id, > > @@ -112,7 +100,7 @@ static int scmi_voltd_get_voltage_level(struct udevice *dev) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(dev, &msg); > > if (ret < 0) > > return ret; > > > > @@ -140,7 +128,6 @@ static int scmi_regulator_of_to_plat(struct udevice *dev) > > static int scmi_regulator_probe(struct udevice *dev) > > { > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > struct scmi_voltd_attr_in in = { 0 }; > > struct scmi_voltd_attr_out out = { 0 }; > > struct scmi_msg scmi_msg = { > > @@ -153,14 +140,14 @@ static int scmi_regulator_probe(struct udevice *dev) > > }; > > int ret; > > > > - ret = devm_scmi_of_get_channel(dev->parent, &priv->channel); > > + ret = devm_scmi_of_get_channel(dev); > > if (ret) > > return ret; > > > > /* Check voltage domain is known from SCMI server */ > > in.domain_id = pdata->domain_id; > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &scmi_msg); > > + ret = devm_scmi_process_msg(dev, &scmi_msg); > > if (ret) { > > dev_err(dev, "Failed to query voltage domain %u: %d\n", > > pdata->domain_id, ret); > > @@ -184,7 +171,6 @@ U_BOOT_DRIVER(scmi_regulator) = { > > .probe = scmi_regulator_probe, > > .of_to_plat = scmi_regulator_of_to_plat, > > .plat_auto = sizeof(struct scmi_regulator_platdata), > > - .priv_auto = sizeof(struct scmi_regulator_priv *), > > }; > > > > static int scmi_regulator_bind(struct udevice *dev) > > @@ -209,4 +195,5 @@ U_BOOT_DRIVER(scmi_voltage_domain) = { > > .name = "scmi_voltage_domain", > > .id = UCLASS_NOP, > > .bind = scmi_regulator_bind, > > + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), > > }; > > diff --git a/drivers/reset/reset-scmi.c b/drivers/reset/reset-scmi.c > > index 122556162ec3..b76711f0a8fb 100644 > > --- a/drivers/reset/reset-scmi.c > > +++ b/drivers/reset/reset-scmi.c > > @@ -13,17 +13,8 @@ > > #include <scmi_protocols.h> > > #include <asm/types.h> > > > > -/** > > - * struct scmi_reset_priv - Private data for SCMI reset controller > > - * @channel: Reference to the SCMI channel to use > > - */ > > -struct scmi_reset_priv { > > - struct scmi_channel *channel; > > -}; > > - > > static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) > > { > > - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); > > struct scmi_rd_reset_in in = { > > .domain_id = rst->id, > > .flags = assert_not_deassert ? SCMI_RD_RESET_FLAG_ASSERT : 0, > > @@ -35,7 +26,7 @@ static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) > > in, out); > > int ret; > > > > - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(rst->dev, &msg); > > if (ret) > > return ret; > > > > @@ -54,7 +45,6 @@ static int scmi_reset_deassert(struct reset_ctl *rst) > > > > static int scmi_reset_request(struct reset_ctl *rst) > > { > > - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); > > struct scmi_rd_attr_in in = { > > .domain_id = rst->id, > > }; > > @@ -68,7 +58,7 @@ static int scmi_reset_request(struct reset_ctl *rst) > > * We don't really care about the attribute, just check > > * the reset domain exists. > > */ > > - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); > > + ret = devm_scmi_process_msg(rst->dev, &msg); > > if (ret) > > return ret; > > > > @@ -83,9 +73,7 @@ static const struct reset_ops scmi_reset_domain_ops = { > > > > static int scmi_reset_probe(struct udevice *dev) > > { > > - struct scmi_reset_priv *priv = dev_get_priv(dev); > > - > > - return devm_scmi_of_get_channel(dev, &priv->channel); > > + return devm_scmi_of_get_channel(dev); > > } > > > > U_BOOT_DRIVER(scmi_reset_domain) = { > > @@ -93,5 +81,4 @@ U_BOOT_DRIVER(scmi_reset_domain) = { > > .id = UCLASS_RESET, > > .ops = &scmi_reset_domain_ops, > > .probe = scmi_reset_probe, > > - .priv_auto = sizeof(struct scmi_reset_priv *), > > }; > > diff --git a/include/scmi_agent.h b/include/scmi_agent.h > > index ee6286366df7..577892029ff8 100644 > > --- a/include/scmi_agent.h > > +++ b/include/scmi_agent.h > > @@ -15,6 +15,14 @@ > > struct udevice; > > struct scmi_channel; > > > > +/** > > + * struct scmi_agent_proto_priv - Private data in device for SCMI agent > > + * @channel: Reference to the SCMI channel to use > > + */ > > +struct scmi_agent_proto_priv { > > + struct scmi_channel *channel; > > +}; > > + > > /* > > * struct scmi_msg - Context of a SCMI message sent and the response received > > * > > @@ -49,10 +57,9 @@ struct scmi_msg { > > * devm_scmi_of_get_channel() - Get SCMI channel handle from SCMI agent DT node > > * > > * @dev: Device requesting a channel > > - * @channel: Output reference to the SCMI channel upon success > > * @return 0 on success and a negative errno on failure > > */ > > -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel); > > +int devm_scmi_of_get_channel(struct udevice *dev); > > > > /** > > * devm_scmi_process_msg() - Send and process an SCMI message > > @@ -62,12 +69,10 @@ int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > > * On return, scmi_msg::out_msg_sz stores the response payload size. > > * > > * @dev: SCMI device > > - * @channel: Communication channel for the device > > * @msg: Message structure reference > > * Return: 0 on success and a negative errno on failure > > */ > > -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > > - struct scmi_msg *msg); > > +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg); > > > > /** > > * scmi_to_linux_errno() - Convert an SCMI error code into a Linux errno code > > -- > > 2.41.0
Hello Akashi-san, > From: AKASHI Takahiro > Sent: Monday, August 7, 2023 12:03 PM > > Hi Etienne, > > On Thu, Aug 03, 2023 at 09:24:51AM +0000, Etienne CARRIERE wrote: > > Hello Takahiro-san, > > > > Sorry again for this late feedback. > > Testing the series against stm32mp135f-dk board which is using SCMI resources, I see the board fails to boot. > > > > > From: AKASHI Takahiro <takahiro.akashi@linaro.org> > > > To: trini@konsulko.com, sjg@chromium.org > > > Cc: etienne.carriere@st.com, u-boot@lists.denx.de, > > > AKASHI Takahiro <takahiro.akashi@linaro.org> > > > Subject: [PATCH v2 01/12] scmi: refactor the code to hide a channel from devices > > > Date: Wed, 26 Jul 2023 17:37:57 +0900 [thread overview] > > > Message-ID: <20230726083808.140780-2-takahiro.akashi@linaro.org> (raw) > > > In-Reply-To: <20230726083808.140780-1-takahiro.akashi@linaro.org> > > > > > > The commit 85dc58289238 ("firmware: scmi: prepare uclass to pass channel > > > reference") added an explicit parameter, channel, but it seems to make > > > the code complex. > > > > > > Hiding this parameter will allow for adding a generic (protocol-agnostic) > > > helper function, i.e. for PROTOCOL_VERSION, in a later patch. > > > > > > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > > > > > > --- > > > v2 > > > * new patch > > > --- > > > drivers/clk/clk_scmi.c | 27 ++------- > > > drivers/firmware/scmi/scmi_agent-uclass.c | 74 ++++++++++++++++++----- > > > drivers/power/regulator/scmi_regulator.c | 27 +++------ > > > drivers/reset/reset-scmi.c | 19 +----- > > > include/scmi_agent.h | 15 +++-- > > > 5 files changed, 86 insertions(+), 76 deletions(-) > > > > > > diff --git a/drivers/clk/clk_scmi.c b/drivers/clk/clk_scmi.c > > > index d172fed24c9d..34a49363a51a 100644 > > > --- a/drivers/clk/clk_scmi.c > > > +++ b/drivers/clk/clk_scmi.c > > > @@ -13,17 +13,8 @@ > > > #include <asm/types.h> > > > #include <linux/clk-provider.h> > > > > > > -/** > > > - * struct scmi_clk_priv - Private data for SCMI clocks > > > - * @channel: Reference to the SCMI channel to use > > > - */ > > > -struct scmi_clk_priv { > > > - struct scmi_channel *channel; > > > -}; > > > - > > > static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > > { > > > - struct scmi_clk_priv *priv = dev_get_priv(dev); > > > struct scmi_clk_protocol_attr_out out; > > > struct scmi_msg msg = { > > > .protocol_id = SCMI_PROTOCOL_ID_CLOCK, > > > @@ -33,7 +24,7 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > > }; > > > int ret; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(dev, &msg); > > > if (ret) > > > return ret; > > > > > > @@ -44,7 +35,6 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) > > > > > > static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > > { > > > - struct scmi_clk_priv *priv = dev_get_priv(dev); > > > struct scmi_clk_attribute_in in = { > > > .clock_id = clkid, > > > }; > > > @@ -59,7 +49,7 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > > }; > > > int ret; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(dev, &msg); > > > if (ret) > > > return ret; > > > > > > @@ -70,7 +60,6 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) > > > > > > static int scmi_clk_gate(struct clk *clk, int enable) > > > { > > > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > > > struct scmi_clk_state_in in = { > > > .clock_id = clk->id, > > > .attributes = enable, > > > @@ -81,7 +70,7 @@ static int scmi_clk_gate(struct clk *clk, int enable) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(clk->dev, &msg); > > > if (ret) > > > return ret; > > > > > > @@ -100,7 +89,6 @@ static int scmi_clk_disable(struct clk *clk) > > > > > > static ulong scmi_clk_get_rate(struct clk *clk) > > > { > > > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > > > struct scmi_clk_rate_get_in in = { > > > .clock_id = clk->id, > > > }; > > > @@ -110,7 +98,7 @@ static ulong scmi_clk_get_rate(struct clk *clk) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(clk->dev, &msg); > > > if (ret < 0) > > > return ret; > > > > > > @@ -123,7 +111,6 @@ static ulong scmi_clk_get_rate(struct clk *clk) > > > > > > static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > > { > > > - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); > > > struct scmi_clk_rate_set_in in = { > > > .clock_id = clk->id, > > > .flags = SCMI_CLK_RATE_ROUND_CLOSEST, > > > @@ -136,7 +123,7 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(clk->dev, &msg); > > > if (ret < 0) > > > return ret; > > > > > > @@ -149,12 +136,11 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) > > > > > > static int scmi_clk_probe(struct udevice *dev) > > > { > > > - struct scmi_clk_priv *priv = dev_get_priv(dev); > > > struct clk *clk; > > > size_t num_clocks, i; > > > int ret; > > > > > > - ret = devm_scmi_of_get_channel(dev, &priv->channel); > > > + ret = devm_scmi_of_get_channel(dev); > > > if (ret) > > > return ret; > > > > > > @@ -205,5 +191,4 @@ U_BOOT_DRIVER(scmi_clock) = { > > > .id = UCLASS_CLK, > > > .ops = &scmi_clk_ops, > > > .probe = scmi_clk_probe, > > > - .priv_auto = sizeof(struct scmi_clk_priv *), > > > }; > > > diff --git a/drivers/firmware/scmi/scmi_agent-uclass.c b/drivers/firmware/scmi/scmi_agent-uclass.c > > > index 02de692d66f3..39cf15c88f75 100644 > > > --- a/drivers/firmware/scmi/scmi_agent-uclass.c > > > +++ b/drivers/firmware/scmi/scmi_agent-uclass.c > > > @@ -8,6 +8,7 @@ > > > #include <common.h> > > > #include <dm.h> > > > #include <errno.h> > > > +#include <scmi_agent.h> > > > #include <scmi_agent-uclass.h> > > > #include <scmi_protocols.h> > > > #include <dm/device_compat.h> > > > @@ -129,43 +130,88 @@ static const struct scmi_agent_ops *transport_dev_ops(struct udevice *dev) > > > return (const struct scmi_agent_ops *)dev->driver->ops; > > > } > > > > > > -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > > > +/** > > > + * scmi_of_get_channel() - Get SCMI channel handle > > > + * > > > + * @dev: SCMI agent device > > > + * @channel: Output reference to the SCMI channel upon success > > > + * > > > + * On return, @channel will be set. > > > + * Return 0 on success and a negative errno on failure > > > + */ > > > +static int scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > > > +{ > > > + const struct scmi_agent_ops *ops; > > > + > > > + ops = transport_dev_ops(dev); > > > + if (ops->of_get_channel) > > > + return ops->of_get_channel(dev, channel); > > > + else > > > + return -EPROTONOSUPPORT; > > > +} > > > + > > > +int devm_scmi_of_get_channel(struct udevice *dev) > > > { > > > struct udevice *parent; > > > + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); > > > > This parent lookup does not match when CONFIG_CLK_CCF is enabled. When so, > > the scmi clock device is on the clock itself but a child udevice of some > > udevice held by the common clock framework (CCF). > > Platforms from the stm32mp13 family do enable SCMI clock and CCF. > > I tested my patch on sandbox, where CCF is also enabled, > and "ut dm scmi_clocks" is passed without any error. > So I'm not sure if the point you mentioned above is a root cause. Ok, i'll better investigate where stm32mp13 gets lost. and why sandbox passes wher I would expect it to fail. Best regards, etienne > > -Takahiro Akashi > > > It can propose a way to go, using find_scmi_transport_device() to provide > > the agent's 1st child private data as an output argument: > > //patch-start > > -static struct udevice *find_scmi_transport_device(struct udevice *dev) > > +static struct udevice * > > +find_scmi_transport_device(struct udevice *dev, > > + struct scmi_agent_proto_priv **priv) > > { > > struct udevice *parent = dev; > > + struct scmi_agent_proto_priv *parent_priv; > > > > do { > > + parent_priv = dev_get_parent_priv(parent); > > parent = dev_get_parent(parent); > > } while (parent && device_get_uclass_id(parent) != UCLASS_SCMI_AGENT); > > > > if (!parent) > > dev_err(dev, "Invalid SCMI device, agent not found\n"); > > + else if (priv) > > + *priv = parent_priv; > > > > return parent; > > } > > //end patch > > > > Tested OK on stm32mp13 with the remaining patch series applied (minor conflics). > > > > BR, > > etienne > > > > > > > + int ret; > > > > > > parent = find_scmi_transport_device(dev); > > > if (!parent) > > > return -ENODEV; > > > > > > - if (transport_dev_ops(parent)->of_get_channel) > > > - return transport_dev_ops(parent)->of_get_channel(parent, channel); > > > + ret = scmi_of_get_channel(parent, &priv->channel); > > > + if (ret == -EPROTONOSUPPORT) { > > > + /* Drivers without a get_channel operator don't need a channel ref */ > > > + priv->channel = NULL; > > > > > > - /* Drivers without a get_channel operator don't need a channel ref */ > > > - *channel = NULL; > > > + return 0; > > > + } > > > > > > - return 0; > > > + return ret; > > > } > > > > > > -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > > > - struct scmi_msg *msg) > > > +/** > > > + * scmi_process_msg() - Send and process an SCMI message > > > + * > > > + * Send a message to an SCMI server. > > > + * Caller sets scmi_msg::out_msg_sz to the output message buffer size. > > > + * > > > + * @dev: SCMI agent device > > > + * @channel: Communication channel for the device > > > + * @msg: Message structure reference > > > + * > > > + * On return, scmi_msg::out_msg_sz stores the response payload size. > > > + * Return: 0 on success and a negative errno on failure > > > + */ > > > +static int scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > > > + struct scmi_msg *msg) > > > { > > > const struct scmi_agent_ops *ops; > > > + > > > + ops = transport_dev_ops(dev); > > > + if (ops->process_msg) > > > + return ops->process_msg(dev, channel, msg); > > > + else > > > + return -EPROTONOSUPPORT; > > > +} > > > + > > > +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg) > > > +{ > > > struct udevice *parent; > > > + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); > > > > > > parent = find_scmi_transport_device(dev); > > > if (!parent) > > > return -ENODEV; > > > > > > - ops = transport_dev_ops(parent); > > > - > > > - if (ops->process_msg) > > > - return ops->process_msg(parent, channel, msg); > > > - > > > - return -EPROTONOSUPPORT; > > > + return scmi_process_msg(parent, priv->channel, msg); > > > } > > > > > > UCLASS_DRIVER(scmi_agent) = { > > > .id = UCLASS_SCMI_AGENT, > > > .name = "scmi_agent", > > > .post_bind = scmi_bind_protocols, > > > + .per_device_plat_auto = sizeof(struct scmi_agent_priv), > > > > Typo: this added line belong to patch 04/12. > > > > Regards, > > Etienne > > > > > + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), > > > }; > > > diff --git a/drivers/power/regulator/scmi_regulator.c b/drivers/power/regulator/scmi_regulator.c > > > index 801148036ff6..08918b20872c 100644 > > > --- a/drivers/power/regulator/scmi_regulator.c > > > +++ b/drivers/power/regulator/scmi_regulator.c > > > @@ -25,18 +25,9 @@ struct scmi_regulator_platdata { > > > u32 domain_id; > > > }; > > > > > > -/** > > > - * struct scmi_regulator_priv - Private data for SCMI voltage regulator > > > - * @channel: Reference to the SCMI channel to use > > > - */ > > > -struct scmi_regulator_priv { > > > - struct scmi_channel *channel; > > > -}; > > > - > > > static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > > > { > > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > > struct scmi_voltd_config_set_in in = { > > > .domain_id = pdata->domain_id, > > > .config = enable ? SCMI_VOLTD_CONFIG_ON : SCMI_VOLTD_CONFIG_OFF, > > > @@ -47,7 +38,7 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(dev, &msg); > > > if (ret) > > > return ret; > > > > > > @@ -57,7 +48,6 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) > > > static int scmi_voltd_get_enable(struct udevice *dev) > > > { > > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > > struct scmi_voltd_config_get_in in = { > > > .domain_id = pdata->domain_id, > > > }; > > > @@ -67,7 +57,7 @@ static int scmi_voltd_get_enable(struct udevice *dev) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(dev, &msg); > > > if (ret < 0) > > > return ret; > > > > > > @@ -80,7 +70,6 @@ static int scmi_voltd_get_enable(struct udevice *dev) > > > > > > static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > > { > > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > > struct scmi_voltd_level_set_in in = { > > > .domain_id = pdata->domain_id, > > > @@ -92,7 +81,7 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(dev, &msg); > > > if (ret < 0) > > > return ret; > > > > > > @@ -101,7 +90,6 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) > > > > > > static int scmi_voltd_get_voltage_level(struct udevice *dev) > > > { > > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > > struct scmi_voltd_level_get_in in = { > > > .domain_id = pdata->domain_id, > > > @@ -112,7 +100,7 @@ static int scmi_voltd_get_voltage_level(struct udevice *dev) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(dev, &msg); > > > if (ret < 0) > > > return ret; > > > > > > @@ -140,7 +128,6 @@ static int scmi_regulator_of_to_plat(struct udevice *dev) > > > static int scmi_regulator_probe(struct udevice *dev) > > > { > > > struct scmi_regulator_platdata *pdata = dev_get_plat(dev); > > > - struct scmi_regulator_priv *priv = dev_get_priv(dev); > > > struct scmi_voltd_attr_in in = { 0 }; > > > struct scmi_voltd_attr_out out = { 0 }; > > > struct scmi_msg scmi_msg = { > > > @@ -153,14 +140,14 @@ static int scmi_regulator_probe(struct udevice *dev) > > > }; > > > int ret; > > > > > > - ret = devm_scmi_of_get_channel(dev->parent, &priv->channel); > > > + ret = devm_scmi_of_get_channel(dev); > > > if (ret) > > > return ret; > > > > > > /* Check voltage domain is known from SCMI server */ > > > in.domain_id = pdata->domain_id; > > > > > > - ret = devm_scmi_process_msg(dev, priv->channel, &scmi_msg); > > > + ret = devm_scmi_process_msg(dev, &scmi_msg); > > > if (ret) { > > > dev_err(dev, "Failed to query voltage domain %u: %d\n", > > > pdata->domain_id, ret); > > > @@ -184,7 +171,6 @@ U_BOOT_DRIVER(scmi_regulator) = { > > > .probe = scmi_regulator_probe, > > > .of_to_plat = scmi_regulator_of_to_plat, > > > .plat_auto = sizeof(struct scmi_regulator_platdata), > > > - .priv_auto = sizeof(struct scmi_regulator_priv *), > > > }; > > > > > > static int scmi_regulator_bind(struct udevice *dev) > > > @@ -209,4 +195,5 @@ U_BOOT_DRIVER(scmi_voltage_domain) = { > > > .name = "scmi_voltage_domain", > > > .id = UCLASS_NOP, > > > .bind = scmi_regulator_bind, > > > + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), > > > }; > > > diff --git a/drivers/reset/reset-scmi.c b/drivers/reset/reset-scmi.c > > > index 122556162ec3..b76711f0a8fb 100644 > > > --- a/drivers/reset/reset-scmi.c > > > +++ b/drivers/reset/reset-scmi.c > > > @@ -13,17 +13,8 @@ > > > #include <scmi_protocols.h> > > > #include <asm/types.h> > > > > > > -/** > > > - * struct scmi_reset_priv - Private data for SCMI reset controller > > > - * @channel: Reference to the SCMI channel to use > > > - */ > > > -struct scmi_reset_priv { > > > - struct scmi_channel *channel; > > > -}; > > > - > > > static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) > > > { > > > - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); > > > struct scmi_rd_reset_in in = { > > > .domain_id = rst->id, > > > .flags = assert_not_deassert ? SCMI_RD_RESET_FLAG_ASSERT : 0, > > > @@ -35,7 +26,7 @@ static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) > > > in, out); > > > int ret; > > > > > > - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(rst->dev, &msg); > > > if (ret) > > > return ret; > > > > > > @@ -54,7 +45,6 @@ static int scmi_reset_deassert(struct reset_ctl *rst) > > > > > > static int scmi_reset_request(struct reset_ctl *rst) > > > { > > > - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); > > > struct scmi_rd_attr_in in = { > > > .domain_id = rst->id, > > > }; > > > @@ -68,7 +58,7 @@ static int scmi_reset_request(struct reset_ctl *rst) > > > * We don't really care about the attribute, just check > > > * the reset domain exists. > > > */ > > > - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); > > > + ret = devm_scmi_process_msg(rst->dev, &msg); > > > if (ret) > > > return ret; > > > > > > @@ -83,9 +73,7 @@ static const struct reset_ops scmi_reset_domain_ops = { > > > > > > static int scmi_reset_probe(struct udevice *dev) > > > { > > > - struct scmi_reset_priv *priv = dev_get_priv(dev); > > > - > > > - return devm_scmi_of_get_channel(dev, &priv->channel); > > > + return devm_scmi_of_get_channel(dev); > > > } > > > > > > U_BOOT_DRIVER(scmi_reset_domain) = { > > > @@ -93,5 +81,4 @@ U_BOOT_DRIVER(scmi_reset_domain) = { > > > .id = UCLASS_RESET, > > > .ops = &scmi_reset_domain_ops, > > > .probe = scmi_reset_probe, > > > - .priv_auto = sizeof(struct scmi_reset_priv *), > > > }; > > > diff --git a/include/scmi_agent.h b/include/scmi_agent.h > > > index ee6286366df7..577892029ff8 100644 > > > --- a/include/scmi_agent.h > > > +++ b/include/scmi_agent.h > > > @@ -15,6 +15,14 @@ > > > struct udevice; > > > struct scmi_channel; > > > > > > +/** > > > + * struct scmi_agent_proto_priv - Private data in device for SCMI agent > > > + * @channel: Reference to the SCMI channel to use > > > + */ > > > +struct scmi_agent_proto_priv { > > > + struct scmi_channel *channel; > > > +}; > > > + > > > /* > > > * struct scmi_msg - Context of a SCMI message sent and the response received > > > * > > > @@ -49,10 +57,9 @@ struct scmi_msg { > > > * devm_scmi_of_get_channel() - Get SCMI channel handle from SCMI agent DT node > > > * > > > * @dev: Device requesting a channel > > > - * @channel: Output reference to the SCMI channel upon success > > > * @return 0 on success and a negative errno on failure > > > */ > > > -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel); > > > +int devm_scmi_of_get_channel(struct udevice *dev); > > > > > > /** > > > * devm_scmi_process_msg() - Send and process an SCMI message > > > @@ -62,12 +69,10 @@ int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) > > > * On return, scmi_msg::out_msg_sz stores the response payload size. > > > * > > > * @dev: SCMI device > > > - * @channel: Communication channel for the device > > > * @msg: Message structure reference > > > * Return: 0 on success and a negative errno on failure > > > */ > > > -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, > > > - struct scmi_msg *msg); > > > +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg); > > > > > > /** > > > * scmi_to_linux_errno() - Convert an SCMI error code into a Linux errno code > > > -- > > > 2.41.0 >
diff --git a/drivers/clk/clk_scmi.c b/drivers/clk/clk_scmi.c index d172fed24c9d..34a49363a51a 100644 --- a/drivers/clk/clk_scmi.c +++ b/drivers/clk/clk_scmi.c @@ -13,17 +13,8 @@ #include <asm/types.h> #include <linux/clk-provider.h> -/** - * struct scmi_clk_priv - Private data for SCMI clocks - * @channel: Reference to the SCMI channel to use - */ -struct scmi_clk_priv { - struct scmi_channel *channel; -}; - static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) { - struct scmi_clk_priv *priv = dev_get_priv(dev); struct scmi_clk_protocol_attr_out out; struct scmi_msg msg = { .protocol_id = SCMI_PROTOCOL_ID_CLOCK, @@ -33,7 +24,7 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) }; int ret; - ret = devm_scmi_process_msg(dev, priv->channel, &msg); + ret = devm_scmi_process_msg(dev, &msg); if (ret) return ret; @@ -44,7 +35,6 @@ static int scmi_clk_get_num_clock(struct udevice *dev, size_t *num_clocks) static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) { - struct scmi_clk_priv *priv = dev_get_priv(dev); struct scmi_clk_attribute_in in = { .clock_id = clkid, }; @@ -59,7 +49,7 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) }; int ret; - ret = devm_scmi_process_msg(dev, priv->channel, &msg); + ret = devm_scmi_process_msg(dev, &msg); if (ret) return ret; @@ -70,7 +60,6 @@ static int scmi_clk_get_attibute(struct udevice *dev, int clkid, char **name) static int scmi_clk_gate(struct clk *clk, int enable) { - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); struct scmi_clk_state_in in = { .clock_id = clk->id, .attributes = enable, @@ -81,7 +70,7 @@ static int scmi_clk_gate(struct clk *clk, int enable) in, out); int ret; - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); + ret = devm_scmi_process_msg(clk->dev, &msg); if (ret) return ret; @@ -100,7 +89,6 @@ static int scmi_clk_disable(struct clk *clk) static ulong scmi_clk_get_rate(struct clk *clk) { - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); struct scmi_clk_rate_get_in in = { .clock_id = clk->id, }; @@ -110,7 +98,7 @@ static ulong scmi_clk_get_rate(struct clk *clk) in, out); int ret; - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); + ret = devm_scmi_process_msg(clk->dev, &msg); if (ret < 0) return ret; @@ -123,7 +111,6 @@ static ulong scmi_clk_get_rate(struct clk *clk) static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) { - struct scmi_clk_priv *priv = dev_get_priv(clk->dev); struct scmi_clk_rate_set_in in = { .clock_id = clk->id, .flags = SCMI_CLK_RATE_ROUND_CLOSEST, @@ -136,7 +123,7 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) in, out); int ret; - ret = devm_scmi_process_msg(clk->dev, priv->channel, &msg); + ret = devm_scmi_process_msg(clk->dev, &msg); if (ret < 0) return ret; @@ -149,12 +136,11 @@ static ulong scmi_clk_set_rate(struct clk *clk, ulong rate) static int scmi_clk_probe(struct udevice *dev) { - struct scmi_clk_priv *priv = dev_get_priv(dev); struct clk *clk; size_t num_clocks, i; int ret; - ret = devm_scmi_of_get_channel(dev, &priv->channel); + ret = devm_scmi_of_get_channel(dev); if (ret) return ret; @@ -205,5 +191,4 @@ U_BOOT_DRIVER(scmi_clock) = { .id = UCLASS_CLK, .ops = &scmi_clk_ops, .probe = scmi_clk_probe, - .priv_auto = sizeof(struct scmi_clk_priv *), }; diff --git a/drivers/firmware/scmi/scmi_agent-uclass.c b/drivers/firmware/scmi/scmi_agent-uclass.c index 02de692d66f3..39cf15c88f75 100644 --- a/drivers/firmware/scmi/scmi_agent-uclass.c +++ b/drivers/firmware/scmi/scmi_agent-uclass.c @@ -8,6 +8,7 @@ #include <common.h> #include <dm.h> #include <errno.h> +#include <scmi_agent.h> #include <scmi_agent-uclass.h> #include <scmi_protocols.h> #include <dm/device_compat.h> @@ -129,43 +130,88 @@ static const struct scmi_agent_ops *transport_dev_ops(struct udevice *dev) return (const struct scmi_agent_ops *)dev->driver->ops; } -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) +/** + * scmi_of_get_channel() - Get SCMI channel handle + * + * @dev: SCMI agent device + * @channel: Output reference to the SCMI channel upon success + * + * On return, @channel will be set. + * Return 0 on success and a negative errno on failure + */ +static int scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) +{ + const struct scmi_agent_ops *ops; + + ops = transport_dev_ops(dev); + if (ops->of_get_channel) + return ops->of_get_channel(dev, channel); + else + return -EPROTONOSUPPORT; +} + +int devm_scmi_of_get_channel(struct udevice *dev) { struct udevice *parent; + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); + int ret; parent = find_scmi_transport_device(dev); if (!parent) return -ENODEV; - if (transport_dev_ops(parent)->of_get_channel) - return transport_dev_ops(parent)->of_get_channel(parent, channel); + ret = scmi_of_get_channel(parent, &priv->channel); + if (ret == -EPROTONOSUPPORT) { + /* Drivers without a get_channel operator don't need a channel ref */ + priv->channel = NULL; - /* Drivers without a get_channel operator don't need a channel ref */ - *channel = NULL; + return 0; + } - return 0; + return ret; } -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, - struct scmi_msg *msg) +/** + * scmi_process_msg() - Send and process an SCMI message + * + * Send a message to an SCMI server. + * Caller sets scmi_msg::out_msg_sz to the output message buffer size. + * + * @dev: SCMI agent device + * @channel: Communication channel for the device + * @msg: Message structure reference + * + * On return, scmi_msg::out_msg_sz stores the response payload size. + * Return: 0 on success and a negative errno on failure + */ +static int scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, + struct scmi_msg *msg) { const struct scmi_agent_ops *ops; + + ops = transport_dev_ops(dev); + if (ops->process_msg) + return ops->process_msg(dev, channel, msg); + else + return -EPROTONOSUPPORT; +} + +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg) +{ struct udevice *parent; + struct scmi_agent_proto_priv *priv = dev_get_parent_priv(dev); parent = find_scmi_transport_device(dev); if (!parent) return -ENODEV; - ops = transport_dev_ops(parent); - - if (ops->process_msg) - return ops->process_msg(parent, channel, msg); - - return -EPROTONOSUPPORT; + return scmi_process_msg(parent, priv->channel, msg); } UCLASS_DRIVER(scmi_agent) = { .id = UCLASS_SCMI_AGENT, .name = "scmi_agent", .post_bind = scmi_bind_protocols, + .per_device_plat_auto = sizeof(struct scmi_agent_priv), + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), }; diff --git a/drivers/power/regulator/scmi_regulator.c b/drivers/power/regulator/scmi_regulator.c index 801148036ff6..08918b20872c 100644 --- a/drivers/power/regulator/scmi_regulator.c +++ b/drivers/power/regulator/scmi_regulator.c @@ -25,18 +25,9 @@ struct scmi_regulator_platdata { u32 domain_id; }; -/** - * struct scmi_regulator_priv - Private data for SCMI voltage regulator - * @channel: Reference to the SCMI channel to use - */ -struct scmi_regulator_priv { - struct scmi_channel *channel; -}; - static int scmi_voltd_set_enable(struct udevice *dev, bool enable) { struct scmi_regulator_platdata *pdata = dev_get_plat(dev); - struct scmi_regulator_priv *priv = dev_get_priv(dev); struct scmi_voltd_config_set_in in = { .domain_id = pdata->domain_id, .config = enable ? SCMI_VOLTD_CONFIG_ON : SCMI_VOLTD_CONFIG_OFF, @@ -47,7 +38,7 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) in, out); int ret; - ret = devm_scmi_process_msg(dev, priv->channel, &msg); + ret = devm_scmi_process_msg(dev, &msg); if (ret) return ret; @@ -57,7 +48,6 @@ static int scmi_voltd_set_enable(struct udevice *dev, bool enable) static int scmi_voltd_get_enable(struct udevice *dev) { struct scmi_regulator_platdata *pdata = dev_get_plat(dev); - struct scmi_regulator_priv *priv = dev_get_priv(dev); struct scmi_voltd_config_get_in in = { .domain_id = pdata->domain_id, }; @@ -67,7 +57,7 @@ static int scmi_voltd_get_enable(struct udevice *dev) in, out); int ret; - ret = devm_scmi_process_msg(dev, priv->channel, &msg); + ret = devm_scmi_process_msg(dev, &msg); if (ret < 0) return ret; @@ -80,7 +70,6 @@ static int scmi_voltd_get_enable(struct udevice *dev) static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) { - struct scmi_regulator_priv *priv = dev_get_priv(dev); struct scmi_regulator_platdata *pdata = dev_get_plat(dev); struct scmi_voltd_level_set_in in = { .domain_id = pdata->domain_id, @@ -92,7 +81,7 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) in, out); int ret; - ret = devm_scmi_process_msg(dev, priv->channel, &msg); + ret = devm_scmi_process_msg(dev, &msg); if (ret < 0) return ret; @@ -101,7 +90,6 @@ static int scmi_voltd_set_voltage_level(struct udevice *dev, int uV) static int scmi_voltd_get_voltage_level(struct udevice *dev) { - struct scmi_regulator_priv *priv = dev_get_priv(dev); struct scmi_regulator_platdata *pdata = dev_get_plat(dev); struct scmi_voltd_level_get_in in = { .domain_id = pdata->domain_id, @@ -112,7 +100,7 @@ static int scmi_voltd_get_voltage_level(struct udevice *dev) in, out); int ret; - ret = devm_scmi_process_msg(dev, priv->channel, &msg); + ret = devm_scmi_process_msg(dev, &msg); if (ret < 0) return ret; @@ -140,7 +128,6 @@ static int scmi_regulator_of_to_plat(struct udevice *dev) static int scmi_regulator_probe(struct udevice *dev) { struct scmi_regulator_platdata *pdata = dev_get_plat(dev); - struct scmi_regulator_priv *priv = dev_get_priv(dev); struct scmi_voltd_attr_in in = { 0 }; struct scmi_voltd_attr_out out = { 0 }; struct scmi_msg scmi_msg = { @@ -153,14 +140,14 @@ static int scmi_regulator_probe(struct udevice *dev) }; int ret; - ret = devm_scmi_of_get_channel(dev->parent, &priv->channel); + ret = devm_scmi_of_get_channel(dev); if (ret) return ret; /* Check voltage domain is known from SCMI server */ in.domain_id = pdata->domain_id; - ret = devm_scmi_process_msg(dev, priv->channel, &scmi_msg); + ret = devm_scmi_process_msg(dev, &scmi_msg); if (ret) { dev_err(dev, "Failed to query voltage domain %u: %d\n", pdata->domain_id, ret); @@ -184,7 +171,6 @@ U_BOOT_DRIVER(scmi_regulator) = { .probe = scmi_regulator_probe, .of_to_plat = scmi_regulator_of_to_plat, .plat_auto = sizeof(struct scmi_regulator_platdata), - .priv_auto = sizeof(struct scmi_regulator_priv *), }; static int scmi_regulator_bind(struct udevice *dev) @@ -209,4 +195,5 @@ U_BOOT_DRIVER(scmi_voltage_domain) = { .name = "scmi_voltage_domain", .id = UCLASS_NOP, .bind = scmi_regulator_bind, + .per_child_auto = sizeof(struct scmi_agent_proto_priv *), }; diff --git a/drivers/reset/reset-scmi.c b/drivers/reset/reset-scmi.c index 122556162ec3..b76711f0a8fb 100644 --- a/drivers/reset/reset-scmi.c +++ b/drivers/reset/reset-scmi.c @@ -13,17 +13,8 @@ #include <scmi_protocols.h> #include <asm/types.h> -/** - * struct scmi_reset_priv - Private data for SCMI reset controller - * @channel: Reference to the SCMI channel to use - */ -struct scmi_reset_priv { - struct scmi_channel *channel; -}; - static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) { - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); struct scmi_rd_reset_in in = { .domain_id = rst->id, .flags = assert_not_deassert ? SCMI_RD_RESET_FLAG_ASSERT : 0, @@ -35,7 +26,7 @@ static int scmi_reset_set_level(struct reset_ctl *rst, bool assert_not_deassert) in, out); int ret; - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); + ret = devm_scmi_process_msg(rst->dev, &msg); if (ret) return ret; @@ -54,7 +45,6 @@ static int scmi_reset_deassert(struct reset_ctl *rst) static int scmi_reset_request(struct reset_ctl *rst) { - struct scmi_reset_priv *priv = dev_get_priv(rst->dev); struct scmi_rd_attr_in in = { .domain_id = rst->id, }; @@ -68,7 +58,7 @@ static int scmi_reset_request(struct reset_ctl *rst) * We don't really care about the attribute, just check * the reset domain exists. */ - ret = devm_scmi_process_msg(rst->dev, priv->channel, &msg); + ret = devm_scmi_process_msg(rst->dev, &msg); if (ret) return ret; @@ -83,9 +73,7 @@ static const struct reset_ops scmi_reset_domain_ops = { static int scmi_reset_probe(struct udevice *dev) { - struct scmi_reset_priv *priv = dev_get_priv(dev); - - return devm_scmi_of_get_channel(dev, &priv->channel); + return devm_scmi_of_get_channel(dev); } U_BOOT_DRIVER(scmi_reset_domain) = { @@ -93,5 +81,4 @@ U_BOOT_DRIVER(scmi_reset_domain) = { .id = UCLASS_RESET, .ops = &scmi_reset_domain_ops, .probe = scmi_reset_probe, - .priv_auto = sizeof(struct scmi_reset_priv *), }; diff --git a/include/scmi_agent.h b/include/scmi_agent.h index ee6286366df7..577892029ff8 100644 --- a/include/scmi_agent.h +++ b/include/scmi_agent.h @@ -15,6 +15,14 @@ struct udevice; struct scmi_channel; +/** + * struct scmi_agent_proto_priv - Private data in device for SCMI agent + * @channel: Reference to the SCMI channel to use + */ +struct scmi_agent_proto_priv { + struct scmi_channel *channel; +}; + /* * struct scmi_msg - Context of a SCMI message sent and the response received * @@ -49,10 +57,9 @@ struct scmi_msg { * devm_scmi_of_get_channel() - Get SCMI channel handle from SCMI agent DT node * * @dev: Device requesting a channel - * @channel: Output reference to the SCMI channel upon success * @return 0 on success and a negative errno on failure */ -int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel); +int devm_scmi_of_get_channel(struct udevice *dev); /** * devm_scmi_process_msg() - Send and process an SCMI message @@ -62,12 +69,10 @@ int devm_scmi_of_get_channel(struct udevice *dev, struct scmi_channel **channel) * On return, scmi_msg::out_msg_sz stores the response payload size. * * @dev: SCMI device - * @channel: Communication channel for the device * @msg: Message structure reference * Return: 0 on success and a negative errno on failure */ -int devm_scmi_process_msg(struct udevice *dev, struct scmi_channel *channel, - struct scmi_msg *msg); +int devm_scmi_process_msg(struct udevice *dev, struct scmi_msg *msg); /** * scmi_to_linux_errno() - Convert an SCMI error code into a Linux errno code
The commit 85dc58289238 ("firmware: scmi: prepare uclass to pass channel reference") added an explicit parameter, channel, but it seems to make the code complex. Hiding this parameter will allow for adding a generic (protocol-agnostic) helper function, i.e. for PROTOCOL_VERSION, in a later patch. Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> --- v2 * new patch --- drivers/clk/clk_scmi.c | 27 ++------- drivers/firmware/scmi/scmi_agent-uclass.c | 74 ++++++++++++++++++----- drivers/power/regulator/scmi_regulator.c | 27 +++------ drivers/reset/reset-scmi.c | 19 +----- include/scmi_agent.h | 15 +++-- 5 files changed, 86 insertions(+), 76 deletions(-)