#ifndef __ASM_ARM_IRQFLAGS_H #define __ASM_ARM_IRQFLAGS_H #ifdef __KERNEL__ #include /* * CPU interrupt mask handling. */ #ifdef CONFIG_CPU_V7M #define IRQMASK_REG_NAME_R "primask" #define IRQMASK_REG_NAME_W "primask" #define IRQMASK_I_BIT 1 #else #define IRQMASK_REG_NAME_R "cpsr" #define IRQMASK_REG_NAME_W "cpsr_c" #define IRQMASK_I_BIT PSR_I_BIT #endif #if __LINUX_ARM_ARCH__ >= 6 #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) #include #define _local_RET_IP_ (unsigned long)__builtin_return_address(0) #endif /* * Save the current interrupt enable state. */ #define arch_local_save_flags arch_local_save_flags static inline unsigned long arch_local_save_flags(void) { unsigned long flags; asm volatile( " mrs %0, " IRQMASK_REG_NAME_R " @ local_save_flags" : "=r" (flags) : : "memory", "cc"); return flags; } #define arch_local_irq_save arch_local_irq_save static inline unsigned long arch_local_irq_save(void) { unsigned long flags, pc_addr __maybe_unused; asm volatile( " mov %1, pc\n" " mrs %0, " IRQMASK_REG_NAME_R " @ arch_local_irq_save\n" " cpsid i" : "=r" (flags), "=r" (pc_addr) : : "memory", "cc"); #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) avm_simple_profiling_irq_disabled(pc_addr, _local_RET_IP_, flags & PSR_I_BIT, /* old-status */ PSR_I_BIT /* new-status */); #endif/*--- #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) ---*/ return flags; } static inline void arch_local_irq_enable(void) { #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) unsigned long flags = arch_local_save_flags(); avm_simple_profiling_irq_disabled((unsigned long)&&here, _local_RET_IP_, flags & PSR_I_BIT, /* old-status */ 0 /* new-status */); here: #endif/*--- #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) ---*/ asm volatile( " cpsie i @ arch_local_irq_enable" : : : "memory", "cc"); } static inline void arch_local_irq_disable(void) { #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) arch_local_irq_save(); return; #endif/*--- #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) ---*/ asm volatile( " cpsid i @ arch_local_irq_disable" : : : "memory", "cc"); } #define local_fiq_enable() __asm__("cpsie f @ __stf" : : : "memory", "cc") #define local_fiq_disable() __asm__("cpsid f @ __clf" : : : "memory", "cc") #else /* * Save the current interrupt enable state & disable IRQs */ static inline unsigned long arch_local_irq_save(void) { unsigned long flags, temp; asm volatile( " mrs %0, cpsr @ arch_local_irq_save\n" " orr %1, %0, #128\n" " msr cpsr_c, %1" : "=r" (flags), "=r" (temp) : : "memory", "cc"); return flags; } /* * Enable IRQs */ static inline void arch_local_irq_enable(void) { unsigned long temp; asm volatile( " mrs %0, cpsr @ arch_local_irq_enable\n" " bic %0, %0, #128\n" " msr cpsr_c, %0" : "=r" (temp) : : "memory", "cc"); } /* * Disable IRQs */ static inline void arch_local_irq_disable(void) { unsigned long temp; asm volatile( " mrs %0, cpsr @ arch_local_irq_disable\n" " orr %0, %0, #128\n" " msr cpsr_c, %0" : "=r" (temp) : : "memory", "cc"); } /* * Enable FIQs */ #define local_fiq_enable() \ ({ \ unsigned long temp; \ __asm__ __volatile__( \ "mrs %0, cpsr @ stf\n" \ " bic %0, %0, #64\n" \ " msr cpsr_c, %0" \ : "=r" (temp) \ : \ : "memory", "cc"); \ }) /* * Disable FIQs */ #define local_fiq_disable() \ ({ \ unsigned long temp; \ __asm__ __volatile__( \ "mrs %0, cpsr @ clf\n" \ " orr %0, %0, #64\n" \ " msr cpsr_c, %0" \ : "=r" (temp) \ : \ : "memory", "cc"); \ }) #endif /* * restore saved IRQ state */ #define arch_local_irq_restore arch_local_irq_restore static inline void arch_local_irq_restore(unsigned long flags) { #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) avm_simple_profiling_irq_disabled((unsigned long)&&here, _local_RET_IP_, PSR_I_BIT, /* old-status */ (flags & PSR_I_BIT) /* new-status */); here: #endif/*--- #if defined(CONFIG_ENABLE_IRQ_PROFILING_HOOKS) ---*/ asm volatile( " msr " IRQMASK_REG_NAME_W ", %0 @ local_irq_restore" : : "r" (flags) : "memory", "cc"); } static inline int arch_irqs_disabled_flags(unsigned long flags) { return flags & IRQMASK_I_BIT; } #endif /* ifdef __KERNEL__ */ #endif /* ifndef __ASM_ARM_IRQFLAGS_H */