xref: /openbmc/u-boot/drivers/sysreset/sysreset_sandbox.c (revision 2a055ea53260ac8addeeb94eb671172844bc9106)
183d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
2b25732c2SMax Filippov /*
3b25732c2SMax Filippov  * Copyright (c) 2015 Google, Inc
4b25732c2SMax Filippov  * Written by Simon Glass <sjg@chromium.org>
5b25732c2SMax Filippov  */
6b25732c2SMax Filippov 
7b25732c2SMax Filippov #include <common.h>
8b25732c2SMax Filippov #include <dm.h>
9b25732c2SMax Filippov #include <errno.h>
10b25732c2SMax Filippov #include <sysreset.h>
11b25732c2SMax Filippov #include <asm/state.h>
12b25732c2SMax Filippov #include <asm/test.h>
13b25732c2SMax Filippov 
sandbox_warm_sysreset_request(struct udevice * dev,enum sysreset_t type)14b25732c2SMax Filippov static int sandbox_warm_sysreset_request(struct udevice *dev,
15b25732c2SMax Filippov 					 enum sysreset_t type)
16b25732c2SMax Filippov {
17b25732c2SMax Filippov 	struct sandbox_state *state = state_get_current();
18b25732c2SMax Filippov 
19b25732c2SMax Filippov 	switch (type) {
20b25732c2SMax Filippov 	case SYSRESET_WARM:
21b25732c2SMax Filippov 		state->last_sysreset = type;
22b25732c2SMax Filippov 		break;
23b25732c2SMax Filippov 	default:
24b25732c2SMax Filippov 		return -ENOSYS;
25b25732c2SMax Filippov 	}
26b25732c2SMax Filippov 	if (!state->sysreset_allowed[type])
27b25732c2SMax Filippov 		return -EACCES;
28b25732c2SMax Filippov 
29b25732c2SMax Filippov 	return -EINPROGRESS;
30b25732c2SMax Filippov }
31b25732c2SMax Filippov 
sandbox_warm_sysreset_get_status(struct udevice * dev,char * buf,int size)32cda4688cSMario Six int sandbox_warm_sysreset_get_status(struct udevice *dev, char *buf, int size)
33cda4688cSMario Six {
34cda4688cSMario Six 	strlcpy(buf, "Reset Status: WARM", size);
35cda4688cSMario Six 
36cda4688cSMario Six 	return 0;
37cda4688cSMario Six }
38cda4688cSMario Six 
sandbox_warm_sysreset_get_last(struct udevice * dev)39751fed42SSimon Glass int sandbox_warm_sysreset_get_last(struct udevice *dev)
40751fed42SSimon Glass {
41751fed42SSimon Glass 	return SYSRESET_WARM;
42751fed42SSimon Glass }
43751fed42SSimon Glass 
sandbox_sysreset_request(struct udevice * dev,enum sysreset_t type)44b25732c2SMax Filippov static int sandbox_sysreset_request(struct udevice *dev, enum sysreset_t type)
45b25732c2SMax Filippov {
46b25732c2SMax Filippov 	struct sandbox_state *state = state_get_current();
47b25732c2SMax Filippov 
48b25732c2SMax Filippov 	/*
49b25732c2SMax Filippov 	 * If we have a device tree, the device we created from platform data
50b25732c2SMax Filippov 	 * (see the U_BOOT_DEVICE() declaration below) should not do anything.
51b25732c2SMax Filippov 	 * If we are that device, return an error.
52b25732c2SMax Filippov 	 */
5303753c9aSSimon Glass 	if (state->fdt_fname && !dev_of_valid(dev))
54b25732c2SMax Filippov 		return -ENODEV;
55b25732c2SMax Filippov 
56b25732c2SMax Filippov 	switch (type) {
57b25732c2SMax Filippov 	case SYSRESET_COLD:
58b25732c2SMax Filippov 		state->last_sysreset = type;
59b25732c2SMax Filippov 		break;
60b25732c2SMax Filippov 	case SYSRESET_POWER:
61b25732c2SMax Filippov 		state->last_sysreset = type;
62b25732c2SMax Filippov 		if (!state->sysreset_allowed[type])
63b25732c2SMax Filippov 			return -EACCES;
64b25732c2SMax Filippov 		sandbox_exit();
65b25732c2SMax Filippov 		break;
66751fed42SSimon Glass 	case SYSRESET_POWER_OFF:
67751fed42SSimon Glass 		if (!state->sysreset_allowed[type])
68751fed42SSimon Glass 			return -EACCES;
69b25732c2SMax Filippov 	default:
70b25732c2SMax Filippov 		return -ENOSYS;
71b25732c2SMax Filippov 	}
72b25732c2SMax Filippov 	if (!state->sysreset_allowed[type])
73b25732c2SMax Filippov 		return -EACCES;
74b25732c2SMax Filippov 
75b25732c2SMax Filippov 	return -EINPROGRESS;
76b25732c2SMax Filippov }
77b25732c2SMax Filippov 
sandbox_sysreset_get_status(struct udevice * dev,char * buf,int size)78cda4688cSMario Six int sandbox_sysreset_get_status(struct udevice *dev, char *buf, int size)
79cda4688cSMario Six {
80cda4688cSMario Six 	strlcpy(buf, "Reset Status: COLD", size);
81cda4688cSMario Six 
82cda4688cSMario Six 	return 0;
83cda4688cSMario Six }
84cda4688cSMario Six 
sandbox_sysreset_get_last(struct udevice * dev)85751fed42SSimon Glass int sandbox_sysreset_get_last(struct udevice *dev)
86751fed42SSimon Glass {
87*2a072690SSimon Glass 	struct sandbox_state *state = state_get_current();
88*2a072690SSimon Glass 
89*2a072690SSimon Glass 	/*
90*2a072690SSimon Glass 	 * The first phase is a power reset, after that we assume we don't
91*2a072690SSimon Glass 	 * know.
92*2a072690SSimon Glass 	 */
93*2a072690SSimon Glass 	return state->jumped_fname ? SYSRESET_WARM : SYSRESET_POWER;
94751fed42SSimon Glass }
95751fed42SSimon Glass 
96b25732c2SMax Filippov static struct sysreset_ops sandbox_sysreset_ops = {
97b25732c2SMax Filippov 	.request	= sandbox_sysreset_request,
98cda4688cSMario Six 	.get_status	= sandbox_sysreset_get_status,
99751fed42SSimon Glass 	.get_last	= sandbox_sysreset_get_last,
100b25732c2SMax Filippov };
101b25732c2SMax Filippov 
102b25732c2SMax Filippov static const struct udevice_id sandbox_sysreset_ids[] = {
103b25732c2SMax Filippov 	{ .compatible = "sandbox,reset" },
104b25732c2SMax Filippov 	{ }
105b25732c2SMax Filippov };
106b25732c2SMax Filippov 
107b25732c2SMax Filippov U_BOOT_DRIVER(sysreset_sandbox) = {
108b25732c2SMax Filippov 	.name		= "sysreset_sandbox",
109b25732c2SMax Filippov 	.id		= UCLASS_SYSRESET,
110b25732c2SMax Filippov 	.of_match	= sandbox_sysreset_ids,
111b25732c2SMax Filippov 	.ops		= &sandbox_sysreset_ops,
112b25732c2SMax Filippov };
113b25732c2SMax Filippov 
114b25732c2SMax Filippov static struct sysreset_ops sandbox_warm_sysreset_ops = {
115b25732c2SMax Filippov 	.request	= sandbox_warm_sysreset_request,
116cda4688cSMario Six 	.get_status	= sandbox_warm_sysreset_get_status,
117751fed42SSimon Glass 	.get_last	= sandbox_warm_sysreset_get_last,
118b25732c2SMax Filippov };
119b25732c2SMax Filippov 
120b25732c2SMax Filippov static const struct udevice_id sandbox_warm_sysreset_ids[] = {
121b25732c2SMax Filippov 	{ .compatible = "sandbox,warm-reset" },
122b25732c2SMax Filippov 	{ }
123b25732c2SMax Filippov };
124b25732c2SMax Filippov 
125b25732c2SMax Filippov U_BOOT_DRIVER(warm_sysreset_sandbox) = {
126b25732c2SMax Filippov 	.name		= "warm_sysreset_sandbox",
127b25732c2SMax Filippov 	.id		= UCLASS_SYSRESET,
128b25732c2SMax Filippov 	.of_match	= sandbox_warm_sysreset_ids,
129b25732c2SMax Filippov 	.ops		= &sandbox_warm_sysreset_ops,
130b25732c2SMax Filippov };
131b25732c2SMax Filippov 
132b25732c2SMax Filippov /* This is here in case we don't have a device tree */
133b25732c2SMax Filippov U_BOOT_DEVICE(sysreset_sandbox_non_fdt) = {
134b25732c2SMax Filippov 	.name = "sysreset_sandbox",
135b25732c2SMax Filippov };
136