xref: /linux/drivers/firmware/imx/sm-lmm.c (revision 297d9111e9fcf47dd1dcc6f79bba915f35378d01)
1*7242bbf4SPeng Fan // SPDX-License-Identifier: GPL-2.0
2*7242bbf4SPeng Fan /*
3*7242bbf4SPeng Fan  * Copyright 2025 NXP
4*7242bbf4SPeng Fan  */
5*7242bbf4SPeng Fan 
6*7242bbf4SPeng Fan #include <linux/firmware/imx/sm.h>
7*7242bbf4SPeng Fan #include <linux/module.h>
8*7242bbf4SPeng Fan #include <linux/of.h>
9*7242bbf4SPeng Fan #include <linux/platform_device.h>
10*7242bbf4SPeng Fan #include <linux/scmi_protocol.h>
11*7242bbf4SPeng Fan #include <linux/scmi_imx_protocol.h>
12*7242bbf4SPeng Fan 
13*7242bbf4SPeng Fan static const struct scmi_imx_lmm_proto_ops *imx_lmm_ops;
14*7242bbf4SPeng Fan static struct scmi_protocol_handle *ph;
15*7242bbf4SPeng Fan 
scmi_imx_lmm_info(u32 lmid,struct scmi_imx_lmm_info * info)16*7242bbf4SPeng Fan int scmi_imx_lmm_info(u32 lmid, struct scmi_imx_lmm_info *info)
17*7242bbf4SPeng Fan {
18*7242bbf4SPeng Fan 	if (!ph)
19*7242bbf4SPeng Fan 		return -EPROBE_DEFER;
20*7242bbf4SPeng Fan 
21*7242bbf4SPeng Fan 	if (!info)
22*7242bbf4SPeng Fan 		return -EINVAL;
23*7242bbf4SPeng Fan 
24*7242bbf4SPeng Fan 	return imx_lmm_ops->lmm_info(ph, lmid, info);
25*7242bbf4SPeng Fan };
26*7242bbf4SPeng Fan EXPORT_SYMBOL(scmi_imx_lmm_info);
27*7242bbf4SPeng Fan 
scmi_imx_lmm_reset_vector_set(u32 lmid,u32 cpuid,u32 flags,u64 vector)28*7242bbf4SPeng Fan int scmi_imx_lmm_reset_vector_set(u32 lmid, u32 cpuid, u32 flags, u64 vector)
29*7242bbf4SPeng Fan {
30*7242bbf4SPeng Fan 	if (!ph)
31*7242bbf4SPeng Fan 		return -EPROBE_DEFER;
32*7242bbf4SPeng Fan 
33*7242bbf4SPeng Fan 	return imx_lmm_ops->lmm_reset_vector_set(ph, lmid, cpuid, flags, vector);
34*7242bbf4SPeng Fan }
35*7242bbf4SPeng Fan EXPORT_SYMBOL(scmi_imx_lmm_reset_vector_set);
36*7242bbf4SPeng Fan 
scmi_imx_lmm_operation(u32 lmid,enum scmi_imx_lmm_op op,u32 flags)37*7242bbf4SPeng Fan int scmi_imx_lmm_operation(u32 lmid, enum scmi_imx_lmm_op op, u32 flags)
38*7242bbf4SPeng Fan {
39*7242bbf4SPeng Fan 	if (!ph)
40*7242bbf4SPeng Fan 		return -EPROBE_DEFER;
41*7242bbf4SPeng Fan 
42*7242bbf4SPeng Fan 	switch (op) {
43*7242bbf4SPeng Fan 	case SCMI_IMX_LMM_BOOT:
44*7242bbf4SPeng Fan 		return imx_lmm_ops->lmm_power_boot(ph, lmid, true);
45*7242bbf4SPeng Fan 	case SCMI_IMX_LMM_POWER_ON:
46*7242bbf4SPeng Fan 		return imx_lmm_ops->lmm_power_boot(ph, lmid, false);
47*7242bbf4SPeng Fan 	case SCMI_IMX_LMM_SHUTDOWN:
48*7242bbf4SPeng Fan 		return imx_lmm_ops->lmm_shutdown(ph, lmid, flags);
49*7242bbf4SPeng Fan 	default:
50*7242bbf4SPeng Fan 		break;
51*7242bbf4SPeng Fan 	}
52*7242bbf4SPeng Fan 
53*7242bbf4SPeng Fan 	return -EINVAL;
54*7242bbf4SPeng Fan }
55*7242bbf4SPeng Fan EXPORT_SYMBOL(scmi_imx_lmm_operation);
56*7242bbf4SPeng Fan 
scmi_imx_lmm_probe(struct scmi_device * sdev)57*7242bbf4SPeng Fan static int scmi_imx_lmm_probe(struct scmi_device *sdev)
58*7242bbf4SPeng Fan {
59*7242bbf4SPeng Fan 	const struct scmi_handle *handle = sdev->handle;
60*7242bbf4SPeng Fan 
61*7242bbf4SPeng Fan 	if (!handle)
62*7242bbf4SPeng Fan 		return -ENODEV;
63*7242bbf4SPeng Fan 
64*7242bbf4SPeng Fan 	if (imx_lmm_ops) {
65*7242bbf4SPeng Fan 		dev_err(&sdev->dev, "lmm already initialized\n");
66*7242bbf4SPeng Fan 		return -EEXIST;
67*7242bbf4SPeng Fan 	}
68*7242bbf4SPeng Fan 
69*7242bbf4SPeng Fan 	imx_lmm_ops = handle->devm_protocol_get(sdev, SCMI_PROTOCOL_IMX_LMM, &ph);
70*7242bbf4SPeng Fan 	if (IS_ERR(imx_lmm_ops))
71*7242bbf4SPeng Fan 		return PTR_ERR(imx_lmm_ops);
72*7242bbf4SPeng Fan 
73*7242bbf4SPeng Fan 	return 0;
74*7242bbf4SPeng Fan }
75*7242bbf4SPeng Fan 
76*7242bbf4SPeng Fan static const struct scmi_device_id scmi_id_table[] = {
77*7242bbf4SPeng Fan 	{ SCMI_PROTOCOL_IMX_LMM, "imx-lmm" },
78*7242bbf4SPeng Fan 	{ },
79*7242bbf4SPeng Fan };
80*7242bbf4SPeng Fan MODULE_DEVICE_TABLE(scmi, scmi_id_table);
81*7242bbf4SPeng Fan 
82*7242bbf4SPeng Fan static struct scmi_driver scmi_imx_lmm_driver = {
83*7242bbf4SPeng Fan 	.name = "scmi-imx-lmm",
84*7242bbf4SPeng Fan 	.probe = scmi_imx_lmm_probe,
85*7242bbf4SPeng Fan 	.id_table = scmi_id_table,
86*7242bbf4SPeng Fan };
87*7242bbf4SPeng Fan module_scmi_driver(scmi_imx_lmm_driver);
88*7242bbf4SPeng Fan 
89*7242bbf4SPeng Fan MODULE_AUTHOR("Peng Fan <peng.fan@nxp.com>");
90*7242bbf4SPeng Fan MODULE_DESCRIPTION("IMX SM LMM driver");
91*7242bbf4SPeng Fan MODULE_LICENSE("GPL");
92