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