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