xref: /optee_os/core/drivers/imx_wdog.c (revision 922308b3a080bdd59000e4ed283bb53bb2d423ff)
11bb92983SJerome Forissier // SPDX-License-Identifier: BSD-2-Clause
2d1ee5145SPeng Fan /*
3*922308b3SPeng Fan  * Copyright 2017-2019 NXP
4d1ee5145SPeng Fan  *
5d1ee5145SPeng Fan  * Peng Fan <peng.fan@nxp.com>
6d1ee5145SPeng Fan  *
7d1ee5145SPeng Fan  * Redistribution and use in source and binary forms, with or without
8d1ee5145SPeng Fan  * modification, are permitted provided that the following conditions are met:
9d1ee5145SPeng Fan  *
10d1ee5145SPeng Fan  * 1. Redistributions of source code must retain the above copyright notice,
11d1ee5145SPeng Fan  * this list of conditions and the following disclaimer.
12d1ee5145SPeng Fan  *
13d1ee5145SPeng Fan  * 2. Redistributions in binary form must reproduce the above copyright notice,
14d1ee5145SPeng Fan  * this list of conditions and the following disclaimer in the documentation
15d1ee5145SPeng Fan  * and/or other materials provided with the distribution.
16d1ee5145SPeng Fan  *
17d1ee5145SPeng Fan  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18d1ee5145SPeng Fan  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19d1ee5145SPeng Fan  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20d1ee5145SPeng Fan  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21d1ee5145SPeng Fan  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22d1ee5145SPeng Fan  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23d1ee5145SPeng Fan  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24d1ee5145SPeng Fan  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25d1ee5145SPeng Fan  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26d1ee5145SPeng Fan  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27d1ee5145SPeng Fan  * POSSIBILITY OF SUCH DAMAGE.
28d1ee5145SPeng Fan  */
29d1ee5145SPeng Fan 
30d1ee5145SPeng Fan #include <assert.h>
31d1ee5145SPeng Fan #include <drivers/imx_wdog.h>
32d1ee5145SPeng Fan #include <io.h>
33d1ee5145SPeng Fan #include <keep.h>
34d1ee5145SPeng Fan #include <kernel/dt.h>
35d1ee5145SPeng Fan #include <kernel/generic_boot.h>
36d1ee5145SPeng Fan #include <kernel/panic.h>
37bcf6d6c9SJordan Rhee #ifdef CFG_DT
38d1ee5145SPeng Fan #include <libfdt.h>
39bcf6d6c9SJordan Rhee #endif
40d1ee5145SPeng Fan #include <mm/core_mmu.h>
41d1ee5145SPeng Fan #include <mm/core_memprot.h>
42d1ee5145SPeng Fan #include <util.h>
43d1ee5145SPeng Fan 
44d1ee5145SPeng Fan static bool ext_reset;
45d1ee5145SPeng Fan static vaddr_t wdog_base;
46d1ee5145SPeng Fan 
47d1ee5145SPeng Fan void imx_wdog_restart(void)
48d1ee5145SPeng Fan {
49d1ee5145SPeng Fan 	uint32_t val;
50d1ee5145SPeng Fan 
51d1ee5145SPeng Fan 	if (!wdog_base) {
52d1ee5145SPeng Fan 		EMSG("No wdog mapped\n");
53d1ee5145SPeng Fan 		panic();
54d1ee5145SPeng Fan 	}
55d1ee5145SPeng Fan 
56*922308b3SPeng Fan #ifdef CFG_MX7ULP
57*922308b3SPeng Fan 	val = io_read32(wdog_base + WDOG_CS);
58*922308b3SPeng Fan 
59*922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CNT, UNLOCK);
60*922308b3SPeng Fan 	/* Enable wdog */
61*922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CS, val | WDOG_CS_EN);
62*922308b3SPeng Fan 
63*922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CNT, UNLOCK);
64*922308b3SPeng Fan 	io_write32(wdog_base + WDOG_TOVAL, 1000);
65*922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CNT, REFRESH);
66*922308b3SPeng Fan #else
67d1ee5145SPeng Fan 	if (ext_reset)
68d1ee5145SPeng Fan 		val = 0x14;
69d1ee5145SPeng Fan 	else
70d1ee5145SPeng Fan 		val = 0x24;
71d1ee5145SPeng Fan 
72d1ee5145SPeng Fan 	DMSG("val %x\n", val);
73d1ee5145SPeng Fan 
74918bb3a5SEtienne Carriere 	io_write16(wdog_base + WDT_WCR, val);
75d1ee5145SPeng Fan 	dsb();
76d1ee5145SPeng Fan 
77918bb3a5SEtienne Carriere 	if (io_read16(wdog_base + WDT_WCR) & WDT_WCR_WDE) {
78918bb3a5SEtienne Carriere 		io_write16(wdog_base + WDT_WSR, WDT_SEQ1);
79918bb3a5SEtienne Carriere 		io_write16(wdog_base + WDT_WSR, WDT_SEQ2);
80d1ee5145SPeng Fan 	}
81d1ee5145SPeng Fan 
82918bb3a5SEtienne Carriere 	io_write16(wdog_base + WDT_WCR, val);
83918bb3a5SEtienne Carriere 	io_write16(wdog_base + WDT_WCR, val);
84*922308b3SPeng Fan #endif
85d1ee5145SPeng Fan 
86d1ee5145SPeng Fan 	while (1)
87d1ee5145SPeng Fan 		;
88d1ee5145SPeng Fan }
89d1ee5145SPeng Fan KEEP_PAGER(imx_wdog_restart);
90d1ee5145SPeng Fan 
91beae1b94SBryan O'Donoghue #if defined(CFG_DT) && !defined(CFG_EXTERNAL_DTB_OVERLAY)
92bcf6d6c9SJordan Rhee static TEE_Result imx_wdog_base(vaddr_t *wdog_vbase)
93d1ee5145SPeng Fan {
94d1ee5145SPeng Fan 	enum teecore_memtypes mtype;
95d1ee5145SPeng Fan 	void *fdt;
96d1ee5145SPeng Fan 	paddr_t pbase;
97d1ee5145SPeng Fan 	vaddr_t vbase;
98d1ee5145SPeng Fan 	ssize_t sz;
99d1ee5145SPeng Fan 	int off;
100d1ee5145SPeng Fan 	int st;
101d1ee5145SPeng Fan 	uint32_t i;
102d1ee5145SPeng Fan 
103d1ee5145SPeng Fan #ifdef CFG_MX7
104d1ee5145SPeng Fan 	static const char * const wdog_path[] = {
105d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@30280000",
106d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@30290000",
107d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@302a0000",
108d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@302b0000",
109d1ee5145SPeng Fan 	};
110*922308b3SPeng Fan #elif defined CFG_MX7ULP
111*922308b3SPeng Fan 	static const char * const wdog_path[] = {
112*922308b3SPeng Fan 		"/ahb-bridge0@40000000/wdog@403D0000",
113*922308b3SPeng Fan 		"/ahb-bridge0@40000000/wdog@40430000",
114*922308b3SPeng Fan 	};
115d1ee5145SPeng Fan #else
116d1ee5145SPeng Fan 	static const char * const wdog_path[] = {
117bed5dcffSPeng Fan 		"/soc/aips-bus@2000000/wdog@20bc000",
118bed5dcffSPeng Fan 		"/soc/aips-bus@2000000/wdog@20c0000",
119d1ee5145SPeng Fan 	};
120d1ee5145SPeng Fan #endif
121d1ee5145SPeng Fan 
1226179ebfaSEtienne Carriere 	fdt = get_dt();
123d1ee5145SPeng Fan 	if (!fdt) {
124d1ee5145SPeng Fan 		EMSG("No DTB\n");
125d1ee5145SPeng Fan 		return TEE_ERROR_NOT_SUPPORTED;
126d1ee5145SPeng Fan 	}
127d1ee5145SPeng Fan 
128d1ee5145SPeng Fan 	/* search the first usable wdog */
129d1ee5145SPeng Fan 	for (i = 0; i < ARRAY_SIZE(wdog_path); i++) {
130d1ee5145SPeng Fan 		off = fdt_path_offset(fdt, wdog_path[i]);
131d1ee5145SPeng Fan 		if (off < 0)
132d1ee5145SPeng Fan 			continue;
133d1ee5145SPeng Fan 
134d1ee5145SPeng Fan 		st = _fdt_get_status(fdt, off);
135d1ee5145SPeng Fan 		if (st & DT_STATUS_OK_SEC)
136d1ee5145SPeng Fan 			break;
137d1ee5145SPeng Fan 	}
138d1ee5145SPeng Fan 
139d1ee5145SPeng Fan 	if (i == ARRAY_SIZE(wdog_path))
140d1ee5145SPeng Fan 		return TEE_ERROR_ITEM_NOT_FOUND;
141d1ee5145SPeng Fan 
142d1ee5145SPeng Fan 	DMSG("path: %s\n", wdog_path[i]);
143d1ee5145SPeng Fan 
144d1ee5145SPeng Fan 	ext_reset = dt_have_prop(fdt, off, "fsl,ext-reset-output");
145d1ee5145SPeng Fan 
146d1ee5145SPeng Fan 	pbase = _fdt_reg_base_address(fdt, off);
147d1ee5145SPeng Fan 	if (pbase == (paddr_t)-1)
148d1ee5145SPeng Fan 		return TEE_ERROR_ITEM_NOT_FOUND;
149d1ee5145SPeng Fan 
150d1ee5145SPeng Fan 	sz = _fdt_reg_size(fdt, off);
151d1ee5145SPeng Fan 	if (sz < 0)
152d1ee5145SPeng Fan 		return TEE_ERROR_ITEM_NOT_FOUND;
153d1ee5145SPeng Fan 
154d1ee5145SPeng Fan 	if ((st & DT_STATUS_OK_SEC) && !(st & DT_STATUS_OK_NSEC))
155d1ee5145SPeng Fan 		mtype = MEM_AREA_IO_SEC;
156d1ee5145SPeng Fan 	else
157d1ee5145SPeng Fan 		mtype = MEM_AREA_IO_NSEC;
158d1ee5145SPeng Fan 
159d1ee5145SPeng Fan 	/*
160d1ee5145SPeng Fan 	 * Check to see whether it has been mapped using
161d1ee5145SPeng Fan 	 * register_phys_mem or not.
162d1ee5145SPeng Fan 	 */
163d1ee5145SPeng Fan 	vbase = (vaddr_t)phys_to_virt(pbase, mtype);
164d1ee5145SPeng Fan 	if (!vbase) {
165d1ee5145SPeng Fan 		if (!core_mmu_add_mapping(mtype, pbase, sz)) {
166d1ee5145SPeng Fan 			EMSG("Failed to map %zu bytes at PA 0x%"PRIxPA,
167d1ee5145SPeng Fan 			     (size_t)sz, pbase);
168d1ee5145SPeng Fan 			return TEE_ERROR_GENERIC;
169d1ee5145SPeng Fan 		}
170d1ee5145SPeng Fan 	}
171d1ee5145SPeng Fan 
172d1ee5145SPeng Fan 	vbase = (vaddr_t)phys_to_virt(pbase, mtype);
173d1ee5145SPeng Fan 	if (!vbase) {
174d1ee5145SPeng Fan 		EMSG("Failed to get VA for PA 0x%"PRIxPA, pbase);
175d1ee5145SPeng Fan 		return TEE_ERROR_GENERIC;
176d1ee5145SPeng Fan 	}
177d1ee5145SPeng Fan 
178bcf6d6c9SJordan Rhee 	*wdog_vbase = vbase;
179d1ee5145SPeng Fan 
180d1ee5145SPeng Fan 	return TEE_SUCCESS;
181d1ee5145SPeng Fan }
182bcf6d6c9SJordan Rhee #else
183a5e82dc7SJerome Forissier register_phys_mem_pgdir(MEM_AREA_IO_SEC, WDOG_BASE, CORE_MMU_PGDIR_SIZE);
184bcf6d6c9SJordan Rhee static TEE_Result imx_wdog_base(vaddr_t *wdog_vbase)
185bcf6d6c9SJordan Rhee {
186bcf6d6c9SJordan Rhee 	*wdog_vbase = (vaddr_t)phys_to_virt(WDOG_BASE, MEM_AREA_IO_SEC);
1877c1ee6aaSBryan O'Donoghue #if defined(CFG_IMX_WDOG_EXT_RESET)
1887c1ee6aaSBryan O'Donoghue 	ext_reset = true;
1897c1ee6aaSBryan O'Donoghue #endif
190bcf6d6c9SJordan Rhee 	return TEE_SUCCESS;
191bcf6d6c9SJordan Rhee }
192bcf6d6c9SJordan Rhee #endif
193bcf6d6c9SJordan Rhee 
194bcf6d6c9SJordan Rhee static TEE_Result imx_wdog_init(void)
195bcf6d6c9SJordan Rhee {
1964e10cbd5SJordan Rhee #if defined(PLATFORM_FLAVOR_mx7dsabresd) || \
1974e10cbd5SJordan Rhee 	defined(PLATFORM_FLAVOR_mx7dclsom)
1984e10cbd5SJordan Rhee 
199bcf6d6c9SJordan Rhee 	ext_reset = true;
200bcf6d6c9SJordan Rhee #endif
201bcf6d6c9SJordan Rhee 	return imx_wdog_base(&wdog_base);
202bcf6d6c9SJordan Rhee }
203d1ee5145SPeng Fan driver_init(imx_wdog_init);
204