summaryrefslogtreecommitdiff
path: root/drivers/watchdog/cortina_wdt.c
blob: 7ab9d7b2db97dd2ed060af1b14a6c805e44a2cbc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
// SPDX-License-Identifier: GPL-2.0+
/*
 * Copyright (C) 2020 Cortina-Access
 *
 */

#include <common.h>
#include <dm.h>
#include <hang.h>
#include <asm/io.h>
#include <wdt.h>
#include <linux/bitops.h>

#define CA_WDT_CTRL		0x00
#define CA_WDT_PS		0x04
#define CA_WDT_DIV		0x08
#define CA_WDT_LD		0x0C
#define CA_WDT_LOADE		0x10
#define CA_WDT_CNT		0x14
#define CA_WDT_IE		0x18
#define CA_WDT_INT		0x1C
#define CA_WDT_STAT		0x20

/* CA_WDT_CTRL */
#define CTL_WDT_EN		BIT(0)
#define CTL_WDT_RSTEN		BIT(1)
#define CTL_WDT_CLK_SEL		BIT(2)
/* CA_WDT_LOADE */
#define WDT_UPD			BIT(0)
#define WDT_UPD_PS		BIT(1)

/* Global config */
#define WDT_RESET_SUB		BIT(4)
#define WDT_RESET_ALL_BLOCK	BIT(6)
#define WDT_RESET_REMAP		BIT(7)
#define WDT_EXT_RESET		BIT(8)
#define WDT_RESET_DEFAULT	(WDT_EXT_RESET | WDT_RESET_REMAP | \
				 WDT_RESET_ALL_BLOCK | WDT_RESET_SUB)

struct ca_wdt_priv {
	void __iomem *base;
	void __iomem *global_config;
};

static void cortina_wdt_set_timeout(struct udevice *dev, u64 timeout_ms)
{
	struct ca_wdt_priv *priv = dev_get_priv(dev);

	/* Prescale using millisecond unit */
	writel(CORTINA_PER_IO_FREQ / 1000, priv->base + CA_WDT_PS);

	/* Millisecond */
	writel(1, priv->base + CA_WDT_DIV);

	writel(timeout_ms, priv->base + CA_WDT_LD);
	writel(WDT_UPD | WDT_UPD_PS, priv->base + CA_WDT_LOADE);
}

static int cortina_wdt_start(struct udevice *dev, u64 timeout, ulong flags)
{
	struct ca_wdt_priv *priv = dev_get_priv(dev);

	cortina_wdt_set_timeout(dev, timeout);

	/* WDT Reset option */
	setbits_32(priv->global_config, WDT_RESET_DEFAULT);

	/* Enable WDT */
	setbits_32(priv->base, CTL_WDT_EN | CTL_WDT_RSTEN | CTL_WDT_CLK_SEL);

	return 0;
}

static int cortina_wdt_stop(struct udevice *dev)
{
	struct ca_wdt_priv *priv = dev_get_priv(dev);

	/* Disable WDT */
	writel(0, priv->base);

	return 0;
}

static int cortina_wdt_reset(struct udevice *dev)
{
	struct ca_wdt_priv *priv = dev_get_priv(dev);

	/* Reload WDT counter */
	writel(WDT_UPD, priv->base + CA_WDT_LOADE);

	return 0;
}

static int cortina_wdt_expire_now(struct udevice *dev, ulong flags)
{
	/* Set 1ms timeout to reset system */
	cortina_wdt_set_timeout(dev, 1);
	hang();

	return 0;
}

static int cortina_wdt_probe(struct udevice *dev)
{
	struct ca_wdt_priv *priv = dev_get_priv(dev);

	priv->base = dev_remap_addr_index(dev, 0);
	if (!priv->base)
		return -ENOENT;

	priv->global_config = dev_remap_addr_index(dev, 1);
	if (!priv->global_config)
		return -ENOENT;

	/* Stop WDT */
	cortina_wdt_stop(dev);

	return 0;
}

static const struct wdt_ops cortina_wdt_ops = {
	.start = cortina_wdt_start,
	.reset = cortina_wdt_reset,
	.stop = cortina_wdt_stop,
	.expire_now = cortina_wdt_expire_now,
};

static const struct udevice_id cortina_wdt_ids[] = {
	{.compatible = "cortina,ca-wdt"},
	{}
};

U_BOOT_DRIVER(cortina_wdt) = {
	.name = "cortina_wdt",
	.id = UCLASS_WDT,
	.probe = cortina_wdt_probe,
	.of_match = cortina_wdt_ids,
	.ops = &cortina_wdt_ops,
};