2005-04-16 15:20:36 -07:00
/*
* arch / sh / mm / tlb - sh3 . c
*
* SH - 3 specific TLB operations
*
* Copyright ( C ) 1999 Niibe Yutaka
* Copyright ( C ) 2002 Paul Mundt
*
* Released under the terms of the GNU GPL v2 .0 .
*/
2007-03-05 19:46:47 +09:00
# include <linux/signal.h>
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/types.h>
# include <linux/ptrace.h>
# include <linux/mman.h>
# include <linux/mm.h>
# include <linux/smp.h>
# include <linux/interrupt.h>
2005-04-16 15:20:36 -07:00
# include <asm/system.h>
2007-03-05 19:46:47 +09:00
# include <asm/io.h>
# include <asm/uaccess.h>
# include <asm/pgalloc.h>
2005-04-16 15:20:36 -07:00
# include <asm/mmu_context.h>
2007-03-05 19:46:47 +09:00
# include <asm/cacheflush.h>
2009-07-29 00:12:17 +09:00
void __update_tlb ( struct vm_area_struct * vma , unsigned long address , pte_t pte )
2007-03-05 19:46:47 +09:00
{
2009-07-29 00:12:17 +09:00
unsigned long flags , pteval , vpn ;
2007-03-05 19:46:47 +09:00
2009-07-29 00:12:17 +09:00
/*
* Handle debugger faulting in for debugee .
*/
2009-07-29 22:06:58 +09:00
if ( vma & & current - > active_mm ! = vma - > vm_mm )
2007-03-05 19:46:47 +09:00
return ;
local_irq_save ( flags ) ;
/* Set PTEH register */
vpn = ( address & MMU_VPN_MASK ) | get_asid ( ) ;
2010-01-26 12:58:40 +09:00
__raw_writel ( vpn , MMU_PTEH ) ;
2007-03-05 19:46:47 +09:00
pteval = pte_val ( pte ) ;
/* Set PTEL register */
pteval & = _PAGE_FLAGS_HARDWARE_MASK ; /* drop software flags */
/* conveniently, we want all the software flags to be 0 anyway */
2010-01-26 12:58:40 +09:00
__raw_writel ( pteval , MMU_PTEL ) ;
2007-03-05 19:46:47 +09:00
/* Load the TLB */
asm volatile ( " ldtlb " : /* no output */ : /* no input */ : " memory " ) ;
local_irq_restore ( flags ) ;
}
2005-04-16 15:20:36 -07:00
2006-12-25 19:28:54 +09:00
void local_flush_tlb_one ( unsigned long asid , unsigned long page )
2005-04-16 15:20:36 -07:00
{
unsigned long addr , data ;
int i , ways = MMU_NTLB_WAYS ;
/*
* NOTE : PTEH . ASID should be set to this MM
* _AND_ we need to write ASID to the array .
*
* It would be simple if we didn ' t need to set PTEH . ASID . . .
*/
addr = MMU_TLB_ADDRESS_ARRAY | ( page & 0x1F000 ) ;
data = ( page & 0xfffe0000 ) | asid ; /* VALID bit is off */
2005-11-07 00:58:28 -08:00
2006-12-25 10:19:56 +09:00
if ( ( current_cpu_data . flags & CPU_HAS_MMU_PAGE_ASSOC ) ) {
2005-04-16 15:20:36 -07:00
addr | = MMU_PAGE_ASSOC_BIT ;
ways = 1 ; /* we already know the way .. */
}
for ( i = 0 ; i < ways ; i + + )
2010-01-26 12:58:40 +09:00
__raw_writel ( data , addr + ( i < < 8 ) ) ;
2005-04-16 15:20:36 -07:00
}
2010-04-02 16:13:27 +09:00
void local_flush_tlb_all ( void )
{
unsigned long flags , status ;
/*
* Flush all the TLB .
*
* Write to the MMU control register ' s bit :
* TF - bit for SH - 3 , TI - bit for SH - 4.
* It ' s same position , bit # 2.
*/
local_irq_save ( flags ) ;
status = __raw_readl ( MMUCR ) ;
status | = 0x04 ;
__raw_writel ( status , MMUCR ) ;
ctrl_barrier ( ) ;
local_irq_restore ( flags ) ;
}