summaryrefslogtreecommitdiff
path: root/sys/dev/fdt/imxdog.c
blob: 73ca0200bb872ddae4181e07586604ce15bc806b (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
140
141
142
/* $OpenBSD: imxdog.c,v 1.3 2021/05/28 13:08:37 patrick Exp $ */
/*
 * Copyright (c) 2012-2013,2021 Patrick Wildt <patrick@blueri.se>
 *
 * Permission to use, copy, modify, and distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */

#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/timeout.h>

#include <machine/bus.h>
#include <machine/fdt.h>

#include <dev/ofw/openfirm.h>
#include <dev/ofw/fdt.h>

extern void (*cpuresetfn)(void);

/* registers */
#define WCR		0x00
#define  WCR_WDE		(1 << 2)
#define  WCR_WT_SEC(x)		(((x) * 2 - 1) << 8)
#define  WCR_WT_MASK		(0xff << 8)
#define WSR		0x02
#define WRSR		0x04
#define WICR		0x06
#define WMCR		0x08

#define WDOG_TIMEOUT_CALLBACK		60
#define WDOG_MAX_TIMEOUT_SEC		128

struct imxdog_softc {
	struct device		sc_dev;
	bus_space_tag_t		sc_iot;
	bus_space_handle_t	sc_ioh;
	struct timeout		sc_tmo;
};

struct imxdog_softc *imxdog_sc;

int	imxdog_match(struct device *, void *, void *);
void	imxdog_attach(struct device *, struct device *, void *);
void	imxdog_reset(void);
void	imxdog_timeout(void *);

struct cfattach	imxdog_ca = {
	sizeof (struct imxdog_softc), imxdog_match, imxdog_attach
};

struct cfdriver imxdog_cd = {
	NULL, "imxdog", DV_DULL
};

int
imxdog_match(struct device *parent, void *match, void *aux)
{
	struct fdt_attach_args *faa = aux;

	return OF_is_compatible(faa->fa_node, "fsl,imx21-wdt");
}

void
imxdog_attach(struct device *parent, struct device *self, void *aux)
{
	struct fdt_attach_args *faa = aux;
	struct imxdog_softc *sc = (struct imxdog_softc *) self;
	uint16_t reg;

	if (faa->fa_nreg < 1)
		return;

	sc->sc_iot = faa->fa_iot;
	if (bus_space_map(sc->sc_iot, faa->fa_reg[0].addr,
	    faa->fa_reg[0].size, 0, &sc->sc_ioh))
		panic("imxdog_attach: bus_space_map failed!");

	printf("\n");

	timeout_set(&sc->sc_tmo, imxdog_timeout, sc);

	/* Adjust timeout to maximum seconds */
	reg = bus_space_read_2(sc->sc_iot, sc->sc_ioh, WCR);
	reg &= ~WCR_WT_MASK;
	reg |= WCR_WT_SEC(WDOG_MAX_TIMEOUT_SEC);
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WCR, reg);

	/* Watchdog cannot be disabled, ping the watchdog if enabled */
	if (bus_space_read_2(sc->sc_iot, sc->sc_ioh, WCR) & WCR_WDE)
		imxdog_timeout(sc);

	imxdog_sc = sc;
	if (cpuresetfn == NULL)
		cpuresetfn = imxdog_reset;
}

void
imxdog_reset(void)
{
	struct imxdog_softc *sc = imxdog_sc;

	if (sc == NULL)
		return;

	/* disable watchdog and set timeout to 0 */
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WCR, 0);

	/* sequence to reset timeout counter */
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WSR, 0x5555);
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WSR, 0xaaaa);

	/* enable watchdog */
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WCR, 1);
	/* errata TKT039676 */
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WCR, 1);

	delay(100000);
}

void
imxdog_timeout(void *args)
{
	struct imxdog_softc *sc = args;

	/* Reload timeout counter */
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WSR, 0x5555);
	bus_space_write_2(sc->sc_iot, sc->sc_ioh, WSR, 0xaaaa);

	/* Schedule reload to trigger before counter runs out */
	timeout_add_sec(&sc->sc_tmo, WDOG_TIMEOUT_CALLBACK);
}