xref: /openbmc/u-boot/arch/arm/mach-socfpga/reset_manager_arria10.c (revision 719afeb0b3c60af82f701f122978b935aa6a5217)
1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0
2827e6a7eSLey Foon Tan /*
3827e6a7eSLey Foon Tan  * Copyright (C) 2016-2017 Intel Corporation
4827e6a7eSLey Foon Tan  */
5827e6a7eSLey Foon Tan 
6827e6a7eSLey Foon Tan #include <asm/io.h>
7827e6a7eSLey Foon Tan #include <asm/arch/fpga_manager.h>
8827e6a7eSLey Foon Tan #include <asm/arch/misc.h>
9827e6a7eSLey Foon Tan #include <asm/arch/reset_manager.h>
10827e6a7eSLey Foon Tan #include <asm/arch/system_manager.h>
11827e6a7eSLey Foon Tan #include <common.h>
12827e6a7eSLey Foon Tan #include <errno.h>
13827e6a7eSLey Foon Tan #include <fdtdec.h>
14827e6a7eSLey Foon Tan #include <wait_bit.h>
15827e6a7eSLey Foon Tan 
16827e6a7eSLey Foon Tan DECLARE_GLOBAL_DATA_PTR;
17827e6a7eSLey Foon Tan 
18827e6a7eSLey Foon Tan static const struct socfpga_reset_manager *reset_manager_base =
19827e6a7eSLey Foon Tan 		(void *)SOCFPGA_RSTMGR_ADDRESS;
20827e6a7eSLey Foon Tan static const struct socfpga_system_manager *sysmgr_regs =
21827e6a7eSLey Foon Tan 		(struct socfpga_system_manager *)SOCFPGA_SYSMGR_ADDRESS;
22827e6a7eSLey Foon Tan 
23827e6a7eSLey Foon Tan struct bridge_cfg {
24827e6a7eSLey Foon Tan 	int compat_id;
25827e6a7eSLey Foon Tan 	u32  mask_noc;
26827e6a7eSLey Foon Tan 	u32  mask_rstmgr;
27827e6a7eSLey Foon Tan };
28827e6a7eSLey Foon Tan 
29827e6a7eSLey Foon Tan static const struct bridge_cfg bridge_cfg_tbl[] = {
30827e6a7eSLey Foon Tan 	{
31827e6a7eSLey Foon Tan 		COMPAT_ALTERA_SOCFPGA_H2F_BRG,
32827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_H2F_SET_MSK,
33827e6a7eSLey Foon Tan 		ALT_RSTMGR_BRGMODRST_H2F_SET_MSK,
34827e6a7eSLey Foon Tan 	},
35827e6a7eSLey Foon Tan 	{
36827e6a7eSLey Foon Tan 		COMPAT_ALTERA_SOCFPGA_LWH2F_BRG,
37827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_LWH2F_SET_MSK,
38827e6a7eSLey Foon Tan 		ALT_RSTMGR_BRGMODRST_LWH2F_SET_MSK,
39827e6a7eSLey Foon Tan 	},
40827e6a7eSLey Foon Tan 	{
41827e6a7eSLey Foon Tan 		COMPAT_ALTERA_SOCFPGA_F2H_BRG,
42827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2H_SET_MSK,
43827e6a7eSLey Foon Tan 		ALT_RSTMGR_BRGMODRST_F2H_SET_MSK,
44827e6a7eSLey Foon Tan 	},
45827e6a7eSLey Foon Tan 	{
46827e6a7eSLey Foon Tan 		COMPAT_ALTERA_SOCFPGA_F2SDR0,
47827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2SDR0_SET_MSK,
48827e6a7eSLey Foon Tan 		ALT_RSTMGR_BRGMODRST_F2SSDRAM0_SET_MSK,
49827e6a7eSLey Foon Tan 	},
50827e6a7eSLey Foon Tan 	{
51827e6a7eSLey Foon Tan 		COMPAT_ALTERA_SOCFPGA_F2SDR1,
52827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2SDR1_SET_MSK,
53827e6a7eSLey Foon Tan 		ALT_RSTMGR_BRGMODRST_F2SSDRAM1_SET_MSK,
54827e6a7eSLey Foon Tan 	},
55827e6a7eSLey Foon Tan 	{
56827e6a7eSLey Foon Tan 		COMPAT_ALTERA_SOCFPGA_F2SDR2,
57827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2SDR2_SET_MSK,
58827e6a7eSLey Foon Tan 		ALT_RSTMGR_BRGMODRST_F2SSDRAM2_SET_MSK,
59827e6a7eSLey Foon Tan 	},
60827e6a7eSLey Foon Tan };
61827e6a7eSLey Foon Tan 
62827e6a7eSLey Foon Tan /* Disable the watchdog (toggle reset to watchdog) */
socfpga_watchdog_disable(void)63827e6a7eSLey Foon Tan void socfpga_watchdog_disable(void)
64827e6a7eSLey Foon Tan {
65827e6a7eSLey Foon Tan 	/* assert reset for watchdog */
66827e6a7eSLey Foon Tan 	setbits_le32(&reset_manager_base->per1modrst,
67827e6a7eSLey Foon Tan 		     ALT_RSTMGR_PER1MODRST_WD0_SET_MSK);
68827e6a7eSLey Foon Tan }
69827e6a7eSLey Foon Tan 
70827e6a7eSLey Foon Tan /* Release NOC ddr scheduler from reset */
socfpga_reset_deassert_noc_ddr_scheduler(void)71827e6a7eSLey Foon Tan void socfpga_reset_deassert_noc_ddr_scheduler(void)
72827e6a7eSLey Foon Tan {
73827e6a7eSLey Foon Tan 	clrbits_le32(&reset_manager_base->brgmodrst,
74827e6a7eSLey Foon Tan 		     ALT_RSTMGR_BRGMODRST_DDRSCH_SET_MSK);
75827e6a7eSLey Foon Tan }
76827e6a7eSLey Foon Tan 
get_bridge_init_val(const void * blob,int compat_id)77827e6a7eSLey Foon Tan static int get_bridge_init_val(const void *blob, int compat_id)
78827e6a7eSLey Foon Tan {
79827e6a7eSLey Foon Tan 	int node;
80827e6a7eSLey Foon Tan 
81827e6a7eSLey Foon Tan 	node = fdtdec_next_compatible(blob, 0, compat_id);
82827e6a7eSLey Foon Tan 	if (node < 0)
83827e6a7eSLey Foon Tan 		return 0;
84827e6a7eSLey Foon Tan 
85827e6a7eSLey Foon Tan 	return fdtdec_get_uint(blob, node, "init-val", 0);
86827e6a7eSLey Foon Tan }
87827e6a7eSLey Foon Tan 
88827e6a7eSLey Foon Tan /* Enable bridges (hps2fpga, lwhps2fpga, fpga2hps, fpga2sdram) per handoff */
socfpga_reset_deassert_bridges_handoff(void)89827e6a7eSLey Foon Tan int socfpga_reset_deassert_bridges_handoff(void)
90827e6a7eSLey Foon Tan {
91827e6a7eSLey Foon Tan 	u32 mask_noc = 0, mask_rstmgr = 0;
92827e6a7eSLey Foon Tan 	int i;
93827e6a7eSLey Foon Tan 
94827e6a7eSLey Foon Tan 	for (i = 0; i < ARRAY_SIZE(bridge_cfg_tbl); i++) {
95827e6a7eSLey Foon Tan 		if (get_bridge_init_val(gd->fdt_blob,
96827e6a7eSLey Foon Tan 					bridge_cfg_tbl[i].compat_id)) {
97827e6a7eSLey Foon Tan 			mask_noc |= bridge_cfg_tbl[i].mask_noc;
98827e6a7eSLey Foon Tan 			mask_rstmgr |= bridge_cfg_tbl[i].mask_rstmgr;
99827e6a7eSLey Foon Tan 		}
100827e6a7eSLey Foon Tan 	}
101827e6a7eSLey Foon Tan 
102827e6a7eSLey Foon Tan 	/* clear idle request to all bridges */
103827e6a7eSLey Foon Tan 	setbits_le32(&sysmgr_regs->noc_idlereq_clr, mask_noc);
104827e6a7eSLey Foon Tan 
105827e6a7eSLey Foon Tan 	/* Release bridges from reset state per handoff value */
106827e6a7eSLey Foon Tan 	clrbits_le32(&reset_manager_base->brgmodrst, mask_rstmgr);
107827e6a7eSLey Foon Tan 
108827e6a7eSLey Foon Tan 	/* Poll until all idleack to 0, timeout at 1000ms */
10948263504SÁlvaro Fernández Rojas 	return wait_for_bit_le32(&sysmgr_regs->noc_idleack, mask_noc,
110827e6a7eSLey Foon Tan 				 false, 1000, false);
111827e6a7eSLey Foon Tan }
112827e6a7eSLey Foon Tan 
113827e6a7eSLey Foon Tan /* Release L4 OSC1 Watchdog Timer 0 from reset through reset manager */
socfpga_reset_deassert_osc1wd0(void)114827e6a7eSLey Foon Tan void socfpga_reset_deassert_osc1wd0(void)
115827e6a7eSLey Foon Tan {
116827e6a7eSLey Foon Tan 	clrbits_le32(&reset_manager_base->per1modrst,
117827e6a7eSLey Foon Tan 		     ALT_RSTMGR_PER1MODRST_WD0_SET_MSK);
118827e6a7eSLey Foon Tan }
119827e6a7eSLey Foon Tan 
120827e6a7eSLey Foon Tan /*
121827e6a7eSLey Foon Tan  * Assert or de-assert SoCFPGA reset manager reset.
122827e6a7eSLey Foon Tan  */
socfpga_per_reset(u32 reset,int set)123827e6a7eSLey Foon Tan void socfpga_per_reset(u32 reset, int set)
124827e6a7eSLey Foon Tan {
125827e6a7eSLey Foon Tan 	const u32 *reg;
126827e6a7eSLey Foon Tan 	u32 rstmgr_bank = RSTMGR_BANK(reset);
127827e6a7eSLey Foon Tan 
128827e6a7eSLey Foon Tan 	switch (rstmgr_bank) {
129827e6a7eSLey Foon Tan 	case 0:
130827e6a7eSLey Foon Tan 		reg = &reset_manager_base->mpumodrst;
131827e6a7eSLey Foon Tan 		break;
132827e6a7eSLey Foon Tan 	case 1:
133827e6a7eSLey Foon Tan 		reg = &reset_manager_base->per0modrst;
134827e6a7eSLey Foon Tan 		break;
135827e6a7eSLey Foon Tan 	case 2:
136827e6a7eSLey Foon Tan 		reg = &reset_manager_base->per1modrst;
137827e6a7eSLey Foon Tan 		break;
138827e6a7eSLey Foon Tan 	case 3:
139827e6a7eSLey Foon Tan 		reg = &reset_manager_base->brgmodrst;
140827e6a7eSLey Foon Tan 		break;
141827e6a7eSLey Foon Tan 	case 4:
142827e6a7eSLey Foon Tan 		reg = &reset_manager_base->sysmodrst;
143827e6a7eSLey Foon Tan 		break;
144827e6a7eSLey Foon Tan 
145827e6a7eSLey Foon Tan 	default:
146827e6a7eSLey Foon Tan 		return;
147827e6a7eSLey Foon Tan 	}
148827e6a7eSLey Foon Tan 
149827e6a7eSLey Foon Tan 	if (set)
150827e6a7eSLey Foon Tan 		setbits_le32(reg, 1 << RSTMGR_RESET(reset));
151827e6a7eSLey Foon Tan 	else
152827e6a7eSLey Foon Tan 		clrbits_le32(reg, 1 << RSTMGR_RESET(reset));
153827e6a7eSLey Foon Tan }
154827e6a7eSLey Foon Tan 
155827e6a7eSLey Foon Tan /*
156827e6a7eSLey Foon Tan  * Assert reset on every peripheral but L4WD0.
157827e6a7eSLey Foon Tan  * Watchdog must be kept intact to prevent glitches
158827e6a7eSLey Foon Tan  * and/or hangs.
159827e6a7eSLey Foon Tan  * For the Arria10, we disable all the peripherals except L4 watchdog0,
160827e6a7eSLey Foon Tan  * L4 Timer 0, and ECC.
161827e6a7eSLey Foon Tan  */
socfpga_per_reset_all(void)162827e6a7eSLey Foon Tan void socfpga_per_reset_all(void)
163827e6a7eSLey Foon Tan {
164827e6a7eSLey Foon Tan 	const u32 l4wd0 = (1 << RSTMGR_RESET(SOCFPGA_RESET(L4WD0)) |
165827e6a7eSLey Foon Tan 			  (1 << RSTMGR_RESET(SOCFPGA_RESET(L4SYSTIMER0))));
166827e6a7eSLey Foon Tan 	unsigned mask_ecc_ocp =
167827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_EMACECC0_SET_MSK |
168827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_EMACECC1_SET_MSK |
169827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_EMACECC2_SET_MSK |
170827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_USBECC0_SET_MSK |
171827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_USBECC1_SET_MSK |
172827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_NANDECC_SET_MSK |
173827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_QSPIECC_SET_MSK |
174827e6a7eSLey Foon Tan 		ALT_RSTMGR_PER0MODRST_SDMMCECC_SET_MSK;
175827e6a7eSLey Foon Tan 
176827e6a7eSLey Foon Tan 	/* disable all components except ECC_OCP, L4 Timer0 and L4 WD0 */
177827e6a7eSLey Foon Tan 	writel(~l4wd0, &reset_manager_base->per1modrst);
178827e6a7eSLey Foon Tan 	setbits_le32(&reset_manager_base->per0modrst, ~mask_ecc_ocp);
179827e6a7eSLey Foon Tan 
180827e6a7eSLey Foon Tan 	/* Finally disable the ECC_OCP */
181827e6a7eSLey Foon Tan 	setbits_le32(&reset_manager_base->per0modrst, mask_ecc_ocp);
182827e6a7eSLey Foon Tan }
183827e6a7eSLey Foon Tan 
socfpga_bridges_reset(void)1846a34af5bSTien Fong Chee int socfpga_bridges_reset(void)
185827e6a7eSLey Foon Tan {
186827e6a7eSLey Foon Tan 	int ret;
187827e6a7eSLey Foon Tan 
188827e6a7eSLey Foon Tan 	/* Disable all the bridges (hps2fpga, lwhps2fpga, fpga2hps,
189827e6a7eSLey Foon Tan 	   fpga2sdram) */
190827e6a7eSLey Foon Tan 	/* set idle request to all bridges */
191827e6a7eSLey Foon Tan 	writel(ALT_SYSMGR_NOC_H2F_SET_MSK |
192827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_LWH2F_SET_MSK |
193827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2H_SET_MSK |
194827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2SDR0_SET_MSK |
195827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2SDR1_SET_MSK |
196827e6a7eSLey Foon Tan 		ALT_SYSMGR_NOC_F2SDR2_SET_MSK,
197827e6a7eSLey Foon Tan 		&sysmgr_regs->noc_idlereq_set);
198827e6a7eSLey Foon Tan 
199827e6a7eSLey Foon Tan 	/* Enable the NOC timeout */
200827e6a7eSLey Foon Tan 	writel(ALT_SYSMGR_NOC_TMO_EN_SET_MSK, &sysmgr_regs->noc_timeout);
201827e6a7eSLey Foon Tan 
202827e6a7eSLey Foon Tan 	/* Poll until all idleack to 1 */
20348263504SÁlvaro Fernández Rojas 	ret = wait_for_bit_le32(&sysmgr_regs->noc_idleack,
204827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_H2F_SET_MSK |
205827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_LWH2F_SET_MSK |
206827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2H_SET_MSK |
207827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2SDR0_SET_MSK |
208827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2SDR1_SET_MSK |
209827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2SDR2_SET_MSK,
210827e6a7eSLey Foon Tan 				true, 10000, false);
211827e6a7eSLey Foon Tan 	if (ret)
212827e6a7eSLey Foon Tan 		return ret;
213827e6a7eSLey Foon Tan 
214827e6a7eSLey Foon Tan 	/* Poll until all idlestatus to 1 */
21548263504SÁlvaro Fernández Rojas 	ret = wait_for_bit_le32(&sysmgr_regs->noc_idlestatus,
216827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_H2F_SET_MSK |
217827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_LWH2F_SET_MSK |
218827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2H_SET_MSK |
219827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2SDR0_SET_MSK |
220827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2SDR1_SET_MSK |
221827e6a7eSLey Foon Tan 				ALT_SYSMGR_NOC_F2SDR2_SET_MSK,
222827e6a7eSLey Foon Tan 				true, 10000, false);
223827e6a7eSLey Foon Tan 	if (ret)
224827e6a7eSLey Foon Tan 		return ret;
225827e6a7eSLey Foon Tan 
226827e6a7eSLey Foon Tan 	/* Put all bridges (except NOR DDR scheduler) into reset state */
227827e6a7eSLey Foon Tan 	setbits_le32(&reset_manager_base->brgmodrst,
228827e6a7eSLey Foon Tan 		     (ALT_RSTMGR_BRGMODRST_H2F_SET_MSK |
229827e6a7eSLey Foon Tan 		     ALT_RSTMGR_BRGMODRST_LWH2F_SET_MSK |
230827e6a7eSLey Foon Tan 		     ALT_RSTMGR_BRGMODRST_F2H_SET_MSK |
231827e6a7eSLey Foon Tan 		     ALT_RSTMGR_BRGMODRST_F2SSDRAM0_SET_MSK |
232827e6a7eSLey Foon Tan 		     ALT_RSTMGR_BRGMODRST_F2SSDRAM1_SET_MSK |
233827e6a7eSLey Foon Tan 		     ALT_RSTMGR_BRGMODRST_F2SSDRAM2_SET_MSK));
234827e6a7eSLey Foon Tan 
235827e6a7eSLey Foon Tan 	/* Disable NOC timeout */
236827e6a7eSLey Foon Tan 	writel(0, &sysmgr_regs->noc_timeout);
237827e6a7eSLey Foon Tan 
238827e6a7eSLey Foon Tan 	return 0;
239827e6a7eSLey Foon Tan }
240