NetBSD/sys/arch/i386/isa/intr.c
1993-10-22 19:33:13 +00:00

314 lines
7.8 KiB
C

/*-
* Copyright (c) 1993 Charles Hannum.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by Charles Hannum.
* 4. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* $Id: intr.c,v 1.2 1993/10/22 19:33:13 mycroft Exp $
*/
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/conf.h>
#include <sys/file.h>
#include <sys/device.h>
#include <sys/syslog.h>
#include <machine/cpu.h>
#include <machine/pio.h>
#include <i386/isa/isa.h>
#include <i386/isa/isavar.h>
#include <i386/isa/icu.h>
#include <i386/isa/timerreg.h>
void isa_defaultirq __P((void));
void isa_flushintrs __P((void));
void isa_intrmaskwickedness __P((void));
int intrmask[NIRQ];
struct intrhand *intrhand[NIRQ];
int fastvec;
#define IDTVEC(name) __CONCAT(X,name)
/* default interrupt vector table entries */
extern *IDTVEC(intr)[];
/* fast interrupt vector table */
extern *IDTVEC(fast)[];
/* out of range default interrupt vector gate entry */
extern IDTVEC(wild);
/*
* Register a fast vector.
*/
void
intr_fasttrap(intr, ih)
int intr;
struct intrhand *ih;
{
int irqnum = ffs(intr) - 1;
#ifdef DIAGNOSTIC
if (intr == IRQUNK || intr == IRQNONE ||
(intr &~ (1 << irqnum)) != IRQNONE)
panic("intr_fasttrap: weird irq");
#endif
if (fastvec & intr)
panic("intr_fasttrap: irq is already fast vector");
if (intrhand[irqnum])
panic("intr_fasttrap: irq is already slow vector");
fastvec |= intr;
ih->ih_count = 0;
ih->ih_next = NULL;
intrhand[irqnum] = ih;
setidt(irqnum, IDTVEC(fast)[irqnum], SDT_SYS386IGT, SEL_KPL);
}
/*
* Register an interrupt handler.
*/
void
intr_establish(intr, ih, class)
int intr;
struct intrhand *ih;
enum devclass class;
{
int irqnum = ffs(intr) - 1;
register struct intrhand **p, *q;
#ifdef DIAGNOSTIC
if (intr == IRQUNK || intr == IRQNONE ||
(intr &~ (1 << irqnum)) != IRQNONE)
panic("intr_establish: weird irq");
#endif
if (fastvec & intr)
panic("intr_establish: irq is already fast vector");
switch (class) {
case DV_DULL:
break;
case DV_DISK:
case DV_TAPE:
biomask |= intr;
break;
case DV_IFNET:
netmask |= intr;
break;
case DV_TTY:
ttymask |= intr;
break;
case DV_CPU:
default:
panic("intr_establish: weird devclass");
}
ih->ih_count = 0;
ih->ih_next = NULL;
/*
* This is O(N^2), but we want to preserve the order, and N is
* always small.
*/
for (p = &intrhand[intr]; (q = *p) != NULL; p = &q->ih_next)
;
*p = ih;
intr_enable(intr);
}
/*
* Set up the masks for each interrupt handler based on the information
* recorded by intr_establish().
*/
void
isa_intrmaskwickedness()
{
int irq, mask;
for (irq = 0; irq < NIRQ; irq++) {
mask = (1 << irq) | astmask;
if (biomask & (1 << irq))
mask |= biomask;
if (ttymask & (1 << irq))
mask |= ttymask;
if (netmask & (1 << irq))
mask |= netmask;
intrmask[irq] = mask;
}
impmask = netmask | ttymask;
printf("biomask %x ttymask %x netmask %x impmask %x astmask %s\n",
biomask, ttymask, netmask, impmask, astmask);
}
/*
* Fill in default interrupt table, and mask all interrupts.
*/
void
isa_defaultirq()
{
int i;
/* icu vectors */
for (i = ICU_OFFSET; i < ICU_OFFSET + ICU_LEN ; i++)
setidt(i, IDTVEC(intr)[i], SDT_SYS386IGT, SEL_KPL);
/* out of range vectors */
for (; i < NIDT; i++)
setidt(i, &IDTVEC(wild), SDT_SYS386IGT, SEL_KPL);
/* initialize 8259's */
outb(IO_ICU1, 0x11); /* reset; program device, four bytes */
outb(IO_ICU1+1, ICU_OFFSET); /* starting at this vector index */
outb(IO_ICU1+1, IRQ_SLAVE);
#ifdef AUTO_EOI_1
outb(IO_ICU1+1, 2 | 1); /* auto EOI, 8086 mode */
#else
outb(IO_ICU1+1, 1); /* 8086 mode */
#endif
outb(IO_ICU1+1, 0xff); /* leave interrupts masked */
outb(IO_ICU1, 0x0a); /* default to IRR on read */
#ifdef REORDER_IRQ
outb(IO_ICU1, 0xc0 | (3 - 1)); /* pri order 3-7, 0-2 (com2 first) */
#endif
outb(IO_ICU2, 0x11); /* reset; program device, four bytes */
outb(IO_ICU2+1, ICU_OFFSET+8); /* staring at this vector index */
outb(IO_ICU2+1, ffs(IRQ_SLAVE)-1);
#ifdef AUTO_EOI_2
outb(IO_ICU2+1, 2 | 1); /* auto EOI, 8086 mode */
#else
outb(IO_ICU2+1, 1); /* 8086 mode */
#endif
outb(IO_ICU2+1, 0xff); /* leave interrupts masked */
outb(IO_ICU2, 0x0a); /* default to IRR on read */
}
/*
* Flush any pending interrupts.
*/
void
isa_flushintrs()
{
register int i;
/* clear any pending interrupts */
for (i = 0; i < 8; i++) {
outb(IO_ICU1, ICU_EOI);
outb(IO_ICU2, ICU_EOI);
}
}
/*
* Determine what IRQ a device is using by trying to force it to generate an
* interrupt and seeing which IRQ line goes high. It is not safe to call
* this function after autoconfig.
*/
u_short
isa_discoverintr(force, aux)
void (*force)();
{
int time = TIMER_FREQ * 1; /* wait up to 1 second */
u_int last, now;
u_short iobase = IO_TIMER1;
isa_flushintrs();
/* attempt to force interrupt */
force(aux);
/* set timer 2 to a known state */
outb(iobase + TIMER_MODE, TIMER_SEL2|TIMER_RATEGEN|TIMER_16BIT);
outb(iobase + TIMER_CNTR2, 0xff);
outb(iobase + TIMER_CNTR2, 0xff);
last = 0xffff;
while (time > 0) {
register u_char irr, lo, hi;
irr = inb(IO_ICU1) & ~IRQ_SLAVE;
if (irr)
return 1 << (ffs(irr) - 1);
irr = inb(IO_ICU2);
if (irr)
return 1 << (ffs(irr) + 7);
outb(iobase + TIMER_MODE, TIMER_SEL2|TIMER_LATCH);
lo = inb(iobase + TIMER_CNTR2);
hi = inb(iobase + TIMER_CNTR2);
now = (hi << 8) | lo;
if (now <= last)
time -= (last - now);
else
time -= 0x10000 - (now - last);
last = now;
}
return IRQNONE;
}
/*
* Caught a stray interrupt, notify
*/
void
isa_strayintr(d)
int d;
{
extern u_long intrcnt_stray[],
intrcnt_wild;
/*
* Stray interrupts on irq 7 occur when an interrupt line is raised
* and then lowered before the CPU acknowledges it. This generally
* means either the device is screwed or something is cli'ing too
* long and it's timing out.
*
* -1 is passed by the generic handler for out of range exceptions,
* since we don't really want 208 little vectors. (It wouldn't be
* all that much code, but why bother?)
*/
if (d == -1) {
intrcnt_wild++;
log(LOG_ERR, "wild interrupt\n");
} else {
if (intrcnt_stray[d]++ < 5)
log(LOG_ERR, "stray interrupt %d%s\n", d,
intrcnt_stray[d] == 5 ? "; stopped logging" : "");
}
}
/*
* Handle a NMI, possibly a machine check.
* return true to panic system, false to ignore.
*
* This implementation is hist[oe]rical.
*/
int
isa_nmi()
{
log(LOG_CRIT, "NMI port 61 %x, port 70 %x\n", inb(0x61), inb(0x70));
return 0;
}