xref: /rk3399_rockchip-uboot/drivers/remoteproc/sandbox_testproc.c (revision a69fdc7787bfa2f27eed74c2ee58c28ce932d502)
1*3df0b8b4SNishanth Menon /*
2*3df0b8b4SNishanth Menon  * (C) Copyright 2015
3*3df0b8b4SNishanth Menon  * Texas Instruments Incorporated - http://www.ti.com/
4*3df0b8b4SNishanth Menon  * SPDX-License-Identifier:	GPL-2.0+
5*3df0b8b4SNishanth Menon  */
6*3df0b8b4SNishanth Menon #define pr_fmt(fmt) "%s: " fmt, __func__
7*3df0b8b4SNishanth Menon #include <common.h>
8*3df0b8b4SNishanth Menon #include <dm.h>
9*3df0b8b4SNishanth Menon #include <errno.h>
10*3df0b8b4SNishanth Menon #include <remoteproc.h>
11*3df0b8b4SNishanth Menon 
12*3df0b8b4SNishanth Menon /**
13*3df0b8b4SNishanth Menon  * enum sandbox_state - different device states
14*3df0b8b4SNishanth Menon  * @sb_booted:	Entry condition, just booted
15*3df0b8b4SNishanth Menon  * @sb_init:	Initialized (basic environment is ready)
16*3df0b8b4SNishanth Menon  * @sb_reset:	Held in reset (accessible, but not running)
17*3df0b8b4SNishanth Menon  * @sb_loaded:	Loaded with image (but not running)
18*3df0b8b4SNishanth Menon  * @sb_running:	Processor is running
19*3df0b8b4SNishanth Menon  */
20*3df0b8b4SNishanth Menon enum sandbox_state {
21*3df0b8b4SNishanth Menon 	sb_booted,
22*3df0b8b4SNishanth Menon 	sb_init,
23*3df0b8b4SNishanth Menon 	sb_reset,
24*3df0b8b4SNishanth Menon 	sb_loaded,
25*3df0b8b4SNishanth Menon 	sb_running
26*3df0b8b4SNishanth Menon };
27*3df0b8b4SNishanth Menon 
28*3df0b8b4SNishanth Menon /**
29*3df0b8b4SNishanth Menon  * struct sandbox_test_devdata - private data per device
30*3df0b8b4SNishanth Menon  * @current_state:	device current state
31*3df0b8b4SNishanth Menon  */
32*3df0b8b4SNishanth Menon struct sandbox_test_devdata {
33*3df0b8b4SNishanth Menon 	enum sandbox_state current_state;
34*3df0b8b4SNishanth Menon };
35*3df0b8b4SNishanth Menon 
36*3df0b8b4SNishanth Menon /**
37*3df0b8b4SNishanth Menon  * sandbox_dev_move_to_state() - statemachine for our dummy device
38*3df0b8b4SNishanth Menon  * @dev:	device to switch state
39*3df0b8b4SNishanth Menon  * @next_state:	next proposed state
40*3df0b8b4SNishanth Menon  *
41*3df0b8b4SNishanth Menon  * This tries to follow the following statemachine:
42*3df0b8b4SNishanth Menon  *           Entry
43*3df0b8b4SNishanth Menon  *            |
44*3df0b8b4SNishanth Menon  *            v
45*3df0b8b4SNishanth Menon  *         +-------+
46*3df0b8b4SNishanth Menon  *     +---+ init  |
47*3df0b8b4SNishanth Menon  *     |   |       | <---------------------+
48*3df0b8b4SNishanth Menon  *     |   +-------+                       |
49*3df0b8b4SNishanth Menon  *     |                                   |
50*3df0b8b4SNishanth Menon  *     |                                   |
51*3df0b8b4SNishanth Menon  *     |   +--------+                      |
52*3df0b8b4SNishanth Menon  * Load|   |  reset |                      |
53*3df0b8b4SNishanth Menon  *     |   |        | <----------+         |
54*3df0b8b4SNishanth Menon  *     |   +--------+            |         |
55*3df0b8b4SNishanth Menon  *     |        |Load            |         |
56*3df0b8b4SNishanth Menon  *     |        |                |         |
57*3df0b8b4SNishanth Menon  *     |   +----v----+   reset   |         |
58*3df0b8b4SNishanth Menon  *     +-> |         |    (opt)  |         |
59*3df0b8b4SNishanth Menon  *         |  Loaded +-----------+         |
60*3df0b8b4SNishanth Menon  *         |         |                     |
61*3df0b8b4SNishanth Menon  *         +----+----+                     |
62*3df0b8b4SNishanth Menon  *              | Start                    |
63*3df0b8b4SNishanth Menon  *          +---v-----+        (opt)       |
64*3df0b8b4SNishanth Menon  *       +->| Running |        Stop        |
65*3df0b8b4SNishanth Menon  * Ping  +- |         +--------------------+
66*3df0b8b4SNishanth Menon  * (opt)    +---------+
67*3df0b8b4SNishanth Menon  *
68*3df0b8b4SNishanth Menon  * (is_running does not change state)
69*3df0b8b4SNishanth Menon  *
70*3df0b8b4SNishanth Menon  * Return: 0 when valid state transition is seen, else returns -EINVAL
71*3df0b8b4SNishanth Menon  */
sandbox_dev_move_to_state(struct udevice * dev,enum sandbox_state next_state)72*3df0b8b4SNishanth Menon static int sandbox_dev_move_to_state(struct udevice *dev,
73*3df0b8b4SNishanth Menon 				     enum sandbox_state next_state)
74*3df0b8b4SNishanth Menon {
75*3df0b8b4SNishanth Menon 	struct sandbox_test_devdata *ddata = dev_get_priv(dev);
76*3df0b8b4SNishanth Menon 
77*3df0b8b4SNishanth Menon 	/* No state transition is OK */
78*3df0b8b4SNishanth Menon 	if (ddata->current_state == next_state)
79*3df0b8b4SNishanth Menon 		return 0;
80*3df0b8b4SNishanth Menon 
81*3df0b8b4SNishanth Menon 	debug("current_state=%d, next_state=%d\n", ddata->current_state,
82*3df0b8b4SNishanth Menon 	      next_state);
83*3df0b8b4SNishanth Menon 	switch (ddata->current_state) {
84*3df0b8b4SNishanth Menon 	case sb_booted:
85*3df0b8b4SNishanth Menon 		if (next_state == sb_init)
86*3df0b8b4SNishanth Menon 			goto ok_state;
87*3df0b8b4SNishanth Menon 		break;
88*3df0b8b4SNishanth Menon 
89*3df0b8b4SNishanth Menon 	case sb_init:
90*3df0b8b4SNishanth Menon 		if (next_state == sb_reset || next_state == sb_loaded)
91*3df0b8b4SNishanth Menon 			goto ok_state;
92*3df0b8b4SNishanth Menon 		break;
93*3df0b8b4SNishanth Menon 
94*3df0b8b4SNishanth Menon 	case sb_reset:
95*3df0b8b4SNishanth Menon 		if (next_state == sb_loaded || next_state == sb_init)
96*3df0b8b4SNishanth Menon 			goto ok_state;
97*3df0b8b4SNishanth Menon 		break;
98*3df0b8b4SNishanth Menon 
99*3df0b8b4SNishanth Menon 	case sb_loaded:
100*3df0b8b4SNishanth Menon 		if (next_state == sb_reset || next_state == sb_init ||
101*3df0b8b4SNishanth Menon 		    next_state == sb_running)
102*3df0b8b4SNishanth Menon 			goto ok_state;
103*3df0b8b4SNishanth Menon 		break;
104*3df0b8b4SNishanth Menon 
105*3df0b8b4SNishanth Menon 	case sb_running:
106*3df0b8b4SNishanth Menon 		if (next_state == sb_reset || next_state == sb_init)
107*3df0b8b4SNishanth Menon 			goto ok_state;
108*3df0b8b4SNishanth Menon 		break;
109*3df0b8b4SNishanth Menon 	};
110*3df0b8b4SNishanth Menon 	return -EINVAL;
111*3df0b8b4SNishanth Menon 
112*3df0b8b4SNishanth Menon ok_state:
113*3df0b8b4SNishanth Menon 	ddata->current_state = next_state;
114*3df0b8b4SNishanth Menon 	return 0;
115*3df0b8b4SNishanth Menon }
116*3df0b8b4SNishanth Menon 
117*3df0b8b4SNishanth Menon /**
118*3df0b8b4SNishanth Menon  * sandbox_testproc_probe() - basic probe function
119*3df0b8b4SNishanth Menon  * @dev:	test proc device that is being probed.
120*3df0b8b4SNishanth Menon  *
121*3df0b8b4SNishanth Menon  * Return: 0 if all went ok, else return appropriate error
122*3df0b8b4SNishanth Menon  */
sandbox_testproc_probe(struct udevice * dev)123*3df0b8b4SNishanth Menon static int sandbox_testproc_probe(struct udevice *dev)
124*3df0b8b4SNishanth Menon {
125*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
126*3df0b8b4SNishanth Menon 	struct sandbox_test_devdata *ddata;
127*3df0b8b4SNishanth Menon 	int ret;
128*3df0b8b4SNishanth Menon 
129*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
130*3df0b8b4SNishanth Menon 	ddata = dev_get_priv(dev);
131*3df0b8b4SNishanth Menon 	if (!ddata) {
132*3df0b8b4SNishanth Menon 		debug("%s: platform private data missing\n", uc_pdata->name);
133*3df0b8b4SNishanth Menon 		return -EINVAL;
134*3df0b8b4SNishanth Menon 	}
135*3df0b8b4SNishanth Menon 	ret = sandbox_dev_move_to_state(dev, sb_booted);
136*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
137*3df0b8b4SNishanth Menon 
138*3df0b8b4SNishanth Menon 	return ret;
139*3df0b8b4SNishanth Menon }
140*3df0b8b4SNishanth Menon 
141*3df0b8b4SNishanth Menon /**
142*3df0b8b4SNishanth Menon  * sandbox_testproc_init() - Simple initialization function
143*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
144*3df0b8b4SNishanth Menon  *
145*3df0b8b4SNishanth Menon  * Return: 0 if all went ok, else return appropriate error
146*3df0b8b4SNishanth Menon  */
sandbox_testproc_init(struct udevice * dev)147*3df0b8b4SNishanth Menon static int sandbox_testproc_init(struct udevice *dev)
148*3df0b8b4SNishanth Menon {
149*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
150*3df0b8b4SNishanth Menon 	int ret;
151*3df0b8b4SNishanth Menon 
152*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
153*3df0b8b4SNishanth Menon 
154*3df0b8b4SNishanth Menon 	ret = sandbox_dev_move_to_state(dev, sb_init);
155*3df0b8b4SNishanth Menon 
156*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
157*3df0b8b4SNishanth Menon 	if (ret)
158*3df0b8b4SNishanth Menon 		debug("%s init failed\n", uc_pdata->name);
159*3df0b8b4SNishanth Menon 
160*3df0b8b4SNishanth Menon 	return ret;
161*3df0b8b4SNishanth Menon }
162*3df0b8b4SNishanth Menon 
163*3df0b8b4SNishanth Menon /**
164*3df0b8b4SNishanth Menon  * sandbox_testproc_reset() - Reset the remote processor
165*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
166*3df0b8b4SNishanth Menon  *
167*3df0b8b4SNishanth Menon  * Return: 0 if all went ok, else return appropriate error
168*3df0b8b4SNishanth Menon  */
sandbox_testproc_reset(struct udevice * dev)169*3df0b8b4SNishanth Menon static int sandbox_testproc_reset(struct udevice *dev)
170*3df0b8b4SNishanth Menon {
171*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
172*3df0b8b4SNishanth Menon 	int ret;
173*3df0b8b4SNishanth Menon 
174*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
175*3df0b8b4SNishanth Menon 
176*3df0b8b4SNishanth Menon 	ret = sandbox_dev_move_to_state(dev, sb_reset);
177*3df0b8b4SNishanth Menon 
178*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
179*3df0b8b4SNishanth Menon 
180*3df0b8b4SNishanth Menon 	if (ret)
181*3df0b8b4SNishanth Menon 		debug("%s reset failed\n", uc_pdata->name);
182*3df0b8b4SNishanth Menon 	return ret;
183*3df0b8b4SNishanth Menon }
184*3df0b8b4SNishanth Menon 
185*3df0b8b4SNishanth Menon /**
186*3df0b8b4SNishanth Menon  * sandbox_testproc_load() - (replace: short desc)
187*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
188*3df0b8b4SNishanth Menon  * @addr:	Address of the binary image to load
189*3df0b8b4SNishanth Menon  * @size:	Size (in bytes) of the binary image to load
190*3df0b8b4SNishanth Menon  *
191*3df0b8b4SNishanth Menon  * Return: 0 if all went ok, else return appropriate error
192*3df0b8b4SNishanth Menon  */
sandbox_testproc_load(struct udevice * dev,ulong addr,ulong size)193*3df0b8b4SNishanth Menon static int sandbox_testproc_load(struct udevice *dev, ulong addr, ulong size)
194*3df0b8b4SNishanth Menon {
195*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
196*3df0b8b4SNishanth Menon 	int ret;
197*3df0b8b4SNishanth Menon 
198*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
199*3df0b8b4SNishanth Menon 
200*3df0b8b4SNishanth Menon 	ret = sandbox_dev_move_to_state(dev, sb_loaded);
201*3df0b8b4SNishanth Menon 
202*3df0b8b4SNishanth Menon 	debug("%s: called(%d) Loading to %08lX %lu size\n",
203*3df0b8b4SNishanth Menon 	      uc_pdata->name, ret, addr, size);
204*3df0b8b4SNishanth Menon 
205*3df0b8b4SNishanth Menon 	if (ret)
206*3df0b8b4SNishanth Menon 		debug("%s load failed\n", uc_pdata->name);
207*3df0b8b4SNishanth Menon 	return ret;
208*3df0b8b4SNishanth Menon }
209*3df0b8b4SNishanth Menon 
210*3df0b8b4SNishanth Menon /**
211*3df0b8b4SNishanth Menon  * sandbox_testproc_start() - Start the remote processor
212*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
213*3df0b8b4SNishanth Menon  *
214*3df0b8b4SNishanth Menon  * Return: 0 if all went ok, else return appropriate error
215*3df0b8b4SNishanth Menon  */
sandbox_testproc_start(struct udevice * dev)216*3df0b8b4SNishanth Menon static int sandbox_testproc_start(struct udevice *dev)
217*3df0b8b4SNishanth Menon {
218*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
219*3df0b8b4SNishanth Menon 	int ret;
220*3df0b8b4SNishanth Menon 
221*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
222*3df0b8b4SNishanth Menon 
223*3df0b8b4SNishanth Menon 	ret = sandbox_dev_move_to_state(dev, sb_running);
224*3df0b8b4SNishanth Menon 
225*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
226*3df0b8b4SNishanth Menon 
227*3df0b8b4SNishanth Menon 	if (ret)
228*3df0b8b4SNishanth Menon 		debug("%s start failed\n", uc_pdata->name);
229*3df0b8b4SNishanth Menon 	return ret;
230*3df0b8b4SNishanth Menon }
231*3df0b8b4SNishanth Menon 
232*3df0b8b4SNishanth Menon /**
233*3df0b8b4SNishanth Menon  * sandbox_testproc_stop() - Stop the remote processor
234*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
235*3df0b8b4SNishanth Menon  *
236*3df0b8b4SNishanth Menon  * Return: 0 if all went ok, else return appropriate error
237*3df0b8b4SNishanth Menon  */
sandbox_testproc_stop(struct udevice * dev)238*3df0b8b4SNishanth Menon static int sandbox_testproc_stop(struct udevice *dev)
239*3df0b8b4SNishanth Menon {
240*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
241*3df0b8b4SNishanth Menon 	int ret;
242*3df0b8b4SNishanth Menon 
243*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
244*3df0b8b4SNishanth Menon 
245*3df0b8b4SNishanth Menon 	ret = sandbox_dev_move_to_state(dev, sb_init);
246*3df0b8b4SNishanth Menon 
247*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
248*3df0b8b4SNishanth Menon 
249*3df0b8b4SNishanth Menon 	if (ret)
250*3df0b8b4SNishanth Menon 		debug("%s stop failed\n", uc_pdata->name);
251*3df0b8b4SNishanth Menon 	return ret;
252*3df0b8b4SNishanth Menon }
253*3df0b8b4SNishanth Menon 
254*3df0b8b4SNishanth Menon /**
255*3df0b8b4SNishanth Menon  * sandbox_testproc_is_running() - Check if remote processor is running
256*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
257*3df0b8b4SNishanth Menon  *
258*3df0b8b4SNishanth Menon  * Return: 0 if running, 1 if not running
259*3df0b8b4SNishanth Menon  */
sandbox_testproc_is_running(struct udevice * dev)260*3df0b8b4SNishanth Menon static int sandbox_testproc_is_running(struct udevice *dev)
261*3df0b8b4SNishanth Menon {
262*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
263*3df0b8b4SNishanth Menon 	struct sandbox_test_devdata *ddata;
264*3df0b8b4SNishanth Menon 	int ret = 1;
265*3df0b8b4SNishanth Menon 
266*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
267*3df0b8b4SNishanth Menon 	ddata = dev_get_priv(dev);
268*3df0b8b4SNishanth Menon 
269*3df0b8b4SNishanth Menon 	if (ddata->current_state == sb_running)
270*3df0b8b4SNishanth Menon 		ret = 0;
271*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
272*3df0b8b4SNishanth Menon 
273*3df0b8b4SNishanth Menon 	return ret;
274*3df0b8b4SNishanth Menon }
275*3df0b8b4SNishanth Menon 
276*3df0b8b4SNishanth Menon /**
277*3df0b8b4SNishanth Menon  * sandbox_testproc_ping() - Try pinging remote processor
278*3df0b8b4SNishanth Menon  * @dev:	device to operate upon
279*3df0b8b4SNishanth Menon  *
280*3df0b8b4SNishanth Menon  * Return: 0 if running, -EINVAL if not running
281*3df0b8b4SNishanth Menon  */
sandbox_testproc_ping(struct udevice * dev)282*3df0b8b4SNishanth Menon static int sandbox_testproc_ping(struct udevice *dev)
283*3df0b8b4SNishanth Menon {
284*3df0b8b4SNishanth Menon 	struct dm_rproc_uclass_pdata *uc_pdata;
285*3df0b8b4SNishanth Menon 	struct sandbox_test_devdata *ddata;
286*3df0b8b4SNishanth Menon 	int ret;
287*3df0b8b4SNishanth Menon 
288*3df0b8b4SNishanth Menon 	uc_pdata = dev_get_uclass_platdata(dev);
289*3df0b8b4SNishanth Menon 	ddata = dev_get_priv(dev);
290*3df0b8b4SNishanth Menon 
291*3df0b8b4SNishanth Menon 	if (ddata->current_state == sb_running)
292*3df0b8b4SNishanth Menon 		ret = 0;
293*3df0b8b4SNishanth Menon 	else
294*3df0b8b4SNishanth Menon 		ret = -EINVAL;
295*3df0b8b4SNishanth Menon 
296*3df0b8b4SNishanth Menon 	debug("%s: called(%d)\n", uc_pdata->name, ret);
297*3df0b8b4SNishanth Menon 	if (ret)
298*3df0b8b4SNishanth Menon 		debug("%s: No response.(Not started?)\n", uc_pdata->name);
299*3df0b8b4SNishanth Menon 
300*3df0b8b4SNishanth Menon 	return ret;
301*3df0b8b4SNishanth Menon }
302*3df0b8b4SNishanth Menon 
303*3df0b8b4SNishanth Menon static const struct dm_rproc_ops sandbox_testproc_ops = {
304*3df0b8b4SNishanth Menon 	.init = sandbox_testproc_init,
305*3df0b8b4SNishanth Menon 	.reset = sandbox_testproc_reset,
306*3df0b8b4SNishanth Menon 	.load = sandbox_testproc_load,
307*3df0b8b4SNishanth Menon 	.start = sandbox_testproc_start,
308*3df0b8b4SNishanth Menon 	.stop = sandbox_testproc_stop,
309*3df0b8b4SNishanth Menon 	.is_running = sandbox_testproc_is_running,
310*3df0b8b4SNishanth Menon 	.ping = sandbox_testproc_ping,
311*3df0b8b4SNishanth Menon };
312*3df0b8b4SNishanth Menon 
313*3df0b8b4SNishanth Menon static const struct udevice_id sandbox_ids[] = {
314*3df0b8b4SNishanth Menon 	{.compatible = "sandbox,test-processor"},
315*3df0b8b4SNishanth Menon 	{}
316*3df0b8b4SNishanth Menon };
317*3df0b8b4SNishanth Menon 
318*3df0b8b4SNishanth Menon U_BOOT_DRIVER(sandbox_testproc) = {
319*3df0b8b4SNishanth Menon 	.name = "sandbox_test_proc",
320*3df0b8b4SNishanth Menon 	.of_match = sandbox_ids,
321*3df0b8b4SNishanth Menon 	.id = UCLASS_REMOTEPROC,
322*3df0b8b4SNishanth Menon 	.ops = &sandbox_testproc_ops,
323*3df0b8b4SNishanth Menon 	.probe = sandbox_testproc_probe,
324*3df0b8b4SNishanth Menon 	.priv_auto_alloc_size = sizeof(struct sandbox_test_devdata),
325*3df0b8b4SNishanth Menon };
326*3df0b8b4SNishanth Menon 
327*3df0b8b4SNishanth Menon /* TODO(nm@ti.com): Remove this along with non-DT support */
328*3df0b8b4SNishanth Menon static struct dm_rproc_uclass_pdata proc_3_test = {
329*3df0b8b4SNishanth Menon 	.name = "proc_3_legacy",
330*3df0b8b4SNishanth Menon 	.mem_type = RPROC_INTERNAL_MEMORY_MAPPED,
331*3df0b8b4SNishanth Menon };
332*3df0b8b4SNishanth Menon 
333*3df0b8b4SNishanth Menon U_BOOT_DEVICE(proc_3_demo) = {
334*3df0b8b4SNishanth Menon 	.name = "sandbox_test_proc",
335*3df0b8b4SNishanth Menon 	.platdata = &proc_3_test,
336*3df0b8b4SNishanth Menon };
337