2012-09-06 11:36:56 -04: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 , 95 , 96 , 97 , 98 , 99 , 2003 by Ralf Baechle
* Copyright ( C ) 1996 by Paul M . Antoine
* Copyright ( C ) 1999 Silicon Graphics
* Copyright ( C ) 2000 MIPS Technologies , Inc .
*/
# include <asm/irqflags.h>
# include <asm/hazards.h>
# include <linux/compiler.h>
# include <linux/preempt.h>
# include <linux/export.h>
2013-02-08 18:13:30 +01:00
# include <linux/stringify.h>
2012-09-06 11:36:56 -04:00
2014-11-13 11:54:31 +00:00
# if !defined(CONFIG_CPU_MIPSR2) && !defined(CONFIG_CPU_MIPSR6)
2012-09-06 11:36:56 -04:00
/*
* For cli ( ) we have to insert nops to make sure that the new value
* has actually arrived in the status register before the end of this
* macro .
* R4000 / R4400 need three nops , the R4600 two nops and the R10000 needs
* no nops at all .
*/
/*
* For TX49 , operating only IE bit is not enough .
*
* If mfc0 $ 12 follows store and the mfc0 is last instruction of a
* page and fetching the next instruction causes TLB miss , the result
* of the mfc0 might wrongly contain EXL bit .
*
* ERT - TX49H2 - 027 , ERT - TX49H3 - 012 , ERT - TX49HL3 - 006 , ERT - TX49H4 - 00 8
*
* Workaround : mask EXL bit of the result or place a nop before mfc0 .
*/
2013-02-08 18:13:30 +01:00
notrace void arch_local_irq_disable ( void )
{
preempt_disable ( ) ;
__asm__ __volatile__ (
2012-09-06 11:36:56 -04:00
" .set push \n "
" .set noat \n "
" mfc0 $1,$12 \n "
" ori $1,0x1f \n "
" xori $1,0x1f \n "
" .set noreorder \n "
" mtc0 $1,$12 \n "
2013-02-08 18:13:30 +01:00
" " __stringify ( __irq_disable_hazard ) " \n "
2012-09-06 11:36:56 -04:00
" .set pop \n "
2013-02-08 18:13:30 +01:00
: /* no outputs */
: /* no inputs */
: " memory " ) ;
2012-09-06 11:36:56 -04:00
preempt_enable ( ) ;
}
EXPORT_SYMBOL ( arch_local_irq_disable ) ;
2013-02-08 18:13:30 +01:00
notrace unsigned long arch_local_irq_save ( void )
{
unsigned long flags ;
preempt_disable ( ) ;
__asm__ __volatile__ (
2012-09-06 11:36:56 -04:00
" .set push \n "
" .set reorder \n "
" .set noat \n "
2013-02-08 18:13:30 +01:00
" mfc0 %[flags], $12 \n "
" ori $1, %[flags], 0x1f \n "
2012-09-06 11:36:56 -04:00
" xori $1, 0x1f \n "
" .set noreorder \n "
" mtc0 $1, $12 \n "
2013-02-08 18:13:30 +01:00
" " __stringify ( __irq_disable_hazard ) " \n "
2012-09-06 11:36:56 -04:00
" .set pop \n "
2013-02-08 18:13:30 +01:00
: [ flags ] " =r " ( flags )
: /* no inputs */
: " memory " ) ;
2012-09-06 11:36:56 -04:00
preempt_enable ( ) ;
2013-02-08 18:13:30 +01:00
2012-09-06 11:36:56 -04:00
return flags ;
}
EXPORT_SYMBOL ( arch_local_irq_save ) ;
2013-02-08 18:13:30 +01:00
notrace void arch_local_irq_restore ( unsigned long flags )
{
unsigned long __tmp1 ;
preempt_disable ( ) ;
2012-09-06 11:36:56 -04:00
2013-02-08 18:13:30 +01:00
__asm__ __volatile__ (
2012-09-06 11:36:56 -04:00
" .set push \n "
" .set noreorder \n "
" .set noat \n "
" mfc0 $1, $12 \n "
2013-02-08 18:13:30 +01:00
" andi %[flags], 1 \n "
2012-09-06 11:36:56 -04:00
" ori $1, 0x1f \n "
" xori $1, 0x1f \n "
2013-02-08 18:13:30 +01:00
" or %[flags], $1 \n "
" mtc0 %[flags], $12 \n "
" " __stringify ( __irq_disable_hazard ) " \n "
2012-09-06 11:36:56 -04:00
" .set pop \n "
2013-02-08 18:13:30 +01:00
: [ flags ] " =r " ( __tmp1 )
: " 0 " ( flags )
: " memory " ) ;
2012-09-06 11:36:56 -04:00
preempt_enable ( ) ;
}
EXPORT_SYMBOL ( arch_local_irq_restore ) ;
2016-01-21 21:09:47 +08:00
# endif /* !CONFIG_CPU_MIPSR2 && !CONFIG_CPU_MIPSR6 */