irq.c 7.5 KB
Newer Older
1
/*
2
 * linux/arch/arm/mach-omap2/irq.c
3 4 5 6 7 8 9 10 11 12 13
 *
 * Interrupt handler for OMAP2 boards.
 *
 * Copyright (C) 2005 Nokia Corporation
 * Author: Paul Mundt <paul.mundt@nokia.com>
 *
 * This file is subject to the terms and conditions of the GNU General Public
 * License. See the file "COPYING" in the main directory of this archive
 * for more details.
 */
#include <linux/kernel.h>
14
#include <linux/module.h>
15 16
#include <linux/init.h>
#include <linux/interrupt.h>
17
#include <linux/io.h>
18

19
#include <asm/exception.h>
20
#include <asm/mach/irq.h>
21 22 23
#include <linux/irqdomain.h>
#include <linux/of.h>
#include <linux/of_address.h>
24
#include <linux/of_irq.h>
25

26
#include "soc.h"
27
#include "iomap.h"
28
#include "common.h"
29 30 31 32 33 34

/* selected INTC register offsets */

#define INTC_REVISION		0x0000
#define INTC_SYSCONFIG		0x0010
#define INTC_SYSSTATUS		0x0014
35
#define INTC_SIR		0x0040
36
#define INTC_CONTROL		0x0048
37 38 39 40
#define INTC_PROTECTION		0x004C
#define INTC_IDLE		0x0050
#define INTC_THRESHOLD		0x0068
#define INTC_MIR0		0x0084
41 42 43
#define INTC_MIR_CLEAR0		0x0088
#define INTC_MIR_SET0		0x008c
#define INTC_PENDING_IRQ0	0x0098
44 45 46
#define INTC_PENDING_IRQ1	0x00b8
#define INTC_PENDING_IRQ2	0x00d8
#define INTC_PENDING_IRQ3	0x00f8
47
#define INTC_ILR0		0x0100
48

49
#define ACTIVEIRQ_MASK		0x7f	/* omap2/3 active interrupt bits */
50
#define INTCPS_NR_ILR_REGS	128
51
#define INTCPS_NR_MIR_REGS	3
52

53 54 55 56 57 58 59
/*
 * OMAP2 has a number of different interrupt controllers, each interrupt
 * controller is identified as its own "bank". Register definitions are
 * fairly consistent for each bank, but not all registers are implemented
 * for each bank.. when in doubt, consult the TRM.
 */

60
/* Structure to save interrupt controller context */
61
struct omap_intc_regs {
62 63 64 65
	u32 sysconfig;
	u32 protection;
	u32 idle;
	u32 threshold;
66
	u32 ilr[INTCPS_NR_ILR_REGS];
67 68
	u32 mir[INTCPS_NR_MIR_REGS];
};
69 70 71 72 73
static struct omap_intc_regs intc_context;

static struct irq_domain *domain;
static void __iomem *omap_irq_base;
static int omap_nr_irqs = 96;
74

75
/* INTC bank register get/set */
76
static void intc_writel(u32 reg, u32 val)
77
{
78
	writel_relaxed(val, omap_irq_base + reg);
79 80
}

81
static u32 intc_readl(u32 reg)
82
{
83
	return readl_relaxed(omap_irq_base + reg);
84 85
}

86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
void omap_intc_save_context(void)
{
	int i;

	intc_context.sysconfig =
		intc_readl(INTC_SYSCONFIG);
	intc_context.protection =
		intc_readl(INTC_PROTECTION);
	intc_context.idle =
		intc_readl(INTC_IDLE);
	intc_context.threshold =
		intc_readl(INTC_THRESHOLD);

	for (i = 0; i < omap_nr_irqs; i++)
		intc_context.ilr[i] =
			intc_readl((INTC_ILR0 + 0x4 * i));
	for (i = 0; i < INTCPS_NR_MIR_REGS; i++)
		intc_context.mir[i] =
			intc_readl(INTC_MIR0 + (0x20 * i));
}

void omap_intc_restore_context(void)
{
	int i;

	intc_writel(INTC_SYSCONFIG, intc_context.sysconfig);
	intc_writel(INTC_PROTECTION, intc_context.protection);
	intc_writel(INTC_IDLE, intc_context.idle);
	intc_writel(INTC_THRESHOLD, intc_context.threshold);

	for (i = 0; i < omap_nr_irqs; i++)
		intc_writel(INTC_ILR0 + 0x4 * i,
				intc_context.ilr[i]);

	for (i = 0; i < INTCPS_NR_MIR_REGS; i++)
		intc_writel(INTC_MIR0 + 0x20 * i,
			intc_context.mir[i]);
	/* MIRs are saved and restore with other PRCM registers */
}

void omap3_intc_prepare_idle(void)
{
	/*
	 * Disable autoidle as it can stall interrupt controller,
	 * cf. errata ID i540 for 3430 (all revisions up to 3.1.x)
	 */
	intc_writel(INTC_SYSCONFIG, 0);
}

void omap3_intc_resume_idle(void)
{
	/* Re-enable autoidle */
	intc_writel(INTC_SYSCONFIG, 1);
}

141
/* XXX: FIQ and additional INTC support (only MPU at the moment) */
142
static void omap_ack_irq(struct irq_data *d)
143
{
144
	intc_writel(INTC_CONTROL, 0x1);
145 146
}

147
static void omap_mask_ack_irq(struct irq_data *d)
148
{
149
	irq_gc_mask_disable_reg(d);
150
	omap_ack_irq(d);
151 152
}

153
static void __init omap_irq_soft_reset(void)
154 155 156
{
	unsigned long tmp;

157
	tmp = intc_readl(INTC_REVISION) & 0xff;
158

P
Paul Walmsley 已提交
159
	pr_info("IRQ: Found an INTC at 0x%p (revision %ld.%ld) with %d interrupts\n",
160
		omap_irq_base, tmp >> 4, tmp & 0xf, omap_nr_irqs);
161

162
	tmp = intc_readl(INTC_SYSCONFIG);
163
	tmp |= 1 << 1;	/* soft reset */
164
	intc_writel(INTC_SYSCONFIG, tmp);
165

166
	while (!(intc_readl(INTC_SYSSTATUS) & 0x1))
167
		/* Wait for reset to complete */;
168 169

	/* Enable autoidle */
170
	intc_writel(INTC_SYSCONFIG, 1 << 0);
171 172
}

173 174
int omap_irq_pending(void)
{
175
	int irq;
176

177 178 179 180
	for (irq = 0; irq < omap_nr_irqs; irq += 32)
		if (intc_readl(INTC_PENDING_IRQ0 +
					((irq >> 5) << 5)))
			return 1;
181 182 183
	return 0;
}

184 185 186 187 188 189
void omap3_intc_suspend(void)
{
	/* A pending interrupt would prevent OMAP from entering suspend */
	omap_ack_irq(NULL);
}

190 191 192 193 194 195 196 197 198 199 200 201
static __init void
omap_alloc_gc(void __iomem *base, unsigned int irq_start, unsigned int num)
{
	struct irq_chip_generic *gc;
	struct irq_chip_type *ct;

	gc = irq_alloc_generic_chip("INTC", 1, irq_start, base,
					handle_level_irq);
	ct = gc->chip_types;
	ct->chip.irq_ack = omap_mask_ack_irq;
	ct->chip.irq_mask = irq_gc_mask_disable_reg;
	ct->chip.irq_unmask = irq_gc_unmask_enable_reg;
202
	ct->chip.flags |= IRQCHIP_SKIP_SET_WAKE;
203 204 205 206 207 208 209

	ct->regs.enable = INTC_MIR_CLEAR0;
	ct->regs.disable = INTC_MIR_SET0;
	irq_setup_generic_chip(gc, IRQ_MSK(num), IRQ_GC_INIT_MASK_CACHE,
				IRQ_NOREQUEST | IRQ_NOPROBE, 0);
}

210 211
static void __init omap_init_irq(u32 base, int nr_irqs,
				 struct device_node *node)
212
{
213
	int j, irq_base;
214

215 216 217 218
	omap_irq_base = ioremap(base, SZ_4K);
	if (WARN_ON(!omap_irq_base))
		return;

219 220
	omap_nr_irqs = nr_irqs;

221 222 223 224 225 226 227
	irq_base = irq_alloc_descs(-1, 0, nr_irqs, 0);
	if (irq_base < 0) {
		pr_warn("Couldn't allocate IRQ numbers\n");
		irq_base = 0;
	}

	domain = irq_domain_add_legacy(node, nr_irqs, irq_base, 0,
228
			&irq_domain_simple_ops, NULL);
229

230
	omap_irq_soft_reset();
231

232 233
	for (j = 0; j < omap_nr_irqs; j += 32)
		omap_alloc_gc(omap_irq_base + j, j + irq_base, 32);
234 235
}

236 237
void __init omap2_init_irq(void)
{
238
	omap_init_irq(OMAP24XX_IC_BASE, 96, NULL);
239 240 241 242
}

void __init omap3_init_irq(void)
{
243
	omap_init_irq(OMAP34XX_IC_BASE, 96, NULL);
244 245
}

246
void __init ti81xx_init_irq(void)
247
{
248
	omap_init_irq(OMAP34XX_IC_BASE, 128, NULL);
249 250
}

251
static inline void omap_intc_handle_irq(struct pt_regs *regs)
252 253
{
	u32 irqnr;
254
	int handled_irq = 0;
255 256

	do {
257
		irqnr = intc_readl(INTC_PENDING_IRQ0);
258 259 260
		if (irqnr)
			goto out;

261
		irqnr = intc_readl(INTC_PENDING_IRQ1);
262 263 264
		if (irqnr)
			goto out;

265
		irqnr = intc_readl(INTC_PENDING_IRQ2);
266
#if IS_ENABLED(CONFIG_SOC_TI81XX) || IS_ENABLED(CONFIG_SOC_AM33XX)
267 268
		if (irqnr)
			goto out;
269
		irqnr = intc_readl(INTC_PENDING_IRQ3);
270 271 272 273 274 275
#endif

out:
		if (!irqnr)
			break;

276
		irqnr = intc_readl(INTC_SIR);
277 278
		irqnr &= ACTIVEIRQ_MASK;

279 280
		if (irqnr) {
			irqnr = irq_find_mapping(domain, irqnr);
281
			handle_IRQ(irqnr, regs);
282
			handled_irq = 1;
283
		}
284
	} while (irqnr);
285 286 287 288 289 290

	/* If an irq is masked or deasserted while active, we will
	 * keep ending up here with no irq handled. So remove it from
	 * the INTC with an ack.*/
	if (!handled_irq)
		omap_ack_irq(NULL);
291 292 293 294
}

asmlinkage void __exception_irq_entry omap2_intc_handle_irq(struct pt_regs *regs)
{
295
	omap_intc_handle_irq(regs);
296 297
}

298
static int __init intc_of_init(struct device_node *node,
299 300 301
			     struct device_node *parent)
{
	struct resource res;
302
	u32 nr_irq = 96;
303 304 305 306 307 308 309 310 311

	if (WARN_ON(!node))
		return -ENODEV;

	if (of_address_to_resource(node, 0, &res)) {
		WARN(1, "unable to get intc registers\n");
		return -EINVAL;
	}

312 313
	if (of_property_read_u32(node, "ti,intc-size", &nr_irq))
		pr_warn("unable to get intc-size, default to %d\n", nr_irq);
314

315
	omap_init_irq(res.start, nr_irq, of_node_get(node));
316 317 318 319

	return 0;
}

320
static const struct of_device_id irq_match[] __initconst = {
321 322 323 324 325 326 327 328 329
	{ .compatible = "ti,omap2-intc", .data = intc_of_init, },
	{ }
};

void __init omap_intc_of_init(void)
{
	of_irq_init(irq_match);
}

330 331
asmlinkage void __exception_irq_entry omap3_intc_handle_irq(struct pt_regs *regs)
{
332
	omap_intc_handle_irq(regs);
333
}