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