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