Linux 5.7.6
[linux/fpc-iii.git] / arch / mips / lasat / interrupt.c
blob7965bbd0d319d580626dacfacf65d2ca49c50836
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Carsten Langgaard, carstenl@mips.com
4 * Copyright (C) 1999,2000 MIPS Technologies, Inc. All rights reserved.
6 * Routines for generic manipulation of the interrupts found on the
7 * Lasat boards.
8 */
9 #include <linux/init.h>
10 #include <linux/interrupt.h>
11 #include <linux/irq.h>
13 #include <asm/irq_cpu.h>
14 #include <asm/lasat/lasat.h>
15 #include <asm/lasat/lasatint.h>
17 #include <irq.h>
19 static volatile int *lasat_int_status;
20 static volatile int *lasat_int_mask;
21 static volatile int lasat_int_mask_shift;
23 void disable_lasat_irq(struct irq_data *d)
25 unsigned int irq_nr = d->irq - LASAT_IRQ_BASE;
27 *lasat_int_mask &= ~(1 << irq_nr) << lasat_int_mask_shift;
30 void enable_lasat_irq(struct irq_data *d)
32 unsigned int irq_nr = d->irq - LASAT_IRQ_BASE;
34 *lasat_int_mask |= (1 << irq_nr) << lasat_int_mask_shift;
37 static struct irq_chip lasat_irq_type = {
38 .name = "Lasat",
39 .irq_mask = disable_lasat_irq,
40 .irq_unmask = enable_lasat_irq,
43 static inline int ls1bit32(unsigned int x)
45 int b = 31, s;
47 s = 16; if (x << 16 == 0) s = 0; b -= s; x <<= s;
48 s = 8; if (x << 8 == 0) s = 0; b -= s; x <<= s;
49 s = 4; if (x << 4 == 0) s = 0; b -= s; x <<= s;
50 s = 2; if (x << 2 == 0) s = 0; b -= s; x <<= s;
51 s = 1; if (x << 1 == 0) s = 0; b -= s;
53 return b;
56 static unsigned long (*get_int_status)(void);
58 static unsigned long get_int_status_100(void)
60 return *lasat_int_status & *lasat_int_mask;
63 static unsigned long get_int_status_200(void)
65 unsigned long int_status;
67 int_status = *lasat_int_status;
68 int_status &= (int_status >> LASATINT_MASK_SHIFT_200) & 0xffff;
69 return int_status;
72 asmlinkage void plat_irq_dispatch(void)
74 unsigned long int_status;
75 unsigned int cause = read_c0_cause();
76 int irq;
78 if (cause & CAUSEF_IP7) { /* R4000 count / compare IRQ */
79 do_IRQ(7);
80 return;
83 int_status = get_int_status();
85 /* if int_status == 0, then the interrupt has already been cleared */
86 if (int_status) {
87 irq = LASAT_IRQ_BASE + ls1bit32(int_status);
89 do_IRQ(irq);
93 void __init arch_init_irq(void)
95 int irq = LASAT_CASCADE_IRQ;
96 int i;
98 if (IS_LASAT_200()) {
99 lasat_int_status = (void *)LASAT_INT_STATUS_REG_200;
100 lasat_int_mask = (void *)LASAT_INT_MASK_REG_200;
101 lasat_int_mask_shift = LASATINT_MASK_SHIFT_200;
102 get_int_status = get_int_status_200;
103 *lasat_int_mask &= 0xffff;
104 } else {
105 lasat_int_status = (void *)LASAT_INT_STATUS_REG_100;
106 lasat_int_mask = (void *)LASAT_INT_MASK_REG_100;
107 lasat_int_mask_shift = LASATINT_MASK_SHIFT_100;
108 get_int_status = get_int_status_100;
109 *lasat_int_mask = 0;
112 mips_cpu_irq_init();
114 for (i = LASAT_IRQ_BASE; i <= LASAT_IRQ_END; i++)
115 irq_set_chip_and_handler(i, &lasat_irq_type, handle_level_irq);
117 if (request_irq(irq, no_action, IRQF_NO_THREAD, "cascade", NULL))
118 pr_err("Failed to request irq %d (cascade)\n", irq);