2008-01-30 13:30:33 +01:00
# ifndef _X86_IRQFLAGS_H_
# define _X86_IRQFLAGS_H_
# include <asm/processor-flags.h>
# ifndef __ASSEMBLY__
/*
* Interrupt control :
*/
static inline unsigned long native_save_fl ( void )
{
unsigned long flags ;
2009-08-03 16:33:40 -07:00
/*
2009-08-25 16:47:16 -07:00
* " =rm " is safe here , because " pop " adjusts the stack before
* it evaluates its effective address - - this is part of the
* documented behavior of the " pop " instruction .
2009-08-03 16:33:40 -07:00
*/
2008-03-23 01:02:30 -07:00
asm volatile ( " # __raw_save_flags \n \t "
" pushf ; pop %0 "
2009-08-25 16:47:16 -07:00
: " =rm " ( flags )
2008-03-23 01:02:30 -07:00
: /* no input */
: " memory " ) ;
2008-01-30 13:30:33 +01:00
return flags ;
}
static inline void native_restore_fl ( unsigned long flags )
{
2008-03-23 01:02:30 -07:00
asm volatile ( " push %0 ; popf "
: /* no output */
: " g " ( flags )
: " memory " , " cc " ) ;
2008-01-30 13:30:33 +01:00
}
static inline void native_irq_disable ( void )
{
asm volatile ( " cli " : : : " memory " ) ;
}
static inline void native_irq_enable ( void )
{
asm volatile ( " sti " : : : " memory " ) ;
}
static inline void native_safe_halt ( void )
{
asm volatile ( " sti; hlt " : : : " memory " ) ;
}
static inline void native_halt ( void )
{
asm volatile ( " hlt " : : : " memory " ) ;
}
# endif
# ifdef CONFIG_PARAVIRT
# include <asm/paravirt.h>
# else
# ifndef __ASSEMBLY__
2011-07-01 23:04:36 -04:00
# include <linux/types.h>
2008-01-30 13:30:33 +01:00
2011-07-01 23:04:36 -04:00
static inline notrace unsigned long arch_local_save_flags ( void )
2008-01-30 13:30:33 +01:00
{
return native_save_fl ( ) ;
}
2011-07-01 23:04:36 -04:00
static inline notrace void arch_local_irq_restore ( unsigned long flags )
2008-01-30 13:30:33 +01:00
{
native_restore_fl ( flags ) ;
}
2011-07-01 23:04:36 -04:00
static inline notrace void arch_local_irq_disable ( void )
2008-01-30 13:30:33 +01:00
{
native_irq_disable ( ) ;
}
2011-07-01 23:04:36 -04:00
static inline notrace void arch_local_irq_enable ( void )
2008-01-30 13:30:33 +01:00
{
native_irq_enable ( ) ;
}
/*
* Used in the idle loop ; sti takes one instruction cycle
* to complete :
*/
2010-10-07 14:08:55 +01:00
static inline void arch_safe_halt ( void )
2008-01-30 13:30:33 +01:00
{
native_safe_halt ( ) ;
}
/*
* Used when interrupts are already enabled or to
* shutdown the processor :
*/
static inline void halt ( void )
{
native_halt ( ) ;
}
/*
* For spinlocks , etc :
*/
2011-07-01 23:04:36 -04:00
static inline notrace unsigned long arch_local_irq_save ( void )
2008-01-30 13:30:33 +01:00
{
2010-10-07 14:08:55 +01:00
unsigned long flags = arch_local_save_flags ( ) ;
arch_local_irq_disable ( ) ;
2008-01-30 13:30:33 +01:00
return flags ;
}
# else
# define ENABLE_INTERRUPTS(x) sti
# define DISABLE_INTERRUPTS(x) cli
# ifdef CONFIG_X86_64
2008-06-27 12:04:03 -07:00
# define SWAPGS swapgs
/*
* Currently paravirt can ' t handle swapgs nicely when we
* don ' t have a stack we can rely on ( such as a user space
* stack ) . So we either find a way around these or just fault
* and emulate if a guest tries to call swapgs directly .
*
* Either way , this is a good way to document that we don ' t
* have a reliable stack . x86_64 only .
*/
2008-06-25 00:19:30 -04:00
# define SWAPGS_UNSAFE_STACK swapgs
2008-06-27 12:04:03 -07:00
# define PARAVIRT_ADJUST_EXCEPTION_FRAME /* */
2008-01-30 13:30:33 +01:00
# define INTERRUPT_RETURN iretq
2008-06-25 00:19:28 -04:00
# define USERGS_SYSRET64 \
swapgs ; \
sysretq ;
# define USERGS_SYSRET32 \
swapgs ; \
sysretl
# define ENABLE_INTERRUPTS_SYSEXIT32 \
swapgs ; \
sti ; \
sysexit
2008-01-30 13:30:33 +01:00
# else
# define INTERRUPT_RETURN iret
2008-06-25 00:19:26 -04:00
# define ENABLE_INTERRUPTS_SYSEXIT sti; sysexit
2008-01-30 13:30:33 +01:00
# define GET_CR0_INTO_EAX movl %cr0, %eax
# endif
# endif /* __ASSEMBLY__ */
# endif /* CONFIG_PARAVIRT */
# ifndef __ASSEMBLY__
2010-10-07 14:08:55 +01:00
static inline int arch_irqs_disabled_flags ( unsigned long flags )
2008-01-30 13:30:33 +01:00
{
return ! ( flags & X86_EFLAGS_IF ) ;
}
2010-10-07 14:08:55 +01:00
static inline int arch_irqs_disabled ( void )
2008-01-30 13:30:33 +01:00
{
2010-10-07 14:08:55 +01:00
unsigned long flags = arch_local_save_flags ( ) ;
2008-01-30 13:30:33 +01:00
2010-10-07 14:08:55 +01:00
return arch_irqs_disabled_flags ( flags ) ;
2008-01-30 13:30:33 +01:00
}
2007-10-11 11:20:03 +02:00
# else
2008-01-30 13:30:33 +01:00
# ifdef CONFIG_X86_64
# define ARCH_LOCKDEP_SYS_EXIT call lockdep_sys_exit_thunk
# define ARCH_LOCKDEP_SYS_EXIT_IRQ \
TRACE_IRQS_ON ; \
sti ; \
SAVE_REST ; \
LOCKDEP_SYS_EXIT ; \
RESTORE_REST ; \
cli ; \
TRACE_IRQS_OFF ;
# else
# define ARCH_LOCKDEP_SYS_EXIT \
pushl % eax ; \
pushl % ecx ; \
pushl % edx ; \
call lockdep_sys_exit ; \
popl % edx ; \
popl % ecx ; \
popl % eax ;
# define ARCH_LOCKDEP_SYS_EXIT_IRQ
# endif
# ifdef CONFIG_TRACE_IRQFLAGS
2008-05-12 21:20:42 +02:00
# define TRACE_IRQS_ON call trace_hardirqs_on_thunk;
# define TRACE_IRQS_OFF call trace_hardirqs_off_thunk;
2008-01-30 13:30:33 +01:00
# else
# define TRACE_IRQS_ON
# define TRACE_IRQS_OFF
# endif
# ifdef CONFIG_DEBUG_LOCK_ALLOC
# define LOCKDEP_SYS_EXIT ARCH_LOCKDEP_SYS_EXIT
# define LOCKDEP_SYS_EXIT_IRQ ARCH_LOCKDEP_SYS_EXIT_IRQ
# else
# define LOCKDEP_SYS_EXIT
# define LOCKDEP_SYS_EXIT_IRQ
# endif
# endif /* __ASSEMBLY__ */
2007-10-11 11:20:03 +02:00
# endif