Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2015 Intel corporation |
| 3 | * |
David B. Kinder | ac74d8b | 2017-01-18 17:01:01 -0800 | [diff] [blame] | 4 | * SPDX-License-Identifier: Apache-2.0 |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 5 | */ |
| 6 | |
| 7 | /** |
| 8 | * @file |
| 9 | * @brief Public interface for configuring interrupts |
| 10 | */ |
| 11 | #ifndef _IRQ_H_ |
| 12 | #define _IRQ_H_ |
| 13 | |
| 14 | /* Pull in the arch-specific implementations */ |
| 15 | #include <arch/cpu.h> |
| 16 | |
Andrew Boie | 15800ce | 2016-03-28 14:49:37 -0700 | [diff] [blame] | 17 | #ifndef _ASMLANGUAGE |
Andrew Boie | 0cf6808 | 2017-08-14 13:25:36 -0700 | [diff] [blame^] | 18 | #include <toolchain.h> |
Andrew Boie | 15800ce | 2016-03-28 14:49:37 -0700 | [diff] [blame] | 19 | |
| 20 | #ifdef __cplusplus |
| 21 | extern "C" { |
| 22 | #endif |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 23 | |
| 24 | /** |
| 25 | * @defgroup isr_apis Interrupt Service Routine APIs |
| 26 | * @ingroup kernel_apis |
| 27 | * @{ |
| 28 | */ |
| 29 | |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 30 | /** |
Allan Stephens | fb513eb | 2016-11-16 16:44:06 -0500 | [diff] [blame] | 31 | * @brief Initialize an interrupt handler. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 32 | * |
Allan Stephens | fb513eb | 2016-11-16 16:44:06 -0500 | [diff] [blame] | 33 | * This routine initializes an interrupt handler for an IRQ. The IRQ must be |
| 34 | * subsequently enabled before the interrupt handler begins servicing |
| 35 | * interrupts. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 36 | * |
Allan Stephens | fb513eb | 2016-11-16 16:44:06 -0500 | [diff] [blame] | 37 | * @warning |
| 38 | * Although this routine is invoked at run-time, all of its arguments must be |
| 39 | * computable by the compiler at build time. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 40 | * |
Allan Stephens | fb513eb | 2016-11-16 16:44:06 -0500 | [diff] [blame] | 41 | * @param irq_p IRQ line number. |
| 42 | * @param priority_p Interrupt priority. |
| 43 | * @param isr_p Address of interrupt service routine. |
| 44 | * @param isr_param_p Parameter passed to interrupt service routine. |
| 45 | * @param flags_p Architecture-specific IRQ configuration flags.. |
| 46 | * |
| 47 | * @return Interrupt vector assigned to this interrupt. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 48 | */ |
| 49 | #define IRQ_CONNECT(irq_p, priority_p, isr_p, isr_param_p, flags_p) \ |
| 50 | _ARCH_IRQ_CONNECT(irq_p, priority_p, isr_p, isr_param_p, flags_p) |
| 51 | |
| 52 | /** |
Andrew Boie | 4fc9606 | 2017-01-18 13:06:59 -0800 | [diff] [blame] | 53 | * @brief Initialize a 'direct' interrupt handler. |
| 54 | * |
| 55 | * This routine initializes an interrupt handler for an IRQ. The IRQ must be |
| 56 | * subsequently enabled via irq_enable() before the interrupt handler begins |
| 57 | * servicing interrupts. |
| 58 | * |
| 59 | * These ISRs are designed for performance-critical interrupt handling and do |
| 60 | * not go through common interrupt handling code. They must be implemented in |
| 61 | * such a way that it is safe to put them directly in the vector table. For |
| 62 | * ISRs written in C, The ISR_DIRECT_DECLARE() macro will do this |
David B. Kinder | 8b986d7 | 2017-04-18 15:56:26 -0700 | [diff] [blame] | 63 | * automatically. For ISRs written in assembly it is entirely up to the |
Andrew Boie | 4fc9606 | 2017-01-18 13:06:59 -0800 | [diff] [blame] | 64 | * developer to ensure that the right steps are taken. |
| 65 | * |
| 66 | * This type of interrupt currently has a few limitations compared to normal |
| 67 | * Zephyr interrupts: |
| 68 | * - No parameters are passed to the ISR. |
| 69 | * - No stack switch is done, the ISR will run on the interrupted context's |
| 70 | * stack, unless the architecture automatically does the stack switch in HW. |
| 71 | * - Interrupt locking state is unchanged from how the HW sets it when the ISR |
| 72 | * runs. On arches that enter ISRs with interrupts locked, they will remain |
| 73 | * locked. |
| 74 | * - Scheduling decisions are now optional, controlled by the return value of |
| 75 | * ISRs implemented with the ISR_DIRECT_DECLARE() macro |
| 76 | * - The call into the OS to exit power management idle state is now optional. |
| 77 | * Normal interrupts always do this before the ISR is run, but when it runs |
| 78 | * is now controlled by the placement of a ISR_DIRECT_PM() macro, or omitted |
| 79 | * entirely. |
| 80 | * |
| 81 | * @warning |
| 82 | * Although this routine is invoked at run-time, all of its arguments must be |
| 83 | * computable by the compiler at build time. |
| 84 | * |
| 85 | * @param irq_p IRQ line number. |
| 86 | * @param priority_p Interrupt priority. |
| 87 | * @param isr_p Address of interrupt service routine. |
| 88 | * @param flags_p Architecture-specific IRQ configuration flags. |
| 89 | * |
| 90 | * @return Interrupt vector assigned to this interrupt. |
| 91 | */ |
| 92 | #define IRQ_DIRECT_CONNECT(irq_p, priority_p, isr_p, flags_p) \ |
| 93 | _ARCH_IRQ_DIRECT_CONNECT(irq_p, priority_p, isr_p, flags_p) |
| 94 | |
| 95 | /** |
| 96 | * @brief Common tasks before executing the body of an ISR |
| 97 | * |
| 98 | * This macro must be at the beginning of all direct interrupts and performs |
| 99 | * minimal architecture-specific tasks before the ISR itself can run. It takes |
| 100 | * no arguments and has no return value. |
| 101 | */ |
| 102 | #define ISR_DIRECT_HEADER() _ARCH_ISR_DIRECT_HEADER() |
| 103 | |
| 104 | /** |
| 105 | * @brief Common tasks before exiting the body of an ISR |
| 106 | * |
| 107 | * This macro must be at the end of all direct interrupts and performs |
| 108 | * minimal architecture-specific tasks like EOI. It has no return value. |
| 109 | * |
| 110 | * In a normal interrupt, a check is done at end of interrupt to invoke |
| 111 | * _Swap() logic if the current thread is preemptible and there is another |
| 112 | * thread ready to run in the kernel's ready queue cache. This is now optional |
| 113 | * and controlled by the check_reschedule argument. If unsure, set to nonzero. |
| 114 | * On systems that do stack switching and nested interrupt tracking in software, |
| 115 | * _Swap() should only be called if this was a non-nested interrupt. |
| 116 | * |
| 117 | * @param check_reschedule If nonzero, additionally invoke scheduling logic |
| 118 | */ |
| 119 | #define ISR_DIRECT_FOOTER(check_reschedule) \ |
| 120 | _ARCH_ISR_DIRECT_FOOTER(check_reschedule) |
| 121 | |
| 122 | /** |
| 123 | * @brief Perform power management idle exit logic |
| 124 | * |
| 125 | * This macro may optionally be invoked somewhere in between IRQ_DIRECT_HEADER() |
| 126 | * and IRQ_DIRECT_FOOTER() invocations. It performs tasks necessary to |
| 127 | * exit power management idle state. It takes no parameters and returns no |
| 128 | * arguments. It may be omitted, but be careful! |
| 129 | */ |
| 130 | #define ISR_DIRECT_PM() _ARCH_ISR_DIRECT_PM() |
| 131 | |
| 132 | /** |
| 133 | * @brief Helper macro to declare a direct interrupt service routine. |
| 134 | * |
| 135 | * This will declare the function in a proper way and automatically include |
| 136 | * the ISR_DIRECT_FOOTER() and ISR_DIRECT_HEADER() macros. The function should |
| 137 | * return nonzero status if a scheduling decision should potentially be made. |
| 138 | * See ISR_DIRECT_FOOTER() for more details on the scheduling decision. |
| 139 | * |
| 140 | * For architectures that support 'regular' and 'fast' interrupt types, where |
| 141 | * these interrupt types require different assembly language handling of |
| 142 | * registers by the ISR, this will always generate code for the 'fast' |
| 143 | * interrupt type. |
| 144 | * |
| 145 | * Example usage: |
| 146 | * |
| 147 | * ISR_DIRECT_DECLARE(my_isr) |
| 148 | * { |
| 149 | * bool done = do_stuff(); |
| 150 | * ISR_DIRECT_PM(); <-- done after do_stuff() due to latency concerns |
| 151 | * if (!done) { |
| 152 | * return 0; <-- Don't bother checking if we have to _Swap() |
| 153 | * } |
| 154 | * k_sem_give(some_sem); |
| 155 | * return 1; |
| 156 | * } |
| 157 | * |
| 158 | * @param name symbol name of the ISR |
| 159 | */ |
| 160 | #define ISR_DIRECT_DECLARE(name) _ARCH_ISR_DIRECT_DECLARE(name) |
| 161 | |
| 162 | /** |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 163 | * @brief Lock interrupts. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 164 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 165 | * This routine disables all interrupts on the CPU. It returns an unsigned |
| 166 | * integer "lock-out key", which is an architecture-dependent indicator of |
| 167 | * whether interrupts were locked prior to the call. The lock-out key must be |
| 168 | * passed to irq_unlock() to re-enable interrupts. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 169 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 170 | * This routine can be called recursively, as long as the caller keeps track |
| 171 | * of each lock-out key that is generated. Interrupts are re-enabled by |
| 172 | * passing each of the keys to irq_unlock() in the reverse order they were |
| 173 | * acquired. (That is, each call to irq_lock() must be balanced by |
| 174 | * a corresponding call to irq_unlock().) |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 175 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 176 | * @note |
| 177 | * This routine can be called by ISRs or by threads. If it is called by a |
| 178 | * thread, the interrupt lock is thread-specific; this means that interrupts |
| 179 | * remain disabled only while the thread is running. If the thread performs an |
| 180 | * operation that allows another thread to run (for example, giving a semaphore |
| 181 | * or sleeping for N milliseconds), the interrupt lock no longer applies and |
| 182 | * interrupts may be re-enabled while other processing occurs. When the thread |
| 183 | * once again becomes the current thread, the kernel re-establishes its |
| 184 | * interrupt lock; this ensures the thread won't be interrupted until it has |
| 185 | * explicitly released the interrupt lock it established. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 186 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 187 | * @warning |
| 188 | * The lock-out key should never be used to manually re-enable interrupts |
| 189 | * or to inspect or manipulate the contents of the CPU's interrupt bits. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 190 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 191 | * @return Lock-out key. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 192 | */ |
| 193 | #define irq_lock() _arch_irq_lock() |
| 194 | |
| 195 | /** |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 196 | * @brief Unlock interrupts. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 197 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 198 | * This routine reverses the effect of a previous call to irq_lock() using |
| 199 | * the associated lock-out key. The caller must call the routine once for |
| 200 | * each time it called irq_lock(), supplying the keys in the reverse order |
| 201 | * they were acquired, before interrupts are enabled. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 202 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 203 | * @note Can be called by ISRs. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 204 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 205 | * @param key Lock-out key generated by irq_lock(). |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 206 | * |
| 207 | * @return N/A |
| 208 | */ |
| 209 | #define irq_unlock(key) _arch_irq_unlock(key) |
| 210 | |
| 211 | /** |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 212 | * @brief Enable an IRQ. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 213 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 214 | * This routine enables interrupts from source @a irq. |
| 215 | * |
| 216 | * @param irq IRQ line. |
| 217 | * |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 218 | * @return N/A |
| 219 | */ |
| 220 | #define irq_enable(irq) _arch_irq_enable(irq) |
| 221 | |
| 222 | /** |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 223 | * @brief Disable an IRQ. |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 224 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 225 | * This routine disables interrupts from source @a irq. |
| 226 | * |
| 227 | * @param irq IRQ line. |
| 228 | * |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 229 | * @return N/A |
| 230 | */ |
| 231 | #define irq_disable(irq) _arch_irq_disable(irq) |
| 232 | |
Vinayak Chettimada | b33aaa4 | 2016-09-10 11:53:49 +0200 | [diff] [blame] | 233 | /** |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 234 | * @brief Get IRQ enable state. |
Vinayak Chettimada | b33aaa4 | 2016-09-10 11:53:49 +0200 | [diff] [blame] | 235 | * |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 236 | * This routine indicates if interrupts from source @a irq are enabled. |
| 237 | * |
| 238 | * @param irq IRQ line. |
| 239 | * |
Vinayak Chettimada | b33aaa4 | 2016-09-10 11:53:49 +0200 | [diff] [blame] | 240 | * @return interrupt enable state, true or false |
| 241 | */ |
| 242 | #define irq_is_enabled(irq) _arch_irq_is_enabled(irq) |
| 243 | |
Allan Stephens | c98da84 | 2016-11-11 15:45:03 -0500 | [diff] [blame] | 244 | /** |
| 245 | * @} |
| 246 | */ |
| 247 | |
Andrew Boie | 15800ce | 2016-03-28 14:49:37 -0700 | [diff] [blame] | 248 | #ifdef __cplusplus |
| 249 | } |
| 250 | #endif |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 251 | |
Andrew Boie | 15800ce | 2016-03-28 14:49:37 -0700 | [diff] [blame] | 252 | #endif /* ASMLANGUAGE */ |
Andrew Boie | e444825 | 2016-02-25 13:21:02 -0800 | [diff] [blame] | 253 | #endif /* _IRQ_H_ */ |