sw_isr_common.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. /*
  2. * Copyright (c) 2018 Intel Corporation.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <sw_isr_table.h>
  7. #include <arch/cpu.h>
  8. #include <sys/__assert.h>
  9. /*
  10. * Common code for arches that use software ISR tables (CONFIG_GEN_ISR_TABLES)
  11. */
  12. #ifdef CONFIG_DYNAMIC_INTERRUPTS
  13. #ifdef CONFIG_MULTI_LEVEL_INTERRUPTS
  14. struct irq_parent_offset {
  15. unsigned int irq;
  16. unsigned int offset;
  17. };
  18. #define INIT_IRQ_PARENT_OFFSET(i, o) { \
  19. .irq = i, \
  20. .offset = o, \
  21. },
  22. #define IRQ_INDEX_TO_OFFSET(i, base) (base + i * CONFIG_MAX_IRQ_PER_AGGREGATOR)
  23. #ifdef CONFIG_2ND_LEVEL_INTERRUPTS
  24. #define CAT_2ND_LVL_LIST(i, base) \
  25. INIT_IRQ_PARENT_OFFSET(CONFIG_2ND_LVL_INTR_0##i##_OFFSET, \
  26. IRQ_INDEX_TO_OFFSET(i, base))
  27. static struct irq_parent_offset lvl2_irq_list[CONFIG_NUM_2ND_LEVEL_AGGREGATORS]
  28. = { UTIL_LISTIFY(CONFIG_NUM_2ND_LEVEL_AGGREGATORS, CAT_2ND_LVL_LIST,
  29. CONFIG_2ND_LVL_ISR_TBL_OFFSET) };
  30. #endif/* CONFIG_2ND_LEVEL_INTERRUPTS */
  31. #ifdef CONFIG_3RD_LEVEL_INTERRUPTS
  32. #define CAT_3RD_LVL_LIST(i, base) \
  33. INIT_IRQ_PARENT_OFFSET(CONFIG_3RD_LVL_INTR_0##i##_OFFSET, \
  34. IRQ_INDEX_TO_OFFSET(i, base))
  35. static struct irq_parent_offset lvl3_irq_list[CONFIG_NUM_3RD_LEVEL_AGGREGATORS]
  36. = { UTIL_LISTIFY(CONFIG_NUM_3RD_LEVEL_AGGREGATORS, CAT_3RD_LVL_LIST,
  37. CONFIG_3RD_LVL_ISR_TBL_OFFSET) };
  38. #endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
  39. unsigned int get_parent_offset(unsigned int parent_irq,
  40. struct irq_parent_offset list[],
  41. unsigned int length)
  42. {
  43. unsigned int i;
  44. unsigned int offset = 0U;
  45. for (i = 0U; i < length; ++i) {
  46. if (list[i].irq == parent_irq) {
  47. offset = list[i].offset;
  48. break;
  49. }
  50. }
  51. __ASSERT(i != length, "Invalid argument: %i", parent_irq);
  52. return offset;
  53. }
  54. #endif /* CONFIG_MULTI_LEVEL_INTERRUPTS */
  55. void z_isr_install(unsigned int irq, void (*routine)(const void *),
  56. const void *param)
  57. {
  58. unsigned int table_idx;
  59. /*
  60. * Do not assert on the IRQ enable status for ARM GIC since the SGI
  61. * type interrupts are always enabled and attempting to install an ISR
  62. * for them will cause the assertion to fail.
  63. */
  64. #ifndef CONFIG_GIC
  65. __ASSERT(!irq_is_enabled(irq), "IRQ %d is enabled", irq);
  66. #endif /* !CONFIG_GIC */
  67. #ifdef CONFIG_MULTI_LEVEL_INTERRUPTS
  68. unsigned int level;
  69. unsigned int parent_irq;
  70. unsigned int parent_offset;
  71. level = irq_get_level(irq);
  72. if (level == 2U) {
  73. parent_irq = irq_parent_level_2(irq);
  74. parent_offset = get_parent_offset(parent_irq,
  75. lvl2_irq_list,
  76. CONFIG_NUM_2ND_LEVEL_AGGREGATORS);
  77. table_idx = parent_offset + irq_from_level_2(irq);
  78. }
  79. #ifdef CONFIG_3RD_LEVEL_INTERRUPTS
  80. else if (level == 3U) {
  81. parent_irq = irq_parent_level_3(irq);
  82. parent_offset = get_parent_offset(parent_irq,
  83. lvl3_irq_list,
  84. CONFIG_NUM_3RD_LEVEL_AGGREGATORS);
  85. table_idx = parent_offset + irq_from_level_3(irq);
  86. }
  87. #endif /* CONFIG_3RD_LEVEL_INTERRUPTS */
  88. else {
  89. table_idx = irq;
  90. }
  91. table_idx -= CONFIG_GEN_IRQ_START_VECTOR;
  92. #else
  93. table_idx = irq - CONFIG_GEN_IRQ_START_VECTOR;
  94. #endif /* CONFIG_MULTI_LEVEL_INTERRUPTS */
  95. /* If dynamic IRQs are enabled, then the _sw_isr_table is in RAM and
  96. * can be modified
  97. */
  98. _sw_isr_table[table_idx].arg = param;
  99. _sw_isr_table[table_idx].isr = routine;
  100. }
  101. /* Some architectures don't/can't interpret flags or priority and have
  102. * no more processing to do than this. Provide a generic fallback.
  103. */
  104. int __weak arch_irq_connect_dynamic(unsigned int irq,
  105. unsigned int priority,
  106. void (*routine)(const void *),
  107. const void *parameter,
  108. uint32_t flags)
  109. {
  110. ARG_UNUSED(flags);
  111. ARG_UNUSED(priority);
  112. z_isr_install(irq, routine, parameter);
  113. return irq;
  114. }
  115. #endif /* CONFIG_DYNAMIC_INTERRUPTS */