blob: a3cc37c0c85e2267497f650611b656af90aaf5aa [file] [log] [blame]
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -07001/*
2 * linux/kernel/irq/chip.c
3 *
4 * Copyright (C) 1992, 1998-2006 Linus Torvalds, Ingo Molnar
5 * Copyright (C) 2005-2006, Thomas Gleixner, Russell King
6 *
7 * This file contains the core interrupt handling code, for irq-chip
8 * based architectures.
9 *
Mauro Carvalho Chehabc0c6e082017-05-14 12:03:39 -030010 * Detailed information is available in Documentation/core-api/genericirq.rst
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070011 */
12
13#include <linux/irq.h>
Michael Ellerman7fe37302007-04-18 19:39:21 +100014#include <linux/msi.h>
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070015#include <linux/module.h>
16#include <linux/interrupt.h>
17#include <linux/kernel_stat.h>
Jiang Liuf8264e32014-11-06 22:20:14 +080018#include <linux/irqdomain.h>
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070019
Steven Rostedtf0696862012-01-25 20:18:55 -050020#include <trace/events/irq.h>
21
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070022#include "internals.h"
23
Mika Westerberge509bd72015-10-05 13:12:15 +030024static irqreturn_t bad_chained_irq(int irq, void *dev_id)
25{
26 WARN_ONCE(1, "Chained irq %d should not call an action\n", irq);
27 return IRQ_NONE;
28}
29
30/*
31 * Chained handlers should never call action on their IRQ. This default
32 * action will emit warning if such thing happens.
33 */
34struct irqaction chained_action = {
35 .handler = bad_chained_irq,
36};
37
Eric W. Biederman3a16d712006-10-04 02:16:37 -070038/**
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010039 * irq_set_chip - set the irq chip for an irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070040 * @irq: irq number
41 * @chip: pointer to irq chip description structure
42 */
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010043int irq_set_chip(unsigned int irq, struct irq_chip *chip)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070044{
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070045 unsigned long flags;
Marc Zyngier31d9d9b2011-09-23 17:03:06 +010046 struct irq_desc *desc = irq_get_desc_lock(irq, &flags, 0);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070047
Thomas Gleixner02725e72011-02-12 10:37:36 +010048 if (!desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070049 return -EINVAL;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070050
51 if (!chip)
52 chip = &no_irq_chip;
53
Thomas Gleixner6b8ff312010-10-01 12:58:38 +020054 desc->irq_data.chip = chip;
Thomas Gleixner02725e72011-02-12 10:37:36 +010055 irq_put_desc_unlock(desc, flags);
David Daneyd72274e2011-03-25 12:38:48 -070056 /*
57 * For !CONFIG_SPARSE_IRQ make the irq show up in
Thomas Gleixnerf63b6a02014-05-07 15:44:21 +000058 * allocated_irqs.
David Daneyd72274e2011-03-25 12:38:48 -070059 */
Thomas Gleixnerf63b6a02014-05-07 15:44:21 +000060 irq_mark_irq(irq);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070061 return 0;
62}
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010063EXPORT_SYMBOL(irq_set_chip);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070064
65/**
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010066 * irq_set_type - set the irq trigger type for an irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070067 * @irq: irq number
David Brownell0c5d1eb2008-10-01 14:46:18 -070068 * @type: IRQ_TYPE_{LEVEL,EDGE}_* value - see include/linux/irq.h
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070069 */
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010070int irq_set_irq_type(unsigned int irq, unsigned int type)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070071{
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070072 unsigned long flags;
Marc Zyngier31d9d9b2011-09-23 17:03:06 +010073 struct irq_desc *desc = irq_get_desc_buslock(irq, &flags, IRQ_GET_DESC_CHECK_GLOBAL);
Thomas Gleixner02725e72011-02-12 10:37:36 +010074 int ret = 0;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070075
Thomas Gleixner02725e72011-02-12 10:37:36 +010076 if (!desc)
77 return -EINVAL;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070078
Jiang Liua1ff5412015-06-23 19:47:29 +020079 ret = __irq_set_trigger(desc, type);
Thomas Gleixner02725e72011-02-12 10:37:36 +010080 irq_put_desc_busunlock(desc, flags);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070081 return ret;
82}
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010083EXPORT_SYMBOL(irq_set_irq_type);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070084
85/**
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010086 * irq_set_handler_data - set irq handler data for an irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070087 * @irq: Interrupt number
88 * @data: Pointer to interrupt specific data
89 *
90 * Set the hardware irq controller data for an irq
91 */
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +010092int irq_set_handler_data(unsigned int irq, void *data)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070093{
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070094 unsigned long flags;
Marc Zyngier31d9d9b2011-09-23 17:03:06 +010095 struct irq_desc *desc = irq_get_desc_lock(irq, &flags, 0);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070096
Thomas Gleixner02725e72011-02-12 10:37:36 +010097 if (!desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -070098 return -EINVAL;
Jiang Liuaf7080e2015-06-01 16:05:21 +080099 desc->irq_common_data.handler_data = data;
Thomas Gleixner02725e72011-02-12 10:37:36 +0100100 irq_put_desc_unlock(desc, flags);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700101 return 0;
102}
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +0100103EXPORT_SYMBOL(irq_set_handler_data);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700104
105/**
Alexander Gordeev51906e72012-11-19 16:01:29 +0100106 * irq_set_msi_desc_off - set MSI descriptor data for an irq at offset
107 * @irq_base: Interrupt number base
108 * @irq_offset: Interrupt number offset
109 * @entry: Pointer to MSI descriptor data
110 *
111 * Set the MSI descriptor entry for an irq at offset
112 */
113int irq_set_msi_desc_off(unsigned int irq_base, unsigned int irq_offset,
114 struct msi_desc *entry)
115{
116 unsigned long flags;
117 struct irq_desc *desc = irq_get_desc_lock(irq_base + irq_offset, &flags, IRQ_GET_DESC_CHECK_GLOBAL);
118
119 if (!desc)
120 return -EINVAL;
Jiang Liub2377212015-06-01 16:05:43 +0800121 desc->irq_common_data.msi_desc = entry;
Alexander Gordeev51906e72012-11-19 16:01:29 +0100122 if (entry && !irq_offset)
123 entry->irq = irq_base;
124 irq_put_desc_unlock(desc, flags);
125 return 0;
126}
127
128/**
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +0100129 * irq_set_msi_desc - set MSI descriptor data for an irq
Eric W. Biederman5b912c12007-01-28 12:52:03 -0700130 * @irq: Interrupt number
Randy Dunlap472900b2007-02-16 01:28:25 -0800131 * @entry: Pointer to MSI descriptor data
Eric W. Biederman5b912c12007-01-28 12:52:03 -0700132 *
Liuweni24b26d42009-11-04 20:11:05 +0800133 * Set the MSI descriptor entry for an irq
Eric W. Biederman5b912c12007-01-28 12:52:03 -0700134 */
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +0100135int irq_set_msi_desc(unsigned int irq, struct msi_desc *entry)
Eric W. Biederman5b912c12007-01-28 12:52:03 -0700136{
Alexander Gordeev51906e72012-11-19 16:01:29 +0100137 return irq_set_msi_desc_off(irq, 0, entry);
Eric W. Biederman5b912c12007-01-28 12:52:03 -0700138}
139
140/**
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +0100141 * irq_set_chip_data - set irq chip data for an irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700142 * @irq: Interrupt number
143 * @data: Pointer to chip specific data
144 *
145 * Set the hardware irq chip data for an irq
146 */
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +0100147int irq_set_chip_data(unsigned int irq, void *data)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700148{
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700149 unsigned long flags;
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100150 struct irq_desc *desc = irq_get_desc_lock(irq, &flags, 0);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700151
Thomas Gleixner02725e72011-02-12 10:37:36 +0100152 if (!desc)
Yinghai Lu7d94f7c2008-08-19 20:50:14 -0700153 return -EINVAL;
Thomas Gleixner6b8ff312010-10-01 12:58:38 +0200154 desc->irq_data.chip_data = data;
Thomas Gleixner02725e72011-02-12 10:37:36 +0100155 irq_put_desc_unlock(desc, flags);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700156 return 0;
157}
Thomas Gleixnera0cd9ca2011-02-10 11:36:33 +0100158EXPORT_SYMBOL(irq_set_chip_data);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700159
Thomas Gleixnerf303a6d2010-09-28 17:34:01 +0200160struct irq_data *irq_get_irq_data(unsigned int irq)
161{
162 struct irq_desc *desc = irq_to_desc(irq);
163
164 return desc ? &desc->irq_data : NULL;
165}
166EXPORT_SYMBOL_GPL(irq_get_irq_data);
167
Thomas Gleixnerc1594b72011-02-07 22:11:30 +0100168static void irq_state_clr_disabled(struct irq_desc *desc)
169{
Thomas Gleixner801a0e92011-03-27 11:02:49 +0200170 irqd_clear(&desc->irq_data, IRQD_IRQ_DISABLED);
Thomas Gleixnerc1594b72011-02-07 22:11:30 +0100171}
172
Thomas Gleixner6e402622011-02-08 12:36:06 +0100173static void irq_state_clr_masked(struct irq_desc *desc)
174{
Thomas Gleixner32f41252011-03-28 14:10:52 +0200175 irqd_clear(&desc->irq_data, IRQD_IRQ_MASKED);
Thomas Gleixner6e402622011-02-08 12:36:06 +0100176}
177
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200178static void irq_state_clr_started(struct irq_desc *desc)
179{
180 irqd_clear(&desc->irq_data, IRQD_IRQ_STARTED);
181}
182
183static void irq_state_set_started(struct irq_desc *desc)
184{
185 irqd_set(&desc->irq_data, IRQD_IRQ_STARTED);
186}
187
Thomas Gleixner761ea382017-06-20 01:37:50 +0200188enum {
189 IRQ_STARTUP_NORMAL,
190 IRQ_STARTUP_MANAGED,
191 IRQ_STARTUP_ABORT,
192};
193
194#ifdef CONFIG_SMP
195static int
196__irq_startup_managed(struct irq_desc *desc, struct cpumask *aff, bool force)
197{
198 struct irq_data *d = irq_desc_get_irq_data(desc);
199
200 if (!irqd_affinity_is_managed(d))
201 return IRQ_STARTUP_NORMAL;
202
203 irqd_clr_managed_shutdown(d);
204
205 if (cpumask_any_and(aff, cpu_online_mask) > nr_cpu_ids) {
206 /*
207 * Catch code which fiddles with enable_irq() on a managed
208 * and potentially shutdown IRQ. Chained interrupt
209 * installment or irq auto probing should not happen on
210 * managed irqs either. Emit a warning, break the affinity
211 * and start it up as a normal interrupt.
212 */
213 if (WARN_ON_ONCE(force))
214 return IRQ_STARTUP_NORMAL;
215 /*
216 * The interrupt was requested, but there is no online CPU
217 * in it's affinity mask. Put it into managed shutdown
218 * state and let the cpu hotplug mechanism start it up once
219 * a CPU in the mask becomes available.
220 */
221 irqd_set_managed_shutdown(d);
222 return IRQ_STARTUP_ABORT;
223 }
224 return IRQ_STARTUP_MANAGED;
225}
226#else
Geert Uytterhoeven2372a512017-07-04 12:06:01 +0200227static __always_inline int
Thomas Gleixner761ea382017-06-20 01:37:50 +0200228__irq_startup_managed(struct irq_desc *desc, struct cpumask *aff, bool force)
229{
230 return IRQ_STARTUP_NORMAL;
231}
232#endif
233
Thomas Gleixner708d1742017-06-20 01:37:48 +0200234static int __irq_startup(struct irq_desc *desc)
235{
236 struct irq_data *d = irq_desc_get_irq_data(desc);
237 int ret = 0;
238
239 irq_domain_activate_irq(d);
240 if (d->chip->irq_startup) {
241 ret = d->chip->irq_startup(d);
242 irq_state_clr_disabled(desc);
243 irq_state_clr_masked(desc);
244 } else {
245 irq_enable(desc);
246 }
247 irq_state_set_started(desc);
248 return ret;
249}
250
Thomas Gleixner4cde9c62017-06-20 01:37:49 +0200251int irq_startup(struct irq_desc *desc, bool resend, bool force)
Thomas Gleixner46999232011-02-02 21:41:14 +0000252{
Thomas Gleixner761ea382017-06-20 01:37:50 +0200253 struct irq_data *d = irq_desc_get_irq_data(desc);
254 struct cpumask *aff = irq_data_get_affinity_mask(d);
Thomas Gleixnerb4bc7242012-02-08 11:57:52 +0100255 int ret = 0;
256
Thomas Gleixner46999232011-02-02 21:41:14 +0000257 desc->depth = 0;
258
Thomas Gleixner761ea382017-06-20 01:37:50 +0200259 if (irqd_is_started(d)) {
Thomas Gleixnerb4bc7242012-02-08 11:57:52 +0100260 irq_enable(desc);
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200261 } else {
Thomas Gleixner761ea382017-06-20 01:37:50 +0200262 switch (__irq_startup_managed(desc, aff, force)) {
263 case IRQ_STARTUP_NORMAL:
264 ret = __irq_startup(desc);
265 irq_setup_affinity(desc);
266 break;
267 case IRQ_STARTUP_MANAGED:
268 ret = __irq_startup(desc);
269 irq_set_affinity_locked(d, aff, false);
270 break;
271 case IRQ_STARTUP_ABORT:
272 return 0;
273 }
Thomas Gleixner3aae9942011-02-04 10:17:52 +0100274 }
Thomas Gleixnerb4bc7242012-02-08 11:57:52 +0100275 if (resend)
Jiang Liu0798abe2015-06-04 12:13:27 +0800276 check_irq_resend(desc);
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200277
Thomas Gleixnerb4bc7242012-02-08 11:57:52 +0100278 return ret;
Thomas Gleixner46999232011-02-02 21:41:14 +0000279}
280
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200281static void __irq_disable(struct irq_desc *desc, bool mask);
282
Thomas Gleixner46999232011-02-02 21:41:14 +0000283void irq_shutdown(struct irq_desc *desc)
284{
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200285 if (irqd_is_started(&desc->irq_data)) {
286 desc->depth = 1;
287 if (desc->irq_data.chip->irq_shutdown) {
288 desc->irq_data.chip->irq_shutdown(&desc->irq_data);
289 irq_state_set_disabled(desc);
290 irq_state_set_masked(desc);
291 } else {
292 __irq_disable(desc, true);
293 }
294 irq_state_clr_started(desc);
295 }
296 /*
297 * This must be called even if the interrupt was never started up,
298 * because the activation can happen before the interrupt is
299 * available for request/startup. It has it's own state tracking so
300 * it's safe to call it unconditionally.
301 */
Jiang Liuf8264e32014-11-06 22:20:14 +0800302 irq_domain_deactivate_irq(&desc->irq_data);
Thomas Gleixner46999232011-02-02 21:41:14 +0000303}
304
Thomas Gleixner87923472011-02-03 12:27:44 +0100305void irq_enable(struct irq_desc *desc)
306{
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800307 if (!irqd_irq_disabled(&desc->irq_data)) {
308 unmask_irq(desc);
309 } else {
310 irq_state_clr_disabled(desc);
311 if (desc->irq_data.chip->irq_enable) {
312 desc->irq_data.chip->irq_enable(&desc->irq_data);
313 irq_state_clr_masked(desc);
314 } else {
315 unmask_irq(desc);
316 }
317 }
Thomas Gleixner87923472011-02-03 12:27:44 +0100318}
319
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200320static void __irq_disable(struct irq_desc *desc, bool mask)
321{
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800322 if (irqd_irq_disabled(&desc->irq_data)) {
323 if (mask)
324 mask_irq(desc);
325 } else {
326 irq_state_set_disabled(desc);
327 if (desc->irq_data.chip->irq_disable) {
328 desc->irq_data.chip->irq_disable(&desc->irq_data);
329 irq_state_set_masked(desc);
330 } else if (mask) {
331 mask_irq(desc);
332 }
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200333 }
334}
335
Andreas Fenkartd671a602013-05-10 12:21:30 +0200336/**
Xie XiuQif788e7b2013-10-18 09:12:04 +0800337 * irq_disable - Mark interrupt disabled
Andreas Fenkartd671a602013-05-10 12:21:30 +0200338 * @desc: irq descriptor which should be disabled
339 *
340 * If the chip does not implement the irq_disable callback, we
341 * use a lazy disable approach. That means we mark the interrupt
342 * disabled, but leave the hardware unmasked. That's an
343 * optimization because we avoid the hardware access for the
344 * common case where no interrupt happens after we marked it
345 * disabled. If an interrupt happens, then the interrupt flow
346 * handler masks the line at the hardware level and marks it
347 * pending.
Thomas Gleixnere9849772015-10-09 23:28:58 +0200348 *
349 * If the interrupt chip does not implement the irq_disable callback,
350 * a driver can disable the lazy approach for a particular irq line by
351 * calling 'irq_set_status_flags(irq, IRQ_DISABLE_UNLAZY)'. This can
352 * be used for devices which cannot disable the interrupt at the
353 * device level under certain circumstances and have to use
354 * disable_irq[_nosync] instead.
Andreas Fenkartd671a602013-05-10 12:21:30 +0200355 */
Thomas Gleixner87923472011-02-03 12:27:44 +0100356void irq_disable(struct irq_desc *desc)
357{
Thomas Gleixner201d7f42017-05-31 11:58:32 +0200358 __irq_disable(desc, irq_settings_disable_unlazy(desc));
Thomas Gleixner89d694b2008-02-18 18:25:17 +0100359}
360
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100361void irq_percpu_enable(struct irq_desc *desc, unsigned int cpu)
362{
363 if (desc->irq_data.chip->irq_enable)
364 desc->irq_data.chip->irq_enable(&desc->irq_data);
365 else
366 desc->irq_data.chip->irq_unmask(&desc->irq_data);
367 cpumask_set_cpu(cpu, desc->percpu_enabled);
368}
369
370void irq_percpu_disable(struct irq_desc *desc, unsigned int cpu)
371{
372 if (desc->irq_data.chip->irq_disable)
373 desc->irq_data.chip->irq_disable(&desc->irq_data);
374 else
375 desc->irq_data.chip->irq_mask(&desc->irq_data);
376 cpumask_clear_cpu(cpu, desc->percpu_enabled);
377}
378
Thomas Gleixner9205e312010-09-27 12:44:50 +0000379static inline void mask_ack_irq(struct irq_desc *desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700380{
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800381 if (desc->irq_data.chip->irq_mask_ack) {
Thomas Gleixner9205e312010-09-27 12:44:50 +0000382 desc->irq_data.chip->irq_mask_ack(&desc->irq_data);
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800383 irq_state_set_masked(desc);
384 } else {
385 mask_irq(desc);
Thomas Gleixner22a49162010-09-27 12:44:47 +0000386 if (desc->irq_data.chip->irq_ack)
387 desc->irq_data.chip->irq_ack(&desc->irq_data);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700388 }
Thomas Gleixner0b1adaa2010-03-09 19:45:54 +0100389}
390
Thomas Gleixnerd4d5e082011-02-10 13:16:14 +0100391void mask_irq(struct irq_desc *desc)
Thomas Gleixner0b1adaa2010-03-09 19:45:54 +0100392{
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800393 if (irqd_irq_masked(&desc->irq_data))
394 return;
395
Thomas Gleixnere2c0f8f2010-09-27 12:44:42 +0000396 if (desc->irq_data.chip->irq_mask) {
397 desc->irq_data.chip->irq_mask(&desc->irq_data);
Thomas Gleixner6e402622011-02-08 12:36:06 +0100398 irq_state_set_masked(desc);
Thomas Gleixner0b1adaa2010-03-09 19:45:54 +0100399 }
400}
401
Thomas Gleixnerd4d5e082011-02-10 13:16:14 +0100402void unmask_irq(struct irq_desc *desc)
Thomas Gleixner0b1adaa2010-03-09 19:45:54 +0100403{
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800404 if (!irqd_irq_masked(&desc->irq_data))
405 return;
406
Thomas Gleixner0eda58b2010-09-27 12:44:44 +0000407 if (desc->irq_data.chip->irq_unmask) {
408 desc->irq_data.chip->irq_unmask(&desc->irq_data);
Thomas Gleixner6e402622011-02-08 12:36:06 +0100409 irq_state_clr_masked(desc);
Thomas Gleixner0b1adaa2010-03-09 19:45:54 +0100410 }
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700411}
412
Thomas Gleixner328a4972014-03-13 19:03:51 +0100413void unmask_threaded_irq(struct irq_desc *desc)
414{
415 struct irq_chip *chip = desc->irq_data.chip;
416
417 if (chip->flags & IRQCHIP_EOI_THREADED)
418 chip->irq_eoi(&desc->irq_data);
419
Jeffy Chenbf22ff452017-06-26 19:33:34 +0800420 unmask_irq(desc);
Thomas Gleixner328a4972014-03-13 19:03:51 +0100421}
422
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200423/*
424 * handle_nested_irq - Handle a nested irq from a irq thread
425 * @irq: the interrupt number
426 *
427 * Handle interrupts which are nested into a threaded interrupt
428 * handler. The handler function is called inside the calling
429 * threads context.
430 */
431void handle_nested_irq(unsigned int irq)
432{
433 struct irq_desc *desc = irq_to_desc(irq);
434 struct irqaction *action;
435 irqreturn_t action_ret;
436
437 might_sleep();
438
Thomas Gleixner239007b2009-11-17 16:46:45 +0100439 raw_spin_lock_irq(&desc->lock);
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200440
Thomas Gleixner293a7a02012-10-16 15:07:49 -0700441 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200442
443 action = desc->action;
Ning Jiang23812b92012-05-22 00:19:20 +0800444 if (unlikely(!action || irqd_irq_disabled(&desc->irq_data))) {
445 desc->istate |= IRQS_PENDING;
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200446 goto out_unlock;
Ning Jiang23812b92012-05-22 00:19:20 +0800447 }
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200448
Sudeep Hollaa946e8c2015-11-04 18:32:37 +0000449 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixner32f41252011-03-28 14:10:52 +0200450 irqd_set(&desc->irq_data, IRQD_IRQ_INPROGRESS);
Thomas Gleixner239007b2009-11-17 16:46:45 +0100451 raw_spin_unlock_irq(&desc->lock);
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200452
Charles Keepax45e52022017-03-07 16:28:18 +0000453 action_ret = IRQ_NONE;
454 for_each_action_of_desc(desc, action)
455 action_ret |= action->thread_fn(action->irq, action->dev_id);
456
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200457 if (!noirqdebug)
Jiang Liu0dcdbc92015-06-04 12:13:28 +0800458 note_interrupt(desc, action_ret);
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200459
Thomas Gleixner239007b2009-11-17 16:46:45 +0100460 raw_spin_lock_irq(&desc->lock);
Thomas Gleixner32f41252011-03-28 14:10:52 +0200461 irqd_clear(&desc->irq_data, IRQD_IRQ_INPROGRESS);
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200462
463out_unlock:
Thomas Gleixner239007b2009-11-17 16:46:45 +0100464 raw_spin_unlock_irq(&desc->lock);
Thomas Gleixner399b5da2009-08-13 13:21:38 +0200465}
466EXPORT_SYMBOL_GPL(handle_nested_irq);
467
Thomas Gleixnerfe200ae2011-02-07 10:34:30 +0100468static bool irq_check_poll(struct irq_desc *desc)
469{
Thomas Gleixner6954b752011-02-07 20:55:35 +0100470 if (!(desc->istate & IRQS_POLL_INPROGRESS))
Thomas Gleixnerfe200ae2011-02-07 10:34:30 +0100471 return false;
472 return irq_wait_for_poll(desc);
473}
474
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200475static bool irq_may_run(struct irq_desc *desc)
476{
Thomas Gleixner9ce7a252014-08-29 14:00:16 +0200477 unsigned int mask = IRQD_IRQ_INPROGRESS | IRQD_WAKEUP_ARMED;
478
479 /*
480 * If the interrupt is not in progress and is not an armed
481 * wakeup interrupt, proceed.
482 */
483 if (!irqd_has_set(&desc->irq_data, mask))
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200484 return true;
Thomas Gleixner9ce7a252014-08-29 14:00:16 +0200485
486 /*
487 * If the interrupt is an armed wakeup source, mark it pending
488 * and suspended, disable it and notify the pm core about the
489 * event.
490 */
491 if (irq_pm_check_wakeup(desc))
492 return false;
493
494 /*
495 * Handle a potential concurrent poll on a different core.
496 */
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200497 return irq_check_poll(desc);
498}
499
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700500/**
501 * handle_simple_irq - Simple and software-decoded IRQs.
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700502 * @desc: the interrupt description structure for this irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700503 *
504 * Simple interrupts are either sent from a demultiplexing interrupt
505 * handler or come from hardware, where no interrupt hardware control
506 * is necessary.
507 *
508 * Note: The caller is expected to handle the ack, clear, mask and
509 * unmask issues if necessary.
510 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200511void handle_simple_irq(struct irq_desc *desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700512{
Thomas Gleixner239007b2009-11-17 16:46:45 +0100513 raw_spin_lock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700514
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200515 if (!irq_may_run(desc))
516 goto out_unlock;
Thomas Gleixnerfe200ae2011-02-07 10:34:30 +0100517
Thomas Gleixner163ef302011-02-08 11:39:15 +0100518 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700519
Ning Jiang23812b92012-05-22 00:19:20 +0800520 if (unlikely(!desc->action || irqd_irq_disabled(&desc->irq_data))) {
521 desc->istate |= IRQS_PENDING;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700522 goto out_unlock;
Ning Jiang23812b92012-05-22 00:19:20 +0800523 }
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700524
Sudeep Hollaa946e8c2015-11-04 18:32:37 +0000525 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixner107781e2011-02-07 01:21:02 +0100526 handle_irq_event(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700527
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700528out_unlock:
Thomas Gleixner239007b2009-11-17 16:46:45 +0100529 raw_spin_unlock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700530}
Jonathan Cameronedf76f82011-05-18 10:39:04 +0100531EXPORT_SYMBOL_GPL(handle_simple_irq);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700532
Keith Buschedd14cf2016-06-17 16:00:20 -0600533/**
534 * handle_untracked_irq - Simple and software-decoded IRQs.
535 * @desc: the interrupt description structure for this irq
536 *
537 * Untracked interrupts are sent from a demultiplexing interrupt
538 * handler when the demultiplexer does not know which device it its
539 * multiplexed irq domain generated the interrupt. IRQ's handled
540 * through here are not subjected to stats tracking, randomness, or
541 * spurious interrupt detection.
542 *
543 * Note: Like handle_simple_irq, the caller is expected to handle
544 * the ack, clear, mask and unmask issues if necessary.
545 */
546void handle_untracked_irq(struct irq_desc *desc)
547{
548 unsigned int flags = 0;
549
550 raw_spin_lock(&desc->lock);
551
552 if (!irq_may_run(desc))
553 goto out_unlock;
554
555 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
556
557 if (unlikely(!desc->action || irqd_irq_disabled(&desc->irq_data))) {
558 desc->istate |= IRQS_PENDING;
559 goto out_unlock;
560 }
561
562 desc->istate &= ~IRQS_PENDING;
563 irqd_set(&desc->irq_data, IRQD_IRQ_INPROGRESS);
564 raw_spin_unlock(&desc->lock);
565
566 __handle_irq_event_percpu(desc, &flags);
567
568 raw_spin_lock(&desc->lock);
569 irqd_clear(&desc->irq_data, IRQD_IRQ_INPROGRESS);
570
571out_unlock:
572 raw_spin_unlock(&desc->lock);
573}
574EXPORT_SYMBOL_GPL(handle_untracked_irq);
575
Thomas Gleixnerac563762012-02-07 17:58:03 +0100576/*
577 * Called unconditionally from handle_level_irq() and only for oneshot
578 * interrupts from handle_fasteoi_irq()
579 */
580static void cond_unmask_irq(struct irq_desc *desc)
581{
582 /*
583 * We need to unmask in the following cases:
584 * - Standard level irq (IRQF_ONESHOT is not set)
585 * - Oneshot irq which did not wake the thread (caused by a
586 * spurious interrupt or a primary handler handling it
587 * completely).
588 */
589 if (!irqd_irq_disabled(&desc->irq_data) &&
590 irqd_irq_masked(&desc->irq_data) && !desc->threads_oneshot)
591 unmask_irq(desc);
592}
593
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700594/**
595 * handle_level_irq - Level type irq handler
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700596 * @desc: the interrupt description structure for this irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700597 *
598 * Level type interrupts are active as long as the hardware line has
599 * the active level. This may require to mask the interrupt and unmask
600 * it after the associated handler has acknowledged the device, so the
601 * interrupt line is back to inactive.
602 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200603void handle_level_irq(struct irq_desc *desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700604{
Thomas Gleixner239007b2009-11-17 16:46:45 +0100605 raw_spin_lock(&desc->lock);
Thomas Gleixner9205e312010-09-27 12:44:50 +0000606 mask_ack_irq(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700607
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200608 if (!irq_may_run(desc))
609 goto out_unlock;
Thomas Gleixnerfe200ae2011-02-07 10:34:30 +0100610
Thomas Gleixner163ef302011-02-08 11:39:15 +0100611 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700612
613 /*
614 * If its disabled or no action available
615 * keep it masked and get out of here
616 */
Thomas Gleixnerd4dc0f92012-04-25 12:54:54 +0200617 if (unlikely(!desc->action || irqd_irq_disabled(&desc->irq_data))) {
618 desc->istate |= IRQS_PENDING;
Ingo Molnar86998aa2006-09-19 11:14:34 +0200619 goto out_unlock;
Thomas Gleixnerd4dc0f92012-04-25 12:54:54 +0200620 }
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700621
Sudeep Hollaa946e8c2015-11-04 18:32:37 +0000622 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixner15298662011-02-07 01:22:17 +0100623 handle_irq_event(desc);
Thomas Gleixnerb25c3402009-08-13 12:17:22 +0200624
Thomas Gleixnerac563762012-02-07 17:58:03 +0100625 cond_unmask_irq(desc);
626
Ingo Molnar86998aa2006-09-19 11:14:34 +0200627out_unlock:
Thomas Gleixner239007b2009-11-17 16:46:45 +0100628 raw_spin_unlock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700629}
Ingo Molnar14819ea2009-01-14 12:34:21 +0100630EXPORT_SYMBOL_GPL(handle_level_irq);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700631
Thomas Gleixner78129572011-02-10 15:14:20 +0100632#ifdef CONFIG_IRQ_PREFLOW_FASTEOI
633static inline void preflow_handler(struct irq_desc *desc)
634{
635 if (desc->preflow_handler)
636 desc->preflow_handler(&desc->irq_data);
637}
638#else
639static inline void preflow_handler(struct irq_desc *desc) { }
640#endif
641
Thomas Gleixner328a4972014-03-13 19:03:51 +0100642static void cond_unmask_eoi_irq(struct irq_desc *desc, struct irq_chip *chip)
643{
644 if (!(desc->istate & IRQS_ONESHOT)) {
645 chip->irq_eoi(&desc->irq_data);
646 return;
647 }
648 /*
649 * We need to unmask in the following cases:
650 * - Oneshot irq which did not wake the thread (caused by a
651 * spurious interrupt or a primary handler handling it
652 * completely).
653 */
654 if (!irqd_irq_disabled(&desc->irq_data) &&
655 irqd_irq_masked(&desc->irq_data) && !desc->threads_oneshot) {
656 chip->irq_eoi(&desc->irq_data);
657 unmask_irq(desc);
658 } else if (!(chip->flags & IRQCHIP_EOI_THREADED)) {
659 chip->irq_eoi(&desc->irq_data);
660 }
661}
662
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700663/**
Ingo Molnar47c2a3a2006-06-29 02:25:03 -0700664 * handle_fasteoi_irq - irq handler for transparent controllers
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700665 * @desc: the interrupt description structure for this irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700666 *
Ingo Molnar47c2a3a2006-06-29 02:25:03 -0700667 * Only a single callback will be issued to the chip: an ->eoi()
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700668 * call when the interrupt has been serviced. This enables support
669 * for modern forms of interrupt handlers, which handle the flow
670 * details in hardware, transparently.
671 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200672void handle_fasteoi_irq(struct irq_desc *desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700673{
Thomas Gleixner328a4972014-03-13 19:03:51 +0100674 struct irq_chip *chip = desc->irq_data.chip;
675
Thomas Gleixner239007b2009-11-17 16:46:45 +0100676 raw_spin_lock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700677
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200678 if (!irq_may_run(desc))
679 goto out;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700680
Thomas Gleixner163ef302011-02-08 11:39:15 +0100681 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700682
683 /*
684 * If its disabled or no action available
Ingo Molnar76d21602007-02-16 01:28:24 -0800685 * then mask it and get out of here:
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700686 */
Thomas Gleixner32f41252011-03-28 14:10:52 +0200687 if (unlikely(!desc->action || irqd_irq_disabled(&desc->irq_data))) {
Thomas Gleixner2a0d6fb2011-02-08 12:17:57 +0100688 desc->istate |= IRQS_PENDING;
Thomas Gleixnere2c0f8f2010-09-27 12:44:42 +0000689 mask_irq(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700690 goto out;
Benjamin Herrenschmidt98bb2442006-06-29 02:25:01 -0700691 }
Thomas Gleixnerc69e3752011-03-02 11:49:21 +0100692
Sudeep Hollaa946e8c2015-11-04 18:32:37 +0000693 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixnerc69e3752011-03-02 11:49:21 +0100694 if (desc->istate & IRQS_ONESHOT)
695 mask_irq(desc);
696
Thomas Gleixner78129572011-02-10 15:14:20 +0100697 preflow_handler(desc);
Thomas Gleixnera7ae4de2011-02-07 01:23:07 +0100698 handle_irq_event(desc);
Thomas Gleixner77694b42011-02-15 10:33:57 +0100699
Thomas Gleixner328a4972014-03-13 19:03:51 +0100700 cond_unmask_eoi_irq(desc, chip);
Thomas Gleixnerac563762012-02-07 17:58:03 +0100701
Thomas Gleixner239007b2009-11-17 16:46:45 +0100702 raw_spin_unlock(&desc->lock);
Thomas Gleixner77694b42011-02-15 10:33:57 +0100703 return;
704out:
Thomas Gleixner328a4972014-03-13 19:03:51 +0100705 if (!(chip->flags & IRQCHIP_EOI_IF_HANDLED))
706 chip->irq_eoi(&desc->irq_data);
707 raw_spin_unlock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700708}
Vincent Stehlé7cad45e2014-08-22 01:31:20 +0200709EXPORT_SYMBOL_GPL(handle_fasteoi_irq);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700710
711/**
712 * handle_edge_irq - edge type IRQ handler
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700713 * @desc: the interrupt description structure for this irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700714 *
715 * Interrupt occures on the falling and/or rising edge of a hardware
Lucas De Marchi25985ed2011-03-30 22:57:33 -0300716 * signal. The occurrence is latched into the irq controller hardware
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700717 * and must be acked in order to be reenabled. After the ack another
718 * interrupt can happen on the same source even before the first one
Uwe Kleine-Königdfff0612010-02-12 21:58:11 +0100719 * is handled by the associated event handler. If this happens it
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700720 * might be necessary to disable (mask) the interrupt depending on the
721 * controller hardware. This requires to reenable the interrupt inside
722 * of the loop which handles the interrupts which have arrived while
723 * the handler was running. If all pending interrupts are handled, the
724 * loop is left.
725 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200726void handle_edge_irq(struct irq_desc *desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700727{
Thomas Gleixner239007b2009-11-17 16:46:45 +0100728 raw_spin_lock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700729
Thomas Gleixner163ef302011-02-08 11:39:15 +0100730 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
Thomas Gleixnerc3d7acd2014-08-29 13:46:08 +0200731
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200732 if (!irq_may_run(desc)) {
733 desc->istate |= IRQS_PENDING;
734 mask_ack_irq(desc);
735 goto out_unlock;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700736 }
Thomas Gleixnerc3d7acd2014-08-29 13:46:08 +0200737
738 /*
739 * If its disabled or no action available then mask it and get
740 * out of here.
741 */
742 if (irqd_irq_disabled(&desc->irq_data) || !desc->action) {
743 desc->istate |= IRQS_PENDING;
744 mask_ack_irq(desc);
745 goto out_unlock;
746 }
747
Jiang Liub51bf952015-06-04 12:13:25 +0800748 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700749
750 /* Start handling the irq */
Thomas Gleixner22a49162010-09-27 12:44:47 +0000751 desc->irq_data.chip->irq_ack(&desc->irq_data);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700752
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700753 do {
Thomas Gleixnera60a5dc2011-02-07 01:24:07 +0100754 if (unlikely(!desc->action)) {
Thomas Gleixnere2c0f8f2010-09-27 12:44:42 +0000755 mask_irq(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700756 goto out_unlock;
757 }
758
759 /*
760 * When another irq arrived while we were handling
761 * one, we could have masked the irq.
762 * Renable it, if it was not disabled in meantime.
763 */
Thomas Gleixner2a0d6fb2011-02-08 12:17:57 +0100764 if (unlikely(desc->istate & IRQS_PENDING)) {
Thomas Gleixner32f41252011-03-28 14:10:52 +0200765 if (!irqd_irq_disabled(&desc->irq_data) &&
766 irqd_irq_masked(&desc->irq_data))
Thomas Gleixnerc1594b72011-02-07 22:11:30 +0100767 unmask_irq(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700768 }
769
Thomas Gleixnera60a5dc2011-02-07 01:24:07 +0100770 handle_irq_event(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700771
Thomas Gleixner2a0d6fb2011-02-08 12:17:57 +0100772 } while ((desc->istate & IRQS_PENDING) &&
Thomas Gleixner32f41252011-03-28 14:10:52 +0200773 !irqd_irq_disabled(&desc->irq_data));
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700774
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700775out_unlock:
Thomas Gleixner239007b2009-11-17 16:46:45 +0100776 raw_spin_unlock(&desc->lock);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700777}
Jiri Kosina3911ff32012-05-13 12:13:15 +0200778EXPORT_SYMBOL(handle_edge_irq);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700779
Thomas Gleixner0521c8f2011-03-28 16:13:24 +0200780#ifdef CONFIG_IRQ_EDGE_EOI_HANDLER
781/**
782 * handle_edge_eoi_irq - edge eoi type IRQ handler
Thomas Gleixner0521c8f2011-03-28 16:13:24 +0200783 * @desc: the interrupt description structure for this irq
784 *
785 * Similar as the above handle_edge_irq, but using eoi and w/o the
786 * mask/unmask logic.
787 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200788void handle_edge_eoi_irq(struct irq_desc *desc)
Thomas Gleixner0521c8f2011-03-28 16:13:24 +0200789{
790 struct irq_chip *chip = irq_desc_get_chip(desc);
791
792 raw_spin_lock(&desc->lock);
793
794 desc->istate &= ~(IRQS_REPLAY | IRQS_WAITING);
Thomas Gleixnerc3d7acd2014-08-29 13:46:08 +0200795
Thomas Gleixnerc7bd3ec02014-08-29 13:39:37 +0200796 if (!irq_may_run(desc)) {
797 desc->istate |= IRQS_PENDING;
798 goto out_eoi;
Thomas Gleixner0521c8f2011-03-28 16:13:24 +0200799 }
Thomas Gleixnerc3d7acd2014-08-29 13:46:08 +0200800
801 /*
802 * If its disabled or no action available then mask it and get
803 * out of here.
804 */
805 if (irqd_irq_disabled(&desc->irq_data) || !desc->action) {
806 desc->istate |= IRQS_PENDING;
807 goto out_eoi;
808 }
809
Jiang Liub51bf952015-06-04 12:13:25 +0800810 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixner0521c8f2011-03-28 16:13:24 +0200811
812 do {
813 if (unlikely(!desc->action))
814 goto out_eoi;
815
816 handle_irq_event(desc);
817
818 } while ((desc->istate & IRQS_PENDING) &&
819 !irqd_irq_disabled(&desc->irq_data));
820
Stephen Rothwellac0e0442011-03-30 10:55:12 +1100821out_eoi:
Thomas Gleixner0521c8f2011-03-28 16:13:24 +0200822 chip->irq_eoi(&desc->irq_data);
823 raw_spin_unlock(&desc->lock);
824}
825#endif
826
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700827/**
Liuweni24b26d42009-11-04 20:11:05 +0800828 * handle_percpu_irq - Per CPU local irq handler
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700829 * @desc: the interrupt description structure for this irq
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700830 *
831 * Per CPU interrupts on SMP machines without locking requirements
832 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200833void handle_percpu_irq(struct irq_desc *desc)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700834{
Thomas Gleixner35e857c2011-02-10 12:20:23 +0100835 struct irq_chip *chip = irq_desc_get_chip(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700836
Jiang Liub51bf952015-06-04 12:13:25 +0800837 kstat_incr_irqs_this_cpu(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700838
Thomas Gleixner849f0612011-02-07 01:25:41 +0100839 if (chip->irq_ack)
840 chip->irq_ack(&desc->irq_data);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700841
Huang Shijie71f64342015-09-02 10:24:55 +0800842 handle_irq_event_percpu(desc);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700843
Thomas Gleixner849f0612011-02-07 01:25:41 +0100844 if (chip->irq_eoi)
845 chip->irq_eoi(&desc->irq_data);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700846}
847
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100848/**
849 * handle_percpu_devid_irq - Per CPU local irq handler with per cpu dev ids
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100850 * @desc: the interrupt description structure for this irq
851 *
852 * Per CPU interrupts on SMP machines without locking requirements. Same as
853 * handle_percpu_irq() above but with the following extras:
854 *
855 * action->percpu_dev_id is a pointer to percpu variables which
856 * contain the real device id for the cpu on which this handler is
857 * called
858 */
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200859void handle_percpu_devid_irq(struct irq_desc *desc)
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100860{
861 struct irq_chip *chip = irq_desc_get_chip(desc);
862 struct irqaction *action = desc->action;
Thomas Gleixnerbd0b9ac2015-09-14 10:42:37 +0200863 unsigned int irq = irq_desc_get_irq(desc);
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100864 irqreturn_t res;
865
Jiang Liub51bf952015-06-04 12:13:25 +0800866 kstat_incr_irqs_this_cpu(desc);
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100867
868 if (chip->irq_ack)
869 chip->irq_ack(&desc->irq_data);
870
Thomas Gleixnerfc590c22016-09-02 14:45:19 +0200871 if (likely(action)) {
872 trace_irq_handler_entry(irq, action);
873 res = action->handler(irq, raw_cpu_ptr(action->percpu_dev_id));
874 trace_irq_handler_exit(irq, action, res);
875 } else {
876 unsigned int cpu = smp_processor_id();
877 bool enabled = cpumask_test_cpu(cpu, desc->percpu_enabled);
878
879 if (enabled)
880 irq_percpu_disable(desc, cpu);
881
882 pr_err_once("Spurious%s percpu IRQ%u on CPU%u\n",
883 enabled ? " and unmasked" : "", irq, cpu);
884 }
Marc Zyngier31d9d9b2011-09-23 17:03:06 +0100885
886 if (chip->irq_eoi)
887 chip->irq_eoi(&desc->irq_data);
888}
889
Wei Yongjunb8129a12016-09-25 15:36:39 +0000890static void
Russell King3b0f95b2015-06-16 23:06:20 +0100891__irq_do_set_handler(struct irq_desc *desc, irq_flow_handler_t handle,
892 int is_chained, const char *name)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700893{
Thomas Gleixner091738a2011-02-14 20:16:43 +0100894 if (!handle) {
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700895 handle = handle_bad_irq;
Thomas Gleixner091738a2011-02-14 20:16:43 +0100896 } else {
Marc Zyngierf86eff22014-11-15 10:49:13 +0000897 struct irq_data *irq_data = &desc->irq_data;
898#ifdef CONFIG_IRQ_DOMAIN_HIERARCHY
899 /*
900 * With hierarchical domains we might run into a
901 * situation where the outermost chip is not yet set
902 * up, but the inner chips are there. Instead of
903 * bailing we install the handler, but obviously we
904 * cannot enable/startup the interrupt at this point.
905 */
906 while (irq_data) {
907 if (irq_data->chip != &no_irq_chip)
908 break;
909 /*
910 * Bail out if the outer chip is not set up
911 * and the interrrupt supposed to be started
912 * right away.
913 */
914 if (WARN_ON(is_chained))
Russell King3b0f95b2015-06-16 23:06:20 +0100915 return;
Marc Zyngierf86eff22014-11-15 10:49:13 +0000916 /* Try the parent */
917 irq_data = irq_data->parent_data;
918 }
919#endif
920 if (WARN_ON(!irq_data || irq_data->chip == &no_irq_chip))
Russell King3b0f95b2015-06-16 23:06:20 +0100921 return;
Thomas Gleixnerf8b54732006-07-01 22:30:08 +0100922 }
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700923
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700924 /* Uninstall? */
925 if (handle == handle_bad_irq) {
Thomas Gleixner6b8ff312010-10-01 12:58:38 +0200926 if (desc->irq_data.chip != &no_irq_chip)
Thomas Gleixner9205e312010-09-27 12:44:50 +0000927 mask_ack_irq(desc);
Thomas Gleixner801a0e92011-03-27 11:02:49 +0200928 irq_state_set_disabled(desc);
Mika Westerberge509bd72015-10-05 13:12:15 +0300929 if (is_chained)
930 desc->action = NULL;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700931 desc->depth = 1;
932 }
933 desc->handle_irq = handle;
Ingo Molnara460e742006-10-17 00:10:03 -0700934 desc->name = name;
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700935
936 if (handle != handle_bad_irq && is_chained) {
Marc Zyngier1984e072016-09-19 09:49:27 +0100937 unsigned int type = irqd_get_trigger_type(&desc->irq_data);
938
Marc Zyngier1e12c4a2016-08-11 14:19:42 +0100939 /*
940 * We're about to start this interrupt immediately,
941 * hence the need to set the trigger configuration.
942 * But the .set_type callback may have overridden the
943 * flow handler, ignoring that we're dealing with a
944 * chained interrupt. Reset it immediately because we
945 * do know better.
946 */
Marc Zyngier1984e072016-09-19 09:49:27 +0100947 if (type != IRQ_TYPE_NONE) {
948 __irq_set_trigger(desc, type);
949 desc->handle_irq = handle;
950 }
Marc Zyngier1e12c4a2016-08-11 14:19:42 +0100951
Thomas Gleixner1ccb4e62011-02-09 14:44:17 +0100952 irq_settings_set_noprobe(desc);
953 irq_settings_set_norequest(desc);
Paul Mundt7f1b1242011-04-07 06:01:44 +0900954 irq_settings_set_nothread(desc);
Mika Westerberge509bd72015-10-05 13:12:15 +0300955 desc->action = &chained_action;
Thomas Gleixner4cde9c62017-06-20 01:37:49 +0200956 irq_startup(desc, IRQ_RESEND, IRQ_START_FORCE);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700957 }
Russell King3b0f95b2015-06-16 23:06:20 +0100958}
959
960void
961__irq_set_handler(unsigned int irq, irq_flow_handler_t handle, int is_chained,
962 const char *name)
963{
964 unsigned long flags;
965 struct irq_desc *desc = irq_get_desc_buslock(irq, &flags, 0);
966
967 if (!desc)
968 return;
969
970 __irq_do_set_handler(desc, handle, is_chained, name);
Thomas Gleixner02725e72011-02-12 10:37:36 +0100971 irq_put_desc_busunlock(desc, flags);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700972}
Thomas Gleixner3836ca02011-02-14 20:09:19 +0100973EXPORT_SYMBOL_GPL(__irq_set_handler);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700974
975void
Russell King3b0f95b2015-06-16 23:06:20 +0100976irq_set_chained_handler_and_data(unsigned int irq, irq_flow_handler_t handle,
977 void *data)
978{
979 unsigned long flags;
980 struct irq_desc *desc = irq_get_desc_buslock(irq, &flags, 0);
981
982 if (!desc)
983 return;
984
Jiang Liuaf7080e2015-06-01 16:05:21 +0800985 desc->irq_common_data.handler_data = data;
Thomas Gleixner2c4569ca2017-05-11 13:54:11 +0200986 __irq_do_set_handler(desc, handle, 1, NULL);
Russell King3b0f95b2015-06-16 23:06:20 +0100987
988 irq_put_desc_busunlock(desc, flags);
989}
990EXPORT_SYMBOL_GPL(irq_set_chained_handler_and_data);
991
992void
Thomas Gleixner3836ca02011-02-14 20:09:19 +0100993irq_set_chip_and_handler_name(unsigned int irq, struct irq_chip *chip,
Ingo Molnara460e742006-10-17 00:10:03 -0700994 irq_flow_handler_t handle, const char *name)
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700995{
Thomas Gleixner35e857c2011-02-10 12:20:23 +0100996 irq_set_chip(irq, chip);
Thomas Gleixner3836ca02011-02-14 20:09:19 +0100997 __irq_set_handler(irq, handle, 0, name);
Thomas Gleixnerdd87eb32006-06-29 02:24:53 -0700998}
Kuninori Morimotob3ae66f2012-07-30 22:39:06 -0700999EXPORT_SYMBOL_GPL(irq_set_chip_and_handler_name);
Ralf Baechle46f4f8f2008-02-08 04:22:01 -08001000
Thomas Gleixner44247182010-09-28 10:40:18 +02001001void irq_modify_status(unsigned int irq, unsigned long clr, unsigned long set)
Ralf Baechle46f4f8f2008-02-08 04:22:01 -08001002{
Ralf Baechle46f4f8f2008-02-08 04:22:01 -08001003 unsigned long flags;
Marc Zyngier31d9d9b2011-09-23 17:03:06 +01001004 struct irq_desc *desc = irq_get_desc_lock(irq, &flags, 0);
Ralf Baechle46f4f8f2008-02-08 04:22:01 -08001005
Thomas Gleixner44247182010-09-28 10:40:18 +02001006 if (!desc)
Ralf Baechle46f4f8f2008-02-08 04:22:01 -08001007 return;
Thomas Gleixner04c848d2017-05-31 11:58:33 +02001008
1009 /*
1010 * Warn when a driver sets the no autoenable flag on an already
1011 * active interrupt.
1012 */
1013 WARN_ON_ONCE(!desc->depth && (set & _IRQ_NOAUTOEN));
1014
Thomas Gleixnera0056772011-02-08 17:11:03 +01001015 irq_settings_clr_and_set(desc, clr, set);
1016
Thomas Gleixner876dbd42011-02-08 17:28:12 +01001017 irqd_clear(&desc->irq_data, IRQD_NO_BALANCING | IRQD_PER_CPU |
Thomas Gleixnere1ef8242011-02-10 22:25:31 +01001018 IRQD_TRIGGER_MASK | IRQD_LEVEL | IRQD_MOVE_PCNTXT);
Thomas Gleixnera0056772011-02-08 17:11:03 +01001019 if (irq_settings_has_no_balance_set(desc))
1020 irqd_set(&desc->irq_data, IRQD_NO_BALANCING);
1021 if (irq_settings_is_per_cpu(desc))
1022 irqd_set(&desc->irq_data, IRQD_PER_CPU);
Thomas Gleixnere1ef8242011-02-10 22:25:31 +01001023 if (irq_settings_can_move_pcntxt(desc))
1024 irqd_set(&desc->irq_data, IRQD_MOVE_PCNTXT);
Thomas Gleixner0ef5ca12011-03-28 21:59:37 +02001025 if (irq_settings_is_level(desc))
1026 irqd_set(&desc->irq_data, IRQD_LEVEL);
Thomas Gleixnera0056772011-02-08 17:11:03 +01001027
Thomas Gleixner876dbd42011-02-08 17:28:12 +01001028 irqd_set(&desc->irq_data, irq_settings_get_trigger_mask(desc));
1029
Thomas Gleixner02725e72011-02-12 10:37:36 +01001030 irq_put_desc_unlock(desc, flags);
Ralf Baechle46f4f8f2008-02-08 04:22:01 -08001031}
Jonathan Cameronedf76f82011-05-18 10:39:04 +01001032EXPORT_SYMBOL_GPL(irq_modify_status);
David Daney0fdb4b22011-03-25 12:38:49 -07001033
1034/**
1035 * irq_cpu_online - Invoke all irq_cpu_online functions.
1036 *
1037 * Iterate through all irqs and invoke the chip.irq_cpu_online()
1038 * for each.
1039 */
1040void irq_cpu_online(void)
1041{
1042 struct irq_desc *desc;
1043 struct irq_chip *chip;
1044 unsigned long flags;
1045 unsigned int irq;
1046
1047 for_each_active_irq(irq) {
1048 desc = irq_to_desc(irq);
1049 if (!desc)
1050 continue;
1051
1052 raw_spin_lock_irqsave(&desc->lock, flags);
1053
1054 chip = irq_data_get_irq_chip(&desc->irq_data);
Thomas Gleixnerb3d42232011-03-27 16:05:36 +02001055 if (chip && chip->irq_cpu_online &&
1056 (!(chip->flags & IRQCHIP_ONOFFLINE_ENABLED) ||
Thomas Gleixner32f41252011-03-28 14:10:52 +02001057 !irqd_irq_disabled(&desc->irq_data)))
David Daney0fdb4b22011-03-25 12:38:49 -07001058 chip->irq_cpu_online(&desc->irq_data);
1059
1060 raw_spin_unlock_irqrestore(&desc->lock, flags);
1061 }
1062}
1063
1064/**
1065 * irq_cpu_offline - Invoke all irq_cpu_offline functions.
1066 *
1067 * Iterate through all irqs and invoke the chip.irq_cpu_offline()
1068 * for each.
1069 */
1070void irq_cpu_offline(void)
1071{
1072 struct irq_desc *desc;
1073 struct irq_chip *chip;
1074 unsigned long flags;
1075 unsigned int irq;
1076
1077 for_each_active_irq(irq) {
1078 desc = irq_to_desc(irq);
1079 if (!desc)
1080 continue;
1081
1082 raw_spin_lock_irqsave(&desc->lock, flags);
1083
1084 chip = irq_data_get_irq_chip(&desc->irq_data);
Thomas Gleixnerb3d42232011-03-27 16:05:36 +02001085 if (chip && chip->irq_cpu_offline &&
1086 (!(chip->flags & IRQCHIP_ONOFFLINE_ENABLED) ||
Thomas Gleixner32f41252011-03-28 14:10:52 +02001087 !irqd_irq_disabled(&desc->irq_data)))
David Daney0fdb4b22011-03-25 12:38:49 -07001088 chip->irq_cpu_offline(&desc->irq_data);
1089
1090 raw_spin_unlock_irqrestore(&desc->lock, flags);
1091 }
1092}
Jiang Liu85f08c12014-11-06 22:20:16 +08001093
1094#ifdef CONFIG_IRQ_DOMAIN_HIERARCHY
1095/**
Stefan Agner3cfeffc2015-05-16 11:44:14 +02001096 * irq_chip_enable_parent - Enable the parent interrupt (defaults to unmask if
1097 * NULL)
1098 * @data: Pointer to interrupt specific data
1099 */
1100void irq_chip_enable_parent(struct irq_data *data)
1101{
1102 data = data->parent_data;
1103 if (data->chip->irq_enable)
1104 data->chip->irq_enable(data);
1105 else
1106 data->chip->irq_unmask(data);
1107}
1108
1109/**
1110 * irq_chip_disable_parent - Disable the parent interrupt (defaults to mask if
1111 * NULL)
1112 * @data: Pointer to interrupt specific data
1113 */
1114void irq_chip_disable_parent(struct irq_data *data)
1115{
1116 data = data->parent_data;
1117 if (data->chip->irq_disable)
1118 data->chip->irq_disable(data);
1119 else
1120 data->chip->irq_mask(data);
1121}
1122
1123/**
Jiang Liu85f08c12014-11-06 22:20:16 +08001124 * irq_chip_ack_parent - Acknowledge the parent interrupt
1125 * @data: Pointer to interrupt specific data
1126 */
1127void irq_chip_ack_parent(struct irq_data *data)
1128{
1129 data = data->parent_data;
1130 data->chip->irq_ack(data);
1131}
Jake Oshinsa4289dc2015-12-10 17:52:59 +00001132EXPORT_SYMBOL_GPL(irq_chip_ack_parent);
Jiang Liu85f08c12014-11-06 22:20:16 +08001133
1134/**
Yingjoe Chen56e8aba2014-11-13 23:37:05 +08001135 * irq_chip_mask_parent - Mask the parent interrupt
1136 * @data: Pointer to interrupt specific data
1137 */
1138void irq_chip_mask_parent(struct irq_data *data)
1139{
1140 data = data->parent_data;
1141 data->chip->irq_mask(data);
1142}
Quan Nguyen52b2a052016-03-03 21:56:52 +07001143EXPORT_SYMBOL_GPL(irq_chip_mask_parent);
Yingjoe Chen56e8aba2014-11-13 23:37:05 +08001144
1145/**
1146 * irq_chip_unmask_parent - Unmask the parent interrupt
1147 * @data: Pointer to interrupt specific data
1148 */
1149void irq_chip_unmask_parent(struct irq_data *data)
1150{
1151 data = data->parent_data;
1152 data->chip->irq_unmask(data);
1153}
Quan Nguyen52b2a052016-03-03 21:56:52 +07001154EXPORT_SYMBOL_GPL(irq_chip_unmask_parent);
Yingjoe Chen56e8aba2014-11-13 23:37:05 +08001155
1156/**
1157 * irq_chip_eoi_parent - Invoke EOI on the parent interrupt
1158 * @data: Pointer to interrupt specific data
1159 */
1160void irq_chip_eoi_parent(struct irq_data *data)
1161{
1162 data = data->parent_data;
1163 data->chip->irq_eoi(data);
1164}
Quan Nguyen52b2a052016-03-03 21:56:52 +07001165EXPORT_SYMBOL_GPL(irq_chip_eoi_parent);
Yingjoe Chen56e8aba2014-11-13 23:37:05 +08001166
1167/**
1168 * irq_chip_set_affinity_parent - Set affinity on the parent interrupt
1169 * @data: Pointer to interrupt specific data
1170 * @dest: The affinity mask to set
1171 * @force: Flag to enforce setting (disable online checks)
1172 *
1173 * Conditinal, as the underlying parent chip might not implement it.
1174 */
1175int irq_chip_set_affinity_parent(struct irq_data *data,
1176 const struct cpumask *dest, bool force)
1177{
1178 data = data->parent_data;
1179 if (data->chip->irq_set_affinity)
1180 return data->chip->irq_set_affinity(data, dest, force);
1181
1182 return -ENOSYS;
1183}
1184
1185/**
Grygorii Strashkob7560de2015-08-14 15:20:26 +03001186 * irq_chip_set_type_parent - Set IRQ type on the parent interrupt
1187 * @data: Pointer to interrupt specific data
1188 * @type: IRQ_TYPE_{LEVEL,EDGE}_* value - see include/linux/irq.h
1189 *
1190 * Conditional, as the underlying parent chip might not implement it.
1191 */
1192int irq_chip_set_type_parent(struct irq_data *data, unsigned int type)
1193{
1194 data = data->parent_data;
1195
1196 if (data->chip->irq_set_type)
1197 return data->chip->irq_set_type(data, type);
1198
1199 return -ENOSYS;
1200}
Quan Nguyen52b2a052016-03-03 21:56:52 +07001201EXPORT_SYMBOL_GPL(irq_chip_set_type_parent);
Grygorii Strashkob7560de2015-08-14 15:20:26 +03001202
1203/**
Jiang Liu85f08c12014-11-06 22:20:16 +08001204 * irq_chip_retrigger_hierarchy - Retrigger an interrupt in hardware
1205 * @data: Pointer to interrupt specific data
1206 *
1207 * Iterate through the domain hierarchy of the interrupt and check
1208 * whether a hw retrigger function exists. If yes, invoke it.
1209 */
1210int irq_chip_retrigger_hierarchy(struct irq_data *data)
1211{
1212 for (data = data->parent_data; data; data = data->parent_data)
1213 if (data->chip && data->chip->irq_retrigger)
1214 return data->chip->irq_retrigger(data);
1215
Grygorii Strashko6d4affe2015-08-14 15:20:25 +03001216 return 0;
Jiang Liu85f08c12014-11-06 22:20:16 +08001217}
Marc Zyngier08b55e22015-03-11 15:43:43 +00001218
1219/**
Jiang Liu0a4377d2015-05-19 17:07:14 +08001220 * irq_chip_set_vcpu_affinity_parent - Set vcpu affinity on the parent interrupt
1221 * @data: Pointer to interrupt specific data
Masanari Iida8505a812015-07-29 19:09:36 +09001222 * @vcpu_info: The vcpu affinity information
Jiang Liu0a4377d2015-05-19 17:07:14 +08001223 */
1224int irq_chip_set_vcpu_affinity_parent(struct irq_data *data, void *vcpu_info)
1225{
1226 data = data->parent_data;
1227 if (data->chip->irq_set_vcpu_affinity)
1228 return data->chip->irq_set_vcpu_affinity(data, vcpu_info);
1229
1230 return -ENOSYS;
1231}
1232
1233/**
Marc Zyngier08b55e22015-03-11 15:43:43 +00001234 * irq_chip_set_wake_parent - Set/reset wake-up on the parent interrupt
1235 * @data: Pointer to interrupt specific data
1236 * @on: Whether to set or reset the wake-up capability of this irq
1237 *
1238 * Conditional, as the underlying parent chip might not implement it.
1239 */
1240int irq_chip_set_wake_parent(struct irq_data *data, unsigned int on)
1241{
1242 data = data->parent_data;
1243 if (data->chip->irq_set_wake)
1244 return data->chip->irq_set_wake(data, on);
1245
1246 return -ENOSYS;
1247}
Jiang Liu85f08c12014-11-06 22:20:16 +08001248#endif
Jiang Liu515085e2014-11-06 22:20:17 +08001249
1250/**
1251 * irq_chip_compose_msi_msg - Componse msi message for a irq chip
1252 * @data: Pointer to interrupt specific data
1253 * @msg: Pointer to the MSI message
1254 *
1255 * For hierarchical domains we find the first chip in the hierarchy
1256 * which implements the irq_compose_msi_msg callback. For non
1257 * hierarchical we use the top level chip.
1258 */
1259int irq_chip_compose_msi_msg(struct irq_data *data, struct msi_msg *msg)
1260{
1261 struct irq_data *pos = NULL;
1262
1263#ifdef CONFIG_IRQ_DOMAIN_HIERARCHY
1264 for (; data; data = data->parent_data)
1265#endif
1266 if (data->chip && data->chip->irq_compose_msi_msg)
1267 pos = data;
1268 if (!pos)
1269 return -ENOSYS;
1270
1271 pos->chip->irq_compose_msi_msg(pos, msg);
1272
1273 return 0;
1274}
Jon Hunterbe45beb2016-06-07 16:12:29 +01001275
1276/**
1277 * irq_chip_pm_get - Enable power for an IRQ chip
1278 * @data: Pointer to interrupt specific data
1279 *
1280 * Enable the power to the IRQ chip referenced by the interrupt data
1281 * structure.
1282 */
1283int irq_chip_pm_get(struct irq_data *data)
1284{
1285 int retval;
1286
1287 if (IS_ENABLED(CONFIG_PM) && data->chip->parent_device) {
1288 retval = pm_runtime_get_sync(data->chip->parent_device);
1289 if (retval < 0) {
1290 pm_runtime_put_noidle(data->chip->parent_device);
1291 return retval;
1292 }
1293 }
1294
1295 return 0;
1296}
1297
1298/**
1299 * irq_chip_pm_put - Disable power for an IRQ chip
1300 * @data: Pointer to interrupt specific data
1301 *
1302 * Disable the power to the IRQ chip referenced by the interrupt data
1303 * structure, belongs. Note that power will only be disabled, once this
1304 * function has been called for all IRQs that have called irq_chip_pm_get().
1305 */
1306int irq_chip_pm_put(struct irq_data *data)
1307{
1308 int retval = 0;
1309
1310 if (IS_ENABLED(CONFIG_PM) && data->chip->parent_device)
1311 retval = pm_runtime_put(data->chip->parent_device);
1312
1313 return (retval < 0) ? retval : 0;
1314}