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