sdk-hwV1.3/lichee/melis-v3.0/source/ekernel/arch/riscv/rv32/e907/cache.c

197 lines
5.8 KiB
C
Raw Normal View History

2024-05-07 10:09:20 +00:00
/*
* Copyright (c) 2019-2025 Allwinner Technology Co., Ltd. ALL rights reserved.
*
* Allwinner is a trademark of Allwinner Technology Co.,Ltd., registered in
* the the People's Republic of China and other countries.
* All Allwinner Technology Co.,Ltd. trademarks are used with permission.
*
* DISCLAIMER
* THIRD PARTY LICENCES MAY BE REQUIRED TO IMPLEMENT THE SOLUTION/PRODUCT.
* IF YOU NEED TO INTEGRATE THIRD PARTYS TECHNOLOGY (SONY, DTS, DOLBY, AVS OR MPEGLA, ETC.)
* IN ALLWINNERSSDK OR PRODUCTS, YOU SHALL BE SOLELY RESPONSIBLE TO OBTAIN
* ALL APPROPRIATELY REQUIRED THIRD PARTY LICENCES.
* ALLWINNER SHALL HAVE NO WARRANTY, INDEMNITY OR OTHER OBLIGATIONS WITH RESPECT TO MATTERS
* COVERED UNDER ANY REQUIRED THIRD PARTY LICENSE.
* YOU ARE SOLELY RESPONSIBLE FOR YOUR USAGE OF THIRD PARTYS TECHNOLOGY.
*
*
* THIS SOFTWARE IS PROVIDED BY ALLWINNER"AS IS" AND TO THE MAXIMUM EXTENT
* PERMITTED BY LAW, ALLWINNER EXPRESSLY DISCLAIMS ALL WARRANTIES OF ANY KIND,
* WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING WITHOUT LIMITATION REGARDING
* THE TITLE, NON-INFRINGEMENT, ACCURACY, CONDITION, COMPLETENESS, PERFORMANCE
* OR MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
* IN NO EVENT SHALL ALLWINNER BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS, OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <csr.h>
#define L1_CACHE_BYTES (32)
static void dcache_wb_range(unsigned long start, unsigned long end)
{
unsigned long i = start & ~(L1_CACHE_BYTES - 1);
for (; i < end; i += L1_CACHE_BYTES)
{
asm volatile("dcache.cpa %0\n"::"r"(i):"memory");
}
asm volatile("fence.i":::"memory");
}
static void dcache_inv_range(unsigned long start, unsigned long end)
{
unsigned long i = start & ~(L1_CACHE_BYTES - 1);
for (; i < end; i += L1_CACHE_BYTES)
{
asm volatile("dcache.ipa %0\n"::"r"(i):"memory");
}
asm volatile("fence.i":::"memory");
}
static void dcache_wbinv_range(unsigned long start, unsigned long end)
{
unsigned long i = start & ~(L1_CACHE_BYTES - 1);
for (; i < end; i += L1_CACHE_BYTES)
{
asm volatile("dcache.cipa %0\n"::"r"(i):"memory");
}
asm volatile("fence.i":::"memory");
}
static void icache_inv_range(unsigned long start, unsigned long end)
{
unsigned long i = start & ~(L1_CACHE_BYTES - 1);
for (; i < end; i += L1_CACHE_BYTES)
{
asm volatile("icache.ipa %0\n"::"r"(i):"memory");
}
asm volatile("fence.i":::"memory");
}
void awos_arch_clean_dcache(void)
{
asm volatile("dcache.call\n":::"memory");
}
void awos_arch_clean_flush_dcache(void)
{
asm volatile("dcache.ciall\n":::"memory");
}
void awos_arch_flush_dcache(void)
{
asm volatile("dcache.iall\n":::"memory");
}
void awos_arch_flush_icache(void)
{
asm volatile("icache.iall\n":::"memory");
}
void awos_arch_mems_flush_icache_region(unsigned long start, unsigned long len)
{
icache_inv_range(start, start + len);
}
void awos_arch_mems_clean_dcache_region(unsigned long start, unsigned long len)
{
dcache_wb_range(start, start + len);
}
void awos_arch_mems_clean_flush_dcache_region(unsigned long start, unsigned long len)
{
dcache_wbinv_range(start, start + len);
}
void awos_arch_mems_flush_dcache_region(unsigned long start, unsigned long len)
{
dcache_inv_range(start, start + len);
}
void awos_arch_clean_flush_cache(void)
{
awos_arch_clean_flush_dcache();
awos_arch_flush_icache();
}
void awos_arch_clean_flush_cache_region(unsigned long start, unsigned long len)
{
awos_arch_mems_clean_flush_dcache_region(start, len);
awos_arch_mems_flush_icache_region(start, len);
}
void awos_arch_flush_cache(void)
{
awos_arch_flush_dcache();
awos_arch_flush_icache();
}
int check_virtual_address(unsigned long vaddr)
{
return 1;
}
void dcache_enable(void)
{
/*
(0:1) CACHE_SEL=2b11时
(4) INV=1
(16) BHT_INV=1
(17) TB_INV=1
*/
//csr_write(CSR_MCOR, 0x70013);
/*
(0) IE=1Icache打开
(1) DE=1Dcache打开
(2) WA=1write allocate模式 (c906不支持)
(3) WB=1 (c906固定为1)
(4) RS=1
(5) BPE=1
(6) BTB=1
(8) WBR=1 c906固定为1
(12) L0BTB=1
*/
csr_write(CSR_MHCR, 0x11ff);
/*
(15) MM为1时支持非对齐访问访
(16) UCME为1时cache操作指令
(17) CLINTEE为1时CLINT发起的超级用户软件中断和计时器中断可以被响应
(21) MAEE为1时MMU的pte中扩展地址属性位
(22) THEADISAEE为1时可以使用C906扩展指令集
*/
csr_set(CSR_MXSTATUS, 0x638000);
/*
(2) DPLD=1dcache预取开启
(3,4,5,6,7) AMR=13L1Cache
(8) IPLD=1ICACHE预取开启
(9) LPE=1
(13,14) DPLD为2时8
*/
//csr_write(CSR_MHINT, 0x16e30c);
csr_write(CSR_MHINT, 0x6e30c);
}
void dcache_disable(void)
{
}
void data_sync_barrier(void)
{
asm volatile("fence.i":::"memory");
}