/* * Copyright (C) 2014-2015 Freie UniversitÀt Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @defgroup cpu_cortexm_common ARM Cortex-M common * @ingroup cpu * @brief Common implementations and headers for Cortex-M family based * micro-controllers * @{ * * @file * @brief Basic definitions for the Cortex-M common module * * When ever you want to do something hardware related, that is accessing MCUs * registers, just include this file. It will then make sure that the MCU * specific headers are included. * * @author Stefan Pfeiffer * @author Hauke Petersen * @author Joakim NohlgÄrd * * @todo remove include irq.h once core was adjusted */ #ifndef CPU_H #define CPU_H #include #include "irq.h" #include "sched.h" #include "thread.h" #include "cpu_conf.h" #ifdef __cplusplus extern "C" { #endif /** * @brief Interrupt stack canary value * * @note 0xe7fe is the ARM Thumb machine code equivalent of asm("bl #-2\n") or * 'while (1);', i.e. an infinite loop. * @internal */ #define STACK_CANARY_WORD (0xE7FEE7FEu) /** * @brief All Cortex-m-based CPUs provide pm_set_lowest * * The pm_set_lowest is provided either by the pm_layered module if used, or * alternatively as fallback by the cortexm's own implementation. */ #define PROVIDES_PM_SET_LOWEST /** * @brief Initialization of the CPU */ void cpu_init(void); /** * @brief Initialize Cortex-M specific core parts of the CPU */ void cortexm_init(void); /** * @brief Prints the current content of the link register (lr) */ static inline void cpu_print_last_instruction(void) { uint32_t *lr_ptr; __asm__ __volatile__("mov %0, lr" : "=r"(lr_ptr)); printf("%p\n", (void*) lr_ptr); } /** * @brief Put the CPU into the 'wait for event' sleep mode * * This function is meant to be used for short periods of time, where it is not * feasible to switch to the idle thread and back. */ static inline void cortexm_sleep_until_event(void) { __WFE(); } /** * @brief Put the CPU into (deep) sleep mode, using the `WFI` instruction * * @param[in] deep !=0 for deep sleep, 0 for light sleep */ static inline void cortexm_sleep(int deep) { if (deep) { SCB->SCR |= (SCB_SCR_SLEEPDEEP_Msk); } else { SCB->SCR &= ~(SCB_SCR_SLEEPDEEP_Msk); } /* ensure that all memory accesses have completed and trigger sleeping */ unsigned state = irq_disable(); __DSB(); __WFI(); #if defined(CPU_MODEL_STM32L152RE) /* STM32L152RE crashes without this __NOP(). See #8518. */ __NOP(); #endif irq_restore(state); } /** * @brief Trigger a conditional context scheduler run / context switch * * This function is supposed to be called in the end of each ISR. */ static inline void cortexm_isr_end(void) { if (sched_context_switch_request) { thread_yield(); } } #ifdef __cplusplus } #endif #endif /* CPU_H */ /** @} */