#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/malloc.h>
#include <sys/syslog.h>
#include <machine/intr.h>
#include <machine/bus.h>
#include <dev/pci/pcireg.h>
#include <dev/pci/pcivar.h>
#include <dev/pci/pcidevs.h>
#include <dev/pci/cy82c693reg.h>
#include <dev/pci/cy82c693var.h>
#include <dev/isa/isareg.h>
#include <dev/isa/isavar.h>
#include <alpha/pci/siovar.h>
#include "sio.h"
#define BROKEN_PROM_CONSOLE
bus_space_tag_t sio_iot;
pci_chipset_tag_t sio_pc;
bus_space_handle_t sio_ioh_icu1, sio_ioh_icu2;
#define ICU_LEN 16
static struct alpha_shared_intr *sio_intr;
#ifndef STRAY_MAX
#define STRAY_MAX 5
#endif
u_int8_t initial_ocw1[2];
u_int8_t initial_elcr[2];
#define INITIALLY_LEVEL_TRIGGERED(irq) \
((initial_elcr[(irq) / 8] & (1 << ((irq) % 8))) != 0)
u_int8_t elcr_override[2] = { 0x00, 0x00 };
void sio_setirqstat(int, int, int);
int sio_intr_alloc(void *, int, int, int *);
int sio_intr_check(void *, int, int);
u_int8_t (*sio_read_elcr)(int);
void (*sio_write_elcr)(int, u_int8_t);
static void specific_eoi(int);
int i82378_setup_elcr(void);
u_int8_t i82378_read_elcr(int);
void i82378_write_elcr(int, u_int8_t);
bus_space_handle_t sio_ioh_elcr;
int
i82378_setup_elcr(void)
{
int device, maxndevs;
pcitag_t tag;
pcireg_t id;
int rv;
rv = bus_space_map(sio_iot, 0x4d0, 2, 0, &sio_ioh_elcr);
if (rv != 0)
return 0;
sio_read_elcr = i82378_read_elcr;
sio_write_elcr = i82378_write_elcr;
maxndevs = pci_bus_maxdevs(sio_pc, 0);
for (device = 0; device < maxndevs; device++) {
tag = pci_make_tag(sio_pc, 0, device, 0);
id = pci_conf_read(sio_pc, tag, PCI_ID_REG);
if (id == PCI_ID_CODE(PCI_VENDOR_ALI, PCI_PRODUCT_ALI_M5237)) {
elcr_override[10 / 8] |= 1 << (10 % 8);
break;
}
}
return (0);
}
u_int8_t
i82378_read_elcr(int elcr)
{
return (bus_space_read_1(sio_iot, sio_ioh_elcr, elcr));
}
void
i82378_write_elcr(int elcr, u_int8_t val)
{
bus_space_write_1(sio_iot, sio_ioh_elcr, elcr, val);
}
int cy82c693_setup_elcr(void);
u_int8_t cy82c693_read_elcr(int);
void cy82c693_write_elcr(int, u_int8_t);
const struct cy82c693_handle *sio_cy82c693_handle;
int
cy82c693_setup_elcr(void)
{
int device, maxndevs;
pcitag_t tag;
pcireg_t id;
maxndevs = pci_bus_maxdevs(sio_pc, 0);
for (device = 0; device < maxndevs; device++) {
tag = pci_make_tag(sio_pc, 0, device, 0);
id = pci_conf_read(sio_pc, tag, PCI_ID_REG);
if (id ==
PCI_ID_CODE(PCI_VENDOR_CONTAQ, PCI_PRODUCT_CONTAQ_82C693)) {
sio_cy82c693_handle = cy82c693_init(sio_iot);
sio_read_elcr = cy82c693_read_elcr;
sio_write_elcr = cy82c693_write_elcr;
return (0);
}
}
return (ENODEV);
}
u_int8_t
cy82c693_read_elcr(int elcr)
{
return (cy82c693_read(sio_cy82c693_handle, CONFIG_ELCR1 + elcr));
}
void
cy82c693_write_elcr(int elcr, u_int8_t val)
{
cy82c693_write(sio_cy82c693_handle, CONFIG_ELCR1 + elcr, val);
}
int (*sio_elcr_setup_funcs[])(void) = {
cy82c693_setup_elcr,
i82378_setup_elcr,
NULL,
};
void
sio_setirqstat(int irq, int enabled, int type)
{
u_int8_t ocw1[2], elcr[2];
int icu, bit;
#if 0
printf("sio_setirqstat: irq %d: %s, %s\n", irq,
enabled ? "enabled" : "disabled", isa_intr_typename(type));
#endif
icu = irq / 8;
bit = irq % 8;
ocw1[0] = bus_space_read_1(sio_iot, sio_ioh_icu1, 1);
ocw1[1] = bus_space_read_1(sio_iot, sio_ioh_icu2, 1);
elcr[0] = (*sio_read_elcr)(0);
elcr[1] = (*sio_read_elcr)(1);
if (enabled)
ocw1[icu] &= ~(1 << bit);
else
ocw1[icu] |= 1 << bit;
if (type == IST_LEVEL)
elcr[icu] |= 1 << bit;
else
elcr[icu] &= ~(1 << bit);
elcr[icu] &= ~elcr_override[icu];
#ifdef not_here
ocw1[0] &= ~0x04;
elcr[0] &= ~0x07;
elcr[1] &= ~0x21;
#endif
bus_space_write_1(sio_iot, sio_ioh_icu1, 1, ocw1[0]);
bus_space_write_1(sio_iot, sio_ioh_icu2, 1, ocw1[1]);
(*sio_write_elcr)(0, elcr[0]);
(*sio_write_elcr)(1, elcr[1]);
}
void
sio_intr_setup(pci_chipset_tag_t pc, bus_space_tag_t iot)
{
int i;
sio_iot = iot;
sio_pc = pc;
if (bus_space_map(sio_iot, IO_ICU1, 2, 0, &sio_ioh_icu1) ||
bus_space_map(sio_iot, IO_ICU2, 2, 0, &sio_ioh_icu2))
panic("sio_intr_setup: can't map ICU I/O ports");
for (i = 0; sio_elcr_setup_funcs[i] != NULL; i++)
if ((*sio_elcr_setup_funcs[i])() == 0)
break;
if (sio_elcr_setup_funcs[i] == NULL)
panic("sio_intr_setup: can't map ELCR");
initial_ocw1[0] = bus_space_read_1(sio_iot, sio_ioh_icu1, 1);
initial_ocw1[1] = bus_space_read_1(sio_iot, sio_ioh_icu2, 1);
initial_elcr[0] = (*sio_read_elcr)(0);
initial_elcr[1] = (*sio_read_elcr)(1);
sio_intr = alpha_shared_intr_alloc(ICU_LEN);
for (i = 0; i < ICU_LEN; i++) {
alpha_shared_intr_set_maxstrays(sio_intr, i, STRAY_MAX);
switch (i) {
case 0:
case 1:
case 8:
case 13:
#ifdef DIAGNOSTIC
if (INITIALLY_LEVEL_TRIGGERED(i))
printf("WARNING: PROM set irq %d"
" level-triggered\n", i);
#endif
sio_setirqstat(i, 0, IST_EDGE);
alpha_shared_intr_set_dfltsharetype(sio_intr, i,
IST_EDGE);
specific_eoi(i);
break;
case 2:
sio_setirqstat(i, 1, IST_EDGE);
alpha_shared_intr_set_dfltsharetype(sio_intr, i,
IST_UNUSABLE);
break;
default:
sio_setirqstat(i, 0, INITIALLY_LEVEL_TRIGGERED(i) ?
IST_LEVEL : IST_NONE);
alpha_shared_intr_set_dfltsharetype(sio_intr, i,
INITIALLY_LEVEL_TRIGGERED(i) ?
IST_LEVEL : IST_NONE);
specific_eoi(i);
break;
}
}
}
void
sio_intr_shutdown(void)
{
#ifdef BROKEN_PROM_CONSOLE
if (sio_write_elcr == NULL)
return;
bus_space_write_1(sio_iot, sio_ioh_icu1, 1, initial_ocw1[0]);
bus_space_write_1(sio_iot, sio_ioh_icu2, 1, initial_ocw1[1]);
(*sio_write_elcr)(0, initial_elcr[0]);
(*sio_write_elcr)(1, initial_elcr[1]);
#endif
}
const char *
sio_intr_string(void *v, int irq)
{
static char irqstr[12];
if (irq == 0 || irq >= ICU_LEN || irq == 2)
panic("sio_intr_string: bogus isa irq 0x%x", irq);
snprintf(irqstr, sizeof irqstr, "isa irq %d", irq);
return (irqstr);
}
int
sio_intr_line(void *v, int irq)
{
return (irq);
}
void *
sio_intr_establish(void *v, int irq, int type, int level, int (*fn)(void *),
void *arg, const char *name)
{
void *cookie;
if (irq >= ICU_LEN || type == IST_NONE)
panic("sio_intr_establish: bogus irq or type");
if (type == IST_EDGE && INITIALLY_LEVEL_TRIGGERED(irq) &&
(irq == 3 || irq == 4))
type = IST_LEVEL;
cookie = alpha_shared_intr_establish(sio_intr, irq, type, level, fn,
arg, name);
if (cookie != NULL &&
alpha_shared_intr_firstactive(sio_intr, irq)) {
scb_set(0x800 + SCB_IDXTOVEC(irq), sio_iointr, NULL);
sio_setirqstat(irq, 1,
alpha_shared_intr_get_sharetype(sio_intr, irq));
}
return (cookie);
}
void
sio_intr_disestablish(void *v, void *cookie)
{
struct alpha_shared_intrhand *ih = cookie;
int s, ist, irq = ih->ih_num;
s = splhigh();
alpha_shared_intr_disestablish(sio_intr, cookie);
if (alpha_shared_intr_isactive(sio_intr, irq) == 0) {
switch (irq) {
case 0:
case 1:
case 8:
case 13:
ist = IST_EDGE;
break;
default:
ist = INITIALLY_LEVEL_TRIGGERED(irq) ?
IST_LEVEL : IST_NONE;
break;
}
sio_setirqstat(irq, 0, ist);
alpha_shared_intr_set_dfltsharetype(sio_intr, irq, ist);
scb_free(0x800 + SCB_IDXTOVEC(irq));
}
splx(s);
}
void
sio_iointr(void *arg, unsigned long vec)
{
int irq;
irq = SCB_VECTOIDX(vec - 0x800);
#ifdef DIAGNOSTIC
if (irq >= ICU_LEN || irq < 0)
panic("sio_iointr: irq out of range (%d)", irq);
#endif
if (!alpha_shared_intr_dispatch(sio_intr, irq))
alpha_shared_intr_stray(sio_intr, irq, "isa irq");
else
alpha_shared_intr_reset_strays(sio_intr, irq);
specific_eoi(irq);
}
#define LEGAL_IRQ(x) ((x) >= 0 && (x) < ICU_LEN && (x) != 2)
int
sio_intr_alloc(void *v, int mask, int type, int *irq)
{
int i, tmp, bestirq, count;
struct alpha_shared_intrhand **p, *q;
if (type == IST_NONE)
panic("intr_alloc: bogus type");
bestirq = -1;
count = -1;
mask &= 0xffff;
mask &= ~((1 << 13) | (1 << 8) | (1 << 2) | (1 << 1) | (1 << 0));
mask &= ~((1 << 12) | (1 << 6));
for (i = 0; i < ICU_LEN; i++) {
if (LEGAL_IRQ(i) == 0 || (mask & (1<<i)) == 0)
continue;
switch (sio_intr[i].intr_sharetype) {
case IST_NONE:
*irq = i;
return (0);
case IST_EDGE:
intr_shared_edge = 1;
case IST_LEVEL:
if (type != sio_intr[i].intr_sharetype)
continue;
for (p = &TAILQ_FIRST(&sio_intr[i].intr_q), tmp = 0;
(q = *p) != NULL; p = &TAILQ_NEXT(q, ih_q), tmp++)
;
if ((bestirq == -1) || (count > tmp)) {
bestirq = i;
count = tmp;
}
break;
case IST_PULSE:
continue;
}
}
if (bestirq == -1)
return (1);
*irq = bestirq;
return (0);
}
int
sio_intr_check(void *v, int irq, int type)
{
if (type == IST_NONE)
return (0);
if (type == IST_EDGE && INITIALLY_LEVEL_TRIGGERED(irq) &&
(irq == 3 || irq == 4))
type = IST_LEVEL;
switch (sio_intr[irq].intr_sharetype) {
case IST_NONE:
return (2);
case IST_EDGE:
case IST_LEVEL:
if (type == sio_intr[irq].intr_sharetype)
return (1);
default:
case IST_PULSE:
return (0);
}
}
static void
specific_eoi(int irq)
{
if (irq > 7) {
bus_space_write_1(sio_iot,
sio_ioh_icu2, 0, 0x60 | (irq & 0x07));
irq = 2;
}
bus_space_write_1(sio_iot, sio_ioh_icu1, 0, 0x60 | irq);
}