blob: 29da432af0907d4839af21d878d9428d53f16f57 [file] [log] [blame]
Andrew Boiee4448252016-02-25 13:21:02 -08001/*
2 * Copyright (c) 2015 Intel corporation
3 *
David B. Kinderac74d8b2017-01-18 17:01:01 -08004 * SPDX-License-Identifier: Apache-2.0
Andrew Boiee4448252016-02-25 13:21:02 -08005 */
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 Boie15800ce2016-03-28 14:49:37 -070017#ifndef _ASMLANGUAGE
Andrew Boie0cf68082017-08-14 13:25:36 -070018#include <toolchain.h>
Andrew Boie15800ce2016-03-28 14:49:37 -070019
20#ifdef __cplusplus
21extern "C" {
22#endif
Allan Stephensc98da842016-11-11 15:45:03 -050023
24/**
25 * @defgroup isr_apis Interrupt Service Routine APIs
26 * @ingroup kernel_apis
27 * @{
28 */
29
Andrew Boiee4448252016-02-25 13:21:02 -080030/**
Allan Stephensfb513eb2016-11-16 16:44:06 -050031 * @brief Initialize an interrupt handler.
Andrew Boiee4448252016-02-25 13:21:02 -080032 *
Allan Stephensfb513eb2016-11-16 16:44:06 -050033 * 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 Boiee4448252016-02-25 13:21:02 -080036 *
Allan Stephensfb513eb2016-11-16 16:44:06 -050037 * @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 Boiee4448252016-02-25 13:21:02 -080040 *
Allan Stephensfb513eb2016-11-16 16:44:06 -050041 * @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 Boiee4448252016-02-25 13:21:02 -080048 */
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 Boie4fc96062017-01-18 13:06:59 -080053 * @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. Kinder8b986d72017-04-18 15:56:26 -070063 * automatically. For ISRs written in assembly it is entirely up to the
Andrew Boie4fc96062017-01-18 13:06:59 -080064 * 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 Stephensc98da842016-11-11 15:45:03 -0500163 * @brief Lock interrupts.
Andrew Boiee4448252016-02-25 13:21:02 -0800164 *
Allan Stephensc98da842016-11-11 15:45:03 -0500165 * 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 Boiee4448252016-02-25 13:21:02 -0800169 *
Allan Stephensc98da842016-11-11 15:45:03 -0500170 * 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 Boiee4448252016-02-25 13:21:02 -0800175 *
Allan Stephensc98da842016-11-11 15:45:03 -0500176 * @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 Boiee4448252016-02-25 13:21:02 -0800186 *
Allan Stephensc98da842016-11-11 15:45:03 -0500187 * @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 Boiee4448252016-02-25 13:21:02 -0800190 *
Allan Stephensc98da842016-11-11 15:45:03 -0500191 * @return Lock-out key.
Andrew Boiee4448252016-02-25 13:21:02 -0800192 */
193#define irq_lock() _arch_irq_lock()
194
195/**
Allan Stephensc98da842016-11-11 15:45:03 -0500196 * @brief Unlock interrupts.
Andrew Boiee4448252016-02-25 13:21:02 -0800197 *
Allan Stephensc98da842016-11-11 15:45:03 -0500198 * 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 Boiee4448252016-02-25 13:21:02 -0800202 *
Allan Stephensc98da842016-11-11 15:45:03 -0500203 * @note Can be called by ISRs.
Andrew Boiee4448252016-02-25 13:21:02 -0800204 *
Allan Stephensc98da842016-11-11 15:45:03 -0500205 * @param key Lock-out key generated by irq_lock().
Andrew Boiee4448252016-02-25 13:21:02 -0800206 *
207 * @return N/A
208 */
209#define irq_unlock(key) _arch_irq_unlock(key)
210
211/**
Allan Stephensc98da842016-11-11 15:45:03 -0500212 * @brief Enable an IRQ.
Andrew Boiee4448252016-02-25 13:21:02 -0800213 *
Allan Stephensc98da842016-11-11 15:45:03 -0500214 * This routine enables interrupts from source @a irq.
215 *
216 * @param irq IRQ line.
217 *
Andrew Boiee4448252016-02-25 13:21:02 -0800218 * @return N/A
219 */
220#define irq_enable(irq) _arch_irq_enable(irq)
221
222/**
Allan Stephensc98da842016-11-11 15:45:03 -0500223 * @brief Disable an IRQ.
Andrew Boiee4448252016-02-25 13:21:02 -0800224 *
Allan Stephensc98da842016-11-11 15:45:03 -0500225 * This routine disables interrupts from source @a irq.
226 *
227 * @param irq IRQ line.
228 *
Andrew Boiee4448252016-02-25 13:21:02 -0800229 * @return N/A
230 */
231#define irq_disable(irq) _arch_irq_disable(irq)
232
Vinayak Chettimadab33aaa42016-09-10 11:53:49 +0200233/**
Allan Stephensc98da842016-11-11 15:45:03 -0500234 * @brief Get IRQ enable state.
Vinayak Chettimadab33aaa42016-09-10 11:53:49 +0200235 *
Allan Stephensc98da842016-11-11 15:45:03 -0500236 * This routine indicates if interrupts from source @a irq are enabled.
237 *
238 * @param irq IRQ line.
239 *
Vinayak Chettimadab33aaa42016-09-10 11:53:49 +0200240 * @return interrupt enable state, true or false
241 */
242#define irq_is_enabled(irq) _arch_irq_is_enabled(irq)
243
Allan Stephensc98da842016-11-11 15:45:03 -0500244/**
245 * @}
246 */
247
Andrew Boie15800ce2016-03-28 14:49:37 -0700248#ifdef __cplusplus
249}
250#endif
Andrew Boiee4448252016-02-25 13:21:02 -0800251
Andrew Boie15800ce2016-03-28 14:49:37 -0700252#endif /* ASMLANGUAGE */
Andrew Boiee4448252016-02-25 13:21:02 -0800253#endif /* _IRQ_H_ */