1*34180863SPeng Fan // SPDX-License-Identifier: GPL-2.0
2*34180863SPeng Fan /*
3*34180863SPeng Fan * System control and Management Interface (SCMI) NXP LMM Protocol
4*34180863SPeng Fan *
5*34180863SPeng Fan * Copyright 2025 NXP
6*34180863SPeng Fan */
7*34180863SPeng Fan
8*34180863SPeng Fan #include <linux/bits.h>
9*34180863SPeng Fan #include <linux/io.h>
10*34180863SPeng Fan #include <linux/module.h>
11*34180863SPeng Fan #include <linux/of.h>
12*34180863SPeng Fan #include <linux/platform_device.h>
13*34180863SPeng Fan #include <linux/scmi_protocol.h>
14*34180863SPeng Fan #include <linux/scmi_imx_protocol.h>
15*34180863SPeng Fan
16*34180863SPeng Fan #include "../../protocols.h"
17*34180863SPeng Fan #include "../../notify.h"
18*34180863SPeng Fan
19*34180863SPeng Fan #define SCMI_PROTOCOL_SUPPORTED_VERSION 0x10000
20*34180863SPeng Fan
21*34180863SPeng Fan enum scmi_imx_lmm_protocol_cmd {
22*34180863SPeng Fan SCMI_IMX_LMM_ATTRIBUTES = 0x3,
23*34180863SPeng Fan SCMI_IMX_LMM_BOOT = 0x4,
24*34180863SPeng Fan SCMI_IMX_LMM_RESET = 0x5,
25*34180863SPeng Fan SCMI_IMX_LMM_SHUTDOWN = 0x6,
26*34180863SPeng Fan SCMI_IMX_LMM_WAKE = 0x7,
27*34180863SPeng Fan SCMI_IMX_LMM_SUSPEND = 0x8,
28*34180863SPeng Fan SCMI_IMX_LMM_NOTIFY = 0x9,
29*34180863SPeng Fan SCMI_IMX_LMM_RESET_REASON = 0xA,
30*34180863SPeng Fan SCMI_IMX_LMM_POWER_ON = 0xB,
31*34180863SPeng Fan SCMI_IMX_LMM_RESET_VECTOR_SET = 0xC,
32*34180863SPeng Fan };
33*34180863SPeng Fan
34*34180863SPeng Fan struct scmi_imx_lmm_priv {
35*34180863SPeng Fan u32 nr_lmm;
36*34180863SPeng Fan };
37*34180863SPeng Fan
38*34180863SPeng Fan #define SCMI_IMX_LMM_NR_LM_MASK GENMASK(5, 0)
39*34180863SPeng Fan #define SCMI_IMX_LMM_NR_MAX 16
40*34180863SPeng Fan struct scmi_msg_imx_lmm_protocol_attributes {
41*34180863SPeng Fan __le32 attributes;
42*34180863SPeng Fan };
43*34180863SPeng Fan
44*34180863SPeng Fan struct scmi_msg_imx_lmm_attributes_out {
45*34180863SPeng Fan __le32 lmid;
46*34180863SPeng Fan __le32 attributes;
47*34180863SPeng Fan __le32 state;
48*34180863SPeng Fan __le32 errstatus;
49*34180863SPeng Fan u8 name[LMM_MAX_NAME];
50*34180863SPeng Fan };
51*34180863SPeng Fan
52*34180863SPeng Fan struct scmi_imx_lmm_reset_vector_set_in {
53*34180863SPeng Fan __le32 lmid;
54*34180863SPeng Fan __le32 cpuid;
55*34180863SPeng Fan __le32 flags; /* reserved for future extension */
56*34180863SPeng Fan __le32 resetvectorlow;
57*34180863SPeng Fan __le32 resetvectorhigh;
58*34180863SPeng Fan };
59*34180863SPeng Fan
60*34180863SPeng Fan struct scmi_imx_lmm_shutdown_in {
61*34180863SPeng Fan __le32 lmid;
62*34180863SPeng Fan #define SCMI_IMX_LMM_SHUTDOWN_GRACEFUL BIT(0)
63*34180863SPeng Fan __le32 flags;
64*34180863SPeng Fan };
65*34180863SPeng Fan
scmi_imx_lmm_validate_lmid(const struct scmi_protocol_handle * ph,u32 lmid)66*34180863SPeng Fan static int scmi_imx_lmm_validate_lmid(const struct scmi_protocol_handle *ph, u32 lmid)
67*34180863SPeng Fan {
68*34180863SPeng Fan struct scmi_imx_lmm_priv *priv = ph->get_priv(ph);
69*34180863SPeng Fan
70*34180863SPeng Fan if (lmid >= priv->nr_lmm)
71*34180863SPeng Fan return -EINVAL;
72*34180863SPeng Fan
73*34180863SPeng Fan return 0;
74*34180863SPeng Fan }
75*34180863SPeng Fan
scmi_imx_lmm_attributes(const struct scmi_protocol_handle * ph,u32 lmid,struct scmi_imx_lmm_info * info)76*34180863SPeng Fan static int scmi_imx_lmm_attributes(const struct scmi_protocol_handle *ph,
77*34180863SPeng Fan u32 lmid, struct scmi_imx_lmm_info *info)
78*34180863SPeng Fan {
79*34180863SPeng Fan struct scmi_msg_imx_lmm_attributes_out *out;
80*34180863SPeng Fan struct scmi_xfer *t;
81*34180863SPeng Fan int ret;
82*34180863SPeng Fan
83*34180863SPeng Fan ret = ph->xops->xfer_get_init(ph, SCMI_IMX_LMM_ATTRIBUTES, sizeof(u32), 0, &t);
84*34180863SPeng Fan if (ret)
85*34180863SPeng Fan return ret;
86*34180863SPeng Fan
87*34180863SPeng Fan put_unaligned_le32(lmid, t->tx.buf);
88*34180863SPeng Fan ret = ph->xops->do_xfer(ph, t);
89*34180863SPeng Fan if (!ret) {
90*34180863SPeng Fan out = t->rx.buf;
91*34180863SPeng Fan info->lmid = le32_to_cpu(out->lmid);
92*34180863SPeng Fan info->state = le32_to_cpu(out->state);
93*34180863SPeng Fan info->errstatus = le32_to_cpu(out->errstatus);
94*34180863SPeng Fan strscpy(info->name, out->name);
95*34180863SPeng Fan dev_dbg(ph->dev, "i.MX LMM: Logical Machine(%d), name: %s\n",
96*34180863SPeng Fan info->lmid, info->name);
97*34180863SPeng Fan } else {
98*34180863SPeng Fan dev_err(ph->dev, "i.MX LMM: Failed to get info of Logical Machine(%u)\n", lmid);
99*34180863SPeng Fan }
100*34180863SPeng Fan
101*34180863SPeng Fan ph->xops->xfer_put(ph, t);
102*34180863SPeng Fan
103*34180863SPeng Fan return ret;
104*34180863SPeng Fan }
105*34180863SPeng Fan
106*34180863SPeng Fan static int
scmi_imx_lmm_power_boot(const struct scmi_protocol_handle * ph,u32 lmid,bool boot)107*34180863SPeng Fan scmi_imx_lmm_power_boot(const struct scmi_protocol_handle *ph, u32 lmid, bool boot)
108*34180863SPeng Fan {
109*34180863SPeng Fan struct scmi_xfer *t;
110*34180863SPeng Fan u8 msg_id;
111*34180863SPeng Fan int ret;
112*34180863SPeng Fan
113*34180863SPeng Fan ret = scmi_imx_lmm_validate_lmid(ph, lmid);
114*34180863SPeng Fan if (ret)
115*34180863SPeng Fan return ret;
116*34180863SPeng Fan
117*34180863SPeng Fan if (boot)
118*34180863SPeng Fan msg_id = SCMI_IMX_LMM_BOOT;
119*34180863SPeng Fan else
120*34180863SPeng Fan msg_id = SCMI_IMX_LMM_POWER_ON;
121*34180863SPeng Fan
122*34180863SPeng Fan ret = ph->xops->xfer_get_init(ph, msg_id, sizeof(u32), 0, &t);
123*34180863SPeng Fan if (ret)
124*34180863SPeng Fan return ret;
125*34180863SPeng Fan
126*34180863SPeng Fan put_unaligned_le32(lmid, t->tx.buf);
127*34180863SPeng Fan ret = ph->xops->do_xfer(ph, t);
128*34180863SPeng Fan
129*34180863SPeng Fan ph->xops->xfer_put(ph, t);
130*34180863SPeng Fan
131*34180863SPeng Fan return ret;
132*34180863SPeng Fan }
133*34180863SPeng Fan
scmi_imx_lmm_reset_vector_set(const struct scmi_protocol_handle * ph,u32 lmid,u32 cpuid,u32 flags,u64 vector)134*34180863SPeng Fan static int scmi_imx_lmm_reset_vector_set(const struct scmi_protocol_handle *ph,
135*34180863SPeng Fan u32 lmid, u32 cpuid, u32 flags, u64 vector)
136*34180863SPeng Fan {
137*34180863SPeng Fan struct scmi_imx_lmm_reset_vector_set_in *in;
138*34180863SPeng Fan struct scmi_xfer *t;
139*34180863SPeng Fan int ret;
140*34180863SPeng Fan
141*34180863SPeng Fan ret = ph->xops->xfer_get_init(ph, SCMI_IMX_LMM_RESET_VECTOR_SET, sizeof(*in),
142*34180863SPeng Fan 0, &t);
143*34180863SPeng Fan if (ret)
144*34180863SPeng Fan return ret;
145*34180863SPeng Fan
146*34180863SPeng Fan in = t->tx.buf;
147*34180863SPeng Fan in->lmid = cpu_to_le32(lmid);
148*34180863SPeng Fan in->cpuid = cpu_to_le32(cpuid);
149*34180863SPeng Fan in->flags = cpu_to_le32(0);
150*34180863SPeng Fan in->resetvectorlow = cpu_to_le32(lower_32_bits(vector));
151*34180863SPeng Fan in->resetvectorhigh = cpu_to_le32(upper_32_bits(vector));
152*34180863SPeng Fan ret = ph->xops->do_xfer(ph, t);
153*34180863SPeng Fan
154*34180863SPeng Fan ph->xops->xfer_put(ph, t);
155*34180863SPeng Fan
156*34180863SPeng Fan return ret;
157*34180863SPeng Fan }
158*34180863SPeng Fan
scmi_imx_lmm_shutdown(const struct scmi_protocol_handle * ph,u32 lmid,u32 flags)159*34180863SPeng Fan static int scmi_imx_lmm_shutdown(const struct scmi_protocol_handle *ph, u32 lmid,
160*34180863SPeng Fan u32 flags)
161*34180863SPeng Fan {
162*34180863SPeng Fan struct scmi_imx_lmm_shutdown_in *in;
163*34180863SPeng Fan struct scmi_xfer *t;
164*34180863SPeng Fan int ret;
165*34180863SPeng Fan
166*34180863SPeng Fan ret = scmi_imx_lmm_validate_lmid(ph, lmid);
167*34180863SPeng Fan if (ret)
168*34180863SPeng Fan return ret;
169*34180863SPeng Fan
170*34180863SPeng Fan ret = ph->xops->xfer_get_init(ph, SCMI_IMX_LMM_SHUTDOWN, sizeof(*in),
171*34180863SPeng Fan 0, &t);
172*34180863SPeng Fan if (ret)
173*34180863SPeng Fan return ret;
174*34180863SPeng Fan
175*34180863SPeng Fan in = t->tx.buf;
176*34180863SPeng Fan in->lmid = cpu_to_le32(lmid);
177*34180863SPeng Fan if (flags & SCMI_IMX_LMM_SHUTDOWN_GRACEFUL)
178*34180863SPeng Fan in->flags = cpu_to_le32(SCMI_IMX_LMM_SHUTDOWN_GRACEFUL);
179*34180863SPeng Fan else
180*34180863SPeng Fan in->flags = cpu_to_le32(0);
181*34180863SPeng Fan ret = ph->xops->do_xfer(ph, t);
182*34180863SPeng Fan
183*34180863SPeng Fan ph->xops->xfer_put(ph, t);
184*34180863SPeng Fan
185*34180863SPeng Fan return ret;
186*34180863SPeng Fan }
187*34180863SPeng Fan
188*34180863SPeng Fan static const struct scmi_imx_lmm_proto_ops scmi_imx_lmm_proto_ops = {
189*34180863SPeng Fan .lmm_power_boot = scmi_imx_lmm_power_boot,
190*34180863SPeng Fan .lmm_info = scmi_imx_lmm_attributes,
191*34180863SPeng Fan .lmm_reset_vector_set = scmi_imx_lmm_reset_vector_set,
192*34180863SPeng Fan .lmm_shutdown = scmi_imx_lmm_shutdown,
193*34180863SPeng Fan };
194*34180863SPeng Fan
scmi_imx_lmm_protocol_attributes_get(const struct scmi_protocol_handle * ph,struct scmi_imx_lmm_priv * priv)195*34180863SPeng Fan static int scmi_imx_lmm_protocol_attributes_get(const struct scmi_protocol_handle *ph,
196*34180863SPeng Fan struct scmi_imx_lmm_priv *priv)
197*34180863SPeng Fan {
198*34180863SPeng Fan struct scmi_msg_imx_lmm_protocol_attributes *attr;
199*34180863SPeng Fan struct scmi_xfer *t;
200*34180863SPeng Fan int ret;
201*34180863SPeng Fan
202*34180863SPeng Fan ret = ph->xops->xfer_get_init(ph, PROTOCOL_ATTRIBUTES, 0,
203*34180863SPeng Fan sizeof(*attr), &t);
204*34180863SPeng Fan if (ret)
205*34180863SPeng Fan return ret;
206*34180863SPeng Fan
207*34180863SPeng Fan attr = t->rx.buf;
208*34180863SPeng Fan
209*34180863SPeng Fan ret = ph->xops->do_xfer(ph, t);
210*34180863SPeng Fan if (!ret) {
211*34180863SPeng Fan priv->nr_lmm = le32_get_bits(attr->attributes, SCMI_IMX_LMM_NR_LM_MASK);
212*34180863SPeng Fan if (priv->nr_lmm > SCMI_IMX_LMM_NR_MAX) {
213*34180863SPeng Fan dev_err(ph->dev, "i.MX LMM: %d:Exceed max supported Logical Machines\n",
214*34180863SPeng Fan priv->nr_lmm);
215*34180863SPeng Fan ret = -EINVAL;
216*34180863SPeng Fan } else {
217*34180863SPeng Fan dev_info(ph->dev, "i.MX LMM: %d Logical Machines\n", priv->nr_lmm);
218*34180863SPeng Fan }
219*34180863SPeng Fan }
220*34180863SPeng Fan
221*34180863SPeng Fan ph->xops->xfer_put(ph, t);
222*34180863SPeng Fan
223*34180863SPeng Fan return ret;
224*34180863SPeng Fan }
225*34180863SPeng Fan
scmi_imx_lmm_protocol_init(const struct scmi_protocol_handle * ph)226*34180863SPeng Fan static int scmi_imx_lmm_protocol_init(const struct scmi_protocol_handle *ph)
227*34180863SPeng Fan {
228*34180863SPeng Fan struct scmi_imx_lmm_priv *info;
229*34180863SPeng Fan u32 version;
230*34180863SPeng Fan int ret;
231*34180863SPeng Fan
232*34180863SPeng Fan ret = ph->xops->version_get(ph, &version);
233*34180863SPeng Fan if (ret)
234*34180863SPeng Fan return ret;
235*34180863SPeng Fan
236*34180863SPeng Fan dev_info(ph->dev, "NXP SM LMM Version %d.%d\n",
237*34180863SPeng Fan PROTOCOL_REV_MAJOR(version), PROTOCOL_REV_MINOR(version));
238*34180863SPeng Fan
239*34180863SPeng Fan info = devm_kzalloc(ph->dev, sizeof(*info), GFP_KERNEL);
240*34180863SPeng Fan if (!info)
241*34180863SPeng Fan return -ENOMEM;
242*34180863SPeng Fan
243*34180863SPeng Fan ret = scmi_imx_lmm_protocol_attributes_get(ph, info);
244*34180863SPeng Fan if (ret)
245*34180863SPeng Fan return ret;
246*34180863SPeng Fan
247*34180863SPeng Fan return ph->set_priv(ph, info, version);
248*34180863SPeng Fan }
249*34180863SPeng Fan
250*34180863SPeng Fan static const struct scmi_protocol scmi_imx_lmm = {
251*34180863SPeng Fan .id = SCMI_PROTOCOL_IMX_LMM,
252*34180863SPeng Fan .owner = THIS_MODULE,
253*34180863SPeng Fan .instance_init = &scmi_imx_lmm_protocol_init,
254*34180863SPeng Fan .ops = &scmi_imx_lmm_proto_ops,
255*34180863SPeng Fan .supported_version = SCMI_PROTOCOL_SUPPORTED_VERSION,
256*34180863SPeng Fan .vendor_id = SCMI_IMX_VENDOR,
257*34180863SPeng Fan .sub_vendor_id = SCMI_IMX_SUBVENDOR,
258*34180863SPeng Fan };
259*34180863SPeng Fan module_scmi_protocol(scmi_imx_lmm);
260*34180863SPeng Fan
261*34180863SPeng Fan MODULE_ALIAS("scmi-protocol-" __stringify(SCMI_PROTOCOL_IMX_LMM) "-" SCMI_IMX_VENDOR);
262*34180863SPeng Fan MODULE_DESCRIPTION("i.MX SCMI LMM driver");
263*34180863SPeng Fan MODULE_LICENSE("GPL");
264