2019-05-28 20:10:14 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-17 02:20:36 +04:00
/*
* arch / sh / mm / tlb - sh4 . c
*
* SH - 4 specific TLB operations
*
* Copyright ( C ) 1999 Niibe Yutaka
2007-09-21 06:55:03 +04:00
* Copyright ( C ) 2002 - 2007 Paul Mundt
2005-04-17 02:20:36 +04:00
*/
2007-03-05 13:46:47 +03:00
# include <linux/kernel.h>
# include <linux/mm.h>
2007-09-21 06:55:03 +04:00
# include <linux/io.h>
2005-04-17 02:20:36 +04:00
# include <asm/mmu_context.h>
2007-03-05 13:46:47 +03:00
# include <asm/cacheflush.h>
2009-07-28 19:12:17 +04:00
void __update_tlb ( struct vm_area_struct * vma , unsigned long address , pte_t pte )
2007-03-05 13:46:47 +03:00
{
2009-07-28 19:12:17 +04:00
unsigned long flags , pteval , vpn ;
2007-03-05 13:46:47 +03:00
2009-07-28 19:12:17 +04:00
/*
* Handle debugger faulting in for debugee .
*/
2009-07-29 17:06:58 +04:00
if ( vma & & current - > active_mm ! = vma - > vm_mm )
2007-03-05 13:46:47 +03:00
return ;
local_irq_save ( flags ) ;
/* Set PTEH register */
vpn = ( address & MMU_VPN_MASK ) | get_asid ( ) ;
2010-01-26 06:58:40 +03:00
__raw_writel ( vpn , MMU_PTEH ) ;
2007-03-05 13:46:47 +03:00
2007-09-21 06:55:03 +04:00
pteval = pte . pte_low ;
2007-03-05 13:46:47 +03:00
/* Set PTEA register */
2007-09-21 06:55:03 +04:00
# ifdef CONFIG_X2TLB
/*
* For the extended mode TLB this is trivial , only the ESZ and
* EPR bits need to be written out to PTEA , with the remainder of
* the protection bits ( with the exception of the compat - mode SZ
* and PR bits , which are cleared ) being written out in PTEL .
*/
2010-01-26 06:58:40 +03:00
__raw_writel ( pte . pte_high , MMU_PTEA ) ;
2007-09-21 06:55:03 +04:00
# else
2009-08-20 08:27:44 +04:00
if ( cpu_data - > flags & CPU_HAS_PTEA ) {
/* The last 3 bits and the first one of pteval contains
* the PTEA timing control and space attribute bits
*/
2010-01-26 06:58:40 +03:00
__raw_writel ( copy_ptea_attributes ( pteval ) , MMU_PTEA ) ;
2009-08-20 08:27:44 +04:00
}
2007-09-21 06:55:03 +04:00
# endif
2007-03-05 13:46:47 +03:00
/* Set PTEL register */
pteval & = _PAGE_FLAGS_HARDWARE_MASK ; /* drop software flags */
2007-07-31 12:07:28 +04:00
# ifdef CONFIG_CACHE_WRITETHROUGH
2007-03-05 13:46:47 +03:00
pteval | = _PAGE_WT ;
# endif
/* conveniently, we want all the software flags to be 0 anyway */
2010-01-26 06:58:40 +03:00
__raw_writel ( pteval , MMU_PTEL ) ;
2007-03-05 13:46:47 +03:00
/* Load the TLB */
asm volatile ( " ldtlb " : /* no output */ : /* no input */ : " memory " ) ;
local_irq_restore ( flags ) ;
}
2005-04-17 02:20:36 +04:00
2010-01-21 10:05:25 +03:00
void local_flush_tlb_one ( unsigned long asid , unsigned long page )
2005-04-17 02:20:36 +04:00
{
unsigned long addr , data ;
/*
* 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_UTLB_ADDRESS_ARRAY | MMU_PAGE_ASSOC_BIT ;
data = page | asid ; /* VALID bit is off */
2007-11-30 11:06:36 +03:00
jump_to_uncached ( ) ;
2010-01-26 06:58:40 +03:00
__raw_writel ( data , addr ) ;
2007-11-30 11:06:36 +03:00
back_to_cached ( ) ;
2005-04-17 02:20:36 +04:00
}
2010-04-02 11:13:27 +04:00
void local_flush_tlb_all ( void )
{
unsigned long flags , status ;
int i ;
/*
* Flush all the TLB .
*/
local_irq_save ( flags ) ;
jump_to_uncached ( ) ;
status = __raw_readl ( MMUCR ) ;
status = ( ( status & MMUCR_URB ) > > MMUCR_URB_SHIFT ) ;
if ( status = = 0 )
status = MMUCR_URB_NENTRIES ;
for ( i = 0 ; i < status ; i + + )
__raw_writel ( 0x0 , MMU_UTLB_ADDRESS_ARRAY | ( i < < 8 ) ) ;
for ( i = 0 ; i < 4 ; i + + )
__raw_writel ( 0x0 , MMU_ITLB_ADDRESS_ARRAY | ( i < < 8 ) ) ;
back_to_cached ( ) ;
ctrl_barrier ( ) ;
local_irq_restore ( flags ) ;
}