blob: f64701aab3c834ad967a30f04b1ef3ca0a217e81 [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>
Simon Glass61b29b82020-02-03 07:36:15 -070016#include <linux/err.h>
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020017
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020018struct syscon_reboot_priv {
19 struct regmap *regmap;
20 unsigned int offset;
21 unsigned int mask;
22};
23
24static int syscon_reboot_request(struct udevice *dev, enum sysreset_t type)
25{
26 struct syscon_reboot_priv *priv = dev_get_priv(dev);
Patrick Delaunay48694212019-05-20 10:58:39 +020027 ulong driver_data = dev_get_driver_data(dev);
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020028
Patrick Delaunay48694212019-05-20 10:58:39 +020029 if (type != driver_data)
Patrick Delaunay50fd31f2019-04-18 17:16:21 +020030 return -EPROTONOSUPPORT;
31
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020032 regmap_write(priv->regmap, priv->offset, priv->mask);
33
34 return -EINPROGRESS;
35}
36
37static struct sysreset_ops syscon_reboot_ops = {
38 .request = syscon_reboot_request,
39};
40
41int syscon_reboot_probe(struct udevice *dev)
42{
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020043 struct syscon_reboot_priv *priv = dev_get_priv(dev);
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020044
Patrick Delaunay662a74a2019-03-07 09:57:14 +010045 priv->regmap = syscon_regmap_lookup_by_phandle(dev, "regmap");
46 if (IS_ERR(priv->regmap)) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090047 pr_err("unable to find regmap\n");
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020048 return -ENODEV;
49 }
50
Álvaro Fernández Rojas06069062018-03-17 12:22:24 +010051 priv->offset = dev_read_u32_default(dev, "offset", 0);
52 priv->mask = dev_read_u32_default(dev, "mask", 0);
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020053
54 return 0;
55}
56
57static const struct udevice_id syscon_reboot_ids[] = {
Patrick Delaunay48694212019-05-20 10:58:39 +020058 { .compatible = "syscon-reboot", .data = SYSRESET_COLD },
59 { .compatible = "syscon-poweroff", .data = SYSRESET_POWER_OFF },
Álvaro Fernández Rojase3889692017-04-25 00:39:14 +020060 { /* sentinel */ }
61};
62
63U_BOOT_DRIVER(syscon_reboot) = {
64 .name = "syscon_reboot",
65 .id = UCLASS_SYSRESET,
66 .of_match = syscon_reboot_ids,
67 .probe = syscon_reboot_probe,
68 .priv_auto_alloc_size = sizeof(struct syscon_reboot_priv),
69 .ops = &syscon_reboot_ops,
70};