/* * Copyright (C) 2012 Andes Technology Corporation * Shawn Lin, Andes Technology Corporation * Macpaul Lin, Andes Technology Corporation * * SPDX-License-Identifier: GPL-2.0+ */ #include static inline unsigned long CACHE_LINE_SIZE(enum cache_t cache) { if (cache == ICACHE) return 8 << (((GET_ICM_CFG() & ICM_CFG_MSK_ISZ) \ >> ICM_CFG_OFF_ISZ) - 1); else return 8 << (((GET_DCM_CFG() & DCM_CFG_MSK_DSZ) \ >> DCM_CFG_OFF_DSZ) - 1); } void flush_dcache_range(unsigned long start, unsigned long end) { unsigned long line_size; line_size = CACHE_LINE_SIZE(DCACHE); while (end > start) { asm volatile ( "\n\tcctl %0, L1D_VA_WB" "\n\tcctl %0, L1D_VA_INVAL" : : "r" (start) ); start += line_size; } } void invalidate_icache_range(unsigned long start, unsigned long end) { unsigned long line_size; line_size = CACHE_LINE_SIZE(ICACHE); while (end > start) { asm volatile ( "\n\tcctl %0, L1I_VA_INVAL" : : "r"(start) ); start += line_size; } } void invalidate_dcache_range(unsigned long start, unsigned long end) { unsigned long line_size; line_size = CACHE_LINE_SIZE(DCACHE); while (end > start) { asm volatile ( "\n\tcctl %0, L1D_VA_INVAL" : : "r"(start) ); start += line_size; } } void flush_cache(unsigned long addr, unsigned long size) { flush_dcache_range(addr, addr + size); invalidate_icache_range(addr, addr + size); } void icache_enable(void) { asm volatile ( "mfsr $p0, $mr8\n\t" "ori $p0, $p0, 0x01\n\t" "mtsr $p0, $mr8\n\t" "isb\n\t" ); } void icache_disable(void) { asm volatile ( "mfsr $p0, $mr8\n\t" "li $p1, ~0x01\n\t" "and $p0, $p0, $p1\n\t" "mtsr $p0, $mr8\n\t" "isb\n\t" ); } int icache_status(void) { int ret; asm volatile ( "mfsr $p0, $mr8\n\t" "andi %0, $p0, 0x01\n\t" : "=r" (ret) : : "memory" ); return ret; } void dcache_enable(void) { asm volatile ( "mfsr $p0, $mr8\n\t" "ori $p0, $p0, 0x02\n\t" "mtsr $p0, $mr8\n\t" "isb\n\t" ); } void dcache_disable(void) { asm volatile ( "mfsr $p0, $mr8\n\t" "li $p1, ~0x02\n\t" "and $p0, $p0, $p1\n\t" "mtsr $p0, $mr8\n\t" "isb\n\t" ); } int dcache_status(void) { int ret; asm volatile ( "mfsr $p0, $mr8\n\t" "andi %0, $p0, 0x02\n\t" : "=r" (ret) : : "memory" ); return ret; }