shithub: purgatorio

ref: a411870ee4640241e3c494367d922847da84f972
dir: /os/boot/puma/trap.c/

View raw version
#include "boot.h"

typedef struct IrqEntry {
		void	(*r)(Ureg*, void*);
		void 	*a;				
} IrqEntry;

IrqEntry Irq[V_MAXNUM+1];

static void dumpstk(ulong *);
void dumpregs(Ureg* ureg);

void
setvec(int v, void (*f)(Ureg*, void*), void* a)
{
	if(v < 0 || v >= V_MAXNUM)
		panic("setvec: interrupt source %d out of range\n", v);
	Irq[v].r = f;
	Irq[v].a = a;
}

ulong irqstack[64];
ulong fiqstack[64];
ulong abtstack[64];
ulong undstack[64];

static void
safeintr(Ureg*, void *a)
{
	int v = (int)a;
//	print("spurious interrupt %d\n", v);
	USED(v);
}

void
trapinit(void)
{
	int offset;
	ulong op;
	int v;
	int s;

	s = splhi();

	/* set up stacks for various exceptions */ 
	setr13(PsrMirq, irqstack+nelem(irqstack)-1);
	setr13(PsrMfiq, fiqstack+nelem(fiqstack)-1);
	setr13(PsrMabt, abtstack+nelem(abtstack)-1);
	setr13(PsrMund, undstack+nelem(undstack)-1);

	for(v = 0; v <= V_MAXNUM; v++) {
		Irq[v].r = safeintr;
		Irq[v].a = (void *)v;
	}

	/* Reset Exception */
	offset = ((((ulong) _vsvccall) - 0x0)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0x0) = op; 

	/* Undefined Instruction Exception */
	offset = ((((ulong) _vundcall) - 0x4)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0x4) = op;

	/* SWI Exception */
	offset = ((((ulong) _vsvccall) - 0x8)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0x8) = op;

	/* Prefetch Abort Exception */
	offset = ((((ulong) _vpabcall) - 0xc)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0xc) = op;

	/* Data Abort Exception */
	offset = ((((ulong) _vdabcall) - 0x10)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0x10) = op;

	/* IRQ Exception */
 	offset = ((((ulong) _virqcall) - 0x18)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0x18) = op;

	/* FIQ Exception */
 	offset = ((((ulong) _vfiqcall) - 0x1c)-8) >> 2;
	op = ( 0xea << 24 ) | offset;
	*((ulong *) 0x1c) = op;


	flushIcache();
	writeBackDC();
	flushDcache();
	flushIcache();
	drainWBuffer();

	splx(s);
}

/*
 *  trap is called splhi().
 */

void
trap(Ureg* ureg)
{
	ushort mask;
	IrqEntry *ip;

	/*
	 * All interrupts/exceptions should be resumed at ureg->pc-4,
	 * except for Data Abort which resumes at ureg->pc-8.
	 */
	ureg->pc -= 4;

	switch(ureg->type) {
	case PsrMirq:				/* Interrupt Request */
		mask = *(uchar*)HARI1 | ((*(uchar*)HARI2) << 8);
		ip = Irq;
		while (mask != 0) {
			if(mask&1)
				ip->r(ureg, ip->a);
			ip++;
			mask >>= 1;
		}
		break;

	case PsrMfiq:					/* FIQ */
		mask = *(uchar*)HARI1 & HARI1_FIQ_MASK;
		ip = Irq;
		while (mask != 0) {
			if(mask&1)
				ip->r(ureg, ip->a);
			ip++;
			mask >>= 1;
		}
		break;

	case PsrMund:			/* Undefined instruction */
		dumpregs(ureg);
		panic("Undefined Instruction Exception\n");
		break;

	case PsrMsvc:				/* Jump through 0 or SWI  */
		dumpregs(ureg);
		panic("SVC/SWI Exception\n");
		break;

	case PsrMabt:					/* Prefetch abort */
		ureg->pc -= 4;
		/* FALLTHROUGH */

	case PsrMabt+1:	{				/* Data abort */
		uint far =0;
		uint fsr =0;

		USED(far,fsr);
		fsr = 0;	/*mmuregr(CpFSR);*/
		far = 0;	/*mmuregr(CpFAR);	*/
		if (ureg->type == PsrMabt)
			print("Prefetch Abort/");
		print("Data Abort\n");
		
		print("Data Abort: FSR %8.8uX FAR %8.8uX\n", fsr, far);
		/* FALLTHROUGH */
	}
	default:
		dumpregs(ureg);
		panic("exception %uX\n", ureg->type);
		break;
	}

	splhi();
}

void
dumpregs(Ureg* ureg)
{
	print("PSR %8.8uX type %2.2uX PC %8.8uX LINK %8.8uX\n",
		ureg->psr, ureg->type, ureg->pc, ureg->link);
	print("R14 %8.8uX R13 %8.8uX R12 %8.8uX R11 %8.8uX R10 %8.8uX\n",
		ureg->r14, ureg->r13, ureg->r12, ureg->r11, ureg->r10);
	print("R9  %8.8uX R8  %8.8uX R7  %8.8uX R6  %8.8uX R5  %8.8uX\n",
		ureg->r9, ureg->r8, ureg->r7, ureg->r6, ureg->r5);
	print("R4  %8.8uX R3  %8.8uX R2  %8.8uX R1  %8.8uX R0  %8.8uX\n",
		ureg->r4, ureg->r3, ureg->r2, ureg->r1, ureg->r0);
	print("Stack is at: %8.8uX\n", ureg);
/*	print("CPSR %8.8uX SPSR %8.8uX\n", cpsrr(), spsrr());*/
}