| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * Copyright 2025 NXP |
| * |
| * Peng Fan <peng.fan@nxp.com> |
| */ |
| |
| #include <dm/uclass.h> |
| #include <scmi_agent.h> |
| |
| int imx_clk_scmi_enable(u32 clock_id, bool enable) |
| { |
| struct scmi_clk_state_in in = { |
| .clock_id = clock_id, |
| .attributes = (enable) ? 1 : 0, |
| }; |
| struct scmi_clk_state_out out; |
| struct scmi_msg msg = SCMI_MSG_IN(SCMI_PROTOCOL_ID_CLOCK, |
| SCMI_CLOCK_CONFIG_SET, |
| in, out); |
| int ret; |
| struct udevice *dev; |
| |
| ret = uclass_get_device_by_name(UCLASS_CLK, "protocol@14", &dev); |
| if (ret) |
| return ret; |
| |
| ret = devm_scmi_process_msg(dev, &msg); |
| if (ret) |
| return ret; |
| |
| return scmi_to_linux_errno(out.status); |
| } |
| |
| ulong imx_clk_scmi_set_rate(u32 clock_id, ulong rate) |
| { |
| struct scmi_clk_rate_set_in in = { |
| .clock_id = clock_id, |
| .flags = SCMI_CLK_RATE_ROUND_CLOSEST, |
| .rate_lsb = (u32)rate, |
| .rate_msb = (u32)((u64)rate >> 32), |
| }; |
| struct scmi_clk_rate_set_out out; |
| struct scmi_msg msg = SCMI_MSG_IN(SCMI_PROTOCOL_ID_CLOCK, |
| SCMI_CLOCK_RATE_SET, |
| in, out); |
| int ret; |
| struct udevice *dev; |
| |
| ret = uclass_get_device_by_name(UCLASS_CLK, "protocol@14", &dev); |
| if (ret) |
| return ret; |
| |
| ret = devm_scmi_process_msg(dev, &msg); |
| if (ret < 0) |
| return ret; |
| |
| ret = scmi_to_linux_errno(out.status); |
| if (ret < 0) |
| return ret; |
| |
| struct scmi_clk_rate_get_in in_rate = { |
| .clock_id = clock_id, |
| }; |
| struct scmi_clk_rate_get_out out_rate; |
| |
| msg = SCMI_MSG_IN(SCMI_PROTOCOL_ID_CLOCK, SCMI_CLOCK_RATE_GET, in_rate, out_rate); |
| |
| ret = devm_scmi_process_msg(dev, &msg); |
| if (ret < 0) |
| return ret; |
| |
| ret = scmi_to_linux_errno(out_rate.status); |
| if (ret < 0) |
| return ret; |
| |
| return (ulong)(((u64)out_rate.rate_msb << 32) | out_rate.rate_lsb); |
| } |
| |
| ulong imx_clk_scmi_get_rate(u32 clock_id) |
| { |
| struct scmi_clk_rate_get_in in = { |
| .clock_id = clock_id, |
| }; |
| struct scmi_clk_rate_get_out out; |
| struct scmi_msg msg = SCMI_MSG_IN(SCMI_PROTOCOL_ID_CLOCK, |
| SCMI_CLOCK_RATE_GET, |
| in, out); |
| int ret; |
| struct udevice *dev; |
| |
| ret = uclass_get_device_by_name(UCLASS_CLK, "protocol@14", &dev); |
| if (ret) |
| return ret; |
| |
| ret = devm_scmi_process_msg(dev, &msg); |
| if (ret < 0) |
| return ret; |
| |
| ret = scmi_to_linux_errno(out.status); |
| if (ret < 0) |
| return ret; |
| |
| return (ulong)(((u64)out.rate_msb << 32) | out.rate_lsb); |
| } |
| |
| int imx_clk_scmi_set_parent(u32 clock_id, u32 parent_id) |
| { |
| struct scmi_clk_parent_set_in in = { |
| .clock_id = clock_id, |
| .parent_clk = parent_id, |
| }; |
| struct scmi_clk_parent_set_out out; |
| struct scmi_msg msg = SCMI_MSG_IN(SCMI_PROTOCOL_ID_CLOCK, |
| SCMI_CLOCK_PARENT_SET, |
| in, out); |
| int ret; |
| struct udevice *dev; |
| |
| ret = uclass_get_device_by_name(UCLASS_CLK, "protocol@14", &dev); |
| if (ret) |
| return ret; |
| |
| ret = devm_scmi_process_msg(dev, &msg); |
| if (ret < 0) |
| return ret; |
| |
| ret = scmi_to_linux_errno(out.status); |
| if (ret < 0 && ret != -EACCES) |
| printf("%s: %d, clock_id %u\n", __func__, ret, clock_id); |
| |
| return ret; |
| } |