2018-05-06 17:58:06 -04:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
2015-06-23 15:39:13 -06:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Google, Inc
|
|
|
|
* Written by Simon Glass <sjg@chromium.org>
|
|
|
|
*/
|
|
|
|
|
2018-10-01 12:22:45 -06:00
|
|
|
#define LOG_CATEGORY UCLASS_SYSRESET
|
|
|
|
|
2015-06-23 15:39:13 -06:00
|
|
|
#include <common.h>
|
2020-05-10 11:40:03 -06:00
|
|
|
#include <command.h>
|
2019-12-28 10:45:01 -07:00
|
|
|
#include <cpu_func.h>
|
2015-06-23 15:39:13 -06:00
|
|
|
#include <dm.h>
|
|
|
|
#include <errno.h>
|
2020-12-23 08:11:14 -07:00
|
|
|
#include <hang.h>
|
|
|
|
#include <log.h>
|
2015-06-23 15:39:13 -06:00
|
|
|
#include <regmap.h>
|
2020-12-23 08:11:14 -07:00
|
|
|
#include <spl.h>
|
|
|
|
#include <sysreset.h>
|
2015-06-23 15:39:13 -06:00
|
|
|
#include <dm/device-internal.h>
|
|
|
|
#include <dm/lists.h>
|
|
|
|
#include <dm/root.h>
|
2020-05-10 11:40:11 -06:00
|
|
|
#include <linux/delay.h>
|
2015-06-23 15:39:13 -06:00
|
|
|
#include <linux/err.h>
|
2020-10-30 21:38:53 -06:00
|
|
|
#include <asm/global_data.h>
|
2015-06-23 15:39:13 -06:00
|
|
|
|
2016-05-12 12:03:35 -06:00
|
|
|
int sysreset_request(struct udevice *dev, enum sysreset_t type)
|
2015-06-23 15:39:13 -06:00
|
|
|
{
|
2016-05-12 12:03:35 -06:00
|
|
|
struct sysreset_ops *ops = sysreset_get_ops(dev);
|
2015-06-23 15:39:13 -06:00
|
|
|
|
|
|
|
if (!ops->request)
|
|
|
|
return -ENOSYS;
|
|
|
|
|
|
|
|
return ops->request(dev, type);
|
|
|
|
}
|
|
|
|
|
2018-08-06 10:23:32 +02:00
|
|
|
int sysreset_get_status(struct udevice *dev, char *buf, int size)
|
|
|
|
{
|
|
|
|
struct sysreset_ops *ops = sysreset_get_ops(dev);
|
|
|
|
|
|
|
|
if (!ops->get_status)
|
|
|
|
return -ENOSYS;
|
|
|
|
|
|
|
|
return ops->get_status(dev, buf, size);
|
|
|
|
}
|
|
|
|
|
2018-10-01 12:22:46 -06:00
|
|
|
int sysreset_get_last(struct udevice *dev)
|
|
|
|
{
|
|
|
|
struct sysreset_ops *ops = sysreset_get_ops(dev);
|
|
|
|
|
|
|
|
if (!ops->get_last)
|
|
|
|
return -ENOSYS;
|
|
|
|
|
|
|
|
return ops->get_last(dev);
|
|
|
|
}
|
|
|
|
|
2016-05-12 12:03:35 -06:00
|
|
|
int sysreset_walk(enum sysreset_t type)
|
2015-06-23 15:39:13 -06:00
|
|
|
{
|
|
|
|
struct udevice *dev;
|
2015-07-06 12:54:27 -06:00
|
|
|
int ret = -ENOSYS;
|
2015-06-23 15:39:13 -06:00
|
|
|
|
2016-05-12 12:03:35 -06:00
|
|
|
while (ret != -EINPROGRESS && type < SYSRESET_COUNT) {
|
|
|
|
for (uclass_first_device(UCLASS_SYSRESET, &dev);
|
2015-07-06 12:54:27 -06:00
|
|
|
dev;
|
|
|
|
uclass_next_device(&dev)) {
|
2016-05-12 12:03:35 -06:00
|
|
|
ret = sysreset_request(dev, type);
|
2015-06-23 15:39:13 -06:00
|
|
|
if (ret == -EINPROGRESS)
|
|
|
|
break;
|
|
|
|
}
|
2015-07-06 12:54:27 -06:00
|
|
|
type++;
|
2015-06-23 15:39:13 -06:00
|
|
|
}
|
|
|
|
|
2015-07-06 12:54:27 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-10-01 12:22:46 -06:00
|
|
|
int sysreset_get_last_walk(void)
|
|
|
|
{
|
|
|
|
struct udevice *dev;
|
|
|
|
int value = -ENOENT;
|
|
|
|
|
|
|
|
for (uclass_first_device(UCLASS_SYSRESET, &dev);
|
|
|
|
dev;
|
|
|
|
uclass_next_device(&dev)) {
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = sysreset_get_last(dev);
|
|
|
|
if (ret >= 0) {
|
|
|
|
value = ret;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2016-05-12 12:03:35 -06:00
|
|
|
void sysreset_walk_halt(enum sysreset_t type)
|
2015-07-06 12:54:27 -06:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2016-05-12 12:03:35 -06:00
|
|
|
ret = sysreset_walk(type);
|
2015-07-06 12:54:27 -06:00
|
|
|
|
2015-06-23 15:39:13 -06:00
|
|
|
/* Wait for the reset to take effect */
|
2015-07-06 12:54:27 -06:00
|
|
|
if (ret == -EINPROGRESS)
|
|
|
|
mdelay(100);
|
2015-06-23 15:39:13 -06:00
|
|
|
|
|
|
|
/* Still no reset? Give up */
|
2020-12-23 08:11:14 -07:00
|
|
|
if (spl_phase() <= PHASE_SPL)
|
|
|
|
log_err("no sysreset\n");
|
|
|
|
else
|
|
|
|
log_err("System reset not supported on this platform\n");
|
2015-06-23 15:39:13 -06:00
|
|
|
hang();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-05-12 12:03:35 -06:00
|
|
|
* reset_cpu() - calls sysreset_walk(SYSRESET_WARM)
|
2015-06-23 15:39:13 -06:00
|
|
|
*/
|
2020-12-15 16:47:52 +01:00
|
|
|
void reset_cpu(void)
|
2015-06-23 15:39:13 -06:00
|
|
|
{
|
2016-05-12 12:03:35 -06:00
|
|
|
sysreset_walk_halt(SYSRESET_WARM);
|
2015-07-06 12:54:27 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-02-25 17:22:52 +08:00
|
|
|
#if IS_ENABLED(CONFIG_SYSRESET_CMD_RESET)
|
2020-05-10 11:40:03 -06:00
|
|
|
int do_reset(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[])
|
2015-07-06 12:54:27 -06:00
|
|
|
{
|
2021-04-01 02:01:55 +03:00
|
|
|
enum sysreset_t reset_type = SYSRESET_COLD;
|
|
|
|
|
|
|
|
if (argc > 2)
|
|
|
|
return CMD_RET_USAGE;
|
|
|
|
|
|
|
|
if (argc == 2 && argv[1][0] == '-' && argv[1][1] == 'w') {
|
|
|
|
reset_type = SYSRESET_WARM;
|
|
|
|
}
|
|
|
|
|
2018-07-19 03:07:31 -07:00
|
|
|
printf("resetting ...\n");
|
2020-07-29 12:13:41 +02:00
|
|
|
mdelay(100);
|
2018-07-19 03:07:31 -07:00
|
|
|
|
2021-04-01 02:01:55 +03:00
|
|
|
sysreset_walk_halt(reset_type);
|
2015-07-06 12:54:27 -06:00
|
|
|
|
|
|
|
return 0;
|
2015-06-23 15:39:13 -06:00
|
|
|
}
|
2021-02-25 17:22:52 +08:00
|
|
|
#endif
|
2015-06-23 15:39:13 -06:00
|
|
|
|
2019-05-16 21:48:42 +00:00
|
|
|
#if IS_ENABLED(CONFIG_SYSRESET_CMD_POWEROFF)
|
2020-05-10 11:40:03 -06:00
|
|
|
int do_poweroff(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[])
|
2019-05-16 21:48:42 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
puts("poweroff ...\n");
|
|
|
|
mdelay(100);
|
|
|
|
|
|
|
|
ret = sysreset_walk(SYSRESET_POWER_OFF);
|
|
|
|
|
|
|
|
if (ret == -EINPROGRESS)
|
|
|
|
mdelay(1000);
|
|
|
|
|
|
|
|
/*NOTREACHED when power off*/
|
|
|
|
return CMD_RET_FAILURE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-07-12 10:36:07 +02:00
|
|
|
static int sysreset_post_bind(struct udevice *dev)
|
|
|
|
{
|
|
|
|
#if defined(CONFIG_NEEDS_MANUAL_RELOC)
|
|
|
|
struct sysreset_ops *ops = sysreset_get_ops(dev);
|
|
|
|
static int reloc_done;
|
|
|
|
|
|
|
|
if (!reloc_done) {
|
|
|
|
if (ops->request)
|
|
|
|
ops->request += gd->reloc_off;
|
|
|
|
reloc_done++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-12 12:03:35 -06:00
|
|
|
UCLASS_DRIVER(sysreset) = {
|
|
|
|
.id = UCLASS_SYSRESET,
|
|
|
|
.name = "sysreset",
|
2018-07-12 10:36:07 +02:00
|
|
|
.post_bind = sysreset_post_bind,
|
2015-06-23 15:39:13 -06:00
|
|
|
};
|