irq-omap-intc.c 9.1 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 21 22
#include <linux/irqdomain.h>
#include <linux/of.h>
#include <linux/of_address.h>
23
#include <linux/of_irq.h>
24

25 26 27 28 29
#include "irqchip.h"

/* Define these here for now until we drop all board-files */
#define OMAP24XX_IC_BASE	0x480fe000
#define OMAP34XX_IC_BASE	0x48200000
30 31 32 33 34 35

/* selected INTC register offsets */

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

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

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

57 58
#define INTC_PROTECTION_ENABLE	(1 << 0)

59 60 61 62 63 64 65
/*
 * 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.
 */

66
/* Structure to save interrupt controller context */
67
struct omap_intc_regs {
68 69 70 71
	u32 sysconfig;
	u32 protection;
	u32 idle;
	u32 threshold;
72
	u32 ilr[INTCPS_NR_ILR_REGS];
73 74
	u32 mir[INTCPS_NR_MIR_REGS];
};
75 76 77 78
static struct omap_intc_regs intc_context;

static struct irq_domain *domain;
static void __iomem *omap_irq_base;
79
static int omap_nr_pending = 3;
80
static int omap_nr_irqs = 96;
81

82
/* INTC bank register get/set */
83
static void intc_writel(u32 reg, u32 val)
84
{
85
	writel_relaxed(val, omap_irq_base + reg);
86 87
}

88
static u32 intc_readl(u32 reg)
89
{
90
	return readl_relaxed(omap_irq_base + reg);
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
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);
140
	intc_writel(INTC_IDLE, INTC_IDLE_TURBO);
141 142 143 144 145 146
}

void omap3_intc_resume_idle(void)
{
	/* Re-enable autoidle */
	intc_writel(INTC_SYSCONFIG, 1);
147
	intc_writel(INTC_IDLE, 0);
148 149
}

150
/* XXX: FIQ and additional INTC support (only MPU at the moment) */
151
static void omap_ack_irq(struct irq_data *d)
152
{
153
	intc_writel(INTC_CONTROL, 0x1);
154 155
}

156
static void omap_mask_ack_irq(struct irq_data *d)
157
{
158
	irq_gc_mask_disable_reg(d);
159
	omap_ack_irq(d);
160 161
}

162
static void __init omap_irq_soft_reset(void)
163 164 165
{
	unsigned long tmp;

166
	tmp = intc_readl(INTC_REVISION) & 0xff;
167

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

171
	tmp = intc_readl(INTC_SYSCONFIG);
172
	tmp |= 1 << 1;	/* soft reset */
173
	intc_writel(INTC_SYSCONFIG, tmp);
174

175
	while (!(intc_readl(INTC_SYSSTATUS) & 0x1))
176
		/* Wait for reset to complete */;
177 178

	/* Enable autoidle */
179
	intc_writel(INTC_SYSCONFIG, 1 << 0);
180 181
}

182 183
int omap_irq_pending(void)
{
184
	int i;
185

186 187
	for (i = 0; i < omap_nr_pending; i++)
		if (intc_readl(INTC_PENDING_IRQ0 + (0x20 * i)))
188
			return 1;
189 190 191
	return 0;
}

192 193 194 195 196 197
void omap3_intc_suspend(void)
{
	/* A pending interrupt would prevent OMAP from entering suspend */
	omap_ack_irq(NULL);
}

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 227 228 229 230 231 232 233 234 235 236
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)
237 238 239 240 241
{
	struct irq_chip_generic *gc;
	struct irq_chip_type *ct;

	gc = irq_alloc_generic_chip("INTC", 1, irq_start, base,
242
			handle_level_irq);
243 244 245 246
	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;
247
	ct->chip.flags |= IRQCHIP_SKIP_SET_WAKE;
248 249 250 251

	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,
252
			IRQ_NOREQUEST | IRQ_NOPROBE, 0);
253 254
}

255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
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;
}

static int __init omap_init_irq_legacy(u32 base)
276
{
277
	int j, irq_base;
278

279 280
	omap_irq_base = ioremap(base, SZ_4K);
	if (WARN_ON(!omap_irq_base))
281
		return -ENOMEM;
282

283
	irq_base = irq_alloc_descs(-1, 0, omap_nr_irqs, 0);
284 285 286 287 288
	if (irq_base < 0) {
		pr_warn("Couldn't allocate IRQ numbers\n");
		irq_base = 0;
	}

289
	domain = irq_domain_add_legacy(NULL, omap_nr_irqs, irq_base, 0,
290
			&irq_domain_simple_ops, NULL);
291

292
	omap_irq_soft_reset();
293

294
	for (j = 0; j < omap_nr_irqs; j += 32)
295 296 297 298 299
		omap_alloc_gc_legacy(omap_irq_base + j, j + irq_base, 32);

	return 0;
}

300 301 302 303 304 305 306 307 308
static void __init omap_irq_enable_protection(void)
{
	u32 reg;

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

309 310
static int __init omap_init_irq(u32 base, struct device_node *node)
{
311 312
	int ret;

313
	if (node)
314
		ret = omap_init_irq_of(node);
315
	else
316 317 318 319 320 321
		ret = omap_init_irq_legacy(base);

	if (ret == 0)
		omap_irq_enable_protection();

	return ret;
322 323
}

324 325
static asmlinkage void __exception_irq_entry
omap_intc_handle_irq(struct pt_regs *regs)
326
{
327
	u32 irqnr = 0;
328
	int handled_irq = 0;
329
	int i;
330 331

	do {
332 333 334 335 336
		for (i = 0; i < omap_nr_pending; i++) {
			irqnr = intc_readl(INTC_PENDING_IRQ0 + (0x20 * i));
			if (irqnr)
				goto out;
		}
337 338 339 340 341

out:
		if (!irqnr)
			break;

342
		irqnr = intc_readl(INTC_SIR);
343 344
		irqnr &= ACTIVEIRQ_MASK;

345 346
		if (irqnr) {
			irqnr = irq_find_mapping(domain, irqnr);
347
			handle_IRQ(irqnr, regs);
348
			handled_irq = 1;
349
		}
350
	} while (irqnr);
351

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

361 362
void __init omap2_init_irq(void)
{
363
	omap_nr_irqs = 96;
364
	omap_nr_pending = 3;
365
	omap_init_irq(OMAP24XX_IC_BASE, NULL);
366
	set_handle_irq(omap_intc_handle_irq);
367 368 369 370
}

void __init omap3_init_irq(void)
{
371
	omap_nr_irqs = 96;
372
	omap_nr_pending = 3;
373
	omap_init_irq(OMAP34XX_IC_BASE, NULL);
374
	set_handle_irq(omap_intc_handle_irq);
375 376 377 378
}

void __init ti81xx_init_irq(void)
{
379
	omap_nr_irqs = 96;
380
	omap_nr_pending = 4;
381
	omap_init_irq(OMAP34XX_IC_BASE, NULL);
382
	set_handle_irq(omap_intc_handle_irq);
383 384
}

385
static int __init intc_of_init(struct device_node *node,
386 387
			     struct device_node *parent)
{
388
	int ret;
389

390
	omap_nr_pending = 3;
391
	omap_nr_irqs = 96;
392 393 394 395

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

396
	if (of_device_is_compatible(node, "ti,am33xx-intc")) {
397
		omap_nr_irqs = 128;
398 399
		omap_nr_pending = 4;
	}
400

401 402 403
	ret = omap_init_irq(-1, of_node_get(node));
	if (ret < 0)
		return ret;
404

405
	set_handle_irq(omap_intc_handle_irq);
406

407 408 409
	return 0;
}

410 411 412
IRQCHIP_DECLARE(omap2_intc, "ti,omap2-intc", intc_of_init);
IRQCHIP_DECLARE(omap3_intc, "ti,omap3-intc", intc_of_init);
IRQCHIP_DECLARE(am33xx_intc, "ti,am33xx-intc", intc_of_init);