blob: 49c061e08803e21619a36b8fea3eb76d25462198 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +02002/*
3 * Copyright (C) 2017 Álvaro Fernández Rojas <noltari@gmail.com>
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +02004 */
5
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +02006#include <dm.h>
Samuel Holland3a8713a2021-11-03 22:55:14 -05007#include <dm/device-internal.h>
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +02008#include <errno.h>
Samuel Holland3a8713a2021-11-03 22:55:14 -05009#include <malloc.h>
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020010#include <sysreset.h>
11#include <wdt.h>
Simon Glassbdd5f812023-09-14 18:21:46 -060012#include <linux/printk.h>
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020013
Samuel Holland90722ba2021-11-03 22:55:13 -050014struct wdt_reboot_plat {
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020015 struct udevice *wdt;
16};
17
18static int wdt_reboot_request(struct udevice *dev, enum sysreset_t type)
19{
Samuel Holland90722ba2021-11-03 22:55:13 -050020 struct wdt_reboot_plat *plat = dev_get_plat(dev);
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020021 int ret;
22
Heinrich Schuchardt9a41ed32021-11-04 10:31:17 +010023 switch (type) {
24 case SYSRESET_COLD:
25 case SYSRESET_WARM:
26 ret = wdt_expire_now(plat->wdt, 0);
27 if (ret)
28 return ret;
29 break;
30 default:
Paul Barkerbdb3a3f2023-11-08 08:51:10 +000031 return -EPROTONOSUPPORT;
Heinrich Schuchardt9a41ed32021-11-04 10:31:17 +010032 }
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020033
34 return -EINPROGRESS;
35}
36
37static struct sysreset_ops wdt_reboot_ops = {
38 .request = wdt_reboot_request,
39};
40
Samuel Holland90722ba2021-11-03 22:55:13 -050041static int wdt_reboot_of_to_plat(struct udevice *dev)
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020042{
Samuel Holland90722ba2021-11-03 22:55:13 -050043 struct wdt_reboot_plat *plat = dev_get_plat(dev);
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020044 int err;
45
46 err = uclass_get_device_by_phandle(UCLASS_WDT, dev,
Samuel Holland90722ba2021-11-03 22:55:13 -050047 "wdt", &plat->wdt);
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020048 if (err) {
Masahiro Yamada81e10422017-09-16 14:10:41 +090049 pr_err("unable to find wdt device\n");
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020050 return err;
51 }
52
53 return 0;
54}
55
56static const struct udevice_id wdt_reboot_ids[] = {
57 { .compatible = "wdt-reboot" },
58 { /* sentinel */ }
59};
60
61U_BOOT_DRIVER(wdt_reboot) = {
62 .name = "wdt_reboot",
63 .id = UCLASS_SYSRESET,
64 .of_match = wdt_reboot_ids,
Samuel Holland90722ba2021-11-03 22:55:13 -050065 .of_to_plat = wdt_reboot_of_to_plat,
66 .plat_auto = sizeof(struct wdt_reboot_plat),
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020067 .ops = &wdt_reboot_ops,
Álvaro Fernández Rojas0da081e2017-05-16 18:29:13 +020068};
Samuel Holland3a8713a2021-11-03 22:55:14 -050069
70#if IS_ENABLED(CONFIG_SYSRESET_WATCHDOG_AUTO)
71int sysreset_register_wdt(struct udevice *dev)
72{
73 struct wdt_reboot_plat *plat = malloc(sizeof(*plat));
74 int ret;
75
76 if (!plat)
77 return -ENOMEM;
78
79 plat->wdt = dev;
80
81 ret = device_bind(dev, DM_DRIVER_GET(wdt_reboot),
82 dev->name, plat, ofnode_null(), NULL);
83 if (ret) {
84 free(plat);
85 return ret;
86 }
87
88 return 0;
89}
90#endif