2005-04-17 02:20:36 +04:00
/*
2007-11-28 14:19:38 +03:00
* arch / sh / mm / cache - sh5 . c
2005-04-17 02:20:36 +04:00
*
2008-02-13 14:14:10 +03:00
* Copyright ( C ) 2000 , 2001 Paolo Alberelli
* Copyright ( C ) 2002 Benedict Gaster
* Copyright ( C ) 2003 Richard Curnow
* Copyright ( C ) 2003 - 2008 Paul Mundt
2007-11-28 14:19:38 +03:00
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
2005-04-17 02:20:36 +04:00
*/
# include <linux/init.h>
# include <linux/mman.h>
# include <linux/mm.h>
2008-02-13 14:14:10 +03:00
# include <asm/tlb.h>
2005-04-17 02:20:36 +04:00
# include <asm/processor.h>
# include <asm/cache.h>
2008-02-13 14:14:10 +03:00
# include <asm/pgalloc.h>
2005-04-17 02:20:36 +04:00
# include <asm/uaccess.h>
# include <asm/mmu_context.h>
2009-08-15 07:29:49 +04:00
extern void __weak sh4__flush_region_init ( void ) ;
2005-04-17 02:20:36 +04:00
/* Wired TLB entry for the D-cache */
static unsigned long long dtlb_cache_slot ;
2008-02-13 14:14:10 +03:00
/*
* The following group of functions deal with mapping and unmapping a
* temporary page into a DTLB slot that has been set aside for exclusive
* use .
*/
static inline void
sh64_setup_dtlb_cache_slot ( unsigned long eaddr , unsigned long asid ,
unsigned long paddr )
2005-04-17 02:20:36 +04:00
{
2009-09-01 16:12:55 +04:00
local_irq_disable ( ) ;
2005-04-17 02:20:36 +04:00
sh64_setup_tlb_slot ( dtlb_cache_slot , eaddr , asid , paddr ) ;
}
static inline void sh64_teardown_dtlb_cache_slot ( void )
{
sh64_teardown_tlb_slot ( dtlb_cache_slot ) ;
2009-09-01 16:12:55 +04:00
local_irq_enable ( ) ;
2005-04-17 02:20:36 +04:00
}
2008-02-13 14:14:10 +03:00
static inline void sh64_icache_inv_all ( void )
2005-04-17 02:20:36 +04:00
{
unsigned long long addr , flag , data ;
2009-09-01 16:12:55 +04:00
unsigned long flags ;
2005-04-17 02:20:36 +04:00
2008-02-13 14:14:10 +03:00
addr = ICCR0 ;
flag = ICCR0_ICI ;
data = 0 ;
2005-04-17 02:20:36 +04:00
2009-09-01 16:12:55 +04:00
/* Make this a critical section for safety (probably not strictly necessary.) */
local_irq_save ( flags ) ;
2005-04-17 02:20:36 +04:00
/* Without %1 it gets unexplicably wrong */
2008-02-13 14:14:10 +03:00
__asm__ __volatile__ (
" getcfg %3, 0, %0 \n \t "
" or %0, %2, %0 \n \t "
" putcfg %3, 0, %0 \n \t "
" synci "
: " =&r " ( data )
: " 0 " ( data ) , " r " ( flag ) , " r " ( addr ) ) ;
2009-09-01 16:12:55 +04:00
local_irq_restore ( flags ) ;
2005-04-17 02:20:36 +04:00
}
static void sh64_icache_inv_kernel_range ( unsigned long start , unsigned long end )
{
/* Invalidate range of addresses [start,end] from the I-cache, where
* the addresses lie in the kernel superpage . */
unsigned long long ullend , addr , aligned_start ;
aligned_start = ( unsigned long long ) ( signed long long ) ( signed long ) start ;
2008-02-13 14:14:10 +03:00
addr = L1_CACHE_ALIGN ( aligned_start ) ;
2005-04-17 02:20:36 +04:00
ullend = ( unsigned long long ) ( signed long long ) ( signed long ) end ;
2008-02-13 14:14:10 +03:00
2005-04-17 02:20:36 +04:00
while ( addr < = ullend ) {
2008-02-13 14:14:10 +03:00
__asm__ __volatile__ ( " icbi %0, 0 " : : " r " ( addr ) ) ;
2005-04-17 02:20:36 +04:00
addr + = L1_CACHE_BYTES ;
}
}
static void sh64_icache_inv_user_page ( struct vm_area_struct * vma , unsigned long eaddr )
{
/* If we get called, we know that vma->vm_flags contains VM_EXEC.
Also , eaddr is page - aligned . */
2008-02-13 14:14:10 +03:00
unsigned int cpu = smp_processor_id ( ) ;
2005-04-17 02:20:36 +04:00
unsigned long long addr , end_addr ;
2009-09-01 16:12:55 +04:00
unsigned long flags = 0 ;
2005-04-17 02:20:36 +04:00
unsigned long running_asid , vma_asid ;
addr = eaddr ;
end_addr = addr + PAGE_SIZE ;
/* Check whether we can use the current ASID for the I-cache
invalidation . For example , if we ' re called via
access_process_vm - > flush_cache_page - > here , ( e . g . when reading from
/ proc ) , ' running_asid ' will be that of the reader , not of the
victim .
Also , note the risk that we might get pre - empted between the ASID
compare and blocking IRQs , and before we regain control , the
pid - > ASID mapping changes . However , the whole cache will get
invalidated when the mapping is renewed , so the worst that can
happen is that the loop below ends up invalidating somebody else ' s
cache entries .
*/
running_asid = get_asid ( ) ;
2008-02-13 14:14:10 +03:00
vma_asid = cpu_asid ( cpu , vma - > vm_mm ) ;
2009-09-01 16:12:55 +04:00
if ( running_asid ! = vma_asid ) {
local_irq_save ( flags ) ;
2005-04-17 02:20:36 +04:00
switch_and_save_asid ( vma_asid ) ;
2009-09-01 16:12:55 +04:00
}
2005-04-17 02:20:36 +04:00
while ( addr < end_addr ) {
/* Worth unrolling a little */
2008-02-13 14:14:10 +03:00
__asm__ __volatile__ ( " icbi %0, 0 " : : " r " ( addr ) ) ;
__asm__ __volatile__ ( " icbi %0, 32 " : : " r " ( addr ) ) ;
__asm__ __volatile__ ( " icbi %0, 64 " : : " r " ( addr ) ) ;
__asm__ __volatile__ ( " icbi %0, 96 " : : " r " ( addr ) ) ;
2005-04-17 02:20:36 +04:00
addr + = 128 ;
}
2009-09-01 16:12:55 +04:00
if ( running_asid ! = vma_asid ) {
2005-04-17 02:20:36 +04:00
switch_and_save_asid ( running_asid ) ;
2009-09-01 16:12:55 +04:00
local_irq_restore ( flags ) ;
}
2005-04-17 02:20:36 +04:00
}
static void sh64_icache_inv_user_page_range ( struct mm_struct * mm ,
unsigned long start , unsigned long end )
{
/* Used for invalidating big chunks of I-cache, i.e. assume the range
is whole pages . If ' start ' or ' end ' is not page aligned , the code
is conservative and invalidates to the ends of the enclosing pages .
This is functionally OK , just a performance loss . */
/* See the comments below in sh64_dcache_purge_user_range() regarding
the choice of algorithm . However , for the I - cache option ( 2 ) isn ' t
available because there are no physical tags so aliases can ' t be
resolved . The icbi instruction has to be used through the user
mapping . Because icbi is cheaper than ocbp on a cache hit , it
would be cheaper to use the selective code for a large range than is
possible with the D - cache . Just assume 64 for now as a working
figure .
*/
int n_pages ;
2008-02-13 14:14:10 +03:00
if ( ! mm )
return ;
2005-04-17 02:20:36 +04:00
n_pages = ( ( end - start ) > > PAGE_SHIFT ) ;
if ( n_pages > = 64 ) {
sh64_icache_inv_all ( ) ;
} else {
unsigned long aligned_start ;
unsigned long eaddr ;
unsigned long after_last_page_start ;
unsigned long mm_asid , current_asid ;
2009-09-01 16:12:55 +04:00
unsigned long flags = 0 ;
2005-04-17 02:20:36 +04:00
2008-02-13 14:14:10 +03:00
mm_asid = cpu_asid ( smp_processor_id ( ) , mm ) ;
2005-04-17 02:20:36 +04:00
current_asid = get_asid ( ) ;
2009-09-01 16:12:55 +04:00
if ( mm_asid ! = current_asid ) {
/* Switch ASID and run the invalidate loop under cli */
local_irq_save ( flags ) ;
2005-04-17 02:20:36 +04:00
switch_and_save_asid ( mm_asid ) ;
2009-09-01 16:12:55 +04:00
}
2005-04-17 02:20:36 +04:00
aligned_start = start & PAGE_MASK ;
after_last_page_start = PAGE_SIZE + ( ( end - 1 ) & PAGE_MASK ) ;
while ( aligned_start < after_last_page_start ) {
struct vm_area_struct * vma ;
unsigned long vma_end ;
vma = find_vma ( mm , aligned_start ) ;
if ( ! vma | | ( aligned_start < = vma - > vm_end ) ) {
/* Avoid getting stuck in an error condition */
aligned_start + = PAGE_SIZE ;
continue ;
}
vma_end = vma - > vm_end ;
if ( vma - > vm_flags & VM_EXEC ) {
/* Executable */
eaddr = aligned_start ;
while ( eaddr < vma_end ) {
sh64_icache_inv_user_page ( vma , eaddr ) ;
eaddr + = PAGE_SIZE ;
}
}
aligned_start = vma - > vm_end ; /* Skip to start of next region */
}
2008-02-13 14:14:10 +03:00
2009-09-01 16:12:55 +04:00
if ( mm_asid ! = current_asid ) {
2005-04-17 02:20:36 +04:00
switch_and_save_asid ( current_asid ) ;
2009-09-01 16:12:55 +04:00
local_irq_restore ( flags ) ;
}
2005-04-17 02:20:36 +04:00
}
}
static void sh64_icache_inv_current_user_range ( unsigned long start , unsigned long end )
{
/* The icbi instruction never raises ITLBMISS. i.e. if there's not a
cache hit on the virtual tag the instruction ends there , without a
TLB lookup . */
unsigned long long aligned_start ;
unsigned long long ull_end ;
unsigned long long addr ;
ull_end = end ;
/* Just invalidate over the range using the natural addresses. TLB
miss handling will be OK ( TBC ) . Since it ' s for the current process ,
either we ' re already in the right ASID context , or the ASIDs have
been recycled since we were last active in which case we might just
invalidate another processes I - cache entries : no worries , just a
performance drop for him . */
2008-02-13 14:14:10 +03:00
aligned_start = L1_CACHE_ALIGN ( start ) ;
2005-04-17 02:20:36 +04:00
addr = aligned_start ;
while ( addr < ull_end ) {
2008-02-13 14:14:10 +03:00
__asm__ __volatile__ ( " icbi %0, 0 " : : " r " ( addr ) ) ;
__asm__ __volatile__ ( " nop " ) ;
__asm__ __volatile__ ( " nop " ) ;
2005-04-17 02:20:36 +04:00
addr + = L1_CACHE_BYTES ;
}
}
/* Buffer used as the target of alloco instructions to purge data from cache
sets by natural eviction . - - RPC */
2008-02-13 14:14:10 +03:00
# define DUMMY_ALLOCO_AREA_SIZE ((L1_CACHE_BYTES << 10) + (1024 * 4))
2005-04-17 02:20:36 +04:00
static unsigned char dummy_alloco_area [ DUMMY_ALLOCO_AREA_SIZE ] __cacheline_aligned = { 0 , } ;
2008-02-13 14:14:10 +03:00
static void inline sh64_dcache_purge_sets ( int sets_to_purge_base , int n_sets )
2005-04-17 02:20:36 +04:00
{
/* Purge all ways in a particular block of sets, specified by the base
set number and number of sets . Can handle wrap - around , if that ' s
needed . */
int dummy_buffer_base_set ;
unsigned long long eaddr , eaddr0 , eaddr1 ;
int j ;
int set_offset ;
2008-02-13 14:14:10 +03:00
dummy_buffer_base_set = ( ( int ) & dummy_alloco_area &
cpu_data - > dcache . entry_mask ) > >
cpu_data - > dcache . entry_shift ;
2005-04-17 02:20:36 +04:00
set_offset = sets_to_purge_base - dummy_buffer_base_set ;
2008-02-13 14:14:10 +03:00
for ( j = 0 ; j < n_sets ; j + + , set_offset + + ) {
2005-04-17 02:20:36 +04:00
set_offset & = ( cpu_data - > dcache . sets - 1 ) ;
2008-02-13 14:14:10 +03:00
eaddr0 = ( unsigned long long ) dummy_alloco_area +
( set_offset < < cpu_data - > dcache . entry_shift ) ;
/*
* Do one alloco which hits the required set per cache
* way . For write - back mode , this will purge the # ways
* resident lines . There ' s little point unrolling this
* loop because the allocos stall more if they ' re too
* close together .
*/
eaddr1 = eaddr0 + cpu_data - > dcache . way_size *
cpu_data - > dcache . ways ;
for ( eaddr = eaddr0 ; eaddr < eaddr1 ;
eaddr + = cpu_data - > dcache . way_size ) {
__asm__ __volatile__ ( " alloco %0, 0 " : : " r " ( eaddr ) ) ;
__asm__ __volatile__ ( " synco " ) ; /* TAKum03020 */
2005-04-17 02:20:36 +04:00
}
2008-02-13 14:14:10 +03:00
eaddr1 = eaddr0 + cpu_data - > dcache . way_size *
cpu_data - > dcache . ways ;
for ( eaddr = eaddr0 ; eaddr < eaddr1 ;
eaddr + = cpu_data - > dcache . way_size ) {
/*
* Load from each address . Required because
* alloco is a NOP if the cache is write - through .
*/
2005-04-17 02:20:36 +04:00
if ( test_bit ( SH_CACHE_MODE_WT , & ( cpu_data - > dcache . flags ) ) )
2009-05-09 09:38:49 +04:00
__raw_readb ( ( unsigned long ) eaddr ) ;
2005-04-17 02:20:36 +04:00
}
}
2008-02-13 14:14:10 +03:00
/*
* Don ' t use OCBI to invalidate the lines . That costs cycles
* directly . If the dummy block is just left resident , it will
* naturally get evicted as required .
*/
2005-04-17 02:20:36 +04:00
}
2008-02-13 14:14:10 +03:00
/*
* Purge the entire contents of the dcache . The most efficient way to
* achieve this is to use alloco instructions on a region of unused
* memory equal in size to the cache , thereby causing the current
* contents to be discarded by natural eviction . The alternative , namely
* reading every tag , setting up a mapping for the corresponding page and
* doing an OCBP for the line , would be much more expensive .
*/
2005-04-17 02:20:36 +04:00
static void sh64_dcache_purge_all ( void )
{
sh64_dcache_purge_sets ( 0 , cpu_data - > dcache . sets ) ;
}
/* Assumes this address (+ (2**n_synbits) pages up from it) aren't used for
anything else in the kernel */
# define MAGIC_PAGE0_START 0xffffffffec000000ULL
2008-02-13 14:14:10 +03:00
/* Purge the physical page 'paddr' from the cache. It's known that any
* cache lines requiring attention have the same page colour as the the
* address ' eaddr ' .
*
* This relies on the fact that the D - cache matches on physical tags when
* no virtual tag matches . So we create an alias for the original page
* and purge through that . ( Alternatively , we could have done this by
* switching ASID to match the original mapping and purged through that ,
* but that involves ASID switching cost + probably a TLBMISS + refill
* anyway . )
*/
static void sh64_dcache_purge_coloured_phy_page ( unsigned long paddr ,
unsigned long eaddr )
2005-04-17 02:20:36 +04:00
{
unsigned long long magic_page_start ;
unsigned long long magic_eaddr , magic_eaddr_end ;
magic_page_start = MAGIC_PAGE0_START + ( eaddr & CACHE_OC_SYN_MASK ) ;
/* As long as the kernel is not pre-emptible, this doesn't need to be
under cli / sti . */
sh64_setup_dtlb_cache_slot ( magic_page_start , get_asid ( ) , paddr ) ;
magic_eaddr = magic_page_start ;
magic_eaddr_end = magic_eaddr + PAGE_SIZE ;
2008-02-13 14:14:10 +03:00
2005-04-17 02:20:36 +04:00
while ( magic_eaddr < magic_eaddr_end ) {
/* Little point in unrolling this loop - the OCBPs are blocking
and won ' t go any quicker ( i . e . the loop overhead is parallel
to part of the OCBP execution . ) */
2008-02-13 14:14:10 +03:00
__asm__ __volatile__ ( " ocbp %0, 0 " : : " r " ( magic_eaddr ) ) ;
2005-04-17 02:20:36 +04:00
magic_eaddr + = L1_CACHE_BYTES ;
}
sh64_teardown_dtlb_cache_slot ( ) ;
}
2008-02-13 14:14:10 +03:00
/*
* Purge a page given its physical start address , by creating a temporary
* 1 page mapping and purging across that . Even if we know the virtual
* address ( & vma or mm ) of the page , the method here is more elegant
* because it avoids issues of coping with page faults on the purge
* instructions ( i . e . no special - case code required in the critical path
* in the TLB miss handling ) .
*/
2005-04-17 02:20:36 +04:00
static void sh64_dcache_purge_phy_page ( unsigned long paddr )
{
unsigned long long eaddr_start , eaddr , eaddr_end ;
int i ;
/* As long as the kernel is not pre-emptible, this doesn't need to be
under cli / sti . */
eaddr_start = MAGIC_PAGE0_START ;
2008-02-13 14:14:10 +03:00
for ( i = 0 ; i < ( 1 < < CACHE_OC_N_SYNBITS ) ; i + + ) {
2005-04-17 02:20:36 +04:00
sh64_setup_dtlb_cache_slot ( eaddr_start , get_asid ( ) , paddr ) ;
eaddr = eaddr_start ;
eaddr_end = eaddr + PAGE_SIZE ;
while ( eaddr < eaddr_end ) {
2008-02-13 14:14:10 +03:00
__asm__ __volatile__ ( " ocbp %0, 0 " : : " r " ( eaddr ) ) ;
2005-04-17 02:20:36 +04:00
eaddr + = L1_CACHE_BYTES ;
}
sh64_teardown_dtlb_cache_slot ( ) ;
eaddr_start + = PAGE_SIZE ;
}
}
2005-10-30 04:16:34 +03:00
static void sh64_dcache_purge_user_pages ( struct mm_struct * mm ,
unsigned long addr , unsigned long end )
2005-04-17 02:20:36 +04:00
{
pgd_t * pgd ;
2008-02-13 14:14:10 +03:00
pud_t * pud ;
2005-04-17 02:20:36 +04:00
pmd_t * pmd ;
pte_t * pte ;
pte_t entry ;
2005-10-30 04:16:34 +03:00
spinlock_t * ptl ;
2005-04-17 02:20:36 +04:00
unsigned long paddr ;
2005-10-30 04:16:34 +03:00
if ( ! mm )
return ; /* No way to find physical address of page */
pgd = pgd_offset ( mm , addr ) ;
if ( pgd_bad ( * pgd ) )
return ;
2008-02-13 14:14:10 +03:00
pud = pud_offset ( pgd , addr ) ;
if ( pud_none ( * pud ) | | pud_bad ( * pud ) )
return ;
pmd = pmd_offset ( pud , addr ) ;
2005-10-30 04:16:34 +03:00
if ( pmd_none ( * pmd ) | | pmd_bad ( * pmd ) )
return ;
pte = pte_offset_map_lock ( mm , pmd , addr , & ptl ) ;
do {
entry = * pte ;
if ( pte_none ( entry ) | | ! pte_present ( entry ) )
continue ;
paddr = pte_val ( entry ) & PAGE_MASK ;
sh64_dcache_purge_coloured_phy_page ( paddr , addr ) ;
} while ( pte + + , addr + = PAGE_SIZE , addr ! = end ) ;
pte_unmap_unlock ( pte - 1 , ptl ) ;
2005-04-17 02:20:36 +04:00
}
2008-02-13 14:14:10 +03:00
/*
* There are at least 5 choices for the implementation of this , with
* pros ( + ) , cons ( - ) , comments ( * ) :
*
* 1. ocbp each line in the range through the original user ' s ASID
* + no lines spuriously evicted
* - tlbmiss handling ( must either handle faults on demand = > extra
* special - case code in tlbmiss critical path ) , or map the page in
* advance ( = > flush_tlb_range in advance to avoid multiple hits )
* - ASID switching
* - expensive for large ranges
*
* 2. temporarily map each page in the range to a special effective
* address and ocbp through the temporary mapping ; relies on the
* fact that SH - 5 OCB * always do TLB lookup and match on ptags ( they
* never look at the etags )
* + no spurious evictions
* - expensive for large ranges
* * surely cheaper than ( 1 )
*
* 3. walk all the lines in the cache , check the tags , if a match
* occurs create a page mapping to ocbp the line through
* + no spurious evictions
* - tag inspection overhead
* - ( especially for small ranges )
* - potential cost of setting up / tearing down page mapping for
* every line that matches the range
* * cost partly independent of range size
*
* 4. walk all the lines in the cache , check the tags , if a match
* occurs use 4 * alloco to purge the line ( + 3 other probably
* innocent victims ) by natural eviction
* + no tlb mapping overheads
* - spurious evictions
* - tag inspection overhead
*
* 5. implement like flush_cache_all
* + no tag inspection overhead
* - spurious evictions
* - bad for small ranges
*
* ( 1 ) can be ruled out as more expensive than ( 2 ) . ( 2 ) appears best
* for small ranges . The choice between ( 3 ) , ( 4 ) and ( 5 ) for large
* ranges and the range size for the large / small boundary need
* benchmarking to determine .
*
* For now use approach ( 2 ) for small ranges and ( 5 ) for large ones .
*/
2005-04-17 02:20:36 +04:00
static void sh64_dcache_purge_user_range ( struct mm_struct * mm ,
unsigned long start , unsigned long end )
{
2008-02-13 14:14:10 +03:00
int n_pages = ( ( end - start ) > > PAGE_SHIFT ) ;
2005-04-17 02:20:36 +04:00
2005-10-30 04:16:34 +03:00
if ( n_pages > = 64 | | ( ( start ^ ( end - 1 ) ) & PMD_MASK ) ) {
2005-04-17 02:20:36 +04:00
sh64_dcache_purge_all ( ) ;
} else {
2005-10-30 04:16:34 +03:00
/* Small range, covered by a single page table page */
start & = PAGE_MASK ; /* should already be so */
end = PAGE_ALIGN ( end ) ; /* should already be so */
sh64_dcache_purge_user_pages ( mm , start , end ) ;
2005-04-17 02:20:36 +04:00
}
}
2008-02-13 14:14:10 +03:00
/*
* Invalidate the entire contents of both caches , after writing back to
* memory any dirty data from the D - cache .
*/
2009-08-21 12:23:14 +04:00
static void sh5_flush_cache_all ( void * unused )
2005-04-17 02:20:36 +04:00
{
sh64_dcache_purge_all ( ) ;
sh64_icache_inv_all ( ) ;
}
2008-02-13 14:14:10 +03:00
/*
* Invalidate an entire user - address space from both caches , after
* writing back dirty data ( e . g . for shared mmap etc ) .
*
* This could be coded selectively by inspecting all the tags then
* doing 4 * alloco on any set containing a match ( as for
* flush_cache_range ) , but fork / exit / execve ( where this is called from )
* are expensive anyway .
*
* Have to do a purge here , despite the comments re I - cache below .
* There could be odd - coloured dirty data associated with the mm still
* in the cache - if this gets written out through natural eviction
* after the kernel has reused the page there will be chaos .
*
* The mm being torn down won ' t ever be active again , so any Icache
* lines tagged with its ASID won ' t be visible for the rest of the
* lifetime of this ASID cycle . Before the ASID gets reused , there
* will be a flush_cache_all . Hence we don ' t need to touch the
* I - cache . This is similar to the lack of action needed in
* flush_tlb_mm - see fault . c .
*/
2009-08-21 12:23:14 +04:00
static void sh5_flush_cache_mm ( void * unused )
2005-04-17 02:20:36 +04:00
{
sh64_dcache_purge_all ( ) ;
}
2008-02-13 14:14:10 +03:00
/*
* Invalidate ( from both caches ) the range [ start , end ) of virtual
* addresses from the user address space specified by mm , after writing
* back any dirty data .
*
* Note , ' end ' is 1 byte beyond the end of the range to flush .
*/
2009-08-21 12:23:14 +04:00
static void sh5_flush_cache_range ( void * args )
2005-04-17 02:20:36 +04:00
{
2009-08-21 12:23:14 +04:00
struct flusher_data * data = args ;
struct vm_area_struct * vma ;
unsigned long start , end ;
2005-04-17 02:20:36 +04:00
2009-08-21 12:23:14 +04:00
vma = data - > vma ;
start = data - > addr1 ;
end = data - > addr2 ;
sh64_dcache_purge_user_range ( vma - > vm_mm , start , end ) ;
sh64_icache_inv_user_page_range ( vma - > vm_mm , start , end ) ;
2005-04-17 02:20:36 +04:00
}
2008-02-13 14:14:10 +03:00
/*
* Invalidate any entries in either cache for the vma within the user
* address space vma - > vm_mm for the page starting at virtual address
* ' eaddr ' . This seems to be used primarily in breaking COW . Note ,
* the I - cache must be searched too in case the page in question is
* both writable and being executed from ( e . g . stack trampolines . )
*
* Note , this is called with pte lock held .
*/
2009-08-21 12:23:14 +04:00
static void sh5_flush_cache_page ( void * args )
2005-04-17 02:20:36 +04:00
{
2009-08-21 12:23:14 +04:00
struct flusher_data * data = args ;
struct vm_area_struct * vma ;
unsigned long eaddr , pfn ;
vma = data - > vma ;
eaddr = data - > addr1 ;
pfn = data - > addr2 ;
2005-04-17 02:20:36 +04:00
sh64_dcache_purge_phy_page ( pfn < < PAGE_SHIFT ) ;
2008-02-13 14:14:10 +03:00
if ( vma - > vm_flags & VM_EXEC )
2005-04-17 02:20:36 +04:00
sh64_icache_inv_user_page ( vma , eaddr ) ;
}
2009-08-21 12:23:14 +04:00
static void sh5_flush_dcache_page ( void * page )
2008-02-13 14:14:10 +03:00
{
sh64_dcache_purge_phy_page ( page_to_phys ( page ) ) ;
wmb ( ) ;
}
2005-04-17 02:20:36 +04:00
2008-02-13 14:14:10 +03:00
/*
* Flush the range [ start , end ] of kernel virtual adddress space from
* the I - cache . The corresponding range must be purged from the
* D - cache also because the SH - 5 doesn ' t have cache snooping between
* the caches . The addresses will be visible through the superpage
* mapping , therefore it ' s guaranteed that there no cache entries for
* the range in cache sets of the wrong colour .
*/
2009-08-21 12:23:14 +04:00
static void sh5_flush_icache_range ( void * args )
2008-02-13 14:14:10 +03:00
{
2009-08-21 12:23:14 +04:00
struct flusher_data * data = args ;
unsigned long start , end ;
start = data - > addr1 ;
end = data - > addr2 ;
2008-02-13 14:14:10 +03:00
__flush_purge_region ( ( void * ) start , end ) ;
wmb ( ) ;
sh64_icache_inv_kernel_range ( start , end ) ;
}
2005-04-17 02:20:36 +04:00
2008-02-13 14:14:10 +03:00
/*
* For the address range [ start , end ) , write back the data from the
* D - cache and invalidate the corresponding region of the I - cache for the
* current process . Used to flush signal trampolines on the stack to
* make them executable .
*/
2009-08-21 12:23:14 +04:00
static void sh5_flush_cache_sigtramp ( void * vaddr )
2008-02-13 14:14:10 +03:00
{
2009-08-21 12:23:14 +04:00
unsigned long end = ( unsigned long ) vaddr + L1_CACHE_BYTES ;
2008-02-13 14:14:10 +03:00
2009-08-21 12:23:14 +04:00
__flush_wback_region ( vaddr , L1_CACHE_BYTES ) ;
2008-02-13 14:14:10 +03:00
wmb ( ) ;
2009-08-21 12:23:14 +04:00
sh64_icache_inv_current_user_range ( ( unsigned long ) vaddr , end ) ;
2005-04-17 02:20:36 +04:00
}
2009-08-15 20:50:17 +04:00
void __init sh5_cache_init ( void )
2005-04-17 02:20:36 +04:00
{
2009-08-21 12:23:14 +04:00
local_flush_cache_all = sh5_flush_cache_all ;
local_flush_cache_mm = sh5_flush_cache_mm ;
local_flush_cache_dup_mm = sh5_flush_cache_mm ;
local_flush_cache_page = sh5_flush_cache_page ;
local_flush_cache_range = sh5_flush_cache_range ;
local_flush_dcache_page = sh5_flush_dcache_page ;
local_flush_icache_range = sh5_flush_icache_range ;
local_flush_cache_sigtramp = sh5_flush_cache_sigtramp ;
2005-04-17 02:20:36 +04:00
2009-08-15 20:50:17 +04:00
/* Reserve a slot for dcache colouring in the DTLB */
dtlb_cache_slot = sh64_get_wired_dtlb_entry ( ) ;
2009-07-27 16:30:17 +04:00
2009-08-15 20:50:17 +04:00
sh4__flush_region_init ( ) ;
2009-07-27 16:30:17 +04:00
}