2005-04-16 15:20:36 -07: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 .
*
* Copyright ( C ) 1994 - 2003 by Ralf Baechle
*/
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/sched.h>
# include <linux/mm.h>
# include <asm/cacheflush.h>
# include <asm/processor.h>
# include <asm/cpu.h>
# include <asm/cpu-features.h>
/* Cache operations. */
void ( * flush_cache_all ) ( void ) ;
void ( * __flush_cache_all ) ( void ) ;
void ( * flush_cache_mm ) ( struct mm_struct * mm ) ;
void ( * flush_cache_range ) ( struct vm_area_struct * vma , unsigned long start ,
unsigned long end ) ;
2005-03-18 17:36:42 +00:00
void ( * flush_cache_page ) ( struct vm_area_struct * vma , unsigned long page ,
unsigned long pfn ) ;
2006-01-29 02:27:51 +09:00
void ( * flush_icache_range ) ( unsigned long start , unsigned long end ) ;
2005-04-16 15:20:36 -07:00
/* MIPS specific cache operations */
void ( * flush_cache_sigtramp ) ( unsigned long addr ) ;
2006-04-05 20:42:04 +01:00
void ( * local_flush_data_cache_page ) ( void * addr ) ;
2005-04-16 15:20:36 -07:00
void ( * flush_data_cache_page ) ( unsigned long addr ) ;
void ( * flush_icache_all ) ( void ) ;
2006-12-10 18:43:59 +00:00
EXPORT_SYMBOL_GPL ( local_flush_data_cache_page ) ;
2005-03-08 14:39:39 +00:00
EXPORT_SYMBOL ( flush_data_cache_page ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_DMA_NONCOHERENT
/* DMA cache operations. */
void ( * _dma_cache_wback_inv ) ( unsigned long start , unsigned long size ) ;
void ( * _dma_cache_wback ) ( unsigned long start , unsigned long size ) ;
void ( * _dma_cache_inv ) ( unsigned long start , unsigned long size ) ;
EXPORT_SYMBOL ( _dma_cache_wback_inv ) ;
EXPORT_SYMBOL ( _dma_cache_wback ) ;
EXPORT_SYMBOL ( _dma_cache_inv ) ;
# endif /* CONFIG_DMA_NONCOHERENT */
/*
* We could optimize the case where the cache argument is not BCACHE but
* that seems very atypical use . . .
*/
2006-01-29 02:27:51 +09:00
asmlinkage int sys_cacheflush ( unsigned long addr ,
2005-03-01 19:22:29 +00:00
unsigned long bytes , unsigned int cache )
2005-04-16 15:20:36 -07:00
{
2005-10-19 19:57:14 +09:00
if ( bytes = = 0 )
return 0 ;
2005-03-01 19:22:29 +00:00
if ( ! access_ok ( VERIFY_WRITE , ( void __user * ) addr , bytes ) )
2005-04-16 15:20:36 -07:00
return - EFAULT ;
flush_icache_range ( addr , addr + bytes ) ;
return 0 ;
}
void __flush_dcache_page ( struct page * page )
{
struct address_space * mapping = page_mapping ( page ) ;
unsigned long addr ;
2006-08-12 16:40:08 +01:00
if ( PageHighMem ( page ) )
return ;
2005-04-16 15:20:36 -07:00
if ( mapping & & ! mapping_mapped ( mapping ) ) {
SetPageDcacheDirty ( page ) ;
return ;
}
/*
* We could delay the flush for the ! page_mapping case too . But that
* case is for exec env / arg pages and those are % 99 certainly going to
* get faulted into the tlb ( and thus flushed ) anyways .
*/
addr = ( unsigned long ) page_address ( page ) ;
flush_data_cache_page ( addr ) ;
}
EXPORT_SYMBOL ( __flush_dcache_page ) ;
void __update_cache ( struct vm_area_struct * vma , unsigned long address ,
pte_t pte )
{
struct page * page ;
unsigned long pfn , addr ;
2006-08-12 16:40:08 +01:00
int exec = ( vma - > vm_flags & VM_EXEC ) & & ! cpu_has_ic_fills_f_dc ;
2005-04-16 15:20:36 -07:00
pfn = pte_pfn ( pte ) ;
2006-08-12 16:40:08 +01:00
if ( unlikely ( ! pfn_valid ( pfn ) ) )
return ;
page = pfn_to_page ( pfn ) ;
if ( page_mapping ( page ) & & Page_dcache_dirty ( page ) ) {
addr = ( unsigned long ) page_address ( page ) ;
if ( exec | | pages_do_alias ( addr , address & PAGE_MASK ) )
2005-04-16 15:20:36 -07:00
flush_data_cache_page ( addr ) ;
ClearPageDcacheDirty ( page ) ;
}
}
2005-10-01 13:06:32 +01:00
static char cache_panic [ ] __initdata = " Yeee, unsupported cache architecture. " ;
2005-04-16 15:20:36 -07:00
void __init cpu_cache_init ( void )
{
2005-10-01 13:06:32 +01:00
if ( cpu_has_3k_cache ) {
extern void __weak r3k_cache_init ( void ) ;
r3k_cache_init ( ) ;
return ;
}
if ( cpu_has_6k_cache ) {
extern void __weak r6k_cache_init ( void ) ;
r6k_cache_init ( ) ;
return ;
2005-04-16 15:20:36 -07:00
}
2005-10-01 13:06:32 +01:00
if ( cpu_has_4k_cache ) {
extern void __weak r4k_cache_init ( void ) ;
r4k_cache_init ( ) ;
return ;
}
if ( cpu_has_8k_cache ) {
extern void __weak r8k_cache_init ( void ) ;
r8k_cache_init ( ) ;
return ;
}
if ( cpu_has_tx39_cache ) {
extern void __weak tx39_cache_init ( void ) ;
tx39_cache_init ( ) ;
return ;
}
if ( cpu_has_sb1_cache ) {
extern void __weak sb1_cache_init ( void ) ;
sb1_cache_init ( ) ;
return ;
}
panic ( cache_panic ) ;
2005-04-16 15:20:36 -07:00
}