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 .
*
2006-03-13 16:16:29 +00:00
* Copyright ( C ) 2003 , 2004 Ralf Baechle < ralf @ linux - mips . org >
* Copyright ( C ) MIPS Technologies , Inc .
* written by Ralf Baechle < ralf @ linux - mips . org >
2005-04-16 15:20:36 -07:00
*/
# ifndef _ASM_HAZARDS_H
# define _ASM_HAZARDS_H
# ifdef __ASSEMBLY__
. macro _ssnop
sll $ 0 , $ 0 , 1
. endm
. macro _ehb
sll $ 0 , $ 0 , 3
. endm
/*
* RM9000 hazards . When the JTLB is updated by tlbwi or tlbwr , a subsequent
* use of the JTLB for instructions should not occur for 4 cpu cycles and use
* for data translations should not occur for 3 cpu cycles .
*/
# ifdef CONFIG_CPU_RM9000
. macro mtc0_tlbw_hazard
. set push
. set mips32
_ssnop ; _ssnop ; _ssnop ; _ssnop
. set pop
. endm
. macro tlbw_eret_hazard
. set push
. set mips32
_ssnop ; _ssnop ; _ssnop ; _ssnop
. set pop
. endm
# else
/*
* The taken branch will result in a two cycle penalty for the two killed
* instructions on R4000 / R4400 . Other processors only have a single cycle
* hazard so this is nice trick to have an optimal code for a range of
* processors .
*/
. macro mtc0_tlbw_hazard
b . + 8
. endm
. macro tlbw_eret_hazard
. endm
# endif
/*
* mtc0 - > mfc0 hazard
* The 24 K has a 2 cycle mtc0 / mfc0 execution hazard .
* It is a MIPS32R2 processor so ehb will clear the hazard .
*/
# ifdef CONFIG_CPU_MIPSR2
/*
* Use a macro for ehb unless explicit support for MIPSR2 is enabled
*/
2006-06-29 20:06:53 +01:00
# define irq_enable_hazard \
2005-04-16 15:20:36 -07:00
_ehb
2006-06-29 20:06:53 +01:00
# define irq_disable_hazard \
2005-04-16 15:20:36 -07:00
_ehb
2006-03-13 16:16:29 +00:00
# elif defined(CONFIG_CPU_R10000) || defined(CONFIG_CPU_RM9000)
2005-04-16 15:20:36 -07:00
/*
* R10000 rocks - all hazards handled in hardware , so this becomes a nobrainer .
*/
# define irq_enable_hazard
# define irq_disable_hazard
# else
/*
* Classic MIPS needs 1 - 3 nops or ssnops
*/
# define irq_enable_hazard
# define irq_disable_hazard \
_ssnop ; _ssnop ; _ssnop
# endif
# else /* __ASSEMBLY__ */
__asm__ (
2006-03-13 16:16:29 +00:00
" .macro _ssnop \n "
" sll $0, $0, 1 \n "
" .endm \n "
" \n "
" .macro _ehb \n "
" sll $0, $0, 3 \n "
" .endm \n " ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_CPU_RM9000
2005-03-02 19:18:46 +00:00
2005-04-16 15:20:36 -07:00
/*
* RM9000 hazards . When the JTLB is updated by tlbwi or tlbwr , a subsequent
* use of the JTLB for instructions should not occur for 4 cpu cycles and use
* for data translations should not occur for 3 cpu cycles .
*/
# define mtc0_tlbw_hazard() \
__asm__ __volatile__ ( \
2006-03-13 16:16:29 +00:00
" .set mips32 \n " \
" _ssnop \n " \
" _ssnop \n " \
" _ssnop \n " \
" _ssnop \n " \
" .set mips0 \n " )
2005-04-16 15:20:36 -07:00
# define tlbw_use_hazard() \
__asm__ __volatile__ ( \
2006-03-13 16:16:29 +00:00
" .set mips32 \n " \
" _ssnop \n " \
" _ssnop \n " \
" _ssnop \n " \
" _ssnop \n " \
" .set mips0 \n " )
2005-03-01 18:12:06 +00:00
2005-04-16 15:20:36 -07:00
# else
/*
* Overkill warning . . .
*/
# define mtc0_tlbw_hazard() \
__asm__ __volatile__ ( \
2006-03-13 16:16:29 +00:00
" .set noreorder \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" .set reorder \n " )
2005-04-16 15:20:36 -07:00
# define tlbw_use_hazard() \
__asm__ __volatile__ ( \
2006-03-13 16:16:29 +00:00
" .set noreorder \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" nop \n " \
" .set reorder \n " )
2005-04-16 15:20:36 -07:00
# endif
/*
2005-07-14 13:25:05 +00:00
* Interrupt enable / disable hazards
* Some processors have hazards when modifying
* the status register to change the interrupt state
2005-04-16 15:20:36 -07:00
*/
# ifdef CONFIG_CPU_MIPSR2
2005-07-14 13:25:05 +00:00
2006-03-13 16:16:29 +00:00
__asm__ ( " .macro irq_enable_hazard \n "
" _ehb \n "
" .endm \n "
" \n "
" .macro irq_disable_hazard \n "
" _ehb \n "
" .endm \n " ) ;
2005-04-16 15:20:36 -07:00
2006-03-13 16:16:29 +00:00
# elif defined(CONFIG_CPU_R10000) || defined(CONFIG_CPU_RM9000)
2005-04-16 15:20:36 -07:00
/*
* R10000 rocks - all hazards handled in hardware , so this becomes a nobrainer .
*/
__asm__ (
2006-03-13 16:16:29 +00:00
" .macro irq_enable_hazard \n "
" .endm \n "
" \n "
" .macro irq_disable_hazard \n "
" .endm \n " ) ;
2005-03-01 18:12:06 +00:00
2005-04-16 15:20:36 -07:00
# else
/*
* Default for classic MIPS processors . Assume worst case hazards but don ' t
* care about the irq_enable_hazard - sooner or later the hardware will
* enable it and we don ' t care when exactly .
*/
__asm__ (
2006-03-13 16:16:29 +00:00
" # \n "
" # There is a hazard but we do not care \n "
" # \n "
" .macro \t irq_enable_hazard \n "
" .endm \n "
" \n "
" .macro \t irq_disable_hazard \n "
" _ssnop \n "
" _ssnop \n "
" _ssnop \n "
" .endm \n " ) ;
2005-04-16 15:20:36 -07:00
2006-03-13 16:16:29 +00:00
# endif
# define irq_enable_hazard() \
__asm__ __volatile__ ( " irq_enable_hazard " )
2005-04-16 15:20:36 -07:00
# define irq_disable_hazard() \
2006-03-13 16:16:29 +00:00
__asm__ __volatile__ ( " irq_disable_hazard " )
2005-04-16 15:20:36 -07:00
2006-03-13 16:16:29 +00:00
/*
* Back - to - back hazards -
*
* What is needed to separate a move to cp0 from a subsequent read from the
* same cp0 register ?
*/
# ifdef CONFIG_CPU_MIPSR2
__asm__ ( " .macro back_to_back_c0_hazard \n "
" _ehb \n "
" .endm \n " ) ;
# elif defined(CONFIG_CPU_R10000) || defined(CONFIG_CPU_RM9000) || \
defined ( CONFIG_CPU_SB1 )
__asm__ ( " .macro back_to_back_c0_hazard \n "
" .endm \n " ) ;
# else
__asm__ ( " .macro back_to_back_c0_hazard \n "
" .set noreorder \n "
" _ssnop \n "
" _ssnop \n "
" _ssnop \n "
" .set reorder \n "
" .endm " ) ;
2005-03-01 18:12:06 +00:00
2005-04-16 15:20:36 -07:00
# endif
2006-03-13 16:16:29 +00:00
# define back_to_back_c0_hazard() \
__asm__ __volatile__ ( " back_to_back_c0_hazard " )
/*
* Instruction execution hazard
*/
2005-10-07 16:58:15 +01:00
# ifdef CONFIG_CPU_MIPSR2
2005-12-22 13:41:29 +01:00
/*
* gcc has a tradition of misscompiling the previous construct using the
* address of a label as argument to inline assembler . Gas otoh has the
* annoying difference between la and dla which are only usable for 32 - bit
* rsp . 64 - bit code , so can ' t be used without conditional compilation .
* The alterantive is switching the assembler to 64 - bit code which happens
* to work right even for 32 - bit code . . .
*/
2005-07-12 18:35:38 +00:00
# define instruction_hazard() \
do { \
2005-12-22 13:41:29 +01:00
unsigned long tmp ; \
\
2005-07-12 18:35:38 +00:00
__asm__ __volatile__ ( \
2005-12-22 13:41:29 +01:00
" .set mips64r2 \n " \
" dla %0, 1f \n " \
2005-07-12 18:35:38 +00:00
" jr.hb %0 \n " \
2005-12-22 13:41:29 +01:00
" .set mips0 \n " \
" 1: \n " \
: " =r " ( tmp ) ) ; \
2005-07-12 18:35:38 +00:00
} while ( 0 )
# else
# define instruction_hazard() do { } while (0)
# endif
2006-04-05 09:45:45 +01:00
extern void mips_ihb ( void ) ;
2005-04-16 15:20:36 -07:00
# endif /* __ASSEMBLY__ */
# endif /* _ASM_HAZARDS_H */