]>
Commit | Line | Data |
---|---|---|
83d290c5 | 1 | // SPDX-License-Identifier: GPL-2.0+ |
abf0cd3d GR |
2 | /* |
3 | * (C) Copyright 2009 | |
dbf7115a | 4 | * Graeme Russ, <[email protected]> |
abf0cd3d GR |
5 | * |
6 | * (C) Copyright 2007 | |
dbf7115a | 7 | * Daniel Hellstrom, Gaisler Research, <[email protected]> |
abf0cd3d GR |
8 | * |
9 | * (C) Copyright 2006 | |
dbf7115a | 10 | * Detlev Zundel, DENX Software Engineering, <[email protected]> |
abf0cd3d GR |
11 | * |
12 | * (C) Copyright -2003 | |
dbf7115a | 13 | * Wolfgang Denk, DENX Software Engineering, <[email protected]> |
abf0cd3d GR |
14 | * |
15 | * (C) Copyright 2002 | |
fa82f871 | 16 | * Daniel Engström, Omicron Ceti AB, <[email protected]> |
abf0cd3d GR |
17 | * |
18 | * (C) Copyright 2001 | |
dbf7115a | 19 | * Josh Huber, Mission Critical Linux, Inc, <[email protected]> |
abf0cd3d GR |
20 | */ |
21 | ||
22 | /* | |
23 | * This file contains the high-level API for the interrupt sub-system | |
dbf7115a | 24 | * of the x86 port of U-Boot. Most of the functionality has been |
abf0cd3d GR |
25 | * shamelessly stolen from the leon2 / leon3 ports of U-Boot. |
26 | * Daniel Hellstrom, Detlev Zundel, Wolfgang Denk and Josh Huber are | |
27 | * credited for the corresponding work on those ports. The original | |
dbf7115a | 28 | * interrupt handling routines for the x86 port were written by |
fa82f871 | 29 | * Daniel Engström |
abf0cd3d GR |
30 | */ |
31 | ||
32 | #include <common.h> | |
33 | #include <asm/interrupt.h> | |
34 | ||
c2bf0dfa SG |
35 | #if !CONFIG_IS_ENABLED(X86_64) |
36 | ||
abf0cd3d GR |
37 | struct irq_action { |
38 | interrupt_handler_t *handler; | |
39 | void *arg; | |
40 | unsigned int count; | |
41 | }; | |
42 | ||
6c505271 | 43 | static struct irq_action irq_handlers[SYS_NUM_IRQS] = { {0} }; |
83088afb GR |
44 | static int spurious_irq_cnt; |
45 | static int spurious_irq; | |
abf0cd3d GR |
46 | |
47 | void irq_install_handler(int irq, interrupt_handler_t *handler, void *arg) | |
48 | { | |
49 | int status; | |
50 | ||
6c505271 | 51 | if (irq < 0 || irq >= SYS_NUM_IRQS) { |
abf0cd3d GR |
52 | printf("irq_install_handler: bad irq number %d\n", irq); |
53 | return; | |
54 | } | |
55 | ||
56 | if (irq_handlers[irq].handler != NULL) | |
57 | printf("irq_install_handler: 0x%08lx replacing 0x%08lx\n", | |
83088afb GR |
58 | (ulong) handler, |
59 | (ulong) irq_handlers[irq].handler); | |
abf0cd3d | 60 | |
83088afb | 61 | status = disable_interrupts(); |
abf0cd3d | 62 | |
1c409bc7 | 63 | irq_handlers[irq].handler = handler; |
abf0cd3d GR |
64 | irq_handlers[irq].arg = arg; |
65 | irq_handlers[irq].count = 0; | |
66 | ||
c6410104 BM |
67 | if (CONFIG_IS_ENABLED(I8259_PIC)) |
68 | unmask_irq(irq); | |
abf0cd3d GR |
69 | |
70 | if (status) | |
71 | enable_interrupts(); | |
72 | ||
73 | return; | |
74 | } | |
75 | ||
76 | void irq_free_handler(int irq) | |
77 | { | |
78 | int status; | |
79 | ||
6c505271 | 80 | if (irq < 0 || irq >= SYS_NUM_IRQS) { |
abf0cd3d GR |
81 | printf("irq_free_handler: bad irq number %d\n", irq); |
82 | return; | |
83 | } | |
84 | ||
83088afb | 85 | status = disable_interrupts(); |
abf0cd3d | 86 | |
c6410104 BM |
87 | if (CONFIG_IS_ENABLED(I8259_PIC)) |
88 | mask_irq(irq); | |
abf0cd3d GR |
89 | |
90 | irq_handlers[irq].handler = NULL; | |
91 | irq_handlers[irq].arg = NULL; | |
92 | ||
93 | if (status) | |
94 | enable_interrupts(); | |
95 | ||
96 | return; | |
97 | } | |
98 | ||
564a9984 | 99 | void do_irq(int hw_irq) |
abf0cd3d | 100 | { |
564a9984 GR |
101 | int irq = hw_irq - 0x20; |
102 | ||
6c505271 | 103 | if (irq < 0 || irq >= SYS_NUM_IRQS) { |
abf0cd3d GR |
104 | printf("do_irq: bad irq number %d\n", irq); |
105 | return; | |
106 | } | |
107 | ||
108 | if (irq_handlers[irq].handler) { | |
c6410104 BM |
109 | if (CONFIG_IS_ENABLED(I8259_PIC)) |
110 | mask_irq(irq); | |
abf0cd3d GR |
111 | |
112 | irq_handlers[irq].handler(irq_handlers[irq].arg); | |
113 | irq_handlers[irq].count++; | |
114 | ||
c6410104 BM |
115 | if (CONFIG_IS_ENABLED(I8259_PIC)) { |
116 | unmask_irq(irq); | |
117 | specific_eoi(irq); | |
118 | } | |
abf0cd3d GR |
119 | } else { |
120 | if ((irq & 7) != 7) { | |
121 | spurious_irq_cnt++; | |
122 | spurious_irq = irq; | |
123 | } | |
124 | } | |
125 | } | |
c2bf0dfa | 126 | #endif |
abf0cd3d GR |
127 | |
128 | #if defined(CONFIG_CMD_IRQ) | |
54841ab5 | 129 | int do_irqinfo(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]) |
abf0cd3d | 130 | { |
c2bf0dfa | 131 | #if !CONFIG_IS_ENABLED(X86_64) |
abf0cd3d GR |
132 | int irq; |
133 | ||
134 | printf("Spurious IRQ: %u, last unknown IRQ: %d\n", | |
83088afb | 135 | spurious_irq_cnt, spurious_irq); |
abf0cd3d | 136 | |
83088afb GR |
137 | printf("Interrupt-Information:\n"); |
138 | printf("Nr Routine Arg Count\n"); | |
abf0cd3d | 139 | |
6c505271 | 140 | for (irq = 0; irq < SYS_NUM_IRQS; irq++) { |
abf0cd3d | 141 | if (irq_handlers[irq].handler != NULL) { |
83088afb | 142 | printf("%02d %08lx %08lx %d\n", |
abf0cd3d GR |
143 | irq, |
144 | (ulong)irq_handlers[irq].handler, | |
145 | (ulong)irq_handlers[irq].arg, | |
146 | irq_handlers[irq].count); | |
147 | } | |
148 | } | |
c2bf0dfa | 149 | #endif |
abf0cd3d GR |
150 | |
151 | return 0; | |
152 | } | |
153 | #endif |