xref: /openbmc/u-boot/drivers/remoteproc/k3_rproc.c (revision c365ed7d4bc4173baad05a0c2d40d48ce8e41394)
1*c365ed7dSLokesh Vutla // SPDX-License-Identifier: GPL-2.0+
2*c365ed7dSLokesh Vutla /*
3*c365ed7dSLokesh Vutla  * Texas Instruments' K3 Remoteproc driver
4*c365ed7dSLokesh Vutla  *
5*c365ed7dSLokesh Vutla  * Copyright (C) 2018 Texas Instruments Incorporated - http://www.ti.com/
6*c365ed7dSLokesh Vutla  *	Lokesh Vutla <lokeshvutla@ti.com>
7*c365ed7dSLokesh Vutla  *
8*c365ed7dSLokesh Vutla  */
9*c365ed7dSLokesh Vutla 
10*c365ed7dSLokesh Vutla #include <common.h>
11*c365ed7dSLokesh Vutla #include <dm.h>
12*c365ed7dSLokesh Vutla #include <remoteproc.h>
13*c365ed7dSLokesh Vutla #include <errno.h>
14*c365ed7dSLokesh Vutla #include <clk.h>
15*c365ed7dSLokesh Vutla #include <reset.h>
16*c365ed7dSLokesh Vutla #include <asm/io.h>
17*c365ed7dSLokesh Vutla #include <power-domain.h>
18*c365ed7dSLokesh Vutla #include <linux/soc/ti/ti_sci_protocol.h>
19*c365ed7dSLokesh Vutla 
20*c365ed7dSLokesh Vutla #define INVALID_ID	0xffff
21*c365ed7dSLokesh Vutla 
22*c365ed7dSLokesh Vutla #define GTC_CNTCR_REG	0x0
23*c365ed7dSLokesh Vutla #define GTC_CNTR_EN	0x3
24*c365ed7dSLokesh Vutla 
25*c365ed7dSLokesh Vutla /**
26*c365ed7dSLokesh Vutla  * struct k3_rproc_privdata - Structure representing Remote processor data.
27*c365ed7dSLokesh Vutla  * @rproc_pwrdmn:	rproc power domain data
28*c365ed7dSLokesh Vutla  * @rproc_rst:		rproc reset control data
29*c365ed7dSLokesh Vutla  * @sci:		Pointer to TISCI handle
30*c365ed7dSLokesh Vutla  * @gtc_base:		Timer base address.
31*c365ed7dSLokesh Vutla  * @proc_id:		TISCI processor ID
32*c365ed7dSLokesh Vutla  * @host_id:		TISCI host id to which the processor gets assigned to.
33*c365ed7dSLokesh Vutla  */
34*c365ed7dSLokesh Vutla struct k3_rproc_privdata {
35*c365ed7dSLokesh Vutla 	struct power_domain rproc_pwrdmn;
36*c365ed7dSLokesh Vutla 	struct power_domain gtc_pwrdmn;
37*c365ed7dSLokesh Vutla 	struct reset_ctl rproc_rst;
38*c365ed7dSLokesh Vutla 	const struct ti_sci_handle *sci;
39*c365ed7dSLokesh Vutla 	void *gtc_base;
40*c365ed7dSLokesh Vutla 	u16 proc_id;
41*c365ed7dSLokesh Vutla 	u16 host_id;
42*c365ed7dSLokesh Vutla };
43*c365ed7dSLokesh Vutla 
44*c365ed7dSLokesh Vutla /**
45*c365ed7dSLokesh Vutla  * k3_rproc_load() - Load up the Remote processor image
46*c365ed7dSLokesh Vutla  * @dev:	rproc device pointer
47*c365ed7dSLokesh Vutla  * @addr:	Address at which image is available
48*c365ed7dSLokesh Vutla  * @size:	size of the image
49*c365ed7dSLokesh Vutla  *
50*c365ed7dSLokesh Vutla  * Return: 0 if all goes good, else appropriate error message.
51*c365ed7dSLokesh Vutla  */
k3_rproc_load(struct udevice * dev,ulong addr,ulong size)52*c365ed7dSLokesh Vutla static int k3_rproc_load(struct udevice *dev, ulong addr, ulong size)
53*c365ed7dSLokesh Vutla {
54*c365ed7dSLokesh Vutla 	struct k3_rproc_privdata *rproc = dev_get_priv(dev);
55*c365ed7dSLokesh Vutla 	const struct ti_sci_proc_ops *pops = &rproc->sci->ops.proc_ops;
56*c365ed7dSLokesh Vutla 	int ret;
57*c365ed7dSLokesh Vutla 
58*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s addr = 0x%lx, size = 0x%lx\n", __func__, addr, size);
59*c365ed7dSLokesh Vutla 
60*c365ed7dSLokesh Vutla 	/* request for the processor */
61*c365ed7dSLokesh Vutla 	ret = pops->proc_request(rproc->sci, rproc->proc_id);
62*c365ed7dSLokesh Vutla 	if (ret) {
63*c365ed7dSLokesh Vutla 		dev_err(dev, "Requesting processor failed %d\n", ret);
64*c365ed7dSLokesh Vutla 		return ret;
65*c365ed7dSLokesh Vutla 	}
66*c365ed7dSLokesh Vutla 
67*c365ed7dSLokesh Vutla 	ret = pops->set_proc_boot_cfg(rproc->sci, rproc->proc_id, addr, 0, 0);
68*c365ed7dSLokesh Vutla 	if (ret) {
69*c365ed7dSLokesh Vutla 		dev_err(dev, "set_proc_boot_cfg failed %d\n", ret);
70*c365ed7dSLokesh Vutla 		return ret;
71*c365ed7dSLokesh Vutla 	}
72*c365ed7dSLokesh Vutla 
73*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s: rproc successfully loaded\n", __func__);
74*c365ed7dSLokesh Vutla 
75*c365ed7dSLokesh Vutla 	return 0;
76*c365ed7dSLokesh Vutla }
77*c365ed7dSLokesh Vutla 
78*c365ed7dSLokesh Vutla /**
79*c365ed7dSLokesh Vutla  * k3_rproc_start() - Start the remote processor
80*c365ed7dSLokesh Vutla  * @dev:	rproc device pointer
81*c365ed7dSLokesh Vutla  *
82*c365ed7dSLokesh Vutla  * Return: 0 if all went ok, else return appropriate error
83*c365ed7dSLokesh Vutla  */
k3_rproc_start(struct udevice * dev)84*c365ed7dSLokesh Vutla static int k3_rproc_start(struct udevice *dev)
85*c365ed7dSLokesh Vutla {
86*c365ed7dSLokesh Vutla 	struct k3_rproc_privdata *rproc = dev_get_priv(dev);
87*c365ed7dSLokesh Vutla 	const struct ti_sci_proc_ops *pops = &rproc->sci->ops.proc_ops;
88*c365ed7dSLokesh Vutla 	int ret;
89*c365ed7dSLokesh Vutla 
90*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s\n", __func__);
91*c365ed7dSLokesh Vutla 
92*c365ed7dSLokesh Vutla 	ret = power_domain_on(&rproc->gtc_pwrdmn);
93*c365ed7dSLokesh Vutla 	if (ret) {
94*c365ed7dSLokesh Vutla 		dev_err(dev, "power_domain_on() failed: %d\n", ret);
95*c365ed7dSLokesh Vutla 		return ret;
96*c365ed7dSLokesh Vutla 	}
97*c365ed7dSLokesh Vutla 
98*c365ed7dSLokesh Vutla 	/* Enable the timer before starting remote core */
99*c365ed7dSLokesh Vutla 	writel(GTC_CNTR_EN, rproc->gtc_base + GTC_CNTCR_REG);
100*c365ed7dSLokesh Vutla 
101*c365ed7dSLokesh Vutla 	/*
102*c365ed7dSLokesh Vutla 	 * Setting the right clock frequency would have taken care by
103*c365ed7dSLokesh Vutla 	 * assigned-clock-rates during the device probe. So no need to
104*c365ed7dSLokesh Vutla 	 * set the frequency again here.
105*c365ed7dSLokesh Vutla 	 */
106*c365ed7dSLokesh Vutla 	ret = power_domain_on(&rproc->rproc_pwrdmn);
107*c365ed7dSLokesh Vutla 	if (ret) {
108*c365ed7dSLokesh Vutla 		dev_err(dev, "power_domain_on() failed: %d\n", ret);
109*c365ed7dSLokesh Vutla 		return ret;
110*c365ed7dSLokesh Vutla 	}
111*c365ed7dSLokesh Vutla 
112*c365ed7dSLokesh Vutla 	if (rproc->host_id != INVALID_ID) {
113*c365ed7dSLokesh Vutla 		ret = pops->proc_handover(rproc->sci, rproc->proc_id,
114*c365ed7dSLokesh Vutla 					  rproc->host_id);
115*c365ed7dSLokesh Vutla 		if (ret) {
116*c365ed7dSLokesh Vutla 			dev_err(dev, "Handover processor failed %d\n", ret);
117*c365ed7dSLokesh Vutla 			return ret;
118*c365ed7dSLokesh Vutla 		}
119*c365ed7dSLokesh Vutla 	} else {
120*c365ed7dSLokesh Vutla 		ret = pops->proc_release(rproc->sci, rproc->proc_id);
121*c365ed7dSLokesh Vutla 		if (ret) {
122*c365ed7dSLokesh Vutla 			dev_err(dev, "Processor release failed %d\n", ret);
123*c365ed7dSLokesh Vutla 			return ret;
124*c365ed7dSLokesh Vutla 		}
125*c365ed7dSLokesh Vutla 	}
126*c365ed7dSLokesh Vutla 
127*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s: rproc successfully started\n", __func__);
128*c365ed7dSLokesh Vutla 
129*c365ed7dSLokesh Vutla 	return 0;
130*c365ed7dSLokesh Vutla }
131*c365ed7dSLokesh Vutla 
132*c365ed7dSLokesh Vutla /**
133*c365ed7dSLokesh Vutla  * k3_rproc_init() - Initialize the remote processor
134*c365ed7dSLokesh Vutla  * @dev:	rproc device pointer
135*c365ed7dSLokesh Vutla  *
136*c365ed7dSLokesh Vutla  * Return: 0 if all went ok, else return appropriate error
137*c365ed7dSLokesh Vutla  */
k3_rproc_init(struct udevice * dev)138*c365ed7dSLokesh Vutla static int k3_rproc_init(struct udevice *dev)
139*c365ed7dSLokesh Vutla {
140*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s\n", __func__);
141*c365ed7dSLokesh Vutla 
142*c365ed7dSLokesh Vutla 	/* Enable the module */
143*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s: rproc successfully initialized\n", __func__);
144*c365ed7dSLokesh Vutla 
145*c365ed7dSLokesh Vutla 	return 0;
146*c365ed7dSLokesh Vutla }
147*c365ed7dSLokesh Vutla 
148*c365ed7dSLokesh Vutla static const struct dm_rproc_ops k3_rproc_ops = {
149*c365ed7dSLokesh Vutla 	.init = k3_rproc_init,
150*c365ed7dSLokesh Vutla 	.load = k3_rproc_load,
151*c365ed7dSLokesh Vutla 	.start = k3_rproc_start,
152*c365ed7dSLokesh Vutla };
153*c365ed7dSLokesh Vutla 
154*c365ed7dSLokesh Vutla /**
155*c365ed7dSLokesh Vutla  * k3_of_to_priv() - generate private data from device tree
156*c365ed7dSLokesh Vutla  * @dev:	corresponding k3 remote processor device
157*c365ed7dSLokesh Vutla  * @priv:	pointer to driver specific private data
158*c365ed7dSLokesh Vutla  *
159*c365ed7dSLokesh Vutla  * Return: 0 if all goes good, else appropriate error message.
160*c365ed7dSLokesh Vutla  */
k3_rproc_of_to_priv(struct udevice * dev,struct k3_rproc_privdata * rproc)161*c365ed7dSLokesh Vutla static int k3_rproc_of_to_priv(struct udevice *dev,
162*c365ed7dSLokesh Vutla 			       struct k3_rproc_privdata *rproc)
163*c365ed7dSLokesh Vutla {
164*c365ed7dSLokesh Vutla 	int ret;
165*c365ed7dSLokesh Vutla 
166*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s\n", __func__);
167*c365ed7dSLokesh Vutla 
168*c365ed7dSLokesh Vutla 	ret = power_domain_get_by_index(dev, &rproc->rproc_pwrdmn, 1);
169*c365ed7dSLokesh Vutla 	if (ret) {
170*c365ed7dSLokesh Vutla 		dev_err(dev, "power_domain_get() failed: %d\n", ret);
171*c365ed7dSLokesh Vutla 		return ret;
172*c365ed7dSLokesh Vutla 	}
173*c365ed7dSLokesh Vutla 
174*c365ed7dSLokesh Vutla 	ret = power_domain_get_by_index(dev, &rproc->gtc_pwrdmn, 0);
175*c365ed7dSLokesh Vutla 	if (ret) {
176*c365ed7dSLokesh Vutla 		dev_err(dev, "power_domain_get() failed: %d\n", ret);
177*c365ed7dSLokesh Vutla 		return ret;
178*c365ed7dSLokesh Vutla 	}
179*c365ed7dSLokesh Vutla 
180*c365ed7dSLokesh Vutla 	ret = reset_get_by_index(dev, 0, &rproc->rproc_rst);
181*c365ed7dSLokesh Vutla 	if (ret) {
182*c365ed7dSLokesh Vutla 		dev_err(dev, "reset_get() failed: %d\n", ret);
183*c365ed7dSLokesh Vutla 		return ret;
184*c365ed7dSLokesh Vutla 	}
185*c365ed7dSLokesh Vutla 
186*c365ed7dSLokesh Vutla 	rproc->sci = ti_sci_get_by_phandle(dev, "ti,sci");
187*c365ed7dSLokesh Vutla 	if (IS_ERR(rproc->sci)) {
188*c365ed7dSLokesh Vutla 		dev_err(dev, "ti_sci get failed: %d\n", ret);
189*c365ed7dSLokesh Vutla 		return PTR_ERR(rproc->sci);
190*c365ed7dSLokesh Vutla 	}
191*c365ed7dSLokesh Vutla 
192*c365ed7dSLokesh Vutla 	rproc->gtc_base = dev_read_addr_ptr(dev);
193*c365ed7dSLokesh Vutla 	if (!rproc->gtc_base) {
194*c365ed7dSLokesh Vutla 		dev_err(dev, "Get address failed\n");
195*c365ed7dSLokesh Vutla 		return -ENODEV;
196*c365ed7dSLokesh Vutla 	}
197*c365ed7dSLokesh Vutla 
198*c365ed7dSLokesh Vutla 	rproc->proc_id = dev_read_u32_default(dev, "ti,sci-proc-id",
199*c365ed7dSLokesh Vutla 					      INVALID_ID);
200*c365ed7dSLokesh Vutla 	rproc->host_id = dev_read_u32_default(dev, "ti,sci-host-id",
201*c365ed7dSLokesh Vutla 					      INVALID_ID);
202*c365ed7dSLokesh Vutla 
203*c365ed7dSLokesh Vutla 	return 0;
204*c365ed7dSLokesh Vutla }
205*c365ed7dSLokesh Vutla 
206*c365ed7dSLokesh Vutla /**
207*c365ed7dSLokesh Vutla  * k3_rproc_probe() - Basic probe
208*c365ed7dSLokesh Vutla  * @dev:	corresponding k3 remote processor device
209*c365ed7dSLokesh Vutla  *
210*c365ed7dSLokesh Vutla  * Return: 0 if all goes good, else appropriate error message.
211*c365ed7dSLokesh Vutla  */
k3_rproc_probe(struct udevice * dev)212*c365ed7dSLokesh Vutla static int k3_rproc_probe(struct udevice *dev)
213*c365ed7dSLokesh Vutla {
214*c365ed7dSLokesh Vutla 	struct k3_rproc_privdata *priv;
215*c365ed7dSLokesh Vutla 	int ret;
216*c365ed7dSLokesh Vutla 
217*c365ed7dSLokesh Vutla 	dev_dbg(dev, "%s\n", __func__);
218*c365ed7dSLokesh Vutla 
219*c365ed7dSLokesh Vutla 	priv = dev_get_priv(dev);
220*c365ed7dSLokesh Vutla 
221*c365ed7dSLokesh Vutla 	ret = k3_rproc_of_to_priv(dev, priv);
222*c365ed7dSLokesh Vutla 	if (ret) {
223*c365ed7dSLokesh Vutla 		dev_dbg(dev, "%s: Probe failed with error %d\n", __func__, ret);
224*c365ed7dSLokesh Vutla 		return ret;
225*c365ed7dSLokesh Vutla 	}
226*c365ed7dSLokesh Vutla 
227*c365ed7dSLokesh Vutla 	dev_dbg(dev, "Remoteproc successfully probed\n");
228*c365ed7dSLokesh Vutla 
229*c365ed7dSLokesh Vutla 	return 0;
230*c365ed7dSLokesh Vutla }
231*c365ed7dSLokesh Vutla 
232*c365ed7dSLokesh Vutla static const struct udevice_id k3_rproc_ids[] = {
233*c365ed7dSLokesh Vutla 	{ .compatible = "ti,am654-rproc"},
234*c365ed7dSLokesh Vutla 	{}
235*c365ed7dSLokesh Vutla };
236*c365ed7dSLokesh Vutla 
237*c365ed7dSLokesh Vutla U_BOOT_DRIVER(k3_rproc) = {
238*c365ed7dSLokesh Vutla 	.name = "k3_rproc",
239*c365ed7dSLokesh Vutla 	.of_match = k3_rproc_ids,
240*c365ed7dSLokesh Vutla 	.id = UCLASS_REMOTEPROC,
241*c365ed7dSLokesh Vutla 	.ops = &k3_rproc_ops,
242*c365ed7dSLokesh Vutla 	.probe = k3_rproc_probe,
243*c365ed7dSLokesh Vutla 	.priv_auto_alloc_size = sizeof(struct k3_rproc_privdata),
244*c365ed7dSLokesh Vutla };
245