Zephyr Project API 4.1.0
A Scalable Open Source RTOS
Loading...
Searching...
No Matches
irq_multilevel.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2023 Meta
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
11#ifndef ZEPHYR_INCLUDE_IRQ_MULTILEVEL_H_
12#define ZEPHYR_INCLUDE_IRQ_MULTILEVEL_H_
13
14#ifndef _ASMLANGUAGE
15#include <zephyr/sys/__assert.h>
17#include <zephyr/types.h>
18
19#ifdef __cplusplus
20extern "C" {
21#endif
22
23#if defined(CONFIG_MULTI_LEVEL_INTERRUPTS) || defined(__DOXYGEN__)
24
25typedef union _z_irq {
26 /* Zephyr multilevel-encoded IRQ */
27 uint32_t irq;
28
29 /* Interrupt bits */
30 struct {
31 /* First level interrupt bits */
32 uint32_t l1: CONFIG_1ST_LEVEL_INTERRUPT_BITS;
33 /* Second level interrupt bits */
34 uint32_t l2: CONFIG_2ND_LEVEL_INTERRUPT_BITS;
35#if defined(CONFIG_3RD_LEVEL_INTERRUPTS) || defined(__DOXYGEN__)
36 /* Third level interrupt bits */
37 uint32_t l3: CONFIG_3RD_LEVEL_INTERRUPT_BITS;
38#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
39 } bits;
40
41#if defined(CONFIG_3RD_LEVEL_INTERRUPTS) || defined(__DOXYGEN__)
42 /* Third level IRQ's interrupt controller */
43 struct {
44 /* IRQ of the third level interrupt aggregator */
45 uint32_t irq: CONFIG_1ST_LEVEL_INTERRUPT_BITS + CONFIG_2ND_LEVEL_INTERRUPT_BITS;
46 } l3_intc;
47#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
48
49 /* Second level IRQ's interrupt controller */
50 struct {
51 /* IRQ of the second level interrupt aggregator */
52 uint32_t irq: CONFIG_1ST_LEVEL_INTERRUPT_BITS;
53 } l2_intc;
54} _z_irq_t;
55
56BUILD_ASSERT(sizeof(_z_irq_t) == sizeof(uint32_t), "Size of `_z_irq_t` must equal to `uint32_t`");
57
58static inline uint32_t _z_l1_irq(_z_irq_t irq)
59{
60 return irq.bits.l1;
61}
62
63static inline uint32_t _z_l2_irq(_z_irq_t irq)
64{
65 return irq.bits.l2 - 1;
66}
67
68#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
69static inline uint32_t _z_l3_irq(_z_irq_t irq)
70{
71 return irq.bits.l3 - 1;
72}
73#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
74
75static inline unsigned int _z_irq_get_level(_z_irq_t z_irq)
76{
77#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
78 if (z_irq.bits.l3 != 0) {
79 return 3;
80 }
81#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
82
83 if (z_irq.bits.l2 != 0) {
84 return 2;
85 }
86
87 return 1;
88}
89
98static inline unsigned int irq_get_level(unsigned int irq)
99{
100 _z_irq_t z_irq = {
101 .irq = irq,
102 };
103
104 return _z_irq_get_level(z_irq);
105}
106
117static inline unsigned int irq_from_level_2(unsigned int irq)
118{
119 _z_irq_t z_irq = {
120 .irq = irq,
121 };
122
123 return _z_l2_irq(z_irq);
124}
125
133#define IRQ_TO_L2(irq) ((irq + 1) << CONFIG_1ST_LEVEL_INTERRUPT_BITS)
134
147static inline unsigned int irq_to_level_2(unsigned int irq)
148{
149 _z_irq_t z_irq = {
150 .bits = {
151 .l1 = 0,
152 .l2 = irq + 1,
153#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
154 .l3 = 0,
155#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
156 },
157 };
158
159 return z_irq.irq;
160}
161
172static inline unsigned int irq_parent_level_2(unsigned int irq)
173{
174 _z_irq_t z_irq = {
175 .irq = irq,
176 };
177
178 return _z_l1_irq(z_irq);
179}
180
181#if defined(CONFIG_3RD_LEVEL_INTERRUPTS) || defined(__DOXYGEN__)
193static inline unsigned int irq_from_level_3(unsigned int irq)
194{
195 _z_irq_t z_irq = {
196 .irq = irq,
197 };
198
199 return _z_l3_irq(z_irq);
200}
201
209#define IRQ_TO_L3(irq) \
210 ((irq + 1) << (CONFIG_1ST_LEVEL_INTERRUPT_BITS + CONFIG_2ND_LEVEL_INTERRUPT_BITS))
211
224static inline unsigned int irq_to_level_3(unsigned int irq)
225{
226 _z_irq_t z_irq = {
227 .bits = {
228 .l1 = 0,
229 .l2 = 0,
230 .l3 = irq + 1,
231 },
232 };
233
234 return z_irq.irq;
235}
236
247static inline unsigned int irq_parent_level_3(unsigned int irq)
248{
249 _z_irq_t z_irq = {
250 .irq = irq,
251 };
252
253 return _z_l2_irq(z_irq);
254}
255#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
256
265static inline unsigned int irq_from_level(unsigned int irq, unsigned int level)
266{
267 if (level == 1) {
268 return irq;
269 } else if (level == 2) {
270 return irq_from_level_2(irq);
271 }
272#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
273 else if (level == 3) {
274 return irq_from_level_3(irq);
275 }
276#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
277
278 /* level is higher than what's supported */
279 __ASSERT_NO_MSG(false);
280 return irq;
281}
282
291static inline unsigned int irq_to_level(unsigned int irq, unsigned int level)
292{
293 if (level == 1) {
294 return irq;
295 } else if (level == 2) {
296 return irq_to_level_2(irq);
297 }
298#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
299 else if (level == 3) {
300 return irq_to_level_3(irq);
301 }
302#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
303
304 /* level is higher than what's supported */
305 __ASSERT_NO_MSG(false);
306 return irq;
307}
308
317static inline unsigned int irq_parent_level(unsigned int irq, unsigned int level)
318{
319 if (level == 1) {
320 /* doesn't really make sense, but return anyway */
321 return irq;
322 } else if (level == 2) {
323 return irq_parent_level_2(irq);
324 }
325#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
326 else if (level == 3) {
327 return irq_parent_level_3(irq);
328 }
329#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
330
331 /* level is higher than what's supported */
332 __ASSERT_NO_MSG(false);
333 return irq;
334}
335
343static inline unsigned int irq_get_intc_irq(unsigned int irq)
344{
345 const unsigned int level = irq_get_level(irq);
346 _z_irq_t z_irq = {
347 .irq = irq,
348 };
349
350#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
351 __ASSERT_NO_MSG(level <= 3);
352 if (level == 3) {
353 return z_irq.l3_intc.irq;
354 }
355#else
356 __ASSERT_NO_MSG(level <= 2);
357#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
358
359 if (level == 2) {
360 return z_irq.l2_intc.irq;
361 }
362
363 return irq;
364}
365
374static inline unsigned int irq_increment(unsigned int irq, unsigned int val)
375{
376 _z_irq_t z_irq = {
377 .irq = irq,
378 };
379
380 if (false) {
381 /* so that it evaluates the next condition */
382#ifdef CONFIG_3RD_LEVEL_INTERRUPTS
383 } else if (z_irq.bits.l3 != 0) {
384 z_irq.bits.l3 += val;
385#endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
386 } else if (z_irq.bits.l2 != 0) {
387 z_irq.bits.l2 += val;
388 } else {
389 z_irq.bits.l1 += val;
390 }
391
392 return z_irq.irq;
393}
394
395#endif /* CONFIG_MULTI_LEVEL_INTERRUPTS */
396#ifdef __cplusplus
397}
398#endif
399
400#endif /* _ASMLANGUAGE */
401#endif /* ZEPHYR_INCLUDE_IRQ_MULTILEVEL_H_ */
static unsigned int irq_from_level(unsigned int irq, unsigned int level)
Return the interrupt number for a given level.
Definition irq_multilevel.h:265
static unsigned int irq_to_level_3(unsigned int irq)
Converts irq from level 1 to level 3 format.
Definition irq_multilevel.h:224
static unsigned int irq_from_level_3(unsigned int irq)
Return the 3rd level interrupt number.
Definition irq_multilevel.h:193
static unsigned int irq_from_level_2(unsigned int irq)
Return the 2nd level interrupt number.
Definition irq_multilevel.h:117
static unsigned int irq_get_level(unsigned int irq)
Return IRQ level This routine returns the interrupt level number of the provided interrupt.
Definition irq_multilevel.h:98
static unsigned int irq_increment(unsigned int irq, unsigned int val)
Increments the multilevel-encoded irq by val.
Definition irq_multilevel.h:374
static unsigned int irq_to_level_2(unsigned int irq)
Converts irq from level 1 to level 2 format.
Definition irq_multilevel.h:147
static unsigned int irq_parent_level(unsigned int irq, unsigned int level)
Returns the parent IRQ of the given level raw IRQ number.
Definition irq_multilevel.h:317
static unsigned int irq_get_intc_irq(unsigned int irq)
Returns the parent interrupt controller IRQ of the given IRQ number.
Definition irq_multilevel.h:343
static unsigned int irq_parent_level_2(unsigned int irq)
Returns the parent IRQ of the level 2 raw IRQ number.
Definition irq_multilevel.h:172
static unsigned int irq_parent_level_3(unsigned int irq)
Returns the parent IRQ of the level 3 raw IRQ number.
Definition irq_multilevel.h:247
static unsigned int irq_to_level(unsigned int irq, unsigned int level)
Converts irq from level 1 to a given level.
Definition irq_multilevel.h:291
__UINT32_TYPE__ uint32_t
Definition stdint.h:90
Macro utilities.