irq-mips-gic.c 17.3 KB
Newer Older
1 2 3 4 5 6 7 8
/*
 * 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.
 *
 * Copyright (C) 2008 Ralf Baechle (ralf@linux-mips.org)
 * Copyright (C) 2012 MIPS Technologies, Inc.  All rights reserved.
 */
9
#include <linux/bitmap.h>
10
#include <linux/clocksource.h>
11
#include <linux/init.h>
12
#include <linux/interrupt.h>
13
#include <linux/irq.h>
14
#include <linux/irqchip/mips-gic.h>
15
#include <linux/sched.h>
16
#include <linux/smp.h>
17

S
Steven J. Hill 已提交
18 19
#include <asm/setup.h>
#include <asm/traps.h>
20

21
unsigned int gic_frequency;
22
unsigned int gic_present;
S
Steven J. Hill 已提交
23

24
struct gic_pcpu_mask {
25
	DECLARE_BITMAP(pcpu_mask, GIC_MAX_INTRS);
26 27
};

28
static void __iomem *gic_base;
29
static struct gic_pcpu_mask pcpu_masks[NR_CPUS];
30
static DEFINE_SPINLOCK(gic_lock);
31
static struct irq_domain *gic_irq_domain;
32
static int gic_shared_intrs;
33
static int gic_vpes;
34
static unsigned int gic_cpu_pin;
35
static struct irq_chip gic_level_irq_controller, gic_edge_irq_controller;
36

37 38
static void __gic_irq_dispatch(void);

39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 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
static inline unsigned int gic_read(unsigned int reg)
{
	return __raw_readl(gic_base + reg);
}

static inline void gic_write(unsigned int reg, unsigned int val)
{
	__raw_writel(val, gic_base + reg);
}

static inline void gic_update_bits(unsigned int reg, unsigned int mask,
				   unsigned int val)
{
	unsigned int regval;

	regval = gic_read(reg);
	regval &= ~mask;
	regval |= val;
	gic_write(reg, regval);
}

static inline void gic_reset_mask(unsigned int intr)
{
	gic_write(GIC_REG(SHARED, GIC_SH_RMASK) + GIC_INTR_OFS(intr),
		  1 << GIC_INTR_BIT(intr));
}

static inline void gic_set_mask(unsigned int intr)
{
	gic_write(GIC_REG(SHARED, GIC_SH_SMASK) + GIC_INTR_OFS(intr),
		  1 << GIC_INTR_BIT(intr));
}

static inline void gic_set_polarity(unsigned int intr, unsigned int pol)
{
	gic_update_bits(GIC_REG(SHARED, GIC_SH_SET_POLARITY) +
			GIC_INTR_OFS(intr), 1 << GIC_INTR_BIT(intr),
			pol << GIC_INTR_BIT(intr));
}

static inline void gic_set_trigger(unsigned int intr, unsigned int trig)
{
	gic_update_bits(GIC_REG(SHARED, GIC_SH_SET_TRIGGER) +
			GIC_INTR_OFS(intr), 1 << GIC_INTR_BIT(intr),
			trig << GIC_INTR_BIT(intr));
}

static inline void gic_set_dual_edge(unsigned int intr, unsigned int dual)
{
	gic_update_bits(GIC_REG(SHARED, GIC_SH_SET_DUAL) + GIC_INTR_OFS(intr),
			1 << GIC_INTR_BIT(intr),
			dual << GIC_INTR_BIT(intr));
}

static inline void gic_map_to_pin(unsigned int intr, unsigned int pin)
{
	gic_write(GIC_REG(SHARED, GIC_SH_INTR_MAP_TO_PIN_BASE) +
		  GIC_SH_MAP_TO_PIN(intr), GIC_MAP_TO_PIN_MSK | pin);
}

static inline void gic_map_to_vpe(unsigned int intr, unsigned int vpe)
{
	gic_write(GIC_REG(SHARED, GIC_SH_INTR_MAP_TO_VPE_BASE) +
		  GIC_SH_MAP_TO_VPE_REG_OFF(intr, vpe),
		  GIC_SH_MAP_TO_VPE_REG_BIT(vpe));
}

106
#if defined(CONFIG_CSRC_GIC) || defined(CONFIG_CEVT_GIC)
107 108 109 110 111
cycle_t gic_read_count(void)
{
	unsigned int hi, hi2, lo;

	do {
112 113 114
		hi = gic_read(GIC_REG(SHARED, GIC_SH_COUNTER_63_32));
		lo = gic_read(GIC_REG(SHARED, GIC_SH_COUNTER_31_00));
		hi2 = gic_read(GIC_REG(SHARED, GIC_SH_COUNTER_63_32));
115 116 117 118
	} while (hi2 != hi);

	return (((cycle_t) hi) << 32) + lo;
}
119

120 121 122 123
unsigned int gic_get_count_width(void)
{
	unsigned int bits, config;

124
	config = gic_read(GIC_REG(SHARED, GIC_SH_CONFIG));
125 126 127 128 129 130
	bits = 32 + 4 * ((config & GIC_SH_CONFIG_COUNTBITS_MSK) >>
			 GIC_SH_CONFIG_COUNTBITS_SHF);

	return bits;
}

131 132
void gic_write_compare(cycle_t cnt)
{
133
	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_HI),
134
				(int)(cnt >> 32));
135
	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_LO),
136 137 138
				(int)(cnt & 0xffffffff));
}

139 140 141 142 143 144
void gic_write_cpu_compare(cycle_t cnt, int cpu)
{
	unsigned long flags;

	local_irq_save(flags);

145 146
	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), cpu);
	gic_write(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_HI),
147
				(int)(cnt >> 32));
148
	gic_write(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_LO),
149 150 151 152 153
				(int)(cnt & 0xffffffff));

	local_irq_restore(flags);
}

154 155 156 157
cycle_t gic_read_compare(void)
{
	unsigned int hi, lo;

158 159
	hi = gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_HI));
	lo = gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_LO));
160 161 162

	return (((cycle_t) hi) << 32) + lo;
}
163 164
#endif

165 166 167 168 169 170 171 172
static bool gic_local_irq_is_routable(int intr)
{
	u32 vpe_ctl;

	/* All local interrupts are routable in EIC mode. */
	if (cpu_has_veic)
		return true;

173
	vpe_ctl = gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_CTL));
174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
	switch (intr) {
	case GIC_LOCAL_INT_TIMER:
		return vpe_ctl & GIC_VPE_CTL_TIMER_RTBL_MSK;
	case GIC_LOCAL_INT_PERFCTR:
		return vpe_ctl & GIC_VPE_CTL_PERFCNT_RTBL_MSK;
	case GIC_LOCAL_INT_FDC:
		return vpe_ctl & GIC_VPE_CTL_FDC_RTBL_MSK;
	case GIC_LOCAL_INT_SWINT0:
	case GIC_LOCAL_INT_SWINT1:
		return vpe_ctl & GIC_VPE_CTL_SWINT_RTBL_MSK;
	default:
		return true;
	}
}

S
Steven J. Hill 已提交
189 190 191 192
unsigned int gic_get_timer_pending(void)
{
	unsigned int vpe_pending;

193
	vpe_pending = gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_PEND));
R
Ralf Baechle 已提交
194
	return vpe_pending & GIC_VPE_PEND_TIMER_MSK;
S
Steven J. Hill 已提交
195 196
}

197
static void gic_bind_eic_interrupt(int irq, int set)
S
Steven J. Hill 已提交
198 199 200 201 202
{
	/* Convert irq vector # to hw int # */
	irq -= GIC_PIN_TO_VEC_OFFSET;

	/* Set irq to use shadow set */
203 204
	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_EIC_SHADOW_SET_BASE) +
		  GIC_VPE_EIC_SS(irq), set);
S
Steven J. Hill 已提交
205 206
}

207 208
void gic_send_ipi(unsigned int intr)
{
209
	gic_write(GIC_REG(SHARED, GIC_SH_WEDGE), 0x80000000 | intr);
210 211
}

212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231
int gic_get_c0_compare_int(void)
{
	if (!gic_local_irq_is_routable(GIC_LOCAL_INT_TIMER))
		return MIPS_CPU_IRQ_BASE + cp0_compare_irq;
	return irq_create_mapping(gic_irq_domain,
				  GIC_LOCAL_TO_HWIRQ(GIC_LOCAL_INT_TIMER));
}

int gic_get_c0_perfcount_int(void)
{
	if (!gic_local_irq_is_routable(GIC_LOCAL_INT_PERFCTR)) {
		/* Is the erformance counter shared with the timer? */
		if (cp0_perfcount_irq < 0)
			return -1;
		return MIPS_CPU_IRQ_BASE + cp0_perfcount_irq;
	}
	return irq_create_mapping(gic_irq_domain,
				  GIC_LOCAL_TO_HWIRQ(GIC_LOCAL_INT_PERFCTR));
}

232
static unsigned int gic_get_int(void)
233 234
{
	unsigned int i;
235
	unsigned long *pcpu_mask;
236
	unsigned long pending_reg, intrmask_reg;
237 238
	DECLARE_BITMAP(pending, GIC_MAX_INTRS);
	DECLARE_BITMAP(intrmask, GIC_MAX_INTRS);
239 240 241 242

	/* Get per-cpu bitmaps */
	pcpu_mask = pcpu_masks[smp_processor_id()].pcpu_mask;

243 244
	pending_reg = GIC_REG(SHARED, GIC_SH_PEND);
	intrmask_reg = GIC_REG(SHARED, GIC_SH_MASK);
245

246
	for (i = 0; i < BITS_TO_LONGS(gic_shared_intrs); i++) {
247 248 249 250
		pending[i] = gic_read(pending_reg);
		intrmask[i] = gic_read(intrmask_reg);
		pending_reg += 0x4;
		intrmask_reg += 0x4;
251 252
	}

253 254
	bitmap_and(pending, pending, intrmask, gic_shared_intrs);
	bitmap_and(pending, pending, pcpu_mask, gic_shared_intrs);
255

256
	return find_first_bit(pending, gic_shared_intrs);
257 258
}

259
static void gic_mask_irq(struct irq_data *d)
260
{
261
	gic_reset_mask(GIC_HWIRQ_TO_SHARED(d->hwirq));
262 263
}

264
static void gic_unmask_irq(struct irq_data *d)
265
{
266
	gic_set_mask(GIC_HWIRQ_TO_SHARED(d->hwirq));
267 268
}

269 270
static void gic_ack_irq(struct irq_data *d)
{
271
	unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
272

273
	gic_write(GIC_REG(SHARED, GIC_SH_WEDGE), irq);
274 275
}

276 277
static int gic_set_type(struct irq_data *d, unsigned int type)
{
278
	unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
279 280 281 282 283 284
	unsigned long flags;
	bool is_edge;

	spin_lock_irqsave(&gic_lock, flags);
	switch (type & IRQ_TYPE_SENSE_MASK) {
	case IRQ_TYPE_EDGE_FALLING:
285 286 287
		gic_set_polarity(irq, GIC_POL_NEG);
		gic_set_trigger(irq, GIC_TRIG_EDGE);
		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE);
288 289 290
		is_edge = true;
		break;
	case IRQ_TYPE_EDGE_RISING:
291 292 293
		gic_set_polarity(irq, GIC_POL_POS);
		gic_set_trigger(irq, GIC_TRIG_EDGE);
		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE);
294 295 296 297
		is_edge = true;
		break;
	case IRQ_TYPE_EDGE_BOTH:
		/* polarity is irrelevant in this case */
298 299
		gic_set_trigger(irq, GIC_TRIG_EDGE);
		gic_set_dual_edge(irq, GIC_TRIG_DUAL_ENABLE);
300 301 302
		is_edge = true;
		break;
	case IRQ_TYPE_LEVEL_LOW:
303 304 305
		gic_set_polarity(irq, GIC_POL_NEG);
		gic_set_trigger(irq, GIC_TRIG_LEVEL);
		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE);
306 307 308 309
		is_edge = false;
		break;
	case IRQ_TYPE_LEVEL_HIGH:
	default:
310 311 312
		gic_set_polarity(irq, GIC_POL_POS);
		gic_set_trigger(irq, GIC_TRIG_LEVEL);
		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE);
313 314 315 316 317
		is_edge = false;
		break;
	}

	if (is_edge) {
318 319 320
		__irq_set_chip_handler_name_locked(d->irq,
						   &gic_edge_irq_controller,
						   handle_edge_irq, NULL);
321
	} else {
322 323 324
		__irq_set_chip_handler_name_locked(d->irq,
						   &gic_level_irq_controller,
						   handle_level_irq, NULL);
325 326
	}
	spin_unlock_irqrestore(&gic_lock, flags);
327

328 329 330 331
	return 0;
}

#ifdef CONFIG_SMP
332 333
static int gic_set_affinity(struct irq_data *d, const struct cpumask *cpumask,
			    bool force)
334
{
335
	unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
336 337 338 339
	cpumask_t	tmp = CPU_MASK_NONE;
	unsigned long	flags;
	int		i;

340
	cpumask_and(&tmp, cpumask, cpu_online_mask);
341
	if (cpus_empty(tmp))
342
		return -EINVAL;
343 344 345 346

	/* Assumption : cpumask refers to a single CPU */
	spin_lock_irqsave(&gic_lock, flags);

347
	/* Re-route this IRQ */
348
	gic_map_to_vpe(irq, first_cpu(tmp));
349 350 351 352 353

	/* Update the pcpu_masks */
	for (i = 0; i < NR_CPUS; i++)
		clear_bit(irq, pcpu_masks[i].pcpu_mask);
	set_bit(irq, pcpu_masks[first_cpu(tmp)].pcpu_mask);
354

355
	cpumask_copy(d->affinity, cpumask);
356 357
	spin_unlock_irqrestore(&gic_lock, flags);

358
	return IRQ_SET_MASK_OK_NOCOPY;
359 360 361
}
#endif

362 363 364 365 366 367 368 369 370 371 372
static struct irq_chip gic_level_irq_controller = {
	.name			=	"MIPS GIC",
	.irq_mask		=	gic_mask_irq,
	.irq_unmask		=	gic_unmask_irq,
	.irq_set_type		=	gic_set_type,
#ifdef CONFIG_SMP
	.irq_set_affinity	=	gic_set_affinity,
#endif
};

static struct irq_chip gic_edge_irq_controller = {
373
	.name			=	"MIPS GIC",
374
	.irq_ack		=	gic_ack_irq,
375 376
	.irq_mask		=	gic_mask_irq,
	.irq_unmask		=	gic_unmask_irq,
377
	.irq_set_type		=	gic_set_type,
378
#ifdef CONFIG_SMP
379
	.irq_set_affinity	=	gic_set_affinity,
380 381 382
#endif
};

383 384 385 386
static unsigned int gic_get_local_int(void)
{
	unsigned long pending, masked;

387 388
	pending = gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_PEND));
	masked = gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_MASK));
389 390 391 392 393 394 395 396 397 398

	bitmap_and(&pending, &pending, &masked, GIC_NUM_LOCAL_INTRS);

	return find_first_bit(&pending, GIC_NUM_LOCAL_INTRS);
}

static void gic_mask_local_irq(struct irq_data *d)
{
	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq);

399
	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_RMASK), 1 << intr);
400 401 402 403 404 405
}

static void gic_unmask_local_irq(struct irq_data *d)
{
	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq);

406
	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_SMASK), 1 << intr);
407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422
}

static struct irq_chip gic_local_irq_controller = {
	.name			=	"MIPS GIC Local",
	.irq_mask		=	gic_mask_local_irq,
	.irq_unmask		=	gic_unmask_local_irq,
};

static void gic_mask_local_irq_all_vpes(struct irq_data *d)
{
	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq);
	int i;
	unsigned long flags;

	spin_lock_irqsave(&gic_lock, flags);
	for (i = 0; i < gic_vpes; i++) {
423 424
		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
		gic_write(GIC_REG(VPE_OTHER, GIC_VPE_RMASK), 1 << intr);
425 426 427 428 429 430 431 432 433 434 435 436
	}
	spin_unlock_irqrestore(&gic_lock, flags);
}

static void gic_unmask_local_irq_all_vpes(struct irq_data *d)
{
	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq);
	int i;
	unsigned long flags;

	spin_lock_irqsave(&gic_lock, flags);
	for (i = 0; i < gic_vpes; i++) {
437 438
		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
		gic_write(GIC_REG(VPE_OTHER, GIC_VPE_SMASK), 1 << intr);
439 440 441 442 443 444 445 446 447 448
	}
	spin_unlock_irqrestore(&gic_lock, flags);
}

static struct irq_chip gic_all_vpes_local_irq_controller = {
	.name			=	"MIPS GIC Local",
	.irq_mask		=	gic_mask_local_irq_all_vpes,
	.irq_unmask		=	gic_unmask_local_irq_all_vpes,
};

449
static void __gic_irq_dispatch(void)
450
{
451
	unsigned int intr, virq;
J
Jeffrey Deans 已提交
452

453 454 455 456 457 458
	while ((intr = gic_get_local_int()) != GIC_NUM_LOCAL_INTRS) {
		virq = irq_linear_revmap(gic_irq_domain,
					 GIC_LOCAL_TO_HWIRQ(intr));
		do_IRQ(virq);
	}

459
	while ((intr = gic_get_int()) != gic_shared_intrs) {
460 461
		virq = irq_linear_revmap(gic_irq_domain,
					 GIC_SHARED_TO_HWIRQ(intr));
462
		do_IRQ(virq);
463
	}
464
}
465

466 467 468 469 470 471 472 473 474 475 476 477 478
static void gic_irq_dispatch(unsigned int irq, struct irq_desc *desc)
{
	__gic_irq_dispatch();
}

#ifdef CONFIG_MIPS_GIC_IPI
static int gic_resched_int_base;
static int gic_call_int_base;

unsigned int plat_ipi_resched_int_xlate(unsigned int cpu)
{
	return gic_resched_int_base + cpu;
}
479

480 481 482 483
unsigned int plat_ipi_call_int_xlate(unsigned int cpu)
{
	return gic_call_int_base + cpu;
}
484

485 486 487 488 489 490 491 492 493 494 495 496 497
static irqreturn_t ipi_resched_interrupt(int irq, void *dev_id)
{
	scheduler_ipi();

	return IRQ_HANDLED;
}

static irqreturn_t ipi_call_interrupt(int irq, void *dev_id)
{
	smp_call_function_interrupt();

	return IRQ_HANDLED;
}
J
Jeffrey Deans 已提交
498

499 500 501 502 503 504 505 506 507 508 509 510 511 512 513
static struct irqaction irq_resched = {
	.handler	= ipi_resched_interrupt,
	.flags		= IRQF_PERCPU,
	.name		= "IPI resched"
};

static struct irqaction irq_call = {
	.handler	= ipi_call_interrupt,
	.flags		= IRQF_PERCPU,
	.name		= "IPI call"
};

static __init void gic_ipi_init_one(unsigned int intr, int cpu,
				    struct irqaction *action)
{
514 515
	int virq = irq_create_mapping(gic_irq_domain,
				      GIC_SHARED_TO_HWIRQ(intr));
516 517
	int i;

518
	gic_map_to_vpe(intr, cpu);
519 520
	for (i = 0; i < NR_CPUS; i++)
		clear_bit(intr, pcpu_masks[i].pcpu_mask);
J
Jeffrey Deans 已提交
521 522
	set_bit(intr, pcpu_masks[cpu].pcpu_mask);

523 524 525 526
	irq_set_irq_type(virq, IRQ_TYPE_EDGE_RISING);

	irq_set_handler(virq, handle_percpu_irq);
	setup_irq(virq, action);
527 528
}

529
static __init void gic_ipi_init(void)
530
{
531 532 533
	int i;

	/* Use last 2 * NR_CPUS interrupts as IPIs */
534
	gic_resched_int_base = gic_shared_intrs - nr_cpu_ids;
535 536 537 538 539 540 541 542 543 544 545 546 547
	gic_call_int_base = gic_resched_int_base - nr_cpu_ids;

	for (i = 0; i < nr_cpu_ids; i++) {
		gic_ipi_init_one(gic_call_int_base + i, i, &irq_call);
		gic_ipi_init_one(gic_resched_int_base + i, i, &irq_resched);
	}
}
#else
static inline void gic_ipi_init(void)
{
}
#endif

548
static void __init gic_basic_init(void)
549 550
{
	unsigned int i;
S
Steven J. Hill 已提交
551 552

	board_bind_eic_interrupt = &gic_bind_eic_interrupt;
553 554

	/* Setup defaults */
555
	for (i = 0; i < gic_shared_intrs; i++) {
556 557 558
		gic_set_polarity(i, GIC_POL_POS);
		gic_set_trigger(i, GIC_TRIG_LEVEL);
		gic_reset_mask(i);
559 560
	}

561 562 563
	for (i = 0; i < gic_vpes; i++) {
		unsigned int j;

564
		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
565 566 567
		for (j = 0; j < GIC_NUM_LOCAL_INTRS; j++) {
			if (!gic_local_irq_is_routable(j))
				continue;
568
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_RMASK), 1 << j);
569 570
		}
	}
571 572
}

573 574
static int gic_local_irq_domain_map(struct irq_domain *d, unsigned int virq,
				    irq_hw_number_t hw)
575
{
576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603
	int intr = GIC_HWIRQ_TO_LOCAL(hw);
	int ret = 0;
	int i;
	unsigned long flags;

	if (!gic_local_irq_is_routable(intr))
		return -EPERM;

	/*
	 * HACK: These are all really percpu interrupts, but the rest
	 * of the MIPS kernel code does not use the percpu IRQ API for
	 * the CP0 timer and performance counter interrupts.
	 */
	if (intr != GIC_LOCAL_INT_TIMER && intr != GIC_LOCAL_INT_PERFCTR) {
		irq_set_chip_and_handler(virq,
					 &gic_local_irq_controller,
					 handle_percpu_devid_irq);
		irq_set_percpu_devid(virq);
	} else {
		irq_set_chip_and_handler(virq,
					 &gic_all_vpes_local_irq_controller,
					 handle_percpu_irq);
	}

	spin_lock_irqsave(&gic_lock, flags);
	for (i = 0; i < gic_vpes; i++) {
		u32 val = GIC_MAP_TO_PIN_MSK | gic_cpu_pin;

604
		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
605 606 607

		switch (intr) {
		case GIC_LOCAL_INT_WD:
608
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_WD_MAP), val);
609 610
			break;
		case GIC_LOCAL_INT_COMPARE:
611
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_MAP), val);
612 613
			break;
		case GIC_LOCAL_INT_TIMER:
614
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_TIMER_MAP), val);
615 616
			break;
		case GIC_LOCAL_INT_PERFCTR:
617
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_PERFCTR_MAP), val);
618 619
			break;
		case GIC_LOCAL_INT_SWINT0:
620
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_SWINT0_MAP), val);
621 622
			break;
		case GIC_LOCAL_INT_SWINT1:
623
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_SWINT1_MAP), val);
624 625
			break;
		case GIC_LOCAL_INT_FDC:
626
			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_FDC_MAP), val);
627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642
			break;
		default:
			pr_err("Invalid local IRQ %d\n", intr);
			ret = -EINVAL;
			break;
		}
	}
	spin_unlock_irqrestore(&gic_lock, flags);

	return ret;
}

static int gic_shared_irq_domain_map(struct irq_domain *d, unsigned int virq,
				     irq_hw_number_t hw)
{
	int intr = GIC_HWIRQ_TO_SHARED(hw);
643 644
	unsigned long flags;

645 646
	irq_set_chip_and_handler(virq, &gic_level_irq_controller,
				 handle_level_irq);
647 648

	spin_lock_irqsave(&gic_lock, flags);
649
	gic_map_to_pin(intr, gic_cpu_pin);
650
	/* Map to VPE 0 by default */
651
	gic_map_to_vpe(intr, 0);
652
	set_bit(intr, pcpu_masks[0].pcpu_mask);
653 654 655 656 657
	spin_unlock_irqrestore(&gic_lock, flags);

	return 0;
}

658 659 660 661 662 663 664 665
static int gic_irq_domain_map(struct irq_domain *d, unsigned int virq,
			      irq_hw_number_t hw)
{
	if (GIC_HWIRQ_TO_LOCAL(hw) < GIC_NUM_LOCAL_INTRS)
		return gic_local_irq_domain_map(d, virq, hw);
	return gic_shared_irq_domain_map(d, virq, hw);
}

666 667 668 669 670
static struct irq_domain_ops gic_irq_domain_ops = {
	.map = gic_irq_domain_map,
	.xlate = irq_domain_xlate_twocell,
};

671
void __init gic_init(unsigned long gic_base_addr,
672
		     unsigned long gic_addrspace_size, unsigned int cpu_vec,
673 674 675 676
		     unsigned int irqbase)
{
	unsigned int gicconfig;

677
	gic_base = ioremap_nocache(gic_base_addr, gic_addrspace_size);
678

679
	gicconfig = gic_read(GIC_REG(SHARED, GIC_SH_CONFIG));
680
	gic_shared_intrs = (gicconfig & GIC_SH_CONFIG_NUMINTRS_MSK) >>
681
		   GIC_SH_CONFIG_NUMINTRS_SHF;
682
	gic_shared_intrs = ((gic_shared_intrs + 1) * 8);
683

684
	gic_vpes = (gicconfig & GIC_SH_CONFIG_NUMVPES_MSK) >>
685
		  GIC_SH_CONFIG_NUMVPES_SHF;
686
	gic_vpes = gic_vpes + 1;
687

688 689 690 691 692 693 694 695 696 697 698
	if (cpu_has_veic) {
		/* Always use vector 1 in EIC mode */
		gic_cpu_pin = 0;
		set_vi_handler(gic_cpu_pin + GIC_PIN_TO_VEC_OFFSET,
			       __gic_irq_dispatch);
	} else {
		gic_cpu_pin = cpu_vec - GIC_CPU_PIN_OFFSET;
		irq_set_chained_handler(MIPS_CPU_IRQ_BASE + cpu_vec,
					gic_irq_dispatch);
	}

699 700
	gic_irq_domain = irq_domain_add_simple(NULL, GIC_NUM_LOCAL_INTRS +
					       gic_shared_intrs, irqbase,
701 702 703
					       &gic_irq_domain_ops, NULL);
	if (!gic_irq_domain)
		panic("Failed to add GIC IRQ domain");
704

705
	gic_basic_init();
706 707

	gic_ipi_init();
708
}