/* ARM AArch32 GCC specific public inline assembler functions and macros */ /* * Copyright (c) 2015, Wind River Systems, Inc. * * SPDX-License-Identifier: Apache-2.0 */ /* Either public functions or macros or invoked by public functions */ #ifndef ZEPHYR_INCLUDE_ARCH_ARM_ASM_INLINE_GCC_H_ #define ZEPHYR_INCLUDE_ARCH_ARM_ASM_INLINE_GCC_H_ /* * The file must not be included directly * Include arch/cpu.h instead */ #ifndef _ASMLANGUAGE #include #include #include #include #if defined(CONFIG_CPU_AARCH32_CORTEX_R) || defined(CONFIG_CPU_AARCH32_CORTEX_A) #include #endif #ifdef __cplusplus extern "C" { #endif /* On ARMv7-M and ARMv8-M Mainline CPUs, this function prevents regular * exceptions (i.e. with interrupt priority lower than or equal to * _EXC_IRQ_DEFAULT_PRIO) from interrupting the CPU. NMI, Faults, SVC, * and Zero Latency IRQs (if supported) may still interrupt the CPU. * * On ARMv6-M and ARMv8-M Baseline CPUs, this function reads the value of * PRIMASK which shows if interrupts are enabled, then disables all interrupts * except NMI. */ static ALWAYS_INLINE unsigned int arch_irq_lock(void) { unsigned int key; #if defined(CONFIG_ARMV6_M_ARMV8_M_BASELINE) #if CONFIG_MP_MAX_NUM_CPUS == 1 || defined(CONFIG_ARMV8_M_BASELINE) key = __get_PRIMASK(); __disable_irq(); #else #error "Cortex-M0 and Cortex-M0+ require SoC specific support for cross core synchronisation." #endif #elif defined(CONFIG_ARMV7_M_ARMV8_M_MAINLINE) key = __get_BASEPRI(); __set_BASEPRI_MAX(_EXC_IRQ_DEFAULT_PRIO); __ISB(); #elif defined(CONFIG_ARMV7_R) || defined(CONFIG_AARCH32_ARMV8_R) \ || defined(CONFIG_ARMV7_A) __asm__ volatile( "mrs %0, cpsr;" "and %0, #" STRINGIFY(I_BIT) ";" "cpsid i;" : "=r" (key) : : "memory", "cc"); #else #error Unknown ARM architecture #endif /* CONFIG_ARMV6_M_ARMV8_M_BASELINE */ return key; } /* On Cortex-M0/M0+, this enables all interrupts if they were not * previously disabled. */ static ALWAYS_INLINE void arch_irq_unlock(unsigned int key) { #if defined(CONFIG_ARMV6_M_ARMV8_M_BASELINE) if (key != 0U) { return; } __enable_irq(); __ISB(); #elif defined(CONFIG_ARMV7_M_ARMV8_M_MAINLINE) __set_BASEPRI(key); __ISB(); #elif defined(CONFIG_ARMV7_R) || defined(CONFIG_AARCH32_ARMV8_R) \ || defined(CONFIG_ARMV7_A) if (key != 0U) { return; } __enable_irq(); #else #error Unknown ARM architecture #endif /* CONFIG_ARMV6_M_ARMV8_M_BASELINE */ } static ALWAYS_INLINE bool arch_irq_unlocked(unsigned int key) { /* This convention works for both PRIMASK and BASEPRI */ return key == 0U; } #ifdef __cplusplus } #endif #endif /* _ASMLANGUAGE */ #endif /* ZEPHYR_INCLUDE_ARCH_ARM_ASM_INLINE_GCC_H_ */