xref: /optee_os/core/drivers/imx_wdog.c (revision c2e4eb43b7b7211345cd38ceceac97773bd78d2c)
11bb92983SJerome Forissier // SPDX-License-Identifier: BSD-2-Clause
2d1ee5145SPeng Fan /*
3922308b3SPeng 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>
3565401337SJens Wiklander #include <kernel/boot.h>
36d1ee5145SPeng Fan #include <kernel/panic.h>
37d1ee5145SPeng Fan #include <libfdt.h>
38d1ee5145SPeng Fan #include <mm/core_memprot.h>
39a2fc83d1SJerome Forissier #include <mm/core_mmu.h>
40d1ee5145SPeng Fan #include <util.h>
41d1ee5145SPeng Fan 
42f85678c1SIgor Opaniuk static bool ext_reset_output;
43d1ee5145SPeng Fan static vaddr_t wdog_base;
44d1ee5145SPeng Fan 
45b6ca39d5SRouven Czerwinski static const char * const dt_wdog_match_table[] = {
46b6ca39d5SRouven Czerwinski 	"fsl,imx21-wdt",
47b6ca39d5SRouven Czerwinski 	"fsl,imx7ulp-wdt",
48b6ca39d5SRouven Czerwinski };
49b6ca39d5SRouven Czerwinski 
50f85678c1SIgor Opaniuk void imx_wdog_restart(bool external_reset __maybe_unused)
51d1ee5145SPeng Fan {
52223f9e05SRouven Czerwinski 	uint32_t val = 0;
53d1ee5145SPeng Fan 
54d1ee5145SPeng Fan 	if (!wdog_base) {
55d1ee5145SPeng Fan 		EMSG("No wdog mapped\n");
56d1ee5145SPeng Fan 		panic();
57d1ee5145SPeng Fan 	}
58d1ee5145SPeng Fan 
59922308b3SPeng Fan #ifdef CFG_MX7ULP
60922308b3SPeng Fan 	val = io_read32(wdog_base + WDOG_CS);
61922308b3SPeng Fan 
62922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CNT, UNLOCK);
63922308b3SPeng Fan 	/* Enable wdog */
64922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CS, val | WDOG_CS_EN);
65922308b3SPeng Fan 
66922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CNT, UNLOCK);
67922308b3SPeng Fan 	io_write32(wdog_base + WDOG_TOVAL, 1000);
68922308b3SPeng Fan 	io_write32(wdog_base + WDOG_CNT, REFRESH);
69922308b3SPeng Fan #else
70f85678c1SIgor Opaniuk 	if (external_reset && ext_reset_output)
71d1ee5145SPeng Fan 		val = 0x14;
72d1ee5145SPeng Fan 	else
73d1ee5145SPeng Fan 		val = 0x24;
74d1ee5145SPeng Fan 
75d1ee5145SPeng Fan 	DMSG("val %x\n", val);
76d1ee5145SPeng Fan 
77918bb3a5SEtienne Carriere 	io_write16(wdog_base + WDT_WCR, val);
78d1ee5145SPeng Fan 	dsb();
79d1ee5145SPeng Fan 
80918bb3a5SEtienne Carriere 	if (io_read16(wdog_base + WDT_WCR) & WDT_WCR_WDE) {
81918bb3a5SEtienne Carriere 		io_write16(wdog_base + WDT_WSR, WDT_SEQ1);
82918bb3a5SEtienne Carriere 		io_write16(wdog_base + WDT_WSR, WDT_SEQ2);
83d1ee5145SPeng Fan 	}
84d1ee5145SPeng Fan 
85918bb3a5SEtienne Carriere 	io_write16(wdog_base + WDT_WCR, val);
86918bb3a5SEtienne Carriere 	io_write16(wdog_base + WDT_WCR, val);
87922308b3SPeng Fan #endif
88d1ee5145SPeng Fan 
89d1ee5145SPeng Fan 	while (1)
90d1ee5145SPeng Fan 		;
91d1ee5145SPeng Fan }
923639b55fSJerome Forissier DECLARE_KEEP_PAGER(imx_wdog_restart);
93d1ee5145SPeng Fan 
94beae1b94SBryan O'Donoghue #if defined(CFG_DT) && !defined(CFG_EXTERNAL_DTB_OVERLAY)
95bcf6d6c9SJordan Rhee static TEE_Result imx_wdog_base(vaddr_t *wdog_vbase)
96d1ee5145SPeng Fan {
97b6ca39d5SRouven Czerwinski 	const char *match = NULL;
98223f9e05SRouven Czerwinski 	void *fdt = NULL;
99223f9e05SRouven Czerwinski 	vaddr_t vbase = 0;
100b6ca39d5SRouven Czerwinski 	int found_off = 0;
101b6ca39d5SRouven Czerwinski 	size_t sz = 0;
102223f9e05SRouven Czerwinski 	int off = 0;
103223f9e05SRouven Czerwinski 	int st = 0;
104223f9e05SRouven Czerwinski 	uint32_t i = 0;
105d1ee5145SPeng Fan 
1066179ebfaSEtienne Carriere 	fdt = get_dt();
107d1ee5145SPeng Fan 	if (!fdt) {
108d1ee5145SPeng Fan 		EMSG("No DTB\n");
109d1ee5145SPeng Fan 		return TEE_ERROR_NOT_SUPPORTED;
110d1ee5145SPeng Fan 	}
111d1ee5145SPeng Fan 
112d1ee5145SPeng Fan 	/* search the first usable wdog */
113b6ca39d5SRouven Czerwinski 	for (i = 0; i < ARRAY_SIZE(dt_wdog_match_table); i++) {
114b6ca39d5SRouven Czerwinski 		match = dt_wdog_match_table[i];
115b6ca39d5SRouven Czerwinski 		off = 0;
116b6ca39d5SRouven Czerwinski 		while (off >= 0) {
117b6ca39d5SRouven Czerwinski 			off = fdt_node_offset_by_compatible(fdt, off, match);
118b6ca39d5SRouven Czerwinski 			if (off > 0) {
119d1ee5145SPeng Fan 				st = _fdt_get_status(fdt, off);
120b6ca39d5SRouven Czerwinski 				if (st & DT_STATUS_OK_SEC) {
121b6ca39d5SRouven Czerwinski 					DMSG("Wdog found at %u", off);
122b6ca39d5SRouven Czerwinski 					found_off = off;
123d1ee5145SPeng Fan 					break;
124d1ee5145SPeng Fan 				}
125b6ca39d5SRouven Czerwinski 			}
126b6ca39d5SRouven Czerwinski 		}
127b6ca39d5SRouven Czerwinski 		if (found_off)
128b6ca39d5SRouven Czerwinski 			break;
129d1ee5145SPeng Fan 		else
130b6ca39d5SRouven Czerwinski 			DMSG("%s not found in DTB", dt_wdog_match_table[i]);
131d1ee5145SPeng Fan 	}
132d1ee5145SPeng Fan 
133b6ca39d5SRouven Czerwinski 	if (!found_off) {
134b6ca39d5SRouven Czerwinski 		EMSG("No Watchdog found in DTB\n");
135b6ca39d5SRouven Czerwinski 		return TEE_ERROR_ITEM_NOT_FOUND;
136b6ca39d5SRouven Czerwinski 	}
137b6ca39d5SRouven Czerwinski 
138f85678c1SIgor Opaniuk 	ext_reset_output = dt_have_prop(fdt, found_off,
139f85678c1SIgor Opaniuk 					"fsl,ext-reset-output");
140b6ca39d5SRouven Czerwinski 
141b6ca39d5SRouven Czerwinski 	if (dt_map_dev(fdt, found_off, &vbase, &sz) < 0) {
142b6ca39d5SRouven Czerwinski 		EMSG("Failed to map Watchdog\n");
143b6ca39d5SRouven Czerwinski 		return TEE_ERROR_ITEM_NOT_FOUND;
144d1ee5145SPeng Fan 	}
145d1ee5145SPeng Fan 
146bcf6d6c9SJordan Rhee 	*wdog_vbase = vbase;
147d1ee5145SPeng Fan 
148d1ee5145SPeng Fan 	return TEE_SUCCESS;
149d1ee5145SPeng Fan }
150bcf6d6c9SJordan Rhee #else
151a5e82dc7SJerome Forissier register_phys_mem_pgdir(MEM_AREA_IO_SEC, WDOG_BASE, CORE_MMU_PGDIR_SIZE);
152bcf6d6c9SJordan Rhee static TEE_Result imx_wdog_base(vaddr_t *wdog_vbase)
153bcf6d6c9SJordan Rhee {
154*c2e4eb43SAnton Rybakov 	*wdog_vbase = (vaddr_t)phys_to_virt(WDOG_BASE, MEM_AREA_IO_SEC, 1);
1557c1ee6aaSBryan O'Donoghue #if defined(CFG_IMX_WDOG_EXT_RESET)
156f85678c1SIgor Opaniuk 	ext_reset_output = true;
1577c1ee6aaSBryan O'Donoghue #endif
158bcf6d6c9SJordan Rhee 	return TEE_SUCCESS;
159bcf6d6c9SJordan Rhee }
160bcf6d6c9SJordan Rhee #endif
161bcf6d6c9SJordan Rhee 
162bcf6d6c9SJordan Rhee static TEE_Result imx_wdog_init(void)
163bcf6d6c9SJordan Rhee {
1644e10cbd5SJordan Rhee #if defined(PLATFORM_FLAVOR_mx7dsabresd) || \
1654e10cbd5SJordan Rhee 	defined(PLATFORM_FLAVOR_mx7dclsom)
1664e10cbd5SJordan Rhee 
167f85678c1SIgor Opaniuk 	ext_reset_output = true;
168bcf6d6c9SJordan Rhee #endif
169bcf6d6c9SJordan Rhee 	return imx_wdog_base(&wdog_base);
170bcf6d6c9SJordan Rhee }
171d1ee5145SPeng Fan driver_init(imx_wdog_init);
172