xref: /optee_os/core/drivers/imx_wdog.c (revision d1ee5145343c1ad5fc32936aeb60346d76187242)
1*d1ee5145SPeng Fan /*
2*d1ee5145SPeng Fan  * Copyright 2017 NXP
3*d1ee5145SPeng Fan  *
4*d1ee5145SPeng Fan  * Peng Fan <peng.fan@nxp.com>
5*d1ee5145SPeng Fan  *
6*d1ee5145SPeng Fan  * Redistribution and use in source and binary forms, with or without
7*d1ee5145SPeng Fan  * modification, are permitted provided that the following conditions are met:
8*d1ee5145SPeng Fan  *
9*d1ee5145SPeng Fan  * 1. Redistributions of source code must retain the above copyright notice,
10*d1ee5145SPeng Fan  * this list of conditions and the following disclaimer.
11*d1ee5145SPeng Fan  *
12*d1ee5145SPeng Fan  * 2. Redistributions in binary form must reproduce the above copyright notice,
13*d1ee5145SPeng Fan  * this list of conditions and the following disclaimer in the documentation
14*d1ee5145SPeng Fan  * and/or other materials provided with the distribution.
15*d1ee5145SPeng Fan  *
16*d1ee5145SPeng Fan  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17*d1ee5145SPeng Fan  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*d1ee5145SPeng Fan  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*d1ee5145SPeng Fan  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
20*d1ee5145SPeng Fan  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21*d1ee5145SPeng Fan  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22*d1ee5145SPeng Fan  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23*d1ee5145SPeng Fan  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24*d1ee5145SPeng Fan  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25*d1ee5145SPeng Fan  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26*d1ee5145SPeng Fan  * POSSIBILITY OF SUCH DAMAGE.
27*d1ee5145SPeng Fan  */
28*d1ee5145SPeng Fan 
29*d1ee5145SPeng Fan #include <assert.h>
30*d1ee5145SPeng Fan #include <drivers/imx_wdog.h>
31*d1ee5145SPeng Fan #include <io.h>
32*d1ee5145SPeng Fan #include <keep.h>
33*d1ee5145SPeng Fan #include <kernel/dt.h>
34*d1ee5145SPeng Fan #include <kernel/generic_boot.h>
35*d1ee5145SPeng Fan #include <kernel/panic.h>
36*d1ee5145SPeng Fan #include <libfdt.h>
37*d1ee5145SPeng Fan #include <mm/core_mmu.h>
38*d1ee5145SPeng Fan #include <mm/core_memprot.h>
39*d1ee5145SPeng Fan #include <util.h>
40*d1ee5145SPeng Fan 
41*d1ee5145SPeng Fan static bool ext_reset;
42*d1ee5145SPeng Fan static vaddr_t wdog_base;
43*d1ee5145SPeng Fan 
44*d1ee5145SPeng Fan void imx_wdog_restart(void)
45*d1ee5145SPeng Fan {
46*d1ee5145SPeng Fan 	uint32_t val;
47*d1ee5145SPeng Fan 
48*d1ee5145SPeng Fan 	if (!wdog_base) {
49*d1ee5145SPeng Fan 		EMSG("No wdog mapped\n");
50*d1ee5145SPeng Fan 		panic();
51*d1ee5145SPeng Fan 	}
52*d1ee5145SPeng Fan 
53*d1ee5145SPeng Fan 	if (ext_reset)
54*d1ee5145SPeng Fan 		val = 0x14;
55*d1ee5145SPeng Fan 	else
56*d1ee5145SPeng Fan 		val = 0x24;
57*d1ee5145SPeng Fan 
58*d1ee5145SPeng Fan 	DMSG("val %x\n", val);
59*d1ee5145SPeng Fan 
60*d1ee5145SPeng Fan 	write16(val, wdog_base + WCR_OFF);
61*d1ee5145SPeng Fan 	dsb();
62*d1ee5145SPeng Fan 
63*d1ee5145SPeng Fan 	if (read32(wdog_base + WDT_WCR) & WDT_WCR_WDE) {
64*d1ee5145SPeng Fan 		write32(WDT_SEQ1, wdog_base + WDT_WSR);
65*d1ee5145SPeng Fan 		write32(WDT_SEQ2, wdog_base + WDT_WSR);
66*d1ee5145SPeng Fan 	}
67*d1ee5145SPeng Fan 
68*d1ee5145SPeng Fan 	write16(val, wdog_base + WCR_OFF);
69*d1ee5145SPeng Fan 	write16(val, wdog_base + WCR_OFF);
70*d1ee5145SPeng Fan 
71*d1ee5145SPeng Fan 	while (1)
72*d1ee5145SPeng Fan 		;
73*d1ee5145SPeng Fan }
74*d1ee5145SPeng Fan KEEP_PAGER(imx_wdog_restart);
75*d1ee5145SPeng Fan 
76*d1ee5145SPeng Fan static TEE_Result imx_wdog_init(void)
77*d1ee5145SPeng Fan {
78*d1ee5145SPeng Fan 	enum teecore_memtypes mtype;
79*d1ee5145SPeng Fan 	void *fdt;
80*d1ee5145SPeng Fan 	paddr_t pbase;
81*d1ee5145SPeng Fan 	vaddr_t vbase;
82*d1ee5145SPeng Fan 	ssize_t sz;
83*d1ee5145SPeng Fan 	int off;
84*d1ee5145SPeng Fan 	int st;
85*d1ee5145SPeng Fan 	uint32_t i;
86*d1ee5145SPeng Fan 
87*d1ee5145SPeng Fan #ifdef CFG_MX7
88*d1ee5145SPeng Fan 	static const char * const wdog_path[] = {
89*d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@30280000",
90*d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@30290000",
91*d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@302a0000",
92*d1ee5145SPeng Fan 		"/soc/aips-bus@30000000/wdog@302b0000",
93*d1ee5145SPeng Fan 	};
94*d1ee5145SPeng Fan #else
95*d1ee5145SPeng Fan 	static const char * const wdog_path[] = {
96*d1ee5145SPeng Fan 		"/soc/aips-bus@02000000/wdog@020bc000",
97*d1ee5145SPeng Fan 		"/soc/aips-bus@02000000/wdog@020c0000",
98*d1ee5145SPeng Fan 	};
99*d1ee5145SPeng Fan #endif
100*d1ee5145SPeng Fan 
101*d1ee5145SPeng Fan 	fdt = get_dt_blob();
102*d1ee5145SPeng Fan 	if (!fdt) {
103*d1ee5145SPeng Fan 		EMSG("No DTB\n");
104*d1ee5145SPeng Fan 		return TEE_ERROR_NOT_SUPPORTED;
105*d1ee5145SPeng Fan 	}
106*d1ee5145SPeng Fan 
107*d1ee5145SPeng Fan 	/* search the first usable wdog */
108*d1ee5145SPeng Fan 	for (i = 0; i < ARRAY_SIZE(wdog_path); i++) {
109*d1ee5145SPeng Fan 		off = fdt_path_offset(fdt, wdog_path[i]);
110*d1ee5145SPeng Fan 		if (off < 0)
111*d1ee5145SPeng Fan 			continue;
112*d1ee5145SPeng Fan 
113*d1ee5145SPeng Fan 		st = _fdt_get_status(fdt, off);
114*d1ee5145SPeng Fan 		if (st & DT_STATUS_OK_SEC)
115*d1ee5145SPeng Fan 			break;
116*d1ee5145SPeng Fan 	}
117*d1ee5145SPeng Fan 
118*d1ee5145SPeng Fan 	if (i == ARRAY_SIZE(wdog_path))
119*d1ee5145SPeng Fan 		return TEE_ERROR_ITEM_NOT_FOUND;
120*d1ee5145SPeng Fan 
121*d1ee5145SPeng Fan 	DMSG("path: %s\n", wdog_path[i]);
122*d1ee5145SPeng Fan 
123*d1ee5145SPeng Fan 	ext_reset = dt_have_prop(fdt, off, "fsl,ext-reset-output");
124*d1ee5145SPeng Fan 
125*d1ee5145SPeng Fan 	pbase = _fdt_reg_base_address(fdt, off);
126*d1ee5145SPeng Fan 	if (pbase == (paddr_t)-1)
127*d1ee5145SPeng Fan 		return TEE_ERROR_ITEM_NOT_FOUND;
128*d1ee5145SPeng Fan 
129*d1ee5145SPeng Fan 	sz = _fdt_reg_size(fdt, off);
130*d1ee5145SPeng Fan 	if (sz < 0)
131*d1ee5145SPeng Fan 		return TEE_ERROR_ITEM_NOT_FOUND;
132*d1ee5145SPeng Fan 
133*d1ee5145SPeng Fan 	if ((st & DT_STATUS_OK_SEC) && !(st & DT_STATUS_OK_NSEC))
134*d1ee5145SPeng Fan 		mtype = MEM_AREA_IO_SEC;
135*d1ee5145SPeng Fan 	else
136*d1ee5145SPeng Fan 		mtype = MEM_AREA_IO_NSEC;
137*d1ee5145SPeng Fan 
138*d1ee5145SPeng Fan 	/*
139*d1ee5145SPeng Fan 	 * Check to see whether it has been mapped using
140*d1ee5145SPeng Fan 	 * register_phys_mem or not.
141*d1ee5145SPeng Fan 	 */
142*d1ee5145SPeng Fan 	vbase = (vaddr_t)phys_to_virt(pbase, mtype);
143*d1ee5145SPeng Fan 	if (!vbase) {
144*d1ee5145SPeng Fan 		if (!core_mmu_add_mapping(mtype, pbase, sz)) {
145*d1ee5145SPeng Fan 			EMSG("Failed to map %zu bytes at PA 0x%"PRIxPA,
146*d1ee5145SPeng Fan 			     (size_t)sz, pbase);
147*d1ee5145SPeng Fan 			return TEE_ERROR_GENERIC;
148*d1ee5145SPeng Fan 		}
149*d1ee5145SPeng Fan 	}
150*d1ee5145SPeng Fan 
151*d1ee5145SPeng Fan 	vbase = (vaddr_t)phys_to_virt(pbase, mtype);
152*d1ee5145SPeng Fan 	if (!vbase) {
153*d1ee5145SPeng Fan 		EMSG("Failed to get VA for PA 0x%"PRIxPA, pbase);
154*d1ee5145SPeng Fan 		return TEE_ERROR_GENERIC;
155*d1ee5145SPeng Fan 	}
156*d1ee5145SPeng Fan 
157*d1ee5145SPeng Fan 	wdog_base = vbase;
158*d1ee5145SPeng Fan 
159*d1ee5145SPeng Fan 	return TEE_SUCCESS;
160*d1ee5145SPeng Fan }
161*d1ee5145SPeng Fan driver_init(imx_wdog_init);
162