| /* MN10300 CPU cache invalidation routines, using direct tag flushing |
| * |
| * Copyright (C) 2011 Red Hat, Inc. All Rights Reserved. |
| * Written by David Howells (dhowells@redhat.com) |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public Licence |
| * as published by the Free Software Foundation; either version |
| * 2 of the Licence, or (at your option) any later version. |
| */ |
| #include <linux/sys.h> |
| #include <linux/linkage.h> |
| #include <asm/smp.h> |
| #include <asm/page.h> |
| #include <asm/cache.h> |
| #include <asm/irqflags.h> |
| #include <asm/cacheflush.h> |
| #include "cache.inc" |
| |
| .am33_2 |
| |
| ############################################################################### |
| # |
| # void debugger_local_cache_flushinv(void) |
| # |
| # Flush the entire data cache back to RAM and invalidate the icache |
| # |
| ############################################################################### |
| ALIGN |
| .globl debugger_local_cache_flushinv |
| .type debugger_local_cache_flushinv,@function |
| debugger_local_cache_flushinv: |
| # |
| # firstly flush the dcache |
| # |
| movhu (CHCTR),d0 |
| btst CHCTR_DCEN|CHCTR_ICEN,d0 |
| beq debugger_local_cache_flushinv_end |
| |
| btst CHCTR_DCEN,d0 |
| beq debugger_local_cache_flushinv_no_dcache |
| |
| # read the addresses tagged in the cache's tag RAM and attempt to flush |
| # those addresses specifically |
| # - we rely on the hardware to filter out invalid tag entry addresses |
| mov DCACHE_TAG(0,0),a0 # dcache tag RAM access address |
| mov DCACHE_PURGE(0,0),a1 # dcache purge request address |
| mov L1_CACHE_NWAYS*L1_CACHE_NENTRIES,e0 # total number of entries |
| |
| mn10300_local_dcache_flush_loop: |
| mov (a0),d0 |
| and L1_CACHE_TAG_MASK,d0 |
| or L1_CACHE_TAG_VALID,d0 # retain valid entries in the |
| # cache |
| mov d0,(a1) # conditional purge |
| |
| add L1_CACHE_BYTES,a0 |
| add L1_CACHE_BYTES,a1 |
| add -1,e0 |
| bne mn10300_local_dcache_flush_loop |
| |
| debugger_local_cache_flushinv_no_dcache: |
| # |
| # secondly, invalidate the icache if it is enabled |
| # |
| mov CHCTR,a0 |
| movhu (a0),d0 |
| btst CHCTR_ICEN,d0 |
| beq debugger_local_cache_flushinv_end |
| |
| invalidate_icache 1 |
| |
| debugger_local_cache_flushinv_end: |
| ret [],0 |
| .size debugger_local_cache_flushinv,.-debugger_local_cache_flushinv |
| |
| ############################################################################### |
| # |
| # void debugger_local_cache_flushinv_one(u8 *addr) |
| # |
| # Invalidate one particular cacheline if it's in the icache |
| # |
| ############################################################################### |
| ALIGN |
| .globl debugger_local_cache_flushinv_one |
| .type debugger_local_cache_flushinv_one,@function |
| debugger_local_cache_flushinv_one: |
| movhu (CHCTR),d1 |
| btst CHCTR_DCEN|CHCTR_ICEN,d1 |
| beq debugger_local_cache_flushinv_one_end |
| btst CHCTR_DCEN,d1 |
| beq debugger_local_cache_flushinv_one_icache |
| |
| # round cacheline addr down |
| and L1_CACHE_TAG_MASK,d0 |
| mov d0,a1 |
| |
| # determine the dcache purge control reg address |
| mov DCACHE_PURGE(0,0),a0 |
| and L1_CACHE_TAG_ENTRY,d0 |
| add d0,a0 |
| |
| # retain valid entries in the cache |
| or L1_CACHE_TAG_VALID,a1 |
| |
| # conditionally purge this line in all ways |
| mov a1,(L1_CACHE_WAYDISP*0,a0) |
| |
| # now go and do the icache |
| bra debugger_local_cache_flushinv_one_icache |
| |
| debugger_local_cache_flushinv_one_end: |
| ret [],0 |
| .size debugger_local_cache_flushinv_one,.-debugger_local_cache_flushinv_one |