1995-03-24 17:59:37 +03:00
|
|
|
/* $NetBSD: tc_3000_300.c,v 1.2 1995/03/24 14:59:37 cgd Exp $ */
|
1995-03-08 03:38:44 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 1994, 1995 Carnegie-Mellon University.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Author: Chris G. Demetriou
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify and distribute this software and
|
|
|
|
* its documentation is hereby granted, provided that both the copyright
|
|
|
|
* notice and this permission notice appear in all copies of the
|
|
|
|
* software, derivative works or modified versions, and any portions
|
|
|
|
* thereof, and that both notices appear in supporting documentation.
|
|
|
|
*
|
|
|
|
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
|
|
|
|
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND
|
|
|
|
* FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
|
|
|
|
*
|
|
|
|
* Carnegie Mellon requests users of this software to return to
|
|
|
|
*
|
|
|
|
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
|
|
|
|
* School of Computer Science
|
|
|
|
* Carnegie Mellon University
|
|
|
|
* Pittsburgh PA 15213-3890
|
|
|
|
*
|
|
|
|
* any improvements or extensions that they make and grant Carnegie the
|
|
|
|
* rights to redistribute these changes.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/device.h>
|
|
|
|
|
|
|
|
#include <machine/autoconf.h>
|
|
|
|
#include <machine/pte.h>
|
|
|
|
|
|
|
|
#include <alpha/tc/tc.h>
|
|
|
|
#include <alpha/tc/tc_3000_300.h>
|
|
|
|
|
|
|
|
/* XXX ESTABLISH, DISESTABLISH */
|
|
|
|
void tc_3000_300_intr_setup __P((void));
|
|
|
|
void tc_3000_300_intr_establish
|
|
|
|
__P((struct confargs *, intr_handler_t, void *));
|
|
|
|
void tc_3000_300_intr_disestablish __P((struct confargs *));
|
|
|
|
void tc_3000_300_iointr __P((void *, int));
|
|
|
|
int tc_3000_300_getdev __P((struct confargs *));
|
|
|
|
|
|
|
|
#define KV(x) ((caddr_t)phystok0seg(x))
|
|
|
|
#define TC_3000_300_NSLOTS 5
|
|
|
|
#define TC_3000_300_MAXDEVS 5
|
|
|
|
|
|
|
|
static struct tc_slot_desc dec_3000_300_slots[TC_3000_300_NSLOTS] = {
|
|
|
|
{ KV(0x100000000), }, /* slot 0 - TC option slot 0 */
|
|
|
|
{ KV(0x120000000), }, /* slot 1 - TC option slot 1 */
|
|
|
|
{ KV(0x180000000), }, /* slot 2 - TCDS ASIC on cpu board */
|
|
|
|
{ KV(0x1a0000000), }, /* slot 3 - IOCTL ASIC on cpu board */
|
|
|
|
{ KV(0x1c0000000), }, /* slot 4 - CXTurbo on cpu board */
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct confargs dec_3000_300_devs[TC_3000_300_MAXDEVS] = {
|
|
|
|
{ "PMAGB-BA", 4, 0x02000000, },
|
|
|
|
{ "IOCTL ", 3, 0x00000000, },
|
|
|
|
{ "PMAZ-DS ", 2, 0x00000000, },
|
|
|
|
{ NULL, 1, 0x0, },
|
|
|
|
{ NULL, 0, 0x0, },
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Indices into the struct confargs array. */
|
|
|
|
#define TC_3000_300_DEV_CXTURBO 0
|
|
|
|
#define TC_3000_300_DEV_IOCTL 1
|
|
|
|
#define TC_3000_300_DEV_TCDS 2
|
|
|
|
#define TC_3000_300_DEV_OPT1 3
|
|
|
|
#define TC_3000_300_DEV_OPT0 4
|
|
|
|
|
|
|
|
struct tc_cpu_desc dec_3000_300_cpu = {
|
|
|
|
dec_3000_300_slots, TC_3000_300_NSLOTS,
|
|
|
|
dec_3000_300_devs, TC_3000_300_MAXDEVS,
|
|
|
|
tc_3000_300_intr_setup,
|
|
|
|
tc_3000_300_intr_establish,
|
|
|
|
tc_3000_300_intr_disestablish,
|
|
|
|
tc_3000_300_iointr,
|
|
|
|
};
|
|
|
|
|
|
|
|
intr_handler_t tc_3000_300_intrhand[TC_3000_300_MAXDEVS];
|
|
|
|
void *tc_3000_300_intrval[TC_3000_300_MAXDEVS];
|
|
|
|
|
|
|
|
void
|
|
|
|
tc_3000_300_intr_setup()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
1995-03-24 17:59:37 +03:00
|
|
|
/* Set up interrupt handlers. */
|
|
|
|
for (i = 0; i < TC_3000_300_MAXDEVS; i++) {
|
|
|
|
tc_3000_300_intrhand[i] = tc_intrnull;
|
|
|
|
tc_3000_300_intrval[i] = (void *)(long)i;
|
|
|
|
}
|
1995-03-08 03:38:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tc_3000_300_intr_establish(ca, handler, val)
|
|
|
|
struct confargs *ca;
|
|
|
|
int (*handler) __P((void *));
|
|
|
|
void *val;
|
|
|
|
{
|
|
|
|
int dev = tc_3000_300_getdev(ca);
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (dev == -1)
|
|
|
|
panic("tc_3000_300_intr_establish: dev == -1");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (tc_3000_300_intrhand[dev] != tc_intrnull)
|
|
|
|
panic("tc_3000_300_intr_establish: dev %d twice", dev);
|
|
|
|
|
|
|
|
tc_3000_300_intrhand[dev] = handler;
|
|
|
|
tc_3000_300_intrval[dev] = val;
|
|
|
|
|
|
|
|
/* XXX ENABLE INTERRUPT MASK FOR DEV */
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tc_3000_300_intr_disestablish(ca)
|
|
|
|
struct confargs *ca;
|
|
|
|
{
|
|
|
|
int dev = tc_3000_300_getdev(ca);
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (dev == -1)
|
|
|
|
panic("tc_3000_300_intr_disestablish: somebody goofed");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (tc_3000_300_intrhand[dev] == tc_intrnull)
|
|
|
|
panic("tc_3000_300_intr_disestablish: dev %d missing intr",
|
|
|
|
dev);
|
|
|
|
|
|
|
|
tc_3000_300_intrhand[dev] = tc_intrnull;
|
|
|
|
tc_3000_300_intrval[dev] = (void *)(long)dev;
|
|
|
|
|
|
|
|
/* XXX DISABLE INTERRUPT MASK FOR DEV */
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tc_3000_300_iointr(framep, vec)
|
1995-03-24 17:59:37 +03:00
|
|
|
void *framep;
|
|
|
|
int vec;
|
1995-03-08 03:38:44 +03:00
|
|
|
{
|
1995-03-24 17:59:37 +03:00
|
|
|
u_int32_t ir;
|
1995-03-08 03:38:44 +03:00
|
|
|
int ifound;
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
int s;
|
|
|
|
if (vec != 0x800)
|
|
|
|
panic("INVALID ASSUMPTION: vec %x, not 0x800", vec);
|
|
|
|
s = splhigh();
|
|
|
|
if (s != PSL_IPL_IO)
|
|
|
|
panic("INVALID ASSUMPTION: IPL %d, not %d", s, PSL_IPL_IO);
|
|
|
|
splx(s);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
do {
|
|
|
|
MAGIC_READ;
|
|
|
|
MB();
|
|
|
|
|
1995-03-24 17:59:37 +03:00
|
|
|
/* find out what interrupts/errors occurred */
|
|
|
|
ir = *(volatile u_int32_t *)TC_3000_300_IR;
|
|
|
|
MB();
|
|
|
|
|
|
|
|
/* clear the interrupts/errors we found. */
|
|
|
|
*(volatile u_int32_t *)TC_3000_300_IR = ir;
|
|
|
|
MB();
|
|
|
|
|
1995-03-08 03:38:44 +03:00
|
|
|
ifound = 0;
|
|
|
|
#define CHECKINTR(slot, bits) \
|
|
|
|
if (ir & bits) { \
|
|
|
|
ifound = 1; \
|
|
|
|
(*tc_3000_300_intrhand[slot]) \
|
|
|
|
(tc_3000_300_intrval[slot]); \
|
|
|
|
}
|
|
|
|
/* Do them in order of priority; highest slot # first. */
|
|
|
|
CHECKINTR(TC_3000_300_DEV_CXTURBO, TC_3000_300_IR_CXTURBO);
|
|
|
|
CHECKINTR(TC_3000_300_DEV_IOCTL, TC_3000_300_IR_IOCTL);
|
|
|
|
CHECKINTR(TC_3000_300_DEV_TCDS, TC_3000_300_IR_TCDS);
|
|
|
|
#if 0
|
|
|
|
CHECKINTR(TC_3000_300_DEV_OPT1, TC_3000_300_IR_OPT1);
|
|
|
|
CHECKINTR(TC_3000_300_DEV_OPT0, TC_3000_300_IR_OPT0);
|
|
|
|
#else
|
|
|
|
/* XXX XXX XXX CHECK OPTION SLOT INTERRUPTS!!! */
|
1995-03-24 17:59:37 +03:00
|
|
|
/* XXX XXX XXX THEIR BITS LIVE IN ANOTHER REG. */
|
1995-03-08 03:38:44 +03:00
|
|
|
#endif
|
|
|
|
#undef CHECKINTR
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
#define PRINTINTR(msg, bits) \
|
|
|
|
if (ir & bits) \
|
|
|
|
printf(msg);
|
|
|
|
PRINTINTR("BCache tag parity error\n",
|
|
|
|
TC_3000_300_IR_BCTAGPARITY);
|
|
|
|
PRINTINTR("TC overrun error\n", TC_3000_300_IR_TCOVERRUN);
|
|
|
|
PRINTINTR("TC I/O timeout\n", TC_3000_300_IR_TCTIMEOUT);
|
|
|
|
PRINTINTR("Bcache parity error\n",
|
|
|
|
TC_3000_300_IR_BCACHEPARITY);
|
|
|
|
PRINTINTR("Memory parity error\n", TC_3000_300_IR_MEMPARITY);
|
|
|
|
#undef PRINTINTR
|
|
|
|
#endif
|
|
|
|
} while (ifound);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tc_3000_300_getdev(ca)
|
|
|
|
struct confargs *ca;
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < TC_3000_300_MAXDEVS; i++)
|
|
|
|
if (ca->ca_slot == dec_3000_300_devs[i].ca_slot &&
|
|
|
|
ca->ca_offset == dec_3000_300_devs[i].ca_offset &&
|
|
|
|
!strncmp(ca->ca_name, dec_3000_300_devs[i].ca_name))
|
|
|
|
return (i);
|
|
|
|
|
|
|
|
return (-1);
|
|
|
|
}
|