Skip to content

Commit d767af5

Browse files
mripardolofj
authored andcommitted
ARM: sunxi: Reintroduce the restart code for A10/A20 SoCs
This partly reverts commits 5536005 (ARM: sunxi: Remove reset code from the platform) and 5e669ec (ARM: sunxi: Remove init_machine callback) for the sun4i, sun5i and sun7i families. This is needed because the watchdog counterpart of these commits was dropped, and didn't make it into 3.16. In order to still be able to reboot the board, we need to reintroduce that code. Of course, the long term view is still to get rid of that code in mach-sunxi. Signed-off-by: Maxime Ripard <[email protected]> Signed-off-by: Olof Johansson <[email protected]>
1 parent 5acd78c commit d767af5

File tree

1 file changed

+77
-0
lines changed

1 file changed

+77
-0
lines changed

arch/arm/mach-sunxi/sunxi.c

Lines changed: 77 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,81 @@
1212

1313
#include <linux/clk-provider.h>
1414
#include <linux/clocksource.h>
15+
#include <linux/delay.h>
16+
#include <linux/kernel.h>
17+
#include <linux/init.h>
18+
#include <linux/of_address.h>
19+
#include <linux/of_irq.h>
20+
#include <linux/of_platform.h>
21+
#include <linux/io.h>
22+
#include <linux/reboot.h>
1523

1624
#include <asm/mach/arch.h>
25+
#include <asm/mach/map.h>
26+
#include <asm/system_misc.h>
27+
28+
#define SUN4I_WATCHDOG_CTRL_REG 0x00
29+
#define SUN4I_WATCHDOG_CTRL_RESTART BIT(0)
30+
#define SUN4I_WATCHDOG_MODE_REG 0x04
31+
#define SUN4I_WATCHDOG_MODE_ENABLE BIT(0)
32+
#define SUN4I_WATCHDOG_MODE_RESET_ENABLE BIT(1)
33+
34+
#define SUN6I_WATCHDOG1_IRQ_REG 0x00
35+
#define SUN6I_WATCHDOG1_CTRL_REG 0x10
36+
#define SUN6I_WATCHDOG1_CTRL_RESTART BIT(0)
37+
#define SUN6I_WATCHDOG1_CONFIG_REG 0x14
38+
#define SUN6I_WATCHDOG1_CONFIG_RESTART BIT(0)
39+
#define SUN6I_WATCHDOG1_CONFIG_IRQ BIT(1)
40+
#define SUN6I_WATCHDOG1_MODE_REG 0x18
41+
#define SUN6I_WATCHDOG1_MODE_ENABLE BIT(0)
42+
43+
static void __iomem *wdt_base;
44+
45+
static void sun4i_restart(enum reboot_mode mode, const char *cmd)
46+
{
47+
if (!wdt_base)
48+
return;
49+
50+
/* Enable timer and set reset bit in the watchdog */
51+
writel(SUN4I_WATCHDOG_MODE_ENABLE | SUN4I_WATCHDOG_MODE_RESET_ENABLE,
52+
wdt_base + SUN4I_WATCHDOG_MODE_REG);
53+
54+
/*
55+
* Restart the watchdog. The default (and lowest) interval
56+
* value for the watchdog is 0.5s.
57+
*/
58+
writel(SUN4I_WATCHDOG_CTRL_RESTART, wdt_base + SUN4I_WATCHDOG_CTRL_REG);
59+
60+
while (1) {
61+
mdelay(5);
62+
writel(SUN4I_WATCHDOG_MODE_ENABLE | SUN4I_WATCHDOG_MODE_RESET_ENABLE,
63+
wdt_base + SUN4I_WATCHDOG_MODE_REG);
64+
}
65+
}
66+
67+
static struct of_device_id sunxi_restart_ids[] = {
68+
{ .compatible = "allwinner,sun4i-a10-wdt" },
69+
{ /*sentinel*/ }
70+
};
71+
72+
static void sunxi_setup_restart(void)
73+
{
74+
struct device_node *np;
75+
76+
np = of_find_matching_node(NULL, sunxi_restart_ids);
77+
if (WARN(!np, "unable to setup watchdog restart"))
78+
return;
79+
80+
wdt_base = of_iomap(np, 0);
81+
WARN(!wdt_base, "failed to map watchdog base address");
82+
}
83+
84+
static void __init sunxi_dt_init(void)
85+
{
86+
sunxi_setup_restart();
87+
88+
of_platform_populate(NULL, of_default_bus_match_table, NULL, NULL);
89+
}
1790

1891
static const char * const sunxi_board_dt_compat[] = {
1992
"allwinner,sun4i-a10",
@@ -23,7 +96,9 @@ static const char * const sunxi_board_dt_compat[] = {
2396
};
2497

2598
DT_MACHINE_START(SUNXI_DT, "Allwinner A1X (Device Tree)")
99+
.init_machine = sunxi_dt_init,
26100
.dt_compat = sunxi_board_dt_compat,
101+
.restart = sun4i_restart,
27102
MACHINE_END
28103

29104
static const char * const sun6i_board_dt_compat[] = {
@@ -51,5 +126,7 @@ static const char * const sun7i_board_dt_compat[] = {
51126
};
52127

53128
DT_MACHINE_START(SUN7I_DT, "Allwinner sun7i (A20) Family")
129+
.init_machine = sunxi_dt_init,
54130
.dt_compat = sun7i_board_dt_compat,
131+
.restart = sun4i_restart,
55132
MACHINE_END

0 commit comments

Comments
 (0)