13a0a0b24SClément Léger // SPDX-License-Identifier: BSD-2-Clause
23a0a0b24SClément Léger /*
33a0a0b24SClément Léger * Copyright (C) 2017 Timesys Corporation.
43a0a0b24SClément Léger * All rights reserved.
53a0a0b24SClément Léger *
63a0a0b24SClément Léger * Redistribution and use in source and binary forms, with or without
73a0a0b24SClément Léger * modification, are permitted provided that the following conditions are met:
83a0a0b24SClément Léger *
93a0a0b24SClément Léger * 1. Redistributions of source code must retain the above copyright notice,
103a0a0b24SClément Léger * this list of conditions and the following disclaimer.
113a0a0b24SClément Léger *
123a0a0b24SClément Léger * 2. Redistributions in binary form must reproduce the above copyright notice,
133a0a0b24SClément Léger * this list of conditions and the following disclaimer in the documentation
143a0a0b24SClément Léger * and/or other materials provided with the distribution.
153a0a0b24SClément Léger *
163a0a0b24SClément Léger * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
173a0a0b24SClément Léger * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
183a0a0b24SClément Léger * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
193a0a0b24SClément Léger * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
203a0a0b24SClément Léger * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
213a0a0b24SClément Léger * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
223a0a0b24SClément Léger * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
233a0a0b24SClément Léger * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
243a0a0b24SClément Léger * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
253a0a0b24SClément Léger * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
263a0a0b24SClément Léger * POSSIBILITY OF SUCH DAMAGE.
273a0a0b24SClément Léger */
283a0a0b24SClément Léger
293a0a0b24SClément Léger #include <io.h>
303a0a0b24SClément Léger #include <kernel/boot.h>
313a0a0b24SClément Léger #include <kernel/tz_ssvce_pl310.h>
323a0a0b24SClément Léger #include <mm/core_memprot.h>
333a0a0b24SClément Léger #include <mm/core_mmu.h>
343a0a0b24SClément Léger #include <sama5d2.h>
353a0a0b24SClément Léger #include <sam_sfr.h>
36*952dbec7STony Han #include <sam_pl310.h>
37*952dbec7STony Han #include <sm/optee_smc.h>
383a0a0b24SClément Léger #include <types_ext.h>
393a0a0b24SClément Léger
40*952dbec7STony Han /* L2 Cache Controller (L2CC) */
41*952dbec7STony Han #define L2CC_DCR_DWB BIT(1) /* Disable Write-back, Force Write-through */
42*952dbec7STony Han #define L2CC_DCR_DCL BIT(0) /* Disable Cache Linefill */
43*952dbec7STony Han
443a0a0b24SClément Léger register_phys_mem_pgdir(MEM_AREA_IO_SEC, PL310_BASE, CORE_MMU_PGDIR_SIZE);
453a0a0b24SClément Léger
pl310_base(void)463a0a0b24SClément Léger vaddr_t pl310_base(void)
473a0a0b24SClément Léger {
483a0a0b24SClément Léger static void *va;
493a0a0b24SClément Léger
503a0a0b24SClément Léger if (cpu_mmu_enabled()) {
513a0a0b24SClément Léger if (!va)
523a0a0b24SClément Léger va = phys_to_virt(PL310_BASE, MEM_AREA_IO_SEC, 1);
533a0a0b24SClément Léger return (vaddr_t)va;
543a0a0b24SClément Léger }
553a0a0b24SClément Léger return PL310_BASE;
563a0a0b24SClément Léger }
573a0a0b24SClément Léger
arm_cl2_config(vaddr_t pl310_base)583a0a0b24SClément Léger void arm_cl2_config(vaddr_t pl310_base)
593a0a0b24SClément Léger {
603a0a0b24SClément Léger io_write32(pl310_base + PL310_CTRL, 0);
61e914243dSClément Léger io_write32(sam_sfr_base() + AT91_SFR_L2CC_HRAMC, 0x1);
623a0a0b24SClément Léger io_write32(pl310_base + PL310_AUX_CTRL, PL310_AUX_CTRL_INIT);
633a0a0b24SClément Léger io_write32(pl310_base + PL310_PREFETCH_CTRL, PL310_PREFETCH_CTRL_INIT);
643a0a0b24SClément Léger io_write32(pl310_base + PL310_POWER_CTRL, PL310_POWER_CTRL_INIT);
653a0a0b24SClément Léger
663a0a0b24SClément Léger /* invalidate all cache ways */
673a0a0b24SClément Léger arm_cl2_invbyway(pl310_base);
683a0a0b24SClément Léger }
693a0a0b24SClément Léger
arm_cl2_enable(vaddr_t pl310_base)703a0a0b24SClément Léger void arm_cl2_enable(vaddr_t pl310_base)
713a0a0b24SClément Léger {
723a0a0b24SClément Léger /* Enable PL310 ctrl -> only set lsb bit */
733a0a0b24SClément Léger io_write32(pl310_base + PL310_CTRL, 1);
743a0a0b24SClément Léger }
75*952dbec7STony Han
76*952dbec7STony Han #ifdef CFG_PL310_SIP_PROTOCOL
pl310_enable(void)77*952dbec7STony Han TEE_Result pl310_enable(void)
78*952dbec7STony Han {
79*952dbec7STony Han vaddr_t base = pl310_base();
80*952dbec7STony Han
81*952dbec7STony Han arm_cl2_config(base);
82*952dbec7STony Han arm_cl2_enable(base);
83*952dbec7STony Han
84*952dbec7STony Han return OPTEE_SMC_RETURN_OK;
85*952dbec7STony Han }
86*952dbec7STony Han
pl310_disable(void)87*952dbec7STony Han TEE_Result pl310_disable(void)
88*952dbec7STony Han {
89*952dbec7STony Han EMSG("not implemented");
90*952dbec7STony Han
91*952dbec7STony Han return OPTEE_SMC_RETURN_ENOTAVAIL;
92*952dbec7STony Han }
93*952dbec7STony Han
pl310_enable_writeback(void)94*952dbec7STony Han TEE_Result pl310_enable_writeback(void)
95*952dbec7STony Han {
96*952dbec7STony Han vaddr_t base = pl310_base();
97*952dbec7STony Han
98*952dbec7STony Han io_write32(base + PL310_DEBUG_CTRL, 0);
99*952dbec7STony Han
100*952dbec7STony Han return OPTEE_SMC_RETURN_OK;
101*952dbec7STony Han }
102*952dbec7STony Han
pl310_disable_writeback(void)103*952dbec7STony Han TEE_Result pl310_disable_writeback(void)
104*952dbec7STony Han {
105*952dbec7STony Han uint32_t val = L2CC_DCR_DWB | L2CC_DCR_DCL;
106*952dbec7STony Han vaddr_t base = pl310_base();
107*952dbec7STony Han
108*952dbec7STony Han io_write32(base + PL310_DEBUG_CTRL, val);
109*952dbec7STony Han
110*952dbec7STony Han return OPTEE_SMC_RETURN_OK;
111*952dbec7STony Han }
112*952dbec7STony Han #endif
113