1b25732c2SMax Filippov /*
2b25732c2SMax Filippov * Copyright (c) 2015 Google, Inc
3b25732c2SMax Filippov * Written by Simon Glass <sjg@chromium.org>
4b25732c2SMax Filippov *
5b25732c2SMax Filippov * SPDX-License-Identifier: GPL-2.0+
6b25732c2SMax Filippov */
7b25732c2SMax Filippov
8b25732c2SMax Filippov #include <common.h>
9b25732c2SMax Filippov #include <dm.h>
10b25732c2SMax Filippov #include <errno.h>
11b25732c2SMax Filippov #include <sysreset.h>
12b25732c2SMax Filippov #include <asm/state.h>
13b25732c2SMax Filippov #include <asm/test.h>
14b25732c2SMax Filippov
15b25732c2SMax Filippov DECLARE_GLOBAL_DATA_PTR;
16b25732c2SMax Filippov
sandbox_warm_sysreset_request(struct udevice * dev,enum sysreset_t type)17b25732c2SMax Filippov static int sandbox_warm_sysreset_request(struct udevice *dev,
18b25732c2SMax Filippov enum sysreset_t type)
19b25732c2SMax Filippov {
20b25732c2SMax Filippov struct sandbox_state *state = state_get_current();
21b25732c2SMax Filippov
22b25732c2SMax Filippov switch (type) {
23b25732c2SMax Filippov case SYSRESET_WARM:
24b25732c2SMax Filippov state->last_sysreset = type;
25b25732c2SMax Filippov break;
26b25732c2SMax Filippov default:
27b25732c2SMax Filippov return -ENOSYS;
28b25732c2SMax Filippov }
29b25732c2SMax Filippov if (!state->sysreset_allowed[type])
30b25732c2SMax Filippov return -EACCES;
31b25732c2SMax Filippov
32b25732c2SMax Filippov return -EINPROGRESS;
33b25732c2SMax Filippov }
34b25732c2SMax Filippov
sandbox_sysreset_request(struct udevice * dev,enum sysreset_t type)35b25732c2SMax Filippov static int sandbox_sysreset_request(struct udevice *dev, enum sysreset_t type)
36b25732c2SMax Filippov {
37b25732c2SMax Filippov struct sandbox_state *state = state_get_current();
38b25732c2SMax Filippov
39b25732c2SMax Filippov /*
40b25732c2SMax Filippov * If we have a device tree, the device we created from platform data
41b25732c2SMax Filippov * (see the U_BOOT_DEVICE() declaration below) should not do anything.
42b25732c2SMax Filippov * If we are that device, return an error.
43b25732c2SMax Filippov */
44*03753c9aSSimon Glass if (state->fdt_fname && !dev_of_valid(dev))
45b25732c2SMax Filippov return -ENODEV;
46b25732c2SMax Filippov
47b25732c2SMax Filippov switch (type) {
48b25732c2SMax Filippov case SYSRESET_COLD:
49b25732c2SMax Filippov state->last_sysreset = type;
50b25732c2SMax Filippov break;
51b25732c2SMax Filippov case SYSRESET_POWER:
52b25732c2SMax Filippov state->last_sysreset = type;
53b25732c2SMax Filippov if (!state->sysreset_allowed[type])
54b25732c2SMax Filippov return -EACCES;
55b25732c2SMax Filippov sandbox_exit();
56b25732c2SMax Filippov break;
57b25732c2SMax Filippov default:
58b25732c2SMax Filippov return -ENOSYS;
59b25732c2SMax Filippov }
60b25732c2SMax Filippov if (!state->sysreset_allowed[type])
61b25732c2SMax Filippov return -EACCES;
62b25732c2SMax Filippov
63b25732c2SMax Filippov return -EINPROGRESS;
64b25732c2SMax Filippov }
65b25732c2SMax Filippov
66b25732c2SMax Filippov static struct sysreset_ops sandbox_sysreset_ops = {
67b25732c2SMax Filippov .request = sandbox_sysreset_request,
68b25732c2SMax Filippov };
69b25732c2SMax Filippov
70b25732c2SMax Filippov static const struct udevice_id sandbox_sysreset_ids[] = {
71b25732c2SMax Filippov { .compatible = "sandbox,reset" },
72b25732c2SMax Filippov { }
73b25732c2SMax Filippov };
74b25732c2SMax Filippov
75b25732c2SMax Filippov U_BOOT_DRIVER(sysreset_sandbox) = {
76b25732c2SMax Filippov .name = "sysreset_sandbox",
77b25732c2SMax Filippov .id = UCLASS_SYSRESET,
78b25732c2SMax Filippov .of_match = sandbox_sysreset_ids,
79b25732c2SMax Filippov .ops = &sandbox_sysreset_ops,
80b25732c2SMax Filippov };
81b25732c2SMax Filippov
82b25732c2SMax Filippov static struct sysreset_ops sandbox_warm_sysreset_ops = {
83b25732c2SMax Filippov .request = sandbox_warm_sysreset_request,
84b25732c2SMax Filippov };
85b25732c2SMax Filippov
86b25732c2SMax Filippov static const struct udevice_id sandbox_warm_sysreset_ids[] = {
87b25732c2SMax Filippov { .compatible = "sandbox,warm-reset" },
88b25732c2SMax Filippov { }
89b25732c2SMax Filippov };
90b25732c2SMax Filippov
91b25732c2SMax Filippov U_BOOT_DRIVER(warm_sysreset_sandbox) = {
92b25732c2SMax Filippov .name = "warm_sysreset_sandbox",
93b25732c2SMax Filippov .id = UCLASS_SYSRESET,
94b25732c2SMax Filippov .of_match = sandbox_warm_sysreset_ids,
95b25732c2SMax Filippov .ops = &sandbox_warm_sysreset_ops,
96b25732c2SMax Filippov };
97b25732c2SMax Filippov
98b25732c2SMax Filippov /* This is here in case we don't have a device tree */
99b25732c2SMax Filippov U_BOOT_DEVICE(sysreset_sandbox_non_fdt) = {
100b25732c2SMax Filippov .name = "sysreset_sandbox",
101b25732c2SMax Filippov };
102