xref: /linux/drivers/firmware/arm_scmi/vendors/imx/imx-sm-lmm.c (revision 297d9111e9fcf47dd1dcc6f79bba915f35378d01)
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