blob: c12eda81d03e7d6b96c6104c2005b554b664ac7b [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Stephen Warren859f2562016-05-12 12:03:35 -06002/*
3 * Copyright (c) 2015 Google, Inc
4 * Written by Simon Glass <sjg@chromium.org>
Stephen Warren859f2562016-05-12 12:03:35 -06005 */
6
7#include <common.h>
8#include <dm.h>
9#include <errno.h>
10#include <sysreset.h>
11#include <asm/state.h>
12#include <asm/test.h>
13
Stephen Warren859f2562016-05-12 12:03:35 -060014static int sandbox_warm_sysreset_request(struct udevice *dev,
15 enum sysreset_t type)
16{
17 struct sandbox_state *state = state_get_current();
18
19 switch (type) {
20 case SYSRESET_WARM:
21 state->last_sysreset = type;
22 break;
23 default:
Paul Barkerbdb3a3f2023-11-08 08:51:10 +000024 return -EPROTONOSUPPORT;
Stephen Warren859f2562016-05-12 12:03:35 -060025 }
26 if (!state->sysreset_allowed[type])
27 return -EACCES;
28
29 return -EINPROGRESS;
30}
31
Mario Sixb6da1fb2018-08-06 10:23:33 +020032int sandbox_warm_sysreset_get_status(struct udevice *dev, char *buf, int size)
33{
34 strlcpy(buf, "Reset Status: WARM", size);
35
36 return 0;
37}
38
Simon Glass07fbba52018-10-01 12:22:46 -060039int sandbox_warm_sysreset_get_last(struct udevice *dev)
40{
41 return SYSRESET_WARM;
42}
43
Stephen Warren859f2562016-05-12 12:03:35 -060044static int sandbox_sysreset_request(struct udevice *dev, enum sysreset_t type)
45{
46 struct sandbox_state *state = state_get_current();
47
48 /*
49 * If we have a device tree, the device we created from platform data
Simon Glass1d8364a2020-12-28 20:34:54 -070050 * (see the U_BOOT_DRVINFO() declaration below) should not do anything.
Stephen Warren859f2562016-05-12 12:03:35 -060051 * If we are that device, return an error.
52 */
Simon Glassf1d50f72020-12-19 10:40:13 -070053 if (state->fdt_fname && !dev_has_ofnode(dev))
Stephen Warren859f2562016-05-12 12:03:35 -060054 return -ENODEV;
55
56 switch (type) {
57 case SYSRESET_COLD:
58 state->last_sysreset = type;
Heinrich Schuchardt1c678442020-10-27 20:29:25 +010059 if (!state->sysreset_allowed[type])
60 return -EACCES;
61 sandbox_reset();
Stephen Warren859f2562016-05-12 12:03:35 -060062 break;
Urja Rannikko25394e72019-05-16 21:48:41 +000063 case SYSRESET_POWER_OFF:
Stephen Warren859f2562016-05-12 12:03:35 -060064 state->last_sysreset = type;
65 if (!state->sysreset_allowed[type])
66 return -EACCES;
67 sandbox_exit();
Urja Rannikko25394e72019-05-16 21:48:41 +000068 case SYSRESET_POWER:
Simon Glass07fbba52018-10-01 12:22:46 -060069 if (!state->sysreset_allowed[type])
70 return -EACCES;
Simon Glass2fe5b912019-05-18 11:59:43 -060071 sandbox_exit();
Stephen Warren859f2562016-05-12 12:03:35 -060072 default:
Paul Barkerbdb3a3f2023-11-08 08:51:10 +000073 return -EPROTONOSUPPORT;
Stephen Warren859f2562016-05-12 12:03:35 -060074 }
75 if (!state->sysreset_allowed[type])
76 return -EACCES;
77
78 return -EINPROGRESS;
79}
80
Mario Sixb6da1fb2018-08-06 10:23:33 +020081int sandbox_sysreset_get_status(struct udevice *dev, char *buf, int size)
82{
83 strlcpy(buf, "Reset Status: COLD", size);
84
85 return 0;
86}
87
Simon Glass07fbba52018-10-01 12:22:46 -060088int sandbox_sysreset_get_last(struct udevice *dev)
89{
Simon Glass7e4d49c2018-11-23 21:29:28 -070090 struct sandbox_state *state = state_get_current();
91
92 /*
93 * The first phase is a power reset, after that we assume we don't
94 * know.
95 */
96 return state->jumped_fname ? SYSRESET_WARM : SYSRESET_POWER;
Simon Glass07fbba52018-10-01 12:22:46 -060097}
98
Stephen Warren859f2562016-05-12 12:03:35 -060099static struct sysreset_ops sandbox_sysreset_ops = {
100 .request = sandbox_sysreset_request,
Mario Sixb6da1fb2018-08-06 10:23:33 +0200101 .get_status = sandbox_sysreset_get_status,
Simon Glass07fbba52018-10-01 12:22:46 -0600102 .get_last = sandbox_sysreset_get_last,
Stephen Warren859f2562016-05-12 12:03:35 -0600103};
104
105static const struct udevice_id sandbox_sysreset_ids[] = {
106 { .compatible = "sandbox,reset" },
107 { }
108};
109
110U_BOOT_DRIVER(sysreset_sandbox) = {
111 .name = "sysreset_sandbox",
112 .id = UCLASS_SYSRESET,
113 .of_match = sandbox_sysreset_ids,
114 .ops = &sandbox_sysreset_ops,
115};
116
117static struct sysreset_ops sandbox_warm_sysreset_ops = {
118 .request = sandbox_warm_sysreset_request,
Mario Sixb6da1fb2018-08-06 10:23:33 +0200119 .get_status = sandbox_warm_sysreset_get_status,
Simon Glass07fbba52018-10-01 12:22:46 -0600120 .get_last = sandbox_warm_sysreset_get_last,
Stephen Warren859f2562016-05-12 12:03:35 -0600121};
122
123static const struct udevice_id sandbox_warm_sysreset_ids[] = {
124 { .compatible = "sandbox,warm-reset" },
125 { }
126};
127
128U_BOOT_DRIVER(warm_sysreset_sandbox) = {
129 .name = "warm_sysreset_sandbox",
130 .id = UCLASS_SYSRESET,
131 .of_match = sandbox_warm_sysreset_ids,
132 .ops = &sandbox_warm_sysreset_ops,
133};
134
Sean Anderson7bf2f8c2023-10-14 16:47:46 -0400135#if CONFIG_IS_ENABLED(OF_REAL) && !CONFIG_IS_ENABLED(OF_PLATDATA)
Stephen Warren859f2562016-05-12 12:03:35 -0600136/* This is here in case we don't have a device tree */
Simon Glass1d8364a2020-12-28 20:34:54 -0700137U_BOOT_DRVINFO(sysreset_sandbox_non_fdt) = {
Stephen Warren859f2562016-05-12 12:03:35 -0600138 .name = "sysreset_sandbox",
139};
Simon Glasscb90bd32020-10-03 11:31:23 -0600140#endif