irq-omap-intc.c 9.2 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 <linux/irqchip.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 27 28
/* Define these here for now until we drop all board-files */
#define OMAP24XX_IC_BASE	0x480fe000
#define OMAP34XX_IC_BASE	0x48200000
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	4
52

53 54 55
#define INTC_IDLE_FUNCIDLE	(1 << 0)
#define INTC_IDLE_TURBO		(1 << 1)

56 57
#define INTC_PROTECTION_ENABLE	(1 << 0)

58
struct omap_intc_regs {
59 60 61 62
	u32 sysconfig;
	u32 protection;
	u32 idle;
	u32 threshold;
63
	u32 ilr[INTCPS_NR_ILR_REGS];
64 65
	u32 mir[INTCPS_NR_MIR_REGS];
};
66 67 68 69
static struct omap_intc_regs intc_context;

static struct irq_domain *domain;
static void __iomem *omap_irq_base;
70
static int omap_nr_pending = 3;
71
static int omap_nr_irqs = 96;
72

73
static void intc_writel(u32 reg, u32 val)
74
{
75
	writel_relaxed(val, omap_irq_base + reg);
76 77
}

78
static u32 intc_readl(u32 reg)
79
{
80
	return readl_relaxed(omap_irq_base + reg);
81 82
}

83 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
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);
130
	intc_writel(INTC_IDLE, INTC_IDLE_TURBO);
131 132 133 134 135 136
}

void omap3_intc_resume_idle(void)
{
	/* Re-enable autoidle */
	intc_writel(INTC_SYSCONFIG, 1);
137
	intc_writel(INTC_IDLE, 0);
138 139
}

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

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

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

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

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

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

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

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

172 173
int omap_irq_pending(void)
{
174
	int i;
175

176 177
	for (i = 0; i < omap_nr_pending; i++)
		if (intc_readl(INTC_PENDING_IRQ0 + (0x20 * i)))
178
			return 1;
179 180 181
	return 0;
}

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

188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226
static int __init omap_alloc_gc_of(struct irq_domain *d, void __iomem *base)
{
	int ret;
	int i;

	ret = irq_alloc_domain_generic_chips(d, 32, 1, "INTC",
			handle_level_irq, IRQ_NOREQUEST | IRQ_NOPROBE,
			IRQ_LEVEL, 0);
	if (ret) {
		pr_warn("Failed to allocate irq chips\n");
		return ret;
	}

	for (i = 0; i < omap_nr_pending; i++) {
		struct irq_chip_generic *gc;
		struct irq_chip_type *ct;

		gc = irq_get_domain_generic_chip(d, 32 * i);
		gc->reg_base = base;
		ct = gc->chip_types;

		ct->type = IRQ_TYPE_LEVEL_MASK;
		ct->handler = handle_level_irq;

		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;

		ct->chip.flags |= IRQCHIP_SKIP_SET_WAKE;

		ct->regs.enable = INTC_MIR_CLEAR0 + 32 * i;
		ct->regs.disable = INTC_MIR_SET0 + 32 * i;
	}

	return 0;
}

static void __init omap_alloc_gc_legacy(void __iomem *base,
		unsigned int irq_start, unsigned int num)
227 228 229 230 231
{
	struct irq_chip_generic *gc;
	struct irq_chip_type *ct;

	gc = irq_alloc_generic_chip("INTC", 1, irq_start, base,
232
			handle_level_irq);
233 234 235 236
	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;
237
	ct->chip.flags |= IRQCHIP_SKIP_SET_WAKE;
238 239 240 241

	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,
242
			IRQ_NOREQUEST | IRQ_NOPROBE, 0);
243 244
}

245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264
static int __init omap_init_irq_of(struct device_node *node)
{
	int ret;

	omap_irq_base = of_iomap(node, 0);
	if (WARN_ON(!omap_irq_base))
		return -ENOMEM;

	domain = irq_domain_add_linear(node, omap_nr_irqs,
			&irq_generic_chip_ops, NULL);

	omap_irq_soft_reset();

	ret = omap_alloc_gc_of(domain, omap_irq_base);
	if (ret < 0)
		irq_domain_remove(domain);

	return ret;
}

265
static int __init omap_init_irq_legacy(u32 base, struct device_node *node)
266
{
267
	int j, irq_base;
268

269 270
	omap_irq_base = ioremap(base, SZ_4K);
	if (WARN_ON(!omap_irq_base))
271
		return -ENOMEM;
272

273
	irq_base = irq_alloc_descs(-1, 0, omap_nr_irqs, 0);
274 275 276 277 278
	if (irq_base < 0) {
		pr_warn("Couldn't allocate IRQ numbers\n");
		irq_base = 0;
	}

279
	domain = irq_domain_add_legacy(node, omap_nr_irqs, irq_base, 0,
280
			&irq_domain_simple_ops, NULL);
281

282
	omap_irq_soft_reset();
283

284
	for (j = 0; j < omap_nr_irqs; j += 32)
285 286 287 288 289
		omap_alloc_gc_legacy(omap_irq_base + j, j + irq_base, 32);

	return 0;
}

290 291 292 293 294 295 296 297 298
static void __init omap_irq_enable_protection(void)
{
	u32 reg;

	reg = intc_readl(INTC_PROTECTION);
	reg |= INTC_PROTECTION_ENABLE;
	intc_writel(INTC_PROTECTION, reg);
}

299 300
static int __init omap_init_irq(u32 base, struct device_node *node)
{
301 302
	int ret;

303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318
	/*
	 * FIXME legacy OMAP DMA driver sitting under arch/arm/plat-omap/dma.c
	 * depends is still not ready for linear IRQ domains; because of that
	 * we need to temporarily "blacklist" OMAP2 and OMAP3 devices from using
	 * linear IRQ Domain until that driver is finally fixed.
	 */
	if (of_device_is_compatible(node, "ti,omap2-intc") ||
			of_device_is_compatible(node, "ti,omap3-intc")) {
		struct resource res;

		if (of_address_to_resource(node, 0, &res))
			return -ENOMEM;

		base = res.start;
		ret = omap_init_irq_legacy(base, node);
	} else if (node) {
319
		ret = omap_init_irq_of(node);
320 321 322
	} else {
		ret = omap_init_irq_legacy(base, NULL);
	}
323 324 325 326 327

	if (ret == 0)
		omap_irq_enable_protection();

	return ret;
328 329
}

330 331
static asmlinkage void __exception_irq_entry
omap_intc_handle_irq(struct pt_regs *regs)
332
{
333
	u32 irqnr = 0;
334
	int handled_irq = 0;
335
	int i;
336 337

	do {
338 339 340 341 342
		for (i = 0; i < omap_nr_pending; i++) {
			irqnr = intc_readl(INTC_PENDING_IRQ0 + (0x20 * i));
			if (irqnr)
				goto out;
		}
343 344 345 346 347

out:
		if (!irqnr)
			break;

348
		irqnr = intc_readl(INTC_SIR);
349 350
		irqnr &= ACTIVEIRQ_MASK;

351
		if (irqnr) {
352
			handle_domain_irq(domain, irqnr, regs);
353
			handled_irq = 1;
354
		}
355
	} while (irqnr);
356

357 358
	/*
	 * If an irq is masked or deasserted while active, we will
359
	 * keep ending up here with no irq handled. So remove it from
360 361
	 * the INTC with an ack.
	 */
362 363
	if (!handled_irq)
		omap_ack_irq(NULL);
364 365
}

366 367
void __init omap3_init_irq(void)
{
368
	omap_nr_irqs = 96;
369
	omap_nr_pending = 3;
370
	omap_init_irq(OMAP34XX_IC_BASE, NULL);
371
	set_handle_irq(omap_intc_handle_irq);
372 373
}

374
static int __init intc_of_init(struct device_node *node,
375 376
			     struct device_node *parent)
{
377
	int ret;
378

379
	omap_nr_pending = 3;
380
	omap_nr_irqs = 96;
381 382 383 384

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

385 386 387
	if (of_device_is_compatible(node, "ti,dm814-intc") ||
	    of_device_is_compatible(node, "ti,dm816-intc") ||
	    of_device_is_compatible(node, "ti,am33xx-intc")) {
388
		omap_nr_irqs = 128;
389 390
		omap_nr_pending = 4;
	}
391

392 393 394
	ret = omap_init_irq(-1, of_node_get(node));
	if (ret < 0)
		return ret;
395

396
	set_handle_irq(omap_intc_handle_irq);
397

398 399 400
	return 0;
}

401 402
IRQCHIP_DECLARE(omap2_intc, "ti,omap2-intc", intc_of_init);
IRQCHIP_DECLARE(omap3_intc, "ti,omap3-intc", intc_of_init);
403 404
IRQCHIP_DECLARE(dm814x_intc, "ti,dm814-intc", intc_of_init);
IRQCHIP_DECLARE(dm816x_intc, "ti,dm816-intc", intc_of_init);
405
IRQCHIP_DECLARE(am33xx_intc, "ti,am33xx-intc", intc_of_init);