irq.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453
  1. /*
  2. * Copyright (c) 2015 Intel corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Public interface for configuring interrupts
  9. */
  10. #ifndef ZEPHYR_INCLUDE_IRQ_H_
  11. #define ZEPHYR_INCLUDE_IRQ_H_
  12. /* Pull in the arch-specific implementations */
  13. #include <arch/cpu.h>
  14. #ifndef _ASMLANGUAGE
  15. #include <toolchain.h>
  16. #include <zephyr/types.h>
  17. #ifdef __cplusplus
  18. extern "C" {
  19. #endif
  20. /**
  21. * @defgroup isr_apis Interrupt Service Routine APIs
  22. * @ingroup kernel_apis
  23. * @{
  24. */
  25. /**
  26. * @brief Initialize an interrupt handler.
  27. *
  28. * This routine initializes an interrupt handler for an IRQ. The IRQ must be
  29. * subsequently enabled before the interrupt handler begins servicing
  30. * interrupts.
  31. *
  32. * @warning
  33. * Although this routine is invoked at run-time, all of its arguments must be
  34. * computable by the compiler at build time.
  35. *
  36. * @param irq_p IRQ line number.
  37. * @param priority_p Interrupt priority.
  38. * @param isr_p Address of interrupt service routine.
  39. * @param isr_param_p Parameter passed to interrupt service routine.
  40. * @param flags_p Architecture-specific IRQ configuration flags..
  41. */
  42. #define IRQ_CONNECT(irq_p, priority_p, isr_p, isr_param_p, flags_p) \
  43. ARCH_IRQ_CONNECT(irq_p, priority_p, isr_p, isr_param_p, flags_p)
  44. /**
  45. * Configure a dynamic interrupt.
  46. *
  47. * Use this instead of IRQ_CONNECT() if arguments cannot be known at build time.
  48. *
  49. * @param irq IRQ line number
  50. * @param priority Interrupt priority
  51. * @param routine Interrupt service routine
  52. * @param parameter ISR parameter
  53. * @param flags Arch-specific IRQ configuration flags
  54. *
  55. * @return The vector assigned to this interrupt
  56. */
  57. static inline int
  58. irq_connect_dynamic(unsigned int irq, unsigned int priority,
  59. void (*routine)(const void *parameter),
  60. const void *parameter, uint32_t flags)
  61. {
  62. return arch_irq_connect_dynamic(irq, priority, routine, parameter,
  63. flags);
  64. }
  65. /**
  66. * @brief Initialize a 'direct' interrupt handler.
  67. *
  68. * This routine initializes an interrupt handler for an IRQ. The IRQ must be
  69. * subsequently enabled via irq_enable() before the interrupt handler begins
  70. * servicing interrupts.
  71. *
  72. * These ISRs are designed for performance-critical interrupt handling and do
  73. * not go through common interrupt handling code. They must be implemented in
  74. * such a way that it is safe to put them directly in the vector table. For
  75. * ISRs written in C, The ISR_DIRECT_DECLARE() macro will do this
  76. * automatically. For ISRs written in assembly it is entirely up to the
  77. * developer to ensure that the right steps are taken.
  78. *
  79. * This type of interrupt currently has a few limitations compared to normal
  80. * Zephyr interrupts:
  81. * - No parameters are passed to the ISR.
  82. * - No stack switch is done, the ISR will run on the interrupted context's
  83. * stack, unless the architecture automatically does the stack switch in HW.
  84. * - Interrupt locking state is unchanged from how the HW sets it when the ISR
  85. * runs. On arches that enter ISRs with interrupts locked, they will remain
  86. * locked.
  87. * - Scheduling decisions are now optional, controlled by the return value of
  88. * ISRs implemented with the ISR_DIRECT_DECLARE() macro
  89. * - The call into the OS to exit power management idle state is now optional.
  90. * Normal interrupts always do this before the ISR is run, but when it runs
  91. * is now controlled by the placement of a ISR_DIRECT_PM() macro, or omitted
  92. * entirely.
  93. *
  94. * @warning
  95. * Although this routine is invoked at run-time, all of its arguments must be
  96. * computable by the compiler at build time.
  97. *
  98. * @param irq_p IRQ line number.
  99. * @param priority_p Interrupt priority.
  100. * @param isr_p Address of interrupt service routine.
  101. * @param flags_p Architecture-specific IRQ configuration flags.
  102. */
  103. #define IRQ_DIRECT_CONNECT(irq_p, priority_p, isr_p, flags_p) \
  104. ARCH_IRQ_DIRECT_CONNECT(irq_p, priority_p, isr_p, flags_p)
  105. /**
  106. * @brief Common tasks before executing the body of an ISR
  107. *
  108. * This macro must be at the beginning of all direct interrupts and performs
  109. * minimal architecture-specific tasks before the ISR itself can run. It takes
  110. * no arguments and has no return value.
  111. */
  112. #define ISR_DIRECT_HEADER() ARCH_ISR_DIRECT_HEADER()
  113. /**
  114. * @brief Common tasks before exiting the body of an ISR
  115. *
  116. * This macro must be at the end of all direct interrupts and performs
  117. * minimal architecture-specific tasks like EOI. It has no return value.
  118. *
  119. * In a normal interrupt, a check is done at end of interrupt to invoke
  120. * z_swap() logic if the current thread is preemptible and there is another
  121. * thread ready to run in the kernel's ready queue cache. This is now optional
  122. * and controlled by the check_reschedule argument. If unsure, set to nonzero.
  123. * On systems that do stack switching and nested interrupt tracking in software,
  124. * z_swap() should only be called if this was a non-nested interrupt.
  125. *
  126. * @param check_reschedule If nonzero, additionally invoke scheduling logic
  127. */
  128. #define ISR_DIRECT_FOOTER(check_reschedule) \
  129. ARCH_ISR_DIRECT_FOOTER(check_reschedule)
  130. /**
  131. * @brief Perform power management idle exit logic
  132. *
  133. * This macro may optionally be invoked somewhere in between IRQ_DIRECT_HEADER()
  134. * and IRQ_DIRECT_FOOTER() invocations. It performs tasks necessary to
  135. * exit power management idle state. It takes no parameters and returns no
  136. * arguments. It may be omitted, but be careful!
  137. */
  138. #define ISR_DIRECT_PM() ARCH_ISR_DIRECT_PM()
  139. /**
  140. * @brief Helper macro to declare a direct interrupt service routine.
  141. *
  142. * This will declare the function in a proper way and automatically include
  143. * the ISR_DIRECT_FOOTER() and ISR_DIRECT_HEADER() macros. The function should
  144. * return nonzero status if a scheduling decision should potentially be made.
  145. * See ISR_DIRECT_FOOTER() for more details on the scheduling decision.
  146. *
  147. * For architectures that support 'regular' and 'fast' interrupt types, where
  148. * these interrupt types require different assembly language handling of
  149. * registers by the ISR, this will always generate code for the 'fast'
  150. * interrupt type.
  151. *
  152. * Example usage:
  153. *
  154. * ISR_DIRECT_DECLARE(my_isr)
  155. * {
  156. * bool done = do_stuff();
  157. * ISR_DIRECT_PM(); // done after do_stuff() due to latency concerns
  158. * if (!done) {
  159. * return 0; // don't bother checking if we have to z_swap()
  160. * }
  161. *
  162. * k_sem_give(some_sem);
  163. * return 1;
  164. * }
  165. *
  166. * @param name symbol name of the ISR
  167. */
  168. #define ISR_DIRECT_DECLARE(name) ARCH_ISR_DIRECT_DECLARE(name)
  169. /**
  170. * @brief Lock interrupts.
  171. * @def irq_lock()
  172. *
  173. * This routine disables all interrupts on the CPU. It returns an unsigned
  174. * integer "lock-out key", which is an architecture-dependent indicator of
  175. * whether interrupts were locked prior to the call. The lock-out key must be
  176. * passed to irq_unlock() to re-enable interrupts.
  177. *
  178. * @note
  179. * This routine must also serve as a memory barrier to ensure the uniprocessor
  180. * implementation of `k_spinlock_t` is correct.
  181. *
  182. * This routine can be called recursively, as long as the caller keeps track
  183. * of each lock-out key that is generated. Interrupts are re-enabled by
  184. * passing each of the keys to irq_unlock() in the reverse order they were
  185. * acquired. (That is, each call to irq_lock() must be balanced by
  186. * a corresponding call to irq_unlock().)
  187. *
  188. * This routine can only be invoked from supervisor mode. Some architectures
  189. * (for example, ARM) will fail silently if invoked from user mode instead
  190. * of generating an exception.
  191. *
  192. * @note
  193. * This routine can be called by ISRs or by threads. If it is called by a
  194. * thread, the interrupt lock is thread-specific; this means that interrupts
  195. * remain disabled only while the thread is running. If the thread performs an
  196. * operation that allows another thread to run (for example, giving a semaphore
  197. * or sleeping for N milliseconds), the interrupt lock no longer applies and
  198. * interrupts may be re-enabled while other processing occurs. When the thread
  199. * once again becomes the current thread, the kernel re-establishes its
  200. * interrupt lock; this ensures the thread won't be interrupted until it has
  201. * explicitly released the interrupt lock it established.
  202. *
  203. * @warning
  204. * The lock-out key should never be used to manually re-enable interrupts
  205. * or to inspect or manipulate the contents of the CPU's interrupt bits.
  206. *
  207. * @return An architecture-dependent lock-out key representing the
  208. * "interrupt disable state" prior to the call.
  209. */
  210. #ifdef CONFIG_SMP
  211. unsigned int z_smp_global_lock(void);
  212. #define irq_lock() z_smp_global_lock()
  213. #else
  214. #define irq_lock() arch_irq_lock()
  215. #endif
  216. /**
  217. * @brief Unlock interrupts.
  218. * @def irq_unlock()
  219. *
  220. * This routine reverses the effect of a previous call to irq_lock() using
  221. * the associated lock-out key. The caller must call the routine once for
  222. * each time it called irq_lock(), supplying the keys in the reverse order
  223. * they were acquired, before interrupts are enabled.
  224. *
  225. * @note
  226. * This routine must also serve as a memory barrier to ensure the uniprocessor
  227. * implementation of `k_spinlock_t` is correct.
  228. *
  229. * This routine can only be invoked from supervisor mode. Some architectures
  230. * (for example, ARM) will fail silently if invoked from user mode instead
  231. * of generating an exception.
  232. *
  233. * @note Can be called by ISRs.
  234. *
  235. * @param key Lock-out key generated by irq_lock().
  236. *
  237. * @return N/A
  238. */
  239. #ifdef CONFIG_SMP
  240. void z_smp_global_unlock(unsigned int key);
  241. #define irq_unlock(key) z_smp_global_unlock(key)
  242. #else
  243. #define irq_unlock(key) arch_irq_unlock(key)
  244. #endif
  245. /**
  246. * @brief Return IRQ level
  247. * @def irq_get_level()
  248. *
  249. * This routine returns the interrupt level number of the provided interrupt.
  250. *
  251. * @param irq IRQ number in its zephyr format
  252. *
  253. * @return 1 if IRQ level 1, 2 if IRQ level 2, 3 if IRQ level 3
  254. */
  255. static inline unsigned int irq_get_level(unsigned int irq)
  256. {
  257. #if defined(CONFIG_3RD_LEVEL_INTERRUPTS)
  258. return ((irq >> 16) & 0xFF) != 0 ? 3 :
  259. (((irq >> 8) & 0xFF) == 0 ? 1 : 2);
  260. #elif defined(CONFIG_2ND_LEVEL_INTERRUPTS)
  261. return ((irq >> 8) & 0xFF) == 0 ? 1 : 2;
  262. #else
  263. ARG_UNUSED(irq);
  264. return 1;
  265. #endif
  266. }
  267. #ifdef CONFIG_2ND_LEVEL_INTERRUPTS
  268. /**
  269. * @brief Return the 2nd level interrupt number
  270. * @def irq_from_level_2()
  271. *
  272. * This routine returns the second level irq number of the zephyr irq
  273. * number passed in
  274. *
  275. * @param irq IRQ number in its zephyr format
  276. *
  277. * @return 2nd level IRQ number
  278. */
  279. static inline unsigned int irq_from_level_2(unsigned int irq)
  280. {
  281. #ifdef CONFIG_3RD_LEVEL_INTERRUPTS
  282. return ((irq >> 8) & 0xFF) - 1;
  283. #else
  284. return (irq >> 8) - 1;
  285. #endif
  286. }
  287. /**
  288. * @brief Converts irq from level 1 to level 2 format
  289. * @def irq_to_level_2()
  290. *
  291. * This routine converts the input into the level 2 irq number format
  292. *
  293. * @note Values >= 0xFF are invalid
  294. *
  295. * @param irq IRQ number in its zephyr format
  296. *
  297. * @return 2nd level IRQ number
  298. */
  299. static inline unsigned int irq_to_level_2(unsigned int irq)
  300. {
  301. return (irq + 1) << 8;
  302. }
  303. /**
  304. * @brief Returns the parent IRQ of the level 2 raw IRQ number
  305. * @def irq_parent_level_2()
  306. *
  307. * The parent of a 2nd level interrupt is in the 1st byte
  308. *
  309. * @param irq IRQ number in its zephyr format
  310. *
  311. * @return 2nd level IRQ parent
  312. */
  313. static inline unsigned int irq_parent_level_2(unsigned int irq)
  314. {
  315. return irq & 0xFF;
  316. }
  317. #endif
  318. #ifdef CONFIG_3RD_LEVEL_INTERRUPTS
  319. /**
  320. * @brief Return the 3rd level interrupt number
  321. * @def irq_from_level_3()
  322. *
  323. * This routine returns the third level irq number of the zephyr irq
  324. * number passed in
  325. *
  326. * @param irq IRQ number in its zephyr format
  327. *
  328. * @return 3rd level IRQ number
  329. */
  330. static inline unsigned int irq_from_level_3(unsigned int irq)
  331. {
  332. return (irq >> 16) - 1;
  333. }
  334. /**
  335. * @brief Converts irq from level 1 to level 3 format
  336. * @def irq_to_level_3()
  337. *
  338. * This routine converts the input into the level 3 irq number format
  339. *
  340. * @note Values >= 0xFF are invalid
  341. *
  342. * @param irq IRQ number in its zephyr format
  343. *
  344. * @return 3rd level IRQ number
  345. */
  346. static inline unsigned int irq_to_level_3(unsigned int irq)
  347. {
  348. return (irq + 1) << 16;
  349. }
  350. /**
  351. * @brief Returns the parent IRQ of the level 3 raw IRQ number
  352. * @def irq_parent_level_3()
  353. *
  354. * The parent of a 3rd level interrupt is in the 2nd byte
  355. *
  356. * @param irq IRQ number in its zephyr format
  357. *
  358. * @return 3rd level IRQ parent
  359. */
  360. static inline unsigned int irq_parent_level_3(unsigned int irq)
  361. {
  362. return (irq >> 8) & 0xFF;
  363. }
  364. #endif
  365. /**
  366. * @brief Enable an IRQ.
  367. *
  368. * This routine enables interrupts from source @a irq.
  369. *
  370. * @param irq IRQ line.
  371. *
  372. * @return N/A
  373. */
  374. #define irq_enable(irq) arch_irq_enable(irq)
  375. /**
  376. * @brief Disable an IRQ.
  377. *
  378. * This routine disables interrupts from source @a irq.
  379. *
  380. * @param irq IRQ line.
  381. *
  382. * @return N/A
  383. */
  384. #define irq_disable(irq) arch_irq_disable(irq)
  385. /**
  386. * @brief Get IRQ enable state.
  387. *
  388. * This routine indicates if interrupts from source @a irq are enabled.
  389. *
  390. * @param irq IRQ line.
  391. *
  392. * @return interrupt enable state, true or false
  393. */
  394. #define irq_is_enabled(irq) arch_irq_is_enabled(irq)
  395. typedef struct sys_irq_flags {
  396. u32_t keys[2];
  397. } SYS_IRQ_FLAGS;
  398. #define CONFIG_SYS_IRQ_LOCK
  399. #if defined(CONFIG_SYS_IRQ_LOCK)
  400. //void sys_irq_lock(SYS_IRQ_FLAGS *flags);
  401. //void sys_irq_unlock(const SYS_IRQ_FLAGS *flags);
  402. //#else
  403. static inline void sys_irq_lock(SYS_IRQ_FLAGS *flags)
  404. {
  405. flags->keys[0] = irq_lock();
  406. }
  407. static inline void sys_irq_unlock(SYS_IRQ_FLAGS *flags)
  408. {
  409. irq_unlock(flags->keys[0]);
  410. }
  411. #endif
  412. /**
  413. * @}
  414. */
  415. #ifdef __cplusplus
  416. }
  417. #endif
  418. #endif /* ASMLANGUAGE */
  419. #endif /* ZEPHYR_INCLUDE_IRQ_H_ */