blob: d0e586f66f6ddaa4a90eaae202d82d93f8b6efb5 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +02002/*
3 * Copyright (C) 2017 Álvaro Fernández Rojas <noltari@gmail.com>
4 *
5 * Derived from linux/drivers/power/reset/syscon-reboot.c:
6 * Copyright (C) 2013, Applied Micro Circuits Corporation
7 * Author: Feng Kan <fkan@apm.com>
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +02008 */
9
10#include <common.h>
11#include <dm.h>
12#include <errno.h>
13#include <regmap.h>
14#include <sysreset.h>
15#include <syscon.h>
16
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020017struct syscon_reboot_priv {
18 struct regmap *regmap;
19 unsigned int offset;
20 unsigned int mask;
21};
22
23static int syscon_reboot_request(struct udevice *dev, enum sysreset_t type)
24{
25 struct syscon_reboot_priv *priv = dev_get_priv(dev);
Patrick Delaunay48694212019-05-20 10:58:39 +020026 ulong driver_data = dev_get_driver_data(dev);
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020027
Patrick Delaunay48694212019-05-20 10:58:39 +020028 if (type != driver_data)
Patrick Delaunay50fd31f2019-04-18 17:16:21 +020029 return -EPROTONOSUPPORT;
30
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020031 regmap_write(priv->regmap, priv->offset, priv->mask);
32
33 return -EINPROGRESS;
34}
35
36static struct sysreset_ops syscon_reboot_ops = {
37 .request = syscon_reboot_request,
38};
39
40int syscon_reboot_probe(struct udevice *dev)
41{
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020042 struct syscon_reboot_priv *priv = dev_get_priv(dev);
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020043
Patrick Delaunay662a74a2019-03-07 09:57:14 +010044 priv->regmap = syscon_regmap_lookup_by_phandle(dev, "regmap");
45 if (IS_ERR(priv->regmap)) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090046 pr_err("unable to find regmap\n");
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020047 return -ENODEV;
48 }
49
Álvaro Fernández Rojas06069062018-03-17 12:22:24 +010050 priv->offset = dev_read_u32_default(dev, "offset", 0);
51 priv->mask = dev_read_u32_default(dev, "mask", 0);
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020052
53 return 0;
54}
55
56static const struct udevice_id syscon_reboot_ids[] = {
Patrick Delaunay48694212019-05-20 10:58:39 +020057 { .compatible = "syscon-reboot", .data = SYSRESET_COLD },
58 { .compatible = "syscon-poweroff", .data = SYSRESET_POWER_OFF },
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020059 { /* sentinel */ }
60};
61
62U_BOOT_DRIVER(syscon_reboot) = {
63 .name = "syscon_reboot",
64 .id = UCLASS_SYSRESET,
65 .of_match = syscon_reboot_ids,
66 .probe = syscon_reboot_probe,
67 .priv_auto_alloc_size = sizeof(struct syscon_reboot_priv),
68 .ops = &syscon_reboot_ops,
69};