xref: /plan9/sys/src/9/pc/trap.c (revision 219b2ee8daee37f4aad58d63f21287faa8e4ffdc)
13e12c5d1SDavid du Colombier #include	"u.h"
23e12c5d1SDavid du Colombier #include	"../port/lib.h"
33e12c5d1SDavid du Colombier #include	"mem.h"
43e12c5d1SDavid du Colombier #include	"dat.h"
53e12c5d1SDavid du Colombier #include	"fns.h"
63e12c5d1SDavid du Colombier #include	"io.h"
73e12c5d1SDavid du Colombier #include	"ureg.h"
83e12c5d1SDavid du Colombier #include	"../port/error.h"
93e12c5d1SDavid du Colombier 
103e12c5d1SDavid du Colombier void	noted(Ureg*, ulong);
113e12c5d1SDavid du Colombier 
123e12c5d1SDavid du Colombier void	intr0(void), intr1(void), intr2(void), intr3(void);
133e12c5d1SDavid du Colombier void	intr4(void), intr5(void), intr6(void), intr7(void);
143e12c5d1SDavid du Colombier void	intr8(void), intr9(void), intr10(void), intr11(void);
153e12c5d1SDavid du Colombier void	intr12(void), intr13(void), intr14(void), intr15(void);
163e12c5d1SDavid du Colombier void	intr16(void);
173e12c5d1SDavid du Colombier void	intr24(void), intr25(void), intr26(void), intr27(void);
183e12c5d1SDavid du Colombier void	intr28(void), intr29(void), intr30(void), intr31(void);
193e12c5d1SDavid du Colombier void	intr32(void), intr33(void), intr34(void), intr35(void);
203e12c5d1SDavid du Colombier void	intr36(void), intr37(void), intr38(void), intr39(void);
213e12c5d1SDavid du Colombier void	intr64(void);
223e12c5d1SDavid du Colombier void	intrbad(void);
233e12c5d1SDavid du Colombier 
243e12c5d1SDavid du Colombier int	int0mask = 0xff;	/* interrupts enabled for first 8259 */
253e12c5d1SDavid du Colombier int	int1mask = 0xff;	/* interrupts enabled for second 8259 */
263e12c5d1SDavid du Colombier 
273e12c5d1SDavid du Colombier /*
283e12c5d1SDavid du Colombier  *  trap/interrupt gates
293e12c5d1SDavid du Colombier  */
303e12c5d1SDavid du Colombier Segdesc ilt[256];
31*219b2ee8SDavid du Colombier int badintr[16];
32*219b2ee8SDavid du Colombier 
33*219b2ee8SDavid du Colombier enum
34*219b2ee8SDavid du Colombier {
35*219b2ee8SDavid du Colombier 	Maxhandler=	128,		/* max number of interrupt handlers */
36*219b2ee8SDavid du Colombier };
37*219b2ee8SDavid du Colombier 
38*219b2ee8SDavid du Colombier typedef struct Handler	Handler;
39*219b2ee8SDavid du Colombier struct Handler
40*219b2ee8SDavid du Colombier {
41*219b2ee8SDavid du Colombier 	void	(*r)(void*, void*);
42*219b2ee8SDavid du Colombier 	void	*arg;
43*219b2ee8SDavid du Colombier 	Handler	*next;
44*219b2ee8SDavid du Colombier };
45*219b2ee8SDavid du Colombier 
46*219b2ee8SDavid du Colombier struct
47*219b2ee8SDavid du Colombier {
48*219b2ee8SDavid du Colombier 	Lock;
49*219b2ee8SDavid du Colombier 	Handler	*ivec[256];
50*219b2ee8SDavid du Colombier 	Handler	h[Maxhandler];
51*219b2ee8SDavid du Colombier 	int	free;
52*219b2ee8SDavid du Colombier } halloc;
533e12c5d1SDavid du Colombier 
543e12c5d1SDavid du Colombier void
553e12c5d1SDavid du Colombier sethvec(int v, void (*r)(void), int type, int pri)
563e12c5d1SDavid du Colombier {
573e12c5d1SDavid du Colombier 	ilt[v].d0 = ((ulong)r)&0xFFFF|(KESEL<<16);
583e12c5d1SDavid du Colombier 	ilt[v].d1 = ((ulong)r)&0xFFFF0000|SEGP|SEGPL(pri)|type;
593e12c5d1SDavid du Colombier }
603e12c5d1SDavid du Colombier 
613e12c5d1SDavid du Colombier void
62*219b2ee8SDavid du Colombier setvec(int v, void (*r)(Ureg*, void*), void *arg)
633e12c5d1SDavid du Colombier {
64*219b2ee8SDavid du Colombier 	Handler *h;
65*219b2ee8SDavid du Colombier 
66*219b2ee8SDavid du Colombier 	lock(&halloc);
67*219b2ee8SDavid du Colombier 	if(halloc.free >= Maxhandler)
68*219b2ee8SDavid du Colombier 		panic("out of interrupt handlers");
69*219b2ee8SDavid du Colombier 	h = &halloc.h[halloc.free++];
70*219b2ee8SDavid du Colombier 	h->next = halloc.ivec[v];
71*219b2ee8SDavid du Colombier 	h->r = r;
72*219b2ee8SDavid du Colombier 	h->arg = arg;
73*219b2ee8SDavid du Colombier 	halloc.ivec[v] = h;
74*219b2ee8SDavid du Colombier 	unlock(&halloc);
753e12c5d1SDavid du Colombier 
763e12c5d1SDavid du Colombier 	/*
773e12c5d1SDavid du Colombier 	 *  enable corresponding interrupt in 8259
783e12c5d1SDavid du Colombier 	 */
793e12c5d1SDavid du Colombier 	if((v&~0x7) == Int0vec){
803e12c5d1SDavid du Colombier 		int0mask &= ~(1<<(v&7));
813e12c5d1SDavid du Colombier 		outb(Int0aux, int0mask);
823e12c5d1SDavid du Colombier 	} else if((v&~0x7) == Int1vec){
833e12c5d1SDavid du Colombier 		int1mask &= ~(1<<(v&7));
843e12c5d1SDavid du Colombier 		outb(Int1aux, int1mask);
853e12c5d1SDavid du Colombier 	}
863e12c5d1SDavid du Colombier }
873e12c5d1SDavid du Colombier 
883e12c5d1SDavid du Colombier void
89*219b2ee8SDavid du Colombier debugbpt(Ureg *ur, void *a)
903e12c5d1SDavid du Colombier {
913e12c5d1SDavid du Colombier 	char buf[ERRLEN];
923e12c5d1SDavid du Colombier 
93*219b2ee8SDavid du Colombier 	USED(a);
94*219b2ee8SDavid du Colombier 
953e12c5d1SDavid du Colombier 	if(u == 0)
963e12c5d1SDavid du Colombier 		panic("kernel bpt");
973e12c5d1SDavid du Colombier 	/* restore pc to instruction that caused the trap */
983e12c5d1SDavid du Colombier 	ur->pc--;
993e12c5d1SDavid du Colombier 	sprint(buf, "sys: breakpoint");
1003e12c5d1SDavid du Colombier 	postnote(u->p, 1, buf, NDebug);
1013e12c5d1SDavid du Colombier }
1023e12c5d1SDavid du Colombier 
1033e12c5d1SDavid du Colombier /*
1043e12c5d1SDavid du Colombier  *  set up the interrupt/trap gates
1053e12c5d1SDavid du Colombier  */
1063e12c5d1SDavid du Colombier void
1073e12c5d1SDavid du Colombier trapinit(void)
1083e12c5d1SDavid du Colombier {
1093e12c5d1SDavid du Colombier 	int i;
1103e12c5d1SDavid du Colombier 
1113e12c5d1SDavid du Colombier 	/*
1123e12c5d1SDavid du Colombier 	 *  set all interrupts to panics
1133e12c5d1SDavid du Colombier 	 */
1143e12c5d1SDavid du Colombier 	for(i = 0; i < 256; i++)
1153e12c5d1SDavid du Colombier 		sethvec(i, intrbad, SEGTG, 0);
1163e12c5d1SDavid du Colombier 
1173e12c5d1SDavid du Colombier 	/*
1183e12c5d1SDavid du Colombier 	 *  80386 processor (and coprocessor) traps
1193e12c5d1SDavid du Colombier 	 */
1203e12c5d1SDavid du Colombier 	sethvec(0, intr0, SEGTG, 0);
1213e12c5d1SDavid du Colombier 	sethvec(1, intr1, SEGTG, 0);
1223e12c5d1SDavid du Colombier 	sethvec(2, intr2, SEGTG, 0);
1233e12c5d1SDavid du Colombier 	sethvec(4, intr4, SEGTG, 0);
1243e12c5d1SDavid du Colombier 	sethvec(5, intr5, SEGTG, 0);
1253e12c5d1SDavid du Colombier 	sethvec(6, intr6, SEGTG, 0);
1263e12c5d1SDavid du Colombier 	sethvec(7, intr7, SEGTG, 0);
1273e12c5d1SDavid du Colombier 	sethvec(8, intr8, SEGTG, 0);
1283e12c5d1SDavid du Colombier 	sethvec(9, intr9, SEGTG, 0);
1293e12c5d1SDavid du Colombier 	sethvec(10, intr10, SEGTG, 0);
1303e12c5d1SDavid du Colombier 	sethvec(11, intr11, SEGTG, 0);
1313e12c5d1SDavid du Colombier 	sethvec(12, intr12, SEGTG, 0);
1323e12c5d1SDavid du Colombier 	sethvec(13, intr13, SEGTG, 0);
1333e12c5d1SDavid du Colombier 	sethvec(14, intr14, SEGIG, 0);	/* page fault, interrupts off */
1343e12c5d1SDavid du Colombier 	sethvec(15, intr15, SEGTG, 0);
1353e12c5d1SDavid du Colombier 	sethvec(16, intr16, SEGIG, 0);	/* math coprocessor, interrupts off */
1363e12c5d1SDavid du Colombier 
1373e12c5d1SDavid du Colombier 	/*
1383e12c5d1SDavid du Colombier 	 *  device interrupts
1393e12c5d1SDavid du Colombier 	 */
1403e12c5d1SDavid du Colombier 	sethvec(24, intr24, SEGIG, 0);
1413e12c5d1SDavid du Colombier 	sethvec(25, intr25, SEGIG, 0);
1423e12c5d1SDavid du Colombier 	sethvec(26, intr26, SEGIG, 0);
1433e12c5d1SDavid du Colombier 	sethvec(27, intr27, SEGIG, 0);
1443e12c5d1SDavid du Colombier 	sethvec(28, intr28, SEGIG, 0);
1453e12c5d1SDavid du Colombier 	sethvec(29, intr29, SEGIG, 0);
1463e12c5d1SDavid du Colombier 	sethvec(30, intr30, SEGIG, 0);
1473e12c5d1SDavid du Colombier 	sethvec(31, intr31, SEGIG, 0);
1483e12c5d1SDavid du Colombier 	sethvec(32, intr32, SEGIG, 0);
1493e12c5d1SDavid du Colombier 	sethvec(33, intr33, SEGIG, 0);
1503e12c5d1SDavid du Colombier 	sethvec(34, intr34, SEGIG, 0);
1513e12c5d1SDavid du Colombier 	sethvec(35, intr35, SEGIG, 0);
1523e12c5d1SDavid du Colombier 	sethvec(36, intr36, SEGIG, 0);
1533e12c5d1SDavid du Colombier 	sethvec(37, intr37, SEGIG, 0);
1543e12c5d1SDavid du Colombier 	sethvec(38, intr38, SEGIG, 0);
1553e12c5d1SDavid du Colombier 	sethvec(39, intr39, SEGIG, 0);
1563e12c5d1SDavid du Colombier 
1573e12c5d1SDavid du Colombier 	/*
1583e12c5d1SDavid du Colombier 	 *  system calls and break points
1593e12c5d1SDavid du Colombier 	 */
1603e12c5d1SDavid du Colombier 	sethvec(Syscallvec, intr64, SEGTG, 3);
161*219b2ee8SDavid du Colombier 	setvec(Syscallvec, (void (*)(Ureg*, void*))syscall, 0);
1623e12c5d1SDavid du Colombier 	sethvec(Bptvec, intr3, SEGTG, 3);
163*219b2ee8SDavid du Colombier 	setvec(Bptvec, debugbpt, 0);
1643e12c5d1SDavid du Colombier 
1653e12c5d1SDavid du Colombier 	/*
1663e12c5d1SDavid du Colombier 	 *  tell the hardware where the table is (and how long)
1673e12c5d1SDavid du Colombier 	 */
1683e12c5d1SDavid du Colombier 	putidt(ilt, sizeof(ilt));
1693e12c5d1SDavid du Colombier 
1703e12c5d1SDavid du Colombier 	/*
1713e12c5d1SDavid du Colombier 	 *  Set up the first 8259 interrupt processor.
1723e12c5d1SDavid du Colombier 	 *  Make 8259 interrupts start at CPU vector Int0vec.
1733e12c5d1SDavid du Colombier 	 *  Set the 8259 as master with edge triggered
1743e12c5d1SDavid du Colombier 	 *  input with fully nested interrupts.
1753e12c5d1SDavid du Colombier 	 */
176*219b2ee8SDavid du Colombier 	outb(Int0ctl, (1<<4)|(0<<3)|(1<<0));	/* ICW1 - edge triggered, master,
1773e12c5d1SDavid du Colombier 					   ICW4 will be sent */
1783e12c5d1SDavid du Colombier 	outb(Int0aux, Int0vec);		/* ICW2 - interrupt vector offset */
1793e12c5d1SDavid du Colombier 	outb(Int0aux, 0x04);		/* ICW3 - have slave on level 2 */
1803e12c5d1SDavid du Colombier 	outb(Int0aux, 0x01);		/* ICW4 - 8086 mode, not buffered */
1813e12c5d1SDavid du Colombier 
1823e12c5d1SDavid du Colombier 	/*
1833e12c5d1SDavid du Colombier 	 *  Set up the second 8259 interrupt processor.
1843e12c5d1SDavid du Colombier 	 *  Make 8259 interrupts start at CPU vector Int0vec.
1853e12c5d1SDavid du Colombier 	 *  Set the 8259 as master with edge triggered
1863e12c5d1SDavid du Colombier 	 *  input with fully nested interrupts.
1873e12c5d1SDavid du Colombier 	 */
188*219b2ee8SDavid du Colombier 	outb(Int1ctl, (1<<4)|(0<<3)|(1<<0));	/* ICW1 - edge triggered, master,
1893e12c5d1SDavid du Colombier 					   ICW4 will be sent */
1903e12c5d1SDavid du Colombier 	outb(Int1aux, Int1vec);		/* ICW2 - interrupt vector offset */
1913e12c5d1SDavid du Colombier 	outb(Int1aux, 0x02);		/* ICW3 - I am a slave on level 2 */
1923e12c5d1SDavid du Colombier 	outb(Int1aux, 0x01);		/* ICW4 - 8086 mode, not buffered */
1933e12c5d1SDavid du Colombier 
1943e12c5d1SDavid du Colombier 	/*
1953e12c5d1SDavid du Colombier 	 *  pass #2 8259 interrupts to #1
1963e12c5d1SDavid du Colombier 	 */
1973e12c5d1SDavid du Colombier 	int0mask &= ~0x04;
1983e12c5d1SDavid du Colombier 	outb(Int0aux, int0mask);
199*219b2ee8SDavid du Colombier 
200*219b2ee8SDavid du Colombier 	/*
201*219b2ee8SDavid du Colombier 	 * Set Ocw3 to return the ISR when ctl read.
202*219b2ee8SDavid du Colombier 	 */
203*219b2ee8SDavid du Colombier 	outb(Int0ctl, Ocw3|0x03);
204*219b2ee8SDavid du Colombier 	outb(Int1ctl, Ocw3|0x03);
2053e12c5d1SDavid du Colombier }
2063e12c5d1SDavid du Colombier 
207*219b2ee8SDavid du Colombier char *excname[] = {
2083e12c5d1SDavid du Colombier 	[0]	"divide error",
2093e12c5d1SDavid du Colombier 	[1]	"debug exception",
210*219b2ee8SDavid du Colombier 	[2]	" nonmaskable interrupt",
211*219b2ee8SDavid du Colombier 	[3]	"breakpoint",
2123e12c5d1SDavid du Colombier 	[4]	"overflow",
2133e12c5d1SDavid du Colombier 	[5]	"bounds check",
2143e12c5d1SDavid du Colombier 	[6]	"invalid opcode",
215*219b2ee8SDavid du Colombier 	[7]	"coprocessor not available",
2163e12c5d1SDavid du Colombier 	[8]	"double fault",
217*219b2ee8SDavid du Colombier 	[9]	"9 (reserved)",
2183e12c5d1SDavid du Colombier 	[10]	"invalid TSS",
2193e12c5d1SDavid du Colombier 	[11]	"segment not present",
2203e12c5d1SDavid du Colombier 	[12]	"stack exception",
2213e12c5d1SDavid du Colombier 	[13]	"general protection violation",
222*219b2ee8SDavid du Colombier 	[14]	"page fault",
223*219b2ee8SDavid du Colombier 	[15]	"15 (reserved)",
224*219b2ee8SDavid du Colombier 	[16]	"coprocessor error",
2253e12c5d1SDavid du Colombier };
2263e12c5d1SDavid du Colombier 
227*219b2ee8SDavid du Colombier 
2283e12c5d1SDavid du Colombier /*
2293e12c5d1SDavid du Colombier  *  All traps
2303e12c5d1SDavid du Colombier  */
2313e12c5d1SDavid du Colombier void
2323e12c5d1SDavid du Colombier trap(Ureg *ur)
2333e12c5d1SDavid du Colombier {
2343e12c5d1SDavid du Colombier 	int v, user;
2353e12c5d1SDavid du Colombier 	int c;
2363e12c5d1SDavid du Colombier 	char buf[ERRLEN];
237*219b2ee8SDavid du Colombier 	Handler *h;
238*219b2ee8SDavid du Colombier 	static ulong intrtime;
239*219b2ee8SDavid du Colombier 	static int snoozing;
240*219b2ee8SDavid du Colombier 	static int iret_traps;
241*219b2ee8SDavid du Colombier 	ushort isr;
2423e12c5d1SDavid du Colombier 
2433e12c5d1SDavid du Colombier 	v = ur->trap;
2443e12c5d1SDavid du Colombier 
2453e12c5d1SDavid du Colombier 	user = ((ur->cs)&0xffff)!=KESEL && v!=Syscallvec;
2463e12c5d1SDavid du Colombier 	if(user)
2473e12c5d1SDavid du Colombier 		u->dbgreg = ur;
248*219b2ee8SDavid du Colombier 	else if(ur->pc >= KTZERO && ur->pc < (ulong)end && *(uchar*)ur->pc == 0xCF) {
249*219b2ee8SDavid du Colombier 		if(iret_traps++ > 10)
250*219b2ee8SDavid du Colombier 			panic("iret trap");
251*219b2ee8SDavid du Colombier 		return;
252*219b2ee8SDavid du Colombier 	}
253*219b2ee8SDavid du Colombier 	iret_traps = 0;
2543e12c5d1SDavid du Colombier 
2553e12c5d1SDavid du Colombier 	/*
2563e12c5d1SDavid du Colombier 	 *  tell the 8259 that we're done with the
2573e12c5d1SDavid du Colombier 	 *  highest level interrupt (interrupts are still
2583e12c5d1SDavid du Colombier 	 *  off at this point)
2593e12c5d1SDavid du Colombier 	 */
2603e12c5d1SDavid du Colombier 	c = v&~0x7;
261*219b2ee8SDavid du Colombier 	isr = 0;
2623e12c5d1SDavid du Colombier 	if(c==Int0vec || c==Int1vec){
263*219b2ee8SDavid du Colombier 		isr = inb(Int0ctl);
2643e12c5d1SDavid du Colombier 		outb(Int0ctl, EOI);
265*219b2ee8SDavid du Colombier 		if(c == Int1vec){
266*219b2ee8SDavid du Colombier 			isr |= inb(Int1ctl)<<8;
267*219b2ee8SDavid du Colombier 			outb(Int1ctl, EOI);
268*219b2ee8SDavid du Colombier 		}
2693e12c5d1SDavid du Colombier 	}
2703e12c5d1SDavid du Colombier 
271*219b2ee8SDavid du Colombier 	if(v>=256 || (h = halloc.ivec[v]) == 0){
272*219b2ee8SDavid du Colombier 		/* an old 386 generates these fairly often, no idea why */
2733e12c5d1SDavid du Colombier 		if(v == 13)
2743e12c5d1SDavid du Colombier 			return;
275*219b2ee8SDavid du Colombier 
276*219b2ee8SDavid du Colombier 		/* a processor or coprocessor error */
2773e12c5d1SDavid du Colombier 		if(v <= 16){
2783e12c5d1SDavid du Colombier 			if(user){
2793e12c5d1SDavid du Colombier 				sprint(buf, "sys: trap: %s", excname[v]);
2803e12c5d1SDavid du Colombier 				postnote(u->p, 1, buf, NDebug);
2813e12c5d1SDavid du Colombier 				return;
2823e12c5d1SDavid du Colombier 			} else {
2833e12c5d1SDavid du Colombier 				dumpregs(ur);
2843e12c5d1SDavid du Colombier 				panic("%s pc=0x%lux", excname[v], ur->pc);
2853e12c5d1SDavid du Colombier 			}
2863e12c5d1SDavid du Colombier 		}
287*219b2ee8SDavid du Colombier 
288*219b2ee8SDavid du Colombier 		if(v >= Int0vec && v < Int0vec+16){
289*219b2ee8SDavid du Colombier 			/* an unknown interrupt */
290*219b2ee8SDavid du Colombier 			v -= Int0vec;
291*219b2ee8SDavid du Colombier 			/*
292*219b2ee8SDavid du Colombier 			 * Check for a default IRQ7. This can happen when
293*219b2ee8SDavid du Colombier 			 * the IRQ input goes away before the acknowledge.
294*219b2ee8SDavid du Colombier 			 * In this case, a 'default IRQ7' is generated, but
295*219b2ee8SDavid du Colombier 			 * the corresponding bit in the ISR isn't set.
296*219b2ee8SDavid du Colombier 			 * In fact, just ignore all such interrupts.
297*219b2ee8SDavid du Colombier 			 */
298*219b2ee8SDavid du Colombier 			if((isr & (1<<v)) == 0)
299*219b2ee8SDavid du Colombier 				return;
300*219b2ee8SDavid du Colombier 			if(badintr[v]++ == 0 || (badintr[v]%100000) == 0){
301*219b2ee8SDavid du Colombier 				print("unknown interrupt %d pc=0x%lux: total %d\n", v,
302*219b2ee8SDavid du Colombier 					ur->pc, badintr[v]);
303*219b2ee8SDavid du Colombier 				print("isr = 0x%4.4ux\n", isr);
304*219b2ee8SDavid du Colombier 			}
305*219b2ee8SDavid du Colombier 		} else {
306*219b2ee8SDavid du Colombier 			/* unimplemented traps */
307*219b2ee8SDavid du Colombier 			print("illegal trap %d pc=0x%lux\n", v, ur->pc);
308*219b2ee8SDavid du Colombier 		}
3093e12c5d1SDavid du Colombier 		return;
3103e12c5d1SDavid du Colombier 	}
3113e12c5d1SDavid du Colombier 
312*219b2ee8SDavid du Colombier 	/* there may be multiple handlers on one interrupt level */
313*219b2ee8SDavid du Colombier 	do {
314*219b2ee8SDavid du Colombier 		(*h->r)(ur, h->arg);
315*219b2ee8SDavid du Colombier 		h = h->next;
316*219b2ee8SDavid du Colombier 	} while(h);
317*219b2ee8SDavid du Colombier 	splhi();
3183e12c5d1SDavid du Colombier 
319*219b2ee8SDavid du Colombier 	/* power management */
320*219b2ee8SDavid du Colombier 	if(v == Clockvec){
321*219b2ee8SDavid du Colombier 		/* allow power sheding on clock ticks */
322*219b2ee8SDavid du Colombier 		if(arch->snooze)
323*219b2ee8SDavid du Colombier 			snoozing = (*arch->snooze)(intrtime, 0);
324*219b2ee8SDavid du Colombier 	} else {
325*219b2ee8SDavid du Colombier 		/* turn power back on when anything else happens */
326*219b2ee8SDavid du Colombier 		if(snoozing && arch->snooze)
327*219b2ee8SDavid du Colombier 			snoozing = (*arch->snooze)(intrtime, 1);
328*219b2ee8SDavid du Colombier 		intrtime = m->ticks;
329*219b2ee8SDavid du Colombier 	}
330*219b2ee8SDavid du Colombier 
331*219b2ee8SDavid du Colombier 	/* check user since syscall does its own notifying */
3323e12c5d1SDavid du Colombier 	if(user && (u->p->procctl || u->nnote))
3333e12c5d1SDavid du Colombier 		notify(ur);
3343e12c5d1SDavid du Colombier }
3353e12c5d1SDavid du Colombier 
3363e12c5d1SDavid du Colombier /*
3373e12c5d1SDavid du Colombier  *  dump registers
3383e12c5d1SDavid du Colombier  */
3393e12c5d1SDavid du Colombier void
340*219b2ee8SDavid du Colombier dumpregs2(Ureg *ur)
3413e12c5d1SDavid du Colombier {
3423e12c5d1SDavid du Colombier 	if(u)
3433e12c5d1SDavid du Colombier 		print("registers for %s %d\n", u->p->text, u->p->pid);
3443e12c5d1SDavid du Colombier 	else
3453e12c5d1SDavid du Colombier 		print("registers for kernel\n");
346*219b2ee8SDavid du Colombier 	print("FLAGS=%lux TRAP=%lux ECODE=%lux CS=%lux PC=%lux", ur->flags, ur->trap,
3473e12c5d1SDavid du Colombier 		ur->ecode, ur->cs&0xff, ur->pc);
3483e12c5d1SDavid du Colombier 	if(ur == (Ureg*)UREGADDR)
3493e12c5d1SDavid du Colombier 		print(" SS=%lux USP=%lux\n", ur->ss&0xff, ur->usp);
3503e12c5d1SDavid du Colombier 	else
3513e12c5d1SDavid du Colombier 		print("\n");
3523e12c5d1SDavid du Colombier 	print("  AX %8.8lux  BX %8.8lux  CX %8.8lux  DX %8.8lux\n",
3533e12c5d1SDavid du Colombier 		ur->ax, ur->bx, ur->cx, ur->dx);
3543e12c5d1SDavid du Colombier 	print("  SI %8.8lux  DI %8.8lux  BP %8.8lux\n",
3553e12c5d1SDavid du Colombier 		ur->si, ur->di, ur->bp);
3563e12c5d1SDavid du Colombier 	print("  DS %4.4ux  ES %4.4ux  FS %4.4ux  GS %4.4ux\n",
3573e12c5d1SDavid du Colombier 		ur->ds&0xffff, ur->es&0xffff, ur->fs&0xffff, ur->gs&0xffff);
358*219b2ee8SDavid du Colombier 
359*219b2ee8SDavid du Colombier }
360*219b2ee8SDavid du Colombier 
361*219b2ee8SDavid du Colombier void
362*219b2ee8SDavid du Colombier dumpregs(Ureg *ur)
363*219b2ee8SDavid du Colombier {
364*219b2ee8SDavid du Colombier 	dumpregs2(ur);
365*219b2ee8SDavid du Colombier 	print("  ur %lux\n", ur);
3663e12c5d1SDavid du Colombier }
3673e12c5d1SDavid du Colombier 
3683e12c5d1SDavid du Colombier void
3693e12c5d1SDavid du Colombier dumpstack(void)
3703e12c5d1SDavid du Colombier {
3713e12c5d1SDavid du Colombier 	ulong l, v, i;
3723e12c5d1SDavid du Colombier 	extern ulong etext;
3733e12c5d1SDavid du Colombier 
3743e12c5d1SDavid du Colombier 	if(u == 0)
3753e12c5d1SDavid du Colombier 		return;
3763e12c5d1SDavid du Colombier 
3773e12c5d1SDavid du Colombier 	i = 0;
3783e12c5d1SDavid du Colombier 	for(l=(ulong)&l; l<USERADDR+BY2PG; l+=4){
3793e12c5d1SDavid du Colombier 		v = *(ulong*)l;
3803e12c5d1SDavid du Colombier 		if(KTZERO < v && v < (ulong)&etext){
3813e12c5d1SDavid du Colombier 			print("%lux ", v);
3823e12c5d1SDavid du Colombier 			i++;
3833e12c5d1SDavid du Colombier 		}
3843e12c5d1SDavid du Colombier 		if(i == 4){
3853e12c5d1SDavid du Colombier 			i = 0;
3863e12c5d1SDavid du Colombier 			print("\n");
3873e12c5d1SDavid du Colombier 		}
3883e12c5d1SDavid du Colombier 	}
3893e12c5d1SDavid du Colombier 
3903e12c5d1SDavid du Colombier }
3913e12c5d1SDavid du Colombier 
3923e12c5d1SDavid du Colombier long
3933e12c5d1SDavid du Colombier execregs(ulong entry, ulong ssize, ulong nargs)
3943e12c5d1SDavid du Colombier {
3953e12c5d1SDavid du Colombier 	ulong *sp;
3963e12c5d1SDavid du Colombier 
3973e12c5d1SDavid du Colombier 	sp = (ulong*)(USTKTOP - ssize);
3983e12c5d1SDavid du Colombier 	*--sp = nargs;
3993e12c5d1SDavid du Colombier 	((Ureg*)UREGADDR)->usp = (ulong)sp;
4003e12c5d1SDavid du Colombier 	((Ureg*)UREGADDR)->pc = entry;
4013e12c5d1SDavid du Colombier 	return USTKTOP-BY2WD;			/* address of user-level clock */
4023e12c5d1SDavid du Colombier }
4033e12c5d1SDavid du Colombier 
4043e12c5d1SDavid du Colombier ulong
4053e12c5d1SDavid du Colombier userpc(void)
4063e12c5d1SDavid du Colombier {
4073e12c5d1SDavid du Colombier 	return ((Ureg*)UREGADDR)->pc;
4083e12c5d1SDavid du Colombier }
4093e12c5d1SDavid du Colombier 
4103e12c5d1SDavid du Colombier /*
4113e12c5d1SDavid du Colombier  *  system calls
4123e12c5d1SDavid du Colombier  */
4133e12c5d1SDavid du Colombier #include "../port/systab.h"
4143e12c5d1SDavid du Colombier 
4153e12c5d1SDavid du Colombier /*
4163e12c5d1SDavid du Colombier  *  syscall is called spllo()
4173e12c5d1SDavid du Colombier  */
4183e12c5d1SDavid du Colombier long
419*219b2ee8SDavid du Colombier syscall(Ureg *ur, void *a)
4203e12c5d1SDavid du Colombier {
4213e12c5d1SDavid du Colombier 	ulong	sp;
4223e12c5d1SDavid du Colombier 	long	ret;
4233e12c5d1SDavid du Colombier 	int	i;
4243e12c5d1SDavid du Colombier 
425*219b2ee8SDavid du Colombier 	USED(a);
426*219b2ee8SDavid du Colombier 
4273e12c5d1SDavid du Colombier 	u->p->insyscall = 1;
4283e12c5d1SDavid du Colombier 	u->p->pc = ur->pc;
4293e12c5d1SDavid du Colombier 	if((ur->cs)&0xffff == KESEL)
4303e12c5d1SDavid du Colombier 		panic("recursive system call");
4313e12c5d1SDavid du Colombier 
4323e12c5d1SDavid du Colombier 	u->scallnr = ur->ax;
4333e12c5d1SDavid du Colombier 	if(u->scallnr == RFORK && u->p->fpstate == FPactive){
4343e12c5d1SDavid du Colombier 		/*
4353e12c5d1SDavid du Colombier 		 *  so that the child starts out with the
4363e12c5d1SDavid du Colombier 		 *  same registers as the parent
4373e12c5d1SDavid du Colombier 		 */
4383e12c5d1SDavid du Colombier 		splhi();
4393e12c5d1SDavid du Colombier 		if(u->p->fpstate == FPactive){
4403e12c5d1SDavid du Colombier 			fpsave(&u->fpsave);
4413e12c5d1SDavid du Colombier 			u->p->fpstate = FPinactive;
4423e12c5d1SDavid du Colombier 		}
4433e12c5d1SDavid du Colombier 		spllo();
4443e12c5d1SDavid du Colombier 	}
4453e12c5d1SDavid du Colombier 	sp = ur->usp;
4463e12c5d1SDavid du Colombier 	u->nerrlab = 0;
4473e12c5d1SDavid du Colombier 	ret = -1;
4483e12c5d1SDavid du Colombier 	if(!waserror()){
4493e12c5d1SDavid du Colombier 		if(u->scallnr >= sizeof systab/BY2WD){
4503e12c5d1SDavid du Colombier 			pprint("bad sys call number %d pc %lux\n", u->scallnr, ur->pc);
4513e12c5d1SDavid du Colombier 			postnote(u->p, 1, "sys: bad sys call", NDebug);
4523e12c5d1SDavid du Colombier 			error(Ebadarg);
4533e12c5d1SDavid du Colombier 		}
4543e12c5d1SDavid du Colombier 
4553e12c5d1SDavid du Colombier 		if(sp<(USTKTOP-BY2PG) || sp>(USTKTOP-(1+MAXSYSARG)*BY2WD))
4563e12c5d1SDavid du Colombier 			validaddr(sp, (1+MAXSYSARG)*BY2WD, 0);
4573e12c5d1SDavid du Colombier 
4583e12c5d1SDavid du Colombier 		u->s = *((Sargs*)(sp+1*BY2WD));
4593e12c5d1SDavid du Colombier 		u->p->psstate = sysctab[u->scallnr];
4603e12c5d1SDavid du Colombier 
4613e12c5d1SDavid du Colombier 		ret = (*systab[u->scallnr])(u->s.args);
4623e12c5d1SDavid du Colombier 		poperror();
4633e12c5d1SDavid du Colombier 	}
4643e12c5d1SDavid du Colombier 	if(u->nerrlab){
4653e12c5d1SDavid du Colombier 		print("bad errstack [%d]: %d extra\n", u->scallnr, u->nerrlab);
4663e12c5d1SDavid du Colombier 		for(i = 0; i < NERR; i++)
4673e12c5d1SDavid du Colombier 			print("sp=%lux pc=%lux\n", u->errlab[i].sp, u->errlab[i].pc);
4683e12c5d1SDavid du Colombier 		panic("error stack");
4693e12c5d1SDavid du Colombier 	}
4703e12c5d1SDavid du Colombier 
4713e12c5d1SDavid du Colombier 	u->p->insyscall = 0;
4723e12c5d1SDavid du Colombier 	u->p->psstate = 0;
4733e12c5d1SDavid du Colombier 
4743e12c5d1SDavid du Colombier 	/*
4753e12c5d1SDavid du Colombier 	 *  Put return value in frame.  On the safari the syscall is
4763e12c5d1SDavid du Colombier 	 *  just another trap and the return value from syscall is
4773e12c5d1SDavid du Colombier 	 *  ignored.  On other machines the return value is put into
4783e12c5d1SDavid du Colombier 	 *  the results register by caller of syscall.
4793e12c5d1SDavid du Colombier 	 */
4803e12c5d1SDavid du Colombier 	ur->ax = ret;
4813e12c5d1SDavid du Colombier 
4823e12c5d1SDavid du Colombier 	if(u->scallnr == NOTED)
4833e12c5d1SDavid du Colombier 		noted(ur, *(ulong*)(sp+BY2WD));
4843e12c5d1SDavid du Colombier 
4853e12c5d1SDavid du Colombier 	splhi(); /* avoid interrupts during the iret */
4863e12c5d1SDavid du Colombier 	if(u->scallnr!=RFORK && (u->p->procctl || u->nnote))
4873e12c5d1SDavid du Colombier 		notify(ur);
4883e12c5d1SDavid du Colombier 
4893e12c5d1SDavid du Colombier 	return ret;
4903e12c5d1SDavid du Colombier }
4913e12c5d1SDavid du Colombier 
4923e12c5d1SDavid du Colombier /*
4933e12c5d1SDavid du Colombier  *  Call user, if necessary, with note.
4943e12c5d1SDavid du Colombier  *  Pass user the Ureg struct and the note on his stack.
4953e12c5d1SDavid du Colombier  */
4963e12c5d1SDavid du Colombier int
4973e12c5d1SDavid du Colombier notify(Ureg *ur)
4983e12c5d1SDavid du Colombier {
499*219b2ee8SDavid du Colombier 	int l;
5003e12c5d1SDavid du Colombier 	ulong s, sp;
5013e12c5d1SDavid du Colombier 	Note *n;
5023e12c5d1SDavid du Colombier 
5033e12c5d1SDavid du Colombier 	if(u->p->procctl)
5043e12c5d1SDavid du Colombier 		procctl(u->p);
5053e12c5d1SDavid du Colombier 	if(u->nnote == 0)
5063e12c5d1SDavid du Colombier 		return 0;
5073e12c5d1SDavid du Colombier 
5083e12c5d1SDavid du Colombier 	s = spllo();
5093e12c5d1SDavid du Colombier 	qlock(&u->p->debug);
5103e12c5d1SDavid du Colombier 	u->p->notepending = 0;
5113e12c5d1SDavid du Colombier 	n = &u->note[0];
5123e12c5d1SDavid du Colombier 	if(strncmp(n->msg, "sys:", 4) == 0){
5133e12c5d1SDavid du Colombier 		l = strlen(n->msg);
5143e12c5d1SDavid du Colombier 		if(l > ERRLEN-15)	/* " pc=0x12345678\0" */
5153e12c5d1SDavid du Colombier 			l = ERRLEN-15;
5163e12c5d1SDavid du Colombier 		sprint(n->msg+l, " pc=0x%.8lux", ur->pc);
5173e12c5d1SDavid du Colombier 	}
518*219b2ee8SDavid du Colombier 
5193e12c5d1SDavid du Colombier 	if(n->flag!=NUser && (u->notified || u->notify==0)){
520*219b2ee8SDavid du Colombier 		if(n->flag == NDebug){
521*219b2ee8SDavid du Colombier 			qunlock(&u->p->debug);
5223e12c5d1SDavid du Colombier 			pprint("suicide: %s\n", n->msg);
523*219b2ee8SDavid du Colombier 		}else
5243e12c5d1SDavid du Colombier 			qunlock(&u->p->debug);
5253e12c5d1SDavid du Colombier 		pexit(n->msg, n->flag!=NDebug);
5263e12c5d1SDavid du Colombier 	}
527*219b2ee8SDavid du Colombier 
528*219b2ee8SDavid du Colombier 	if(u->notified) {
529*219b2ee8SDavid du Colombier 		qunlock(&u->p->debug);
530*219b2ee8SDavid du Colombier 		splhi();
531*219b2ee8SDavid du Colombier 		return 0;
532*219b2ee8SDavid du Colombier 	}
533*219b2ee8SDavid du Colombier 
534bd389b36SDavid du Colombier 	if(!u->notify){
535bd389b36SDavid du Colombier 		qunlock(&u->p->debug);
536bd389b36SDavid du Colombier 		pexit(n->msg, n->flag!=NDebug);
537bd389b36SDavid du Colombier 	}
5383e12c5d1SDavid du Colombier 	sp = ur->usp;
5393e12c5d1SDavid du Colombier 	sp -= sizeof(Ureg);
540*219b2ee8SDavid du Colombier 
5413e12c5d1SDavid du Colombier 	if(!okaddr((ulong)u->notify, 1, 0)
542*219b2ee8SDavid du Colombier 	|| !okaddr(sp-ERRLEN-4*BY2WD, sizeof(Ureg)+ERRLEN+4*BY2WD, 1)){
5433e12c5d1SDavid du Colombier 		qunlock(&u->p->debug);
544*219b2ee8SDavid du Colombier 		pprint("suicide: bad address in notify\n");
5453e12c5d1SDavid du Colombier 		pexit("Suicide", 0);
5463e12c5d1SDavid du Colombier 	}
547*219b2ee8SDavid du Colombier 
5483e12c5d1SDavid du Colombier 	u->ureg = (void*)sp;
5493e12c5d1SDavid du Colombier 	memmove((Ureg*)sp, ur, sizeof(Ureg));
550*219b2ee8SDavid du Colombier 	*(Ureg**)(sp-BY2WD) = u->ureg;	/* word under Ureg is old u->ureg */
551*219b2ee8SDavid du Colombier 	u->ureg = (void*)sp;
552*219b2ee8SDavid du Colombier 	sp -= BY2WD+ERRLEN;
5533e12c5d1SDavid du Colombier 	memmove((char*)sp, u->note[0].msg, ERRLEN);
5543e12c5d1SDavid du Colombier 	sp -= 3*BY2WD;
5553e12c5d1SDavid du Colombier 	*(ulong*)(sp+2*BY2WD) = sp+3*BY2WD;	/* arg 2 is string */
5563e12c5d1SDavid du Colombier 	*(ulong*)(sp+1*BY2WD) = (ulong)u->ureg;	/* arg 1 is ureg* */
5573e12c5d1SDavid du Colombier 	*(ulong*)(sp+0*BY2WD) = 0;		/* arg 0 is pc */
5583e12c5d1SDavid du Colombier 	ur->usp = sp;
5593e12c5d1SDavid du Colombier 	ur->pc = (ulong)u->notify;
5603e12c5d1SDavid du Colombier 	u->notified = 1;
5613e12c5d1SDavid du Colombier 	u->nnote--;
5623e12c5d1SDavid du Colombier 	memmove(&u->lastnote, &u->note[0], sizeof(Note));
5633e12c5d1SDavid du Colombier 	memmove(&u->note[0], &u->note[1], u->nnote*sizeof(Note));
564*219b2ee8SDavid du Colombier 
5653e12c5d1SDavid du Colombier 	qunlock(&u->p->debug);
5663e12c5d1SDavid du Colombier 	splx(s);
567*219b2ee8SDavid du Colombier 	return 1;
5683e12c5d1SDavid du Colombier }
5693e12c5d1SDavid du Colombier 
5703e12c5d1SDavid du Colombier /*
5713e12c5d1SDavid du Colombier  *   Return user to state before notify()
5723e12c5d1SDavid du Colombier  */
5733e12c5d1SDavid du Colombier void
5743e12c5d1SDavid du Colombier noted(Ureg *ur, ulong arg0)
5753e12c5d1SDavid du Colombier {
5763e12c5d1SDavid du Colombier 	Ureg *nur;
577*219b2ee8SDavid du Colombier 	ulong oureg, sp;
5783e12c5d1SDavid du Colombier 
579*219b2ee8SDavid du Colombier 	qlock(&u->p->debug);
580*219b2ee8SDavid du Colombier 	if(arg0!=NRSTR && !u->notified) {
581*219b2ee8SDavid du Colombier 		qunlock(&u->p->debug);
582*219b2ee8SDavid du Colombier 		pprint("call to noted() when not notified\n");
5833e12c5d1SDavid du Colombier 		pexit("Suicide", 0);
5843e12c5d1SDavid du Colombier 	}
5853e12c5d1SDavid du Colombier 	u->notified = 0;
586*219b2ee8SDavid du Colombier 
587*219b2ee8SDavid du Colombier 	nur = u->ureg;		/* pointer to user returned Ureg struct */
588*219b2ee8SDavid du Colombier 
589*219b2ee8SDavid du Colombier 	/* sanity clause */
590*219b2ee8SDavid du Colombier 	oureg = (ulong)nur;
591*219b2ee8SDavid du Colombier 	if(!okaddr((ulong)oureg-BY2WD, BY2WD+sizeof(Ureg), 0)){
592*219b2ee8SDavid du Colombier 		qunlock(&u->p->debug);
593*219b2ee8SDavid du Colombier 		pprint("bad u->ureg in noted or call to noted() when not notified\n");
594*219b2ee8SDavid du Colombier 		pexit("Suicide", 0);
595*219b2ee8SDavid du Colombier 	}
596*219b2ee8SDavid du Colombier 
597*219b2ee8SDavid du Colombier 	/* don't let user change text or stack segments */
598*219b2ee8SDavid du Colombier 	nur->cs = ur->cs;
599*219b2ee8SDavid du Colombier 	nur->ss = ur->ss;
600*219b2ee8SDavid du Colombier 
601*219b2ee8SDavid du Colombier 	/* don't let user change system flags */
602*219b2ee8SDavid du Colombier 	nur->flags = (ur->flags & ~0xCD5) | (nur->flags & 0xCD5);
603*219b2ee8SDavid du Colombier 
6043e12c5d1SDavid du Colombier 	memmove(ur, nur, sizeof(Ureg));
605*219b2ee8SDavid du Colombier 
6063e12c5d1SDavid du Colombier 	switch(arg0){
6073e12c5d1SDavid du Colombier 	case NCONT:
608*219b2ee8SDavid du Colombier 	case NRSTR:
6093e12c5d1SDavid du Colombier 		if(!okaddr(nur->pc, 1, 0) || !okaddr(nur->usp, BY2WD, 0)){
6103e12c5d1SDavid du Colombier 			qunlock(&u->p->debug);
611*219b2ee8SDavid du Colombier 			pprint("suicide: trap in noted\n");
612*219b2ee8SDavid du Colombier 			pexit("Suicide", 0);
613*219b2ee8SDavid du Colombier 		}
614*219b2ee8SDavid du Colombier 		u->ureg = (Ureg*)(*(ulong*)(oureg-BY2WD));
615*219b2ee8SDavid du Colombier 		qunlock(&u->p->debug);
616*219b2ee8SDavid du Colombier 		break;
617*219b2ee8SDavid du Colombier 
618*219b2ee8SDavid du Colombier 	case NSAVE:
619*219b2ee8SDavid du Colombier 		if(!okaddr(nur->pc, BY2WD, 0) || !okaddr(nur->usp, BY2WD, 0)){
620*219b2ee8SDavid du Colombier 			qunlock(&u->p->debug);
621*219b2ee8SDavid du Colombier 			pprint("suicide: trap in noted\n");
622*219b2ee8SDavid du Colombier 			pexit("Suicide", 0);
6233e12c5d1SDavid du Colombier 		}
6243e12c5d1SDavid du Colombier 		qunlock(&u->p->debug);
625*219b2ee8SDavid du Colombier 		sp = oureg-4*BY2WD-ERRLEN;
626*219b2ee8SDavid du Colombier 		splhi();
627*219b2ee8SDavid du Colombier 		ur->sp = sp;
628*219b2ee8SDavid du Colombier 		((ulong*)sp)[1] = oureg;	/* arg 1 0(FP) is ureg* */
629*219b2ee8SDavid du Colombier 		((ulong*)sp)[0] = 0;		/* arg 0 is pc */
630*219b2ee8SDavid du Colombier 		break;
6313e12c5d1SDavid du Colombier 
6323e12c5d1SDavid du Colombier 	default:
6333e12c5d1SDavid du Colombier 		u->lastnote.flag = NDebug;
634*219b2ee8SDavid du Colombier 		qunlock(&u->p->debug);
635*219b2ee8SDavid du Colombier 		pprint("unknown noted arg 0x%lux\n", arg0);
636*219b2ee8SDavid du Colombier 		pprint("suicide: %s\n", u->lastnote.msg);
637*219b2ee8SDavid du Colombier 		pexit(u->lastnote.msg, 0);
638*219b2ee8SDavid du Colombier 		break;
6393e12c5d1SDavid du Colombier 
6403e12c5d1SDavid du Colombier 	case NDFLT:
641*219b2ee8SDavid du Colombier 		if(u->lastnote.flag == NDebug){
642*219b2ee8SDavid du Colombier 			qunlock(&u->p->debug);
6433e12c5d1SDavid du Colombier 			pprint("suicide: %s\n", u->lastnote.msg);
644*219b2ee8SDavid du Colombier 		}else
6453e12c5d1SDavid du Colombier 			qunlock(&u->p->debug);
6463e12c5d1SDavid du Colombier 		pexit(u->lastnote.msg, u->lastnote.flag!=NDebug);
6473e12c5d1SDavid du Colombier 	}
6483e12c5d1SDavid du Colombier }
6493e12c5d1SDavid du Colombier 
6503e12c5d1SDavid du Colombier /* This routine must save the values of registers the user is not permitted to write
6513e12c5d1SDavid du Colombier  * from devproc and the restore the saved values before returning
6523e12c5d1SDavid du Colombier  */
6533e12c5d1SDavid du Colombier void
6543e12c5d1SDavid du Colombier setregisters(Ureg *xp, char *pureg, char *uva, int n)
6553e12c5d1SDavid du Colombier {
6563e12c5d1SDavid du Colombier 	ulong flags;
6573e12c5d1SDavid du Colombier 	ulong cs;
6583e12c5d1SDavid du Colombier 	ulong ss;
6593e12c5d1SDavid du Colombier 
6603e12c5d1SDavid du Colombier 	flags = xp->flags;
6613e12c5d1SDavid du Colombier 	cs = xp->cs;
6623e12c5d1SDavid du Colombier 	ss = xp->ss;
6633e12c5d1SDavid du Colombier 	memmove(pureg, uva, n);
6643e12c5d1SDavid du Colombier 	xp->flags = (xp->flags & 0xff) | (flags & 0xff00);
6653e12c5d1SDavid du Colombier 	xp->cs = cs;
6663e12c5d1SDavid du Colombier 	xp->ss = ss;
6673e12c5d1SDavid du Colombier }
668