malta-int.c 21.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
/*
 * Carsten Langgaard, carstenl@mips.com
 * Copyright (C) 2000, 2001, 2004 MIPS Technologies, Inc.
 * Copyright (C) 2001 Ralf Baechle
 *
 *  This program is free software; you can distribute it and/or modify it
 *  under the terms of the GNU General Public License (Version 2) as
 *  published by the Free Software Foundation.
 *
 *  This program is distributed in the hope it will be useful, but WITHOUT
 *  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 *  FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 *  for more details.
 *
 *  You should have received a copy of the GNU General Public License along
 *  with this program; if not, write to the Free Software Foundation, Inc.,
 *  59 Temple Place - Suite 330, Boston MA 02111-1307, USA.
 *
 * Routines for generic manipulation of the interrupts found on the MIPS
 * Malta board.
 * The interrupt controller is located in the South Bridge a PIIX4 device
 * with two internal 82C95 interrupt controllers.
 */
#include <linux/init.h>
#include <linux/irq.h>
#include <linux/sched.h>
27
#include <linux/smp.h>
L
Linus Torvalds 已提交
28
#include <linux/interrupt.h>
29
#include <linux/io.h>
L
Linus Torvalds 已提交
30
#include <linux/kernel_stat.h>
31
#include <linux/kernel.h>
L
Linus Torvalds 已提交
32 33
#include <linux/random.h>

34
#include <asm/traps.h>
L
Linus Torvalds 已提交
35
#include <asm/i8259.h>
36
#include <asm/irq_cpu.h>
37
#include <asm/irq_regs.h>
L
Linus Torvalds 已提交
38 39 40 41 42 43
#include <asm/mips-boards/malta.h>
#include <asm/mips-boards/maltaint.h>
#include <asm/mips-boards/piix4.h>
#include <asm/gt64120.h>
#include <asm/mips-boards/generic.h>
#include <asm/mips-boards/msc01_pci.h>
44
#include <asm/msc01_ic.h>
45 46
#include <asm/gic.h>
#include <asm/gcmpregs.h>
47
#include <asm/setup.h>
48 49 50 51 52

int gcmp_present = -1;
static unsigned long _msc01_biu_base;
static unsigned long _gcmp_base;
static unsigned int ipi_map[NR_CPUS];
L
Linus Torvalds 已提交
53

54
static DEFINE_RAW_SPINLOCK(mips_irq_lock);
L
Linus Torvalds 已提交
55 56 57 58 59 60 61 62 63

static inline int mips_pcibios_iack(void)
{
	int irq;

	/*
	 * Determine highest priority pending interrupt by performing
	 * a PCI Interrupt Acknowledge cycle.
	 */
64 65 66 67 68
	switch (mips_revision_sconid) {
	case MIPS_REVISION_SCON_SOCIT:
	case MIPS_REVISION_SCON_ROCIT:
	case MIPS_REVISION_SCON_SOCITSC:
	case MIPS_REVISION_SCON_SOCITSCP:
69
		MSC_READ(MSC01_PCI_IACK, irq);
L
Linus Torvalds 已提交
70 71
		irq &= 0xff;
		break;
72
	case MIPS_REVISION_SCON_GT64120:
L
Linus Torvalds 已提交
73 74 75
		irq = GT_READ(GT_PCI0_IACK_OFS);
		irq &= 0xff;
		break;
76
	case MIPS_REVISION_SCON_BONITO:
L
Linus Torvalds 已提交
77 78 79 80 81 82 83 84
		/* The following will generate a PCI IACK cycle on the
		 * Bonito controller. It's a little bit kludgy, but it
		 * was the easiest way to implement it in hardware at
		 * the given time.
		 */
		BONITO_PCIMAP_CFG = 0x20000;

		/* Flush Bonito register block */
85
		(void) BONITO_PCIMAP_CFG;
R
Ralf Baechle 已提交
86
		iob();	  /* sync */
L
Linus Torvalds 已提交
87

88
		irq = __raw_readl((u32 *)_pcictrl_bonito_pcicfg);
R
Ralf Baechle 已提交
89
		iob();	  /* sync */
L
Linus Torvalds 已提交
90 91 92 93
		irq &= 0xff;
		BONITO_PCIMAP_CFG = 0;
		break;
	default:
94
		printk(KERN_WARNING "Unknown system controller.\n");
L
Linus Torvalds 已提交
95 96 97 98 99
		return -1;
	}
	return irq;
}

100
static inline int get_int(void)
L
Linus Torvalds 已提交
101 102
{
	unsigned long flags;
103
	int irq;
104
	raw_spin_lock_irqsave(&mips_irq_lock, flags);
L
Linus Torvalds 已提交
105

106
	irq = mips_pcibios_iack();
L
Linus Torvalds 已提交
107 108

	/*
R
Ralf Baechle 已提交
109 110 111
	 * The only way we can decide if an interrupt is spurious
	 * is by checking the 8259 registers.  This needs a spinlock
	 * on an SMP system,  so leave it up to the generic code...
L
Linus Torvalds 已提交
112 113
	 */

114
	raw_spin_unlock_irqrestore(&mips_irq_lock, flags);
L
Linus Torvalds 已提交
115

116
	return irq;
L
Linus Torvalds 已提交
117 118
}

119
static void malta_hw0_irqdispatch(void)
L
Linus Torvalds 已提交
120 121 122
{
	int irq;

123
	irq = get_int();
124
	if (irq < 0) {
125 126
		/* interrupt has already been cleared */
		return;
127
	}
L
Linus Torvalds 已提交
128

129
	do_IRQ(MALTA_INT_BASE + irq);
L
Linus Torvalds 已提交
130 131
}

132 133 134 135
static void malta_ipi_irqdispatch(void)
{
	int irq;

136 137 138
	if (gic_compare_int())
		do_IRQ(MIPS_GIC_IRQ_BASE);

139 140
	irq = gic_get_int();
	if (irq < 0)
R
Ralf Baechle 已提交
141
		return;	 /* interrupt has already been cleared */
142 143 144 145

	do_IRQ(MIPS_GIC_IRQ_BASE + irq);
}

146
static void corehi_irqdispatch(void)
L
Linus Torvalds 已提交
147
{
148
	unsigned int intedge, intsteer, pcicmd, pcibadaddr;
149
	unsigned int pcimstat, intisr, inten, intpol;
150
	unsigned int intrcause, datalo, datahi;
151
	struct pt_regs *regs = get_irq_regs();
L
Linus Torvalds 已提交
152

153
	printk(KERN_EMERG "CoreHI interrupt, shouldn't happen, we die here!\n");
R
Ralf Baechle 已提交
154
	printk(KERN_EMERG "epc	 : %08lx\nStatus: %08lx\n"
155 156 157
			"Cause : %08lx\nbadVaddr : %08lx\n",
			regs->cp0_epc, regs->cp0_status,
			regs->cp0_cause, regs->cp0_badvaddr);
158 159 160 161 162 163

	/* Read all the registers and then print them as there is a
	   problem with interspersed printk's upsetting the Bonito controller.
	   Do it for the others too.
	*/

164
	switch (mips_revision_sconid) {
165
	case MIPS_REVISION_SCON_SOCIT:
166 167 168
	case MIPS_REVISION_SCON_ROCIT:
	case MIPS_REVISION_SCON_SOCITSC:
	case MIPS_REVISION_SCON_SOCITSCP:
169 170 171 172 173 174
		ll_msc_irq();
		break;
	case MIPS_REVISION_SCON_GT64120:
		intrcause = GT_READ(GT_INTRCAUSE_OFS);
		datalo = GT_READ(GT_CPUERR_ADDRLO_OFS);
		datahi = GT_READ(GT_CPUERR_ADDRHI_OFS);
175 176 177
		printk(KERN_EMERG "GT_INTRCAUSE = %08x\n", intrcause);
		printk(KERN_EMERG "GT_CPUERR_ADDR = %02x%08x\n",
				datahi, datalo);
178 179 180 181 182 183 184 185 186 187
		break;
	case MIPS_REVISION_SCON_BONITO:
		pcibadaddr = BONITO_PCIBADADDR;
		pcimstat = BONITO_PCIMSTAT;
		intisr = BONITO_INTISR;
		inten = BONITO_INTEN;
		intpol = BONITO_INTPOL;
		intedge = BONITO_INTEDGE;
		intsteer = BONITO_INTSTEER;
		pcicmd = BONITO_PCICMD;
188 189 190 191 192 193 194 195
		printk(KERN_EMERG "BONITO_INTISR = %08x\n", intisr);
		printk(KERN_EMERG "BONITO_INTEN = %08x\n", inten);
		printk(KERN_EMERG "BONITO_INTPOL = %08x\n", intpol);
		printk(KERN_EMERG "BONITO_INTEDGE = %08x\n", intedge);
		printk(KERN_EMERG "BONITO_INTSTEER = %08x\n", intsteer);
		printk(KERN_EMERG "BONITO_PCICMD = %08x\n", pcicmd);
		printk(KERN_EMERG "BONITO_PCIBADADDR = %08x\n", pcibadaddr);
		printk(KERN_EMERG "BONITO_PCIMSTAT = %08x\n", pcimstat);
196 197
		break;
	}
L
Linus Torvalds 已提交
198

199
	die("CoreHi interrupt", regs);
L
Linus Torvalds 已提交
200 201
}

202 203
static inline int clz(unsigned long x)
{
204
	__asm__(
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
	"	.set	push					\n"
	"	.set	mips32					\n"
	"	clz	%0, %1					\n"
	"	.set	pop					\n"
	: "=r" (x)
	: "r" (x));

	return x;
}

/*
 * Version of ffs that only looks at bits 12..15.
 */
static inline unsigned int irq_ffs(unsigned int pending)
{
#if defined(CONFIG_CPU_MIPS32) || defined(CONFIG_CPU_MIPS64)
	return -clz(pending) + 31 - CAUSEB_IP;
#else
	unsigned int a0 = 7;
	unsigned int t0;

226
	t0 = pending & 0xf000;
227 228 229
	t0 = t0 < 1;
	t0 = t0 << 2;
	a0 = a0 - t0;
230
	pending = pending << t0;
231

232
	t0 = pending & 0xc000;
233 234 235
	t0 = t0 < 1;
	t0 = t0 << 1;
	a0 = a0 - t0;
236
	pending = pending << t0;
237

238
	t0 = pending & 0x8000;
239
	t0 = t0 < 1;
240
	/* t0 = t0 << 2; */
241
	a0 = a0 - t0;
242
	/* pending = pending << t0; */
243 244 245 246 247 248 249 250 251 252 253

	return a0;
#endif
}

/*
 * IRQs on the Malta board look basically (barring software IRQs which we
 * don't use at all and all external interrupt sources are combined together
 * on hardware interrupt 0 (MIPS IRQ 2)) like:
 *
 *	MIPS IRQ	Source
R
Ralf Baechle 已提交
254 255 256 257 258 259 260 261 262
 *	--------	------
 *	       0	Software (ignored)
 *	       1	Software (ignored)
 *	       2	Combined hardware interrupt (hw0)
 *	       3	Hardware (ignored)
 *	       4	Hardware (ignored)
 *	       5	Hardware (ignored)
 *	       6	Hardware (ignored)
 *	       7	R4k timer (what we use)
263 264 265
 *
 * We handle the IRQ according to _our_ priority which is:
 *
R
Ralf Baechle 已提交
266 267
 * Highest ----	    R4k Timer
 * Lowest  ----	    Combined hardware interrupt
268 269 270 271 272
 *
 * then we just return, if multiple IRQs are pending then we will just take
 * another exception, big deal.
 */

273
asmlinkage void plat_irq_dispatch(void)
274 275 276 277
{
	unsigned int pending = read_c0_cause() & read_c0_status() & ST0_IM;
	int irq;

278 279 280 281 282
	if (unlikely(!pending)) {
		spurious_interrupt();
		return;
	}

283 284 285
	irq = irq_ffs(pending);

	if (irq == MIPSCPU_INT_I8259A)
286
		malta_hw0_irqdispatch();
287 288
	else if (gic_present && ((1 << irq) & ipi_map[smp_processor_id()]))
		malta_ipi_irqdispatch();
289
	else
290
		do_IRQ(MIPS_CPU_IRQ_BASE + irq);
291 292
}

293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316
#ifdef CONFIG_MIPS_MT_SMP


#define GIC_MIPS_CPU_IPI_RESCHED_IRQ	3
#define GIC_MIPS_CPU_IPI_CALL_IRQ	4

#define MIPS_CPU_IPI_RESCHED_IRQ 0	/* SW int 0 for resched */
#define C_RESCHED C_SW0
#define MIPS_CPU_IPI_CALL_IRQ 1		/* SW int 1 for resched */
#define C_CALL C_SW1
static int cpu_ipi_resched_irq, cpu_ipi_call_irq;

static void ipi_resched_dispatch(void)
{
	do_IRQ(MIPS_CPU_IRQ_BASE + MIPS_CPU_IPI_RESCHED_IRQ);
}

static void ipi_call_dispatch(void)
{
	do_IRQ(MIPS_CPU_IRQ_BASE + MIPS_CPU_IPI_CALL_IRQ);
}

static irqreturn_t ipi_resched_interrupt(int irq, void *dev_id)
{
317 318
	scheduler_ipi();

319 320 321 322 323 324 325 326 327 328 329 330
	return IRQ_HANDLED;
}

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

	return IRQ_HANDLED;
}

static struct irqaction irq_resched = {
	.handler	= ipi_resched_interrupt,
Y
Yong Zhang 已提交
331
	.flags		= IRQF_PERCPU,
332 333 334 335 336
	.name		= "IPI_resched"
};

static struct irqaction irq_call = {
	.handler	= ipi_call_interrupt,
Y
Yong Zhang 已提交
337
	.flags		= IRQF_PERCPU,
338 339
	.name		= "IPI_call"
};
340
#endif /* CONFIG_MIPS_MT_SMP */
341 342 343 344 345

static int gic_resched_int_base;
static int gic_call_int_base;
#define GIC_RESCHED_INT(cpu) (gic_resched_int_base+(cpu))
#define GIC_CALL_INT(cpu) (gic_call_int_base+(cpu))
T
Tim Anderson 已提交
346 347 348 349 350 351 352 353 354 355

unsigned int plat_ipi_call_int_xlate(unsigned int cpu)
{
	return GIC_CALL_INT(cpu);
}

unsigned int plat_ipi_resched_int_xlate(unsigned int cpu)
{
	return GIC_RESCHED_INT(cpu);
}
356

357 358
static struct irqaction i8259irq = {
	.handler = no_action,
359 360
	.name = "XT-PIC cascade",
	.flags = IRQF_NO_THREAD,
361 362 363 364
};

static struct irqaction corehi_irqaction = {
	.handler = no_action,
365 366
	.name = "CoreHi",
	.flags = IRQF_NO_THREAD,
367 368
};

369
static msc_irqmap_t __initdata msc_irqmap[] = {
370 371 372
	{MSC01C_INT_TMR,		MSC01_IRQ_EDGE, 0},
	{MSC01C_INT_PCI,		MSC01_IRQ_LEVEL, 0},
};
373
static int __initdata msc_nr_irqs = ARRAY_SIZE(msc_irqmap);
374

375
static msc_irqmap_t __initdata msc_eicirqmap[] = {
376 377 378 379 380 381 382 383 384 385 386
	{MSC01E_INT_SW0,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_SW1,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_I8259A,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_SMI,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_COREHI,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_CORELO,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_TMR,		MSC01_IRQ_EDGE, 0},
	{MSC01E_INT_PCI,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_PERFCTR,		MSC01_IRQ_LEVEL, 0},
	{MSC01E_INT_CPUCTR,		MSC01_IRQ_LEVEL, 0}
};
387

388
static int __initdata msc_nr_eicirqs = ARRAY_SIZE(msc_eicirqmap);
389

390 391 392 393 394
/*
 * This GIC specific tabular array defines the association between External
 * Interrupts and CPUs/Core Interrupts. The nature of the External
 * Interrupts is also defined here - polarity/trigger.
 */
395 396

#define GIC_CPU_NMI GIC_MAP_TO_NMI_MSK
397 398
#define X GIC_UNUSED

399
static struct gic_intr_map gic_intr_map[GIC_NUM_INTRS] = {
400
	{ X, X,		   X,		X,		0 },
R
Ralf Baechle 已提交
401
	{ X, X,		   X,		X,		0 },
402 403 404 405 406 407 408 409 410 411 412 413 414
	{ X, X,		   X,		X,		0 },
	{ 0, GIC_CPU_INT0, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_INT1, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_INT2, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_INT3, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_INT4, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_INT3, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_INT3, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ X, X,		   X,		X,		0 },
	{ X, X,		   X,		X,		0 },
	{ 0, GIC_CPU_INT3, GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_NMI,  GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
	{ 0, GIC_CPU_NMI,  GIC_POL_POS, GIC_TRIG_LEVEL, GIC_FLAG_TRANSPARENT },
R
Ralf Baechle 已提交
415
	{ X, X,		   X,		X,		0 },
416
	/* The remainder of this table is initialised by fill_ipi_map */
417
};
418
#undef X
419 420 421 422

/*
 * GCMP needs to be detected before any SMP initialisation
 */
423
int __init gcmp_probe(unsigned long addr, unsigned long size)
424
{
425 426 427 428 429
	if (mips_revision_sconid != MIPS_REVISION_SCON_ROCIT) {
		gcmp_present = 0;
		return gcmp_present;
	}

430 431 432 433 434 435 436 437
	if (gcmp_present >= 0)
		return gcmp_present;

	_gcmp_base = (unsigned long) ioremap_nocache(GCMP_BASE_ADDR, GCMP_ADDRSPACE_SZ);
	_msc01_biu_base = (unsigned long) ioremap_nocache(MSC01_BIU_REG_BASE, MSC01_BIU_ADDRSPACE_SZ);
	gcmp_present = (GCMPGCB(GCMPB) & GCMP_GCB_GCMPB_GCMPBASE_MSK) == GCMP_BASE_ADDR;

	if (gcmp_present)
438
		pr_debug("GCMP present\n");
439 440 441
	return gcmp_present;
}

442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457
/* Return the number of IOCU's present */
int __init gcmp_niocu(void)
{
  return gcmp_present ?
    (GCMPGCB(GC) & GCMP_GCB_GC_NUMIOCU_MSK) >> GCMP_GCB_GC_NUMIOCU_SHF :
    0;
}

/* Set GCMP region attributes */
void __init gcmp_setregion(int region, unsigned long base,
			   unsigned long mask, int type)
{
	GCMPGCBn(CMxBASE, region) = base;
	GCMPGCBn(CMxMASK, region) = mask | type;
}

458
#if defined(CONFIG_MIPS_MT_SMP)
459 460 461 462 463 464 465
static void __init fill_ipi_map1(int baseintr, int cpu, int cpupin)
{
	int intr = baseintr + cpu;
	gic_intr_map[intr].cpunum = cpu;
	gic_intr_map[intr].pin = cpupin;
	gic_intr_map[intr].polarity = GIC_POL_POS;
	gic_intr_map[intr].trigtype = GIC_TRIG_EDGE;
466
	gic_intr_map[intr].flags = GIC_FLAG_IPI;
467 468 469
	ipi_map[cpu] |= (1 << (cpupin + 2));
}

470
static void __init fill_ipi_map(void)
471
{
472
	int cpu;
473

474 475 476
	for (cpu = 0; cpu < NR_CPUS; cpu++) {
		fill_ipi_map1(gic_resched_int_base, cpu, GIC_CPU_INT1);
		fill_ipi_map1(gic_call_int_base, cpu, GIC_CPU_INT2);
477 478
	}
}
479
#endif
480

481 482 483
void __init arch_init_ipiirq(int irq, struct irqaction *action)
{
	setup_irq(irq, action);
484
	irq_set_handler(irq, handle_percpu_irq);
485 486
}

L
Linus Torvalds 已提交
487 488 489
void __init arch_init_irq(void)
{
	init_i8259_irqs();
490 491

	if (!cpu_has_veic)
492
		mips_cpu_irq_init();
493

494 495 496 497
	if (gcmp_present)  {
		GCMPGCB(GICBA) = GIC_BASE_ADDR | GCMP_GCB_GICBA_EN_MSK;
		gic_present = 1;
	} else {
498 499 500 501 502 503 504 505
		if (mips_revision_sconid == MIPS_REVISION_SCON_ROCIT) {
			_msc01_biu_base = (unsigned long)
					ioremap_nocache(MSC01_BIU_REG_BASE,
						MSC01_BIU_ADDRSPACE_SZ);
			gic_present = (REG(_msc01_biu_base, MSC01_SC_CFG) &
					MSC01_SC_CFG_GICPRES_MSK) >>
					MSC01_SC_CFG_GICPRES_SHF;
		}
506 507
	}
	if (gic_present)
508
		pr_debug("GIC present\n");
509

510 511 512
	switch (mips_revision_sconid) {
	case MIPS_REVISION_SCON_SOCIT:
	case MIPS_REVISION_SCON_ROCIT:
513
		if (cpu_has_veic)
514 515 516
			init_msc_irqs(MIPS_MSC01_IC_REG_BASE,
					MSC01E_INT_BASE, msc_eicirqmap,
					msc_nr_eicirqs);
517
		else
518 519 520
			init_msc_irqs(MIPS_MSC01_IC_REG_BASE,
					MSC01C_INT_BASE, msc_irqmap,
					msc_nr_irqs);
521 522
		break;

523 524
	case MIPS_REVISION_SCON_SOCITSC:
	case MIPS_REVISION_SCON_SOCITSCP:
525
		if (cpu_has_veic)
526 527 528
			init_msc_irqs(MIPS_SOCITSC_IC_REG_BASE,
					MSC01E_INT_BASE, msc_eicirqmap,
					msc_nr_eicirqs);
529
		else
530 531 532
			init_msc_irqs(MIPS_SOCITSC_IC_REG_BASE,
					MSC01C_INT_BASE, msc_irqmap,
					msc_nr_irqs);
533 534 535
	}

	if (cpu_has_veic) {
536 537 538 539
		set_vi_handler(MSC01E_INT_I8259A, malta_hw0_irqdispatch);
		set_vi_handler(MSC01E_INT_COREHI, corehi_irqdispatch);
		setup_irq(MSC01E_INT_BASE+MSC01E_INT_I8259A, &i8259irq);
		setup_irq(MSC01E_INT_BASE+MSC01E_INT_COREHI, &corehi_irqaction);
540
	} else if (cpu_has_vint) {
541 542
		set_vi_handler(MIPSCPU_INT_I8259A, malta_hw0_irqdispatch);
		set_vi_handler(MIPSCPU_INT_COREHI, corehi_irqdispatch);
543
#ifdef CONFIG_MIPS_MT_SMTC
544
		setup_irq_smtc(MIPS_CPU_IRQ_BASE+MIPSCPU_INT_I8259A, &i8259irq,
545
			(0x100 << MIPSCPU_INT_I8259A));
546
		setup_irq_smtc(MIPS_CPU_IRQ_BASE+MIPSCPU_INT_COREHI,
547
			&corehi_irqaction, (0x100 << MIPSCPU_INT_COREHI));
548 549 550 551 552 553 554 555 556 557 558 559
		/*
		 * Temporary hack to ensure that the subsidiary device
		 * interrupts coing in via the i8259A, but associated
		 * with low IRQ numbers, will restore the Status.IM
		 * value associated with the i8259A.
		 */
		{
			int i;

			for (i = 0; i < 16; i++)
				irq_hwmask[i] = (0x100 << MIPSCPU_INT_I8259A);
		}
560
#else /* Not SMTC */
561
		setup_irq(MIPS_CPU_IRQ_BASE+MIPSCPU_INT_I8259A, &i8259irq);
562 563
		setup_irq(MIPS_CPU_IRQ_BASE+MIPSCPU_INT_COREHI,
						&corehi_irqaction);
564
#endif /* CONFIG_MIPS_MT_SMTC */
565
	} else {
566
		setup_irq(MIPS_CPU_IRQ_BASE+MIPSCPU_INT_I8259A, &i8259irq);
567 568
		setup_irq(MIPS_CPU_IRQ_BASE+MIPSCPU_INT_COREHI,
						&corehi_irqaction);
569
	}
570 571 572 573

	if (gic_present) {
		/* FIXME */
		int i;
574
#if defined(CONFIG_MIPS_MT_SMP)
575 576
		gic_call_int_base = GIC_NUM_INTRS - NR_CPUS;
		gic_resched_int_base = gic_call_int_base - NR_CPUS;
577
		fill_ipi_map();
578 579 580
#endif
		gic_init(GIC_BASE_ADDR, GIC_ADDRSPACE_SZ, gic_intr_map,
				ARRAY_SIZE(gic_intr_map), MIPS_GIC_IRQ_BASE);
581 582 583 584 585 586 587
		if (!gcmp_present) {
			/* Enable the GIC */
			i = REG(_msc01_biu_base, MSC01_SC_CFG);
			REG(_msc01_biu_base, MSC01_SC_CFG) =
				(i | (0x1 << MSC01_SC_CFG_GICENA_SHF));
			pr_debug("GIC Enabled\n");
		}
588
#if defined(CONFIG_MIPS_MT_SMP)
589 590 591 592 593 594 595 596 597 598 599
		/* set up ipi interrupts */
		if (cpu_has_vint) {
			set_vi_handler(MIPSCPU_INT_IPI0, malta_ipi_irqdispatch);
			set_vi_handler(MIPSCPU_INT_IPI1, malta_ipi_irqdispatch);
		}
		/* Argh.. this really needs sorting out.. */
		printk("CPU%d: status register was %08x\n", smp_processor_id(), read_c0_status());
		write_c0_status(read_c0_status() | STATUSF_IP3 | STATUSF_IP4);
		printk("CPU%d: status register now %08x\n", smp_processor_id(), read_c0_status());
		write_c0_status(0x1100dc00);
		printk("CPU%d: status register frc %08x\n", smp_processor_id(), read_c0_status());
600
		for (i = 0; i < NR_CPUS; i++) {
601 602 603 604
			arch_init_ipiirq(MIPS_GIC_IRQ_BASE +
					 GIC_RESCHED_INT(i), &irq_resched);
			arch_init_ipiirq(MIPS_GIC_IRQ_BASE +
					 GIC_CALL_INT(i), &irq_call);
605
		}
606
#endif
607
	} else {
608
#if defined(CONFIG_MIPS_MT_SMP)
609 610 611 612 613 614 615 616 617 618 619 620 621 622
		/* set up ipi interrupts */
		if (cpu_has_veic) {
			set_vi_handler (MSC01E_INT_SW0, ipi_resched_dispatch);
			set_vi_handler (MSC01E_INT_SW1, ipi_call_dispatch);
			cpu_ipi_resched_irq = MSC01E_INT_SW0;
			cpu_ipi_call_irq = MSC01E_INT_SW1;
		} else {
			if (cpu_has_vint) {
				set_vi_handler (MIPS_CPU_IPI_RESCHED_IRQ, ipi_resched_dispatch);
				set_vi_handler (MIPS_CPU_IPI_CALL_IRQ, ipi_call_dispatch);
			}
			cpu_ipi_resched_irq = MIPS_CPU_IRQ_BASE + MIPS_CPU_IPI_RESCHED_IRQ;
			cpu_ipi_call_irq = MIPS_CPU_IRQ_BASE + MIPS_CPU_IPI_CALL_IRQ;
		}
623 624
		arch_init_ipiirq(cpu_ipi_resched_irq, &irq_resched);
		arch_init_ipiirq(cpu_ipi_call_irq, &irq_call);
625
#endif
626
	}
627 628 629 630 631 632 633 634 635 636 637 638
}

void malta_be_init(void)
{
	if (gcmp_present) {
		/* Could change CM error mask register */
	}
}


static char *tr[8] = {
	"mem",	"gcr",	"gic",	"mmio",
R
Ralf Baechle 已提交
639
	"0x04", "0x05", "0x06", "0x07"
640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677
};

static char *mcmd[32] = {
	[0x00] = "0x00",
	[0x01] = "Legacy Write",
	[0x02] = "Legacy Read",
	[0x03] = "0x03",
	[0x04] = "0x04",
	[0x05] = "0x05",
	[0x06] = "0x06",
	[0x07] = "0x07",
	[0x08] = "Coherent Read Own",
	[0x09] = "Coherent Read Share",
	[0x0a] = "Coherent Read Discard",
	[0x0b] = "Coherent Ready Share Always",
	[0x0c] = "Coherent Upgrade",
	[0x0d] = "Coherent Writeback",
	[0x0e] = "0x0e",
	[0x0f] = "0x0f",
	[0x10] = "Coherent Copyback",
	[0x11] = "Coherent Copyback Invalidate",
	[0x12] = "Coherent Invalidate",
	[0x13] = "Coherent Write Invalidate",
	[0x14] = "Coherent Completion Sync",
	[0x15] = "0x15",
	[0x16] = "0x16",
	[0x17] = "0x17",
	[0x18] = "0x18",
	[0x19] = "0x19",
	[0x1a] = "0x1a",
	[0x1b] = "0x1b",
	[0x1c] = "0x1c",
	[0x1d] = "0x1d",
	[0x1e] = "0x1e",
	[0x1f] = "0x1f"
};

static char *core[8] = {
R
Ralf Baechle 已提交
678
	"Invalid/OK",	"Invalid/Data",
679 680
	"Shared/OK",	"Shared/Data",
	"Modified/OK",	"Modified/Data",
R
Ralf Baechle 已提交
681
	"Exclusive/OK", "Exclusive/Data"
682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753
};

static char *causes[32] = {
	"None", "GC_WR_ERR", "GC_RD_ERR", "COH_WR_ERR",
	"COH_RD_ERR", "MMIO_WR_ERR", "MMIO_RD_ERR", "0x07",
	"0x08", "0x09", "0x0a", "0x0b",
	"0x0c", "0x0d", "0x0e", "0x0f",
	"0x10", "0x11", "0x12", "0x13",
	"0x14", "0x15", "0x16", "INTVN_WR_ERR",
	"INTVN_RD_ERR", "0x19", "0x1a", "0x1b",
	"0x1c", "0x1d", "0x1e", "0x1f"
};

int malta_be_handler(struct pt_regs *regs, int is_fixup)
{
	/* This duplicates the handling in do_be which seems wrong */
	int retval = is_fixup ? MIPS_BE_FIXUP : MIPS_BE_FATAL;

	if (gcmp_present) {
		unsigned long cm_error = GCMPGCB(GCMEC);
		unsigned long cm_addr = GCMPGCB(GCMEA);
		unsigned long cm_other = GCMPGCB(GCMEO);
		unsigned long cause, ocause;
		char buf[256];

		cause = (cm_error & GCMP_GCB_GMEC_ERROR_TYPE_MSK);
		if (cause != 0) {
			cause >>= GCMP_GCB_GMEC_ERROR_TYPE_SHF;
			if (cause < 16) {
				unsigned long cca_bits = (cm_error >> 15) & 7;
				unsigned long tr_bits = (cm_error >> 12) & 7;
				unsigned long mcmd_bits = (cm_error >> 7) & 0x1f;
				unsigned long stag_bits = (cm_error >> 3) & 15;
				unsigned long sport_bits = (cm_error >> 0) & 7;

				snprintf(buf, sizeof(buf),
					 "CCA=%lu TR=%s MCmd=%s STag=%lu "
					 "SPort=%lu\n",
					 cca_bits, tr[tr_bits], mcmd[mcmd_bits],
					 stag_bits, sport_bits);
			} else {
				/* glob state & sresp together */
				unsigned long c3_bits = (cm_error >> 18) & 7;
				unsigned long c2_bits = (cm_error >> 15) & 7;
				unsigned long c1_bits = (cm_error >> 12) & 7;
				unsigned long c0_bits = (cm_error >> 9) & 7;
				unsigned long sc_bit = (cm_error >> 8) & 1;
				unsigned long mcmd_bits = (cm_error >> 3) & 0x1f;
				unsigned long sport_bits = (cm_error >> 0) & 7;
				snprintf(buf, sizeof(buf),
					 "C3=%s C2=%s C1=%s C0=%s SC=%s "
					 "MCmd=%s SPort=%lu\n",
					 core[c3_bits], core[c2_bits],
					 core[c1_bits], core[c0_bits],
					 sc_bit ? "True" : "False",
					 mcmd[mcmd_bits], sport_bits);
			}

			ocause = (cm_other & GCMP_GCB_GMEO_ERROR_2ND_MSK) >>
				 GCMP_GCB_GMEO_ERROR_2ND_SHF;

			printk("CM_ERROR=%08lx %s <%s>\n", cm_error,
			       causes[cause], buf);
			printk("CM_ADDR =%08lx\n", cm_addr);
			printk("CM_OTHER=%08lx %s\n", cm_other, causes[ocause]);

			/* reprime cause register */
			GCMPGCB(GCMEC) = 0;
		}
	}

	return retval;
L
Linus Torvalds 已提交
754
}
755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788

void gic_enable_interrupt(int irq_vec)
{
	GIC_SET_INTR_MASK(irq_vec);
}

void gic_disable_interrupt(int irq_vec)
{
	GIC_CLR_INTR_MASK(irq_vec);
}

void gic_irq_ack(struct irq_data *d)
{
	int irq = (d->irq - gic_irq_base);

	GIC_CLR_INTR_MASK(irq);

	if (gic_irq_flags[irq] & GIC_TRIG_EDGE)
		GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), irq);
}

void gic_finish_irq(struct irq_data *d)
{
	/* Enable interrupts. */
	GIC_SET_INTR_MASK(d->irq - gic_irq_base);
}

void __init gic_platform_init(int irqs, struct irq_chip *irq_controller)
{
	int i;

	for (i = gic_irq_base; i < (gic_irq_base + irqs); i++)
		irq_set_chip(i, irq_controller);
}