diff options
Diffstat (limited to 'sys/arch/amiga/isa/cross.c')
-rw-r--r-- | sys/arch/amiga/isa/cross.c | 530 |
1 files changed, 338 insertions, 192 deletions
diff --git a/sys/arch/amiga/isa/cross.c b/sys/arch/amiga/isa/cross.c index 0b98f664472..923f059eccf 100644 --- a/sys/arch/amiga/isa/cross.c +++ b/sys/arch/amiga/isa/cross.c @@ -1,8 +1,7 @@ -/* $OpenBSD: cross.c,v 1.2 1996/02/27 15:40:54 niklas Exp $ */ -/* $NetBSD: cross.c,v 1.0 1994/07/08 23:32:17 niklas Exp $ */ +/* $OpenBSD: cross.c,v 1.3 1996/04/27 18:38:55 niklas Exp $ */ /* - * Copyright (c) 1994 Niklas Hallqvist, Carsten Hammer + * Copyright (c) 1994, 1996 Niklas Hallqvist, Carsten Hammer * All rights reserved. * * Redistribution and use in source and binary forms, with or without @@ -15,7 +14,7 @@ * 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 Christian E. Hopps. + * This product includes software developed by Niklas Hallqvist. * 4. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission * @@ -30,14 +29,16 @@ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ + #include <sys/param.h> #include <sys/device.h> #include <sys/kernel.h> #include <sys/malloc.h> #include <sys/syslog.h> +#include <machine/bus.h> #include <machine/cpu.h> -#include <machine/pio.h> +#include <machine/intr.h> #include <dev/isa/isavar.h> @@ -46,35 +47,93 @@ #include <amiga/amiga/isr.h> #include <amiga/dev/zbusvar.h> #include <amiga/isa/isa_machdep.h> -#include <amiga/isa/isa_intr.h> #include <amiga/isa/crossvar.h> #include <amiga/isa/crossreg.h> +extern int cold; + int crossdebug = 0; -/* This static is OK because we only allow one ISA bus. */ -struct cross_device *crossp; - -void crossattach __P((struct device *, struct device *, void *)); -int crossmatch __P((struct device *, void *, void *)); -int crossprint __P((void *auxp, char *)); -void crossstb __P((struct device *, int, u_char)); -u_char crossldb __P((struct device *, int)); -void crossstw __P((struct device *, int, u_short)); -u_short crossldw __P((struct device *, int)); -void *cross_establish_intr __P((int intr, int type, int level, - int (*ih_fun) (void *), void *ih_arg, - char *ih_what)); -void cross_disestablish_intr __P((void *handler)); - -struct isa_intr_fcns cross_intr_fcns = { - 0 /* cross_intr_setup */, cross_establish_intr, - cross_disestablish_intr, 0 /* cross_iointr */ +void crossattach __P((struct device *, struct device *, void *)); +int crossmatch __P((struct device *, void *, void *)); +int crossprint __P((void *, char *)); + +int cross_io_map(bus_chipset_tag_t, bus_io_addr_t, bus_io_size_t, + bus_io_handle_t *); +int cross_mem_map(bus_chipset_tag_t, bus_mem_addr_t, bus_mem_size_t, int, + bus_mem_handle_t *); + +void cross_io_read_multi_1(bus_io_handle_t, bus_io_size_t, u_int8_t *, + bus_io_size_t); +void cross_io_read_multi_2(bus_io_handle_t, bus_io_size_t, u_int16_t *, + bus_io_size_t); + +void cross_io_write_multi_1(bus_io_handle_t, bus_io_size_t, + const u_int8_t *, bus_io_size_t); +void cross_io_write_multi_2(bus_io_handle_t, bus_io_size_t, + const u_int16_t *, bus_io_size_t); + +/* + * Note that the following unified access functions are prototyped for the + * I/O access case. We use casts to get type correctness. + */ +int cross_unmap(bus_io_handle_t, bus_io_size_t); + +__inline u_int8_t cross_read_1(bus_io_handle_t, bus_io_size_t); +__inline u_int16_t cross_read_2(bus_io_handle_t, bus_io_size_t); + +__inline void cross_write_1(bus_io_handle_t, bus_io_size_t, u_int8_t); +__inline void cross_write_2(bus_io_handle_t, bus_io_size_t, u_int16_t); + +/* + * In order to share the access function implementations for I/O and memory + * access we cast the functions for the memory access case. These typedefs + * make that casting look nicer. + */ +typedef int (*bus_mem_unmap_t)(bus_mem_handle_t, bus_mem_size_t); +typedef u_int8_t (*bus_mem_read_1_t)(bus_mem_handle_t, bus_mem_size_t); +typedef u_int16_t (*bus_mem_read_2_t)(bus_mem_handle_t, bus_mem_size_t); +typedef void (*bus_mem_write_1_t)(bus_mem_handle_t, bus_mem_size_t, u_int8_t); +typedef void (*bus_mem_write_2_t)(bus_mem_handle_t, bus_mem_size_t, u_int16_t); + +void cross_attach_hook(struct device *, struct device *, + struct isabus_attach_args *); +void *cross_intr_establish __P((void *, int, int, int, int (*)(void *), + void *, char *)); +void cross_intr_disestablish __P((void *, void *)); + +static u_int16_t swap __P((u_int16_t)); + +struct amiga_bus_chipset cross_chipset = { + 0 /* bc_data */, + + cross_io_map, cross_unmap, + cross_read_1, cross_read_2, + 0 /* bc_io_read_4 */, 0 /* bc_io_read_8 */, + cross_io_read_multi_1, cross_io_read_multi_2, + 0 /* bc_io_multi_4 */, 0 /* bc_io_multi_8 */, + cross_write_1, cross_write_2, + 0 /* bc_io_write_4 */, 0 /* bc_io_write_8 */, + cross_io_write_multi_1, cross_io_write_multi_2, + 0 /* bc_io_write_multi_4 */, 0 /* bc_io_write_multi_8 */, + + cross_mem_map, (bus_mem_unmap_t)cross_unmap, + (bus_mem_read_1_t)cross_read_1, (bus_mem_read_2_t)cross_read_2, + 0 /* bc_mem_read_4 */, 0 /* bc_mem_read_8 */, + (bus_mem_write_1_t)cross_write_1, (bus_mem_write_2_t)cross_write_2, + 0 /* bc_mem_write_4 */, 0 /* bc_mem_write_8 */, + + /* These are extensions to the general NetBSD bus interface. */ + swap, 0 /* bc_to_host_4 */, 0 /* bc_to_host_8 */, + swap, 0 /* bc_from_host_4 */, 0 /* bc_from_host_8 */, +}; + +struct cfattach cross_ca = { + sizeof(struct cross_softc), crossmatch, crossattach }; -struct cfdriver crosscd = { - NULL, "cross", crossmatch, crossattach, - DV_DULL, sizeof(struct cross_device), 0 +struct cfdriver cross_cd = { + NULL, "cross", DV_DULL, 0 }; int @@ -93,38 +152,36 @@ crossmatch(parent, match, aux) } void -crossattach(pdp, dp, auxp) - struct device *pdp, *dp; - void *auxp; +crossattach(parent, self, aux) + struct device *parent, *self; + void *aux; { - struct zbus_args *zap = auxp; - struct cross_device *cdp = (struct cross_device *)dp; - - crossp = cdp; - bcopy(zap, &cdp->cd_zargs, sizeof(struct zbus_args)); - cdp->cd_link.il_dev = dp; - cdp->cd_link.il_ldb = crossldb; - cdp->cd_link.il_stb = crossstb; - cdp->cd_link.il_ldw = crossldw; - cdp->cd_link.il_stw = crossstw; - cdp->cd_imask = 1 << CROSS_MASTER; - - isa_intr_fcns = &cross_intr_fcns; - isa_pio_fcns = &cross_pio_fcns; - - /* Enable interrupts lazily in crossaddint. */ + struct cross_softc *sc = (struct cross_softc *)self; + struct zbus_args *zap = aux; + struct isabus_attach_args iba; + + bcopy(zap, &sc->sc_zargs, sizeof(struct zbus_args)); + bcopy(&cross_chipset, &sc->sc_bc, sizeof(struct amiga_bus_chipset)); + sc->sc_bc.bc_data = sc; + sc->sc_status = CROSS_STATUS_ADDR(zap->va); + sc->sc_imask = 1 << CROSS_MASTER; + + /* Enable interrupts lazily in cross_intr_establish. */ CROSS_ENABLE_INTS(zap->va, 0); /* Default 16 bit tranfer */ - *(volatile u_short *)(cdp->cd_zargs.va + CROSS_XLP_LATCH) = CROSS_SBHE; - + *CROSS_HANDLE_TO_XLP_LATCH((bus_io_handle_t)zap->va) = CROSS_SBHE; printf(": pa 0x%08x va 0x%08x size 0x%x\n", zap->pa, zap->va, - zap->size); + zap->size); + sc->sc_ic.ic_data = sc; + sc->sc_ic.ic_attach_hook = cross_attach_hook; + sc->sc_ic.ic_intr_establish = cross_intr_establish; + sc->sc_ic.ic_intr_disestablish = cross_intr_disestablish; - /* - * attempt to configure the board. - */ - config_found(dp, &cdp->cd_link, crossprint); + iba.iba_busname = "isa"; + iba.iba_bc = &sc->sc_bc; + iba.iba_ic = &sc->sc_ic; + config_found(self, &iba, crossprint); } int @@ -138,184 +195,273 @@ crossprint(auxp, pnp) } -void -crossstb(dev, ia, b) - struct device *dev; - int ia; - u_char b; +int +cross_io_map(bct, addr, sz, handle) + bus_chipset_tag_t bct; + bus_io_addr_t addr; + bus_io_size_t sz; + bus_io_handle_t *handle; +{ + *handle = (bus_io_handle_t) + ((struct cross_softc *)bct->bc_data)->sc_zargs.va + 2 * addr; +#if 0 + printf("io_map %x %d -> %x\n", addr, sz, *handle); +#endif + return 0; +} + +int +cross_mem_map(bct, addr, sz, cacheable, handle) + bus_chipset_tag_t bct; + bus_mem_addr_t addr; + bus_mem_size_t sz; + int cacheable; + bus_mem_handle_t *handle; +{ + *handle = (bus_mem_handle_t) + ((struct cross_softc *)bct->bc_data)->sc_zargs.va + 2 * addr + + CROSS_MEMORY_OFFSET; +#if 0 + printf("mem_map %x %d -> %x\n", addr, sz, *handle); +#endif + return 0; +} + +int +cross_unmap(handle, sz) + bus_io_handle_t handle; + bus_io_size_t sz; +{ + return 0; +} + +__inline u_int8_t +cross_read_1(handle, addr) + bus_io_handle_t handle; + bus_io_size_t addr; { + u_int8_t val; + /* generate A13-A19 for correct page */ - u_short upper_addressbits = ia >> 13; - struct cross_device *cd = (struct cross_device *)dev; + *CROSS_HANDLE_TO_XLP_LATCH(handle) = addr >> 13 | CROSS_SBHE; + val = *(volatile u_int8_t *)(handle + 2 * addr); - *(volatile u_short *)(cd->cd_zargs.va + CROSS_XLP_LATCH) = - upper_addressbits | CROSS_SBHE; - *(volatile u_char *)(cd->cd_zargs.va + CROSS_MEMORY_OFFSET + 2 * ia) = - b; +#if 0 + printf("read_1 @%x handle %x -> %d\n", addr, handle, val); +#endif + return val; } -u_char -crossldb(dev, ia) - struct device *dev; - int ia; +__inline u_int16_t +cross_read_2(handle, addr) + bus_io_handle_t handle; + bus_io_size_t addr; { /* generate A13-A19 for correct page */ - u_short upper_addressbits = ia >> 13; - struct cross_device *cd = (struct cross_device *)dev; + *CROSS_HANDLE_TO_XLP_LATCH(handle) = addr >> 13 | CROSS_SBHE; + return *(volatile u_int16_t *)(handle + 2 * addr); +} - *(volatile u_short *)(cd->cd_zargs.va + CROSS_XLP_LATCH) = - upper_addressbits | CROSS_SBHE; - return *(volatile u_char *)(cd->cd_zargs.va + CROSS_MEMORY_OFFSET + - 2 * ia); +void +cross_io_read_multi_1(handle, addr, buf, cnt) + bus_io_handle_t handle; + bus_io_size_t addr; + u_int8_t *buf; + bus_io_size_t cnt; +{ + while (cnt--) + *buf++ = cross_read_1(handle, addr); } void -crossstw(dev, ia, w) - struct device *dev; - int ia; - u_short w; +cross_io_read_multi_2(handle, addr, buf, cnt) + bus_io_handle_t handle; + bus_io_size_t addr; + u_int16_t *buf; + bus_io_size_t cnt; { - /* generate A13-A19 for correct page */ - u_short upper_addressbits = ia >> 13; - struct cross_device *cd = (struct cross_device *)dev; - - *(volatile u_short *)(cd->cd_zargs.va + CROSS_XLP_LATCH) = - upper_addressbits | CROSS_SBHE; -#ifdef DEBUG - if (crossdebug) - printf("outw 0x%x,0x%x\n", ia, w); -#endif - *(volatile u_short *)(cd->cd_zargs.va + CROSS_MEMORY_OFFSET + 2 * ia) = - w; + while (cnt--) + *buf++ = cross_read_2(handle, addr); } -u_short -crossldw(dev, ia) - struct device *dev; - int ia; +__inline void +cross_write_1(handle, addr, val) + bus_io_handle_t handle; + bus_io_size_t addr; + u_int8_t val; { /* generate A13-A19 for correct page */ - u_short upper_addressbits = ia >> 13; - struct cross_device *cd = (struct cross_device *)dev; - u_short retval; - - *(volatile u_short *)(cd->cd_zargs.va + CROSS_XLP_LATCH) = - upper_addressbits | CROSS_SBHE; - retval = *(volatile u_short *)(cd->cd_zargs.va + CROSS_MEMORY_OFFSET + - 2 * ia); -#ifdef DEBUG - if (crossdebug) - printf("ldw 0x%x => 0x%x\n", ia, retval); + *CROSS_HANDLE_TO_XLP_LATCH(handle) = addr >> 13 | CROSS_SBHE; +#if 0 + printf("write_1 @%x handle %x: %d\n", addr, handle, val); #endif - return retval; + *(volatile u_int8_t *)(handle + 2 * addr + 1) = val; } -static cross_int_map[] = { - 0, 0, 0, 0, CROSS_IRQ3, CROSS_IRQ4, CROSS_IRQ5, CROSS_IRQ6, CROSS_IRQ7, 0, - CROSS_IRQ9, CROSS_IRQ10, CROSS_IRQ11, CROSS_IRQ12, 0, CROSS_IRQ14, - CROSS_IRQ15 -}; +__inline void +cross_write_2(handle, addr, val) + bus_io_handle_t handle; + bus_io_size_t addr; + u_int16_t val; +{ + /* generate A13-A19 for correct page */ + *CROSS_HANDLE_TO_XLP_LATCH(handle) = addr >> 13 | CROSS_SBHE; + *(volatile u_int16_t *)(handle + 2 * addr) = val; +} -#if 0 -/* XXX We don't care about the priority yet, although we ought to. */ void -crossaddint(dev, irq, func, arg, pri) - struct device *dev; - int irq; - int (*func)(); - void *arg; - int pri; +cross_io_write_multi_1(handle, addr, buf, cnt) + bus_io_handle_t handle; + bus_io_size_t addr; + const u_int8_t *buf; + bus_io_size_t cnt; { - struct cross_device *cd = (struct cross_device *)dev; - int s = splhigh(); - int bit = cross_int_map[irq + 1]; - - if (!bit) { - log(LOG_WARNING, "Registration of unknown ISA interrupt %d\n", - irq); - goto out; - } - if (cd->cd_imask & 1 << bit) { - log(LOG_WARNING, "ISA interrupt %d already handled\n", irq); - goto out; - } - cd->cd_imask |= (1 << bit); - CROSS_ENABLE_INTS (cd->cd_zargs.va, cd->cd_imask); - cd->cd_ifunc[bit] = func; - cd->cd_ipri[bit] = pri; - cd->cd_iarg[bit] = arg; -out: - splx(s); + while (cnt--) + cross_write_1(handle, addr, *buf++); } void -crossremint(dev, irq) - struct device *dev; - int irq; +cross_io_write_multi_2(handle, addr, buf, cnt) + bus_io_handle_t handle; + bus_io_size_t addr; + const u_int16_t *buf; + bus_io_size_t cnt; { - struct cross_device *cd = (struct cross_device *)dev; - int s = splhigh(); - int bit = cross_int_map[irq + 1]; - - cd->cd_imask &= ~(1 << bit); - CROSS_ENABLE_INTS (cd->cd_zargs.va, cd->cd_imask); - splx(s); + while (cnt--) + cross_write_2(handle, addr, *buf++); } -#endif -struct crossintr_desc { - struct isr cid_isr; - int cid_mask; - int (*cid_fun)(void *); - void *cid_arg; -}; -static struct crossintr_desc *crid[16]; /* XXX */ +static cross_int_map[] = { + 0, 0, 0, 0, CROSS_IRQ3, CROSS_IRQ4, CROSS_IRQ5, CROSS_IRQ6, CROSS_IRQ7, 0, + CROSS_IRQ9, CROSS_IRQ10, CROSS_IRQ11, CROSS_IRQ12, 0, CROSS_IRQ14, + CROSS_IRQ15 +}; int -crossintr(cid) - struct crossintr_desc *cid; +crossintr(v) + void *v; +{ + struct intrhand *ih = (struct intrhand *)v; + int handled; + + if (!(*ih->ih_status & ih->ih_mask)) + return 0; + for (handled = 0; ih; ih = ih->ih_next) + if ((*ih->ih_fun)(ih->ih_arg)) + handled = 1; + return handled; +} + +void +cross_attach_hook(parent, self, iba) + struct device *parent, *self; + struct isabus_attach_args *iba; { - return (CROSS_GET_STATUS (crossp->cd_zargs.va) & cid->cid_mask) ? - (*cid->cid_fun)(cid->cid_arg) : 0; } void * -cross_establish_intr(intr, type, level, ih_fun, ih_arg, ih_what) - int intr; - int type; - int level; - int (*ih_fun)(void *); - void *ih_arg; +cross_intr_establish(ic, irq, type, level, ih_fun, ih_arg, ih_what) + void *ic; + int irq; + int type; + int level; + int (*ih_fun)(void *); + void *ih_arg; char *ih_what; { - if (crid[intr]) { - log(LOG_WARNING, "ISA interrupt %d already handled\n", intr); - return 0; - } - MALLOC(crid[intr], struct crossintr_desc *, - sizeof(struct crossintr_desc), M_DEVBUF, M_WAITOK); - crid[intr]->cid_isr.isr_intr = crossintr; - crid[intr]->cid_isr.isr_arg = crid[intr]; - crid[intr]->cid_isr.isr_ipl = 6; - crid[intr]->cid_isr.isr_mapped_ipl = level; - crid[intr]->cid_mask = 1 << cross_int_map[intr + 1]; - crid[intr]->cid_fun = ih_fun; - crid[intr]->cid_arg = ih_arg; - add_isr (&crid[intr]->cid_isr); - crossp->cd_imask |= 1 << cross_int_map[intr + 1]; - CROSS_ENABLE_INTS (crossp->cd_zargs.va, crossp->cd_imask); - return &crid[intr]; + struct intrhand **p, *c, *ih; + struct cross_softc *sc = (struct cross_softc *)ic; + + /* no point in sleeping unless someone can free memory. */ + ih = malloc(sizeof *ih, M_DEVBUF, cold ? M_NOWAIT : M_WAITOK); + if (ih == NULL) + panic("cross_intr_establish: can't malloc handler info"); + + if (irq > ICU_LEN || type == IST_NONE) + panic("cross_intr_establish: bogus irq or type"); + + switch (sc->sc_intrsharetype[irq]) { + case IST_EDGE: + case IST_LEVEL: + if (type == sc->sc_intrsharetype[irq]) + break; + case IST_PULSE: + if (type != IST_NONE) + panic("cross_intr_establish: can't share %s with %s", + isa_intr_typename(sc->sc_intrsharetype[irq]), + isa_intr_typename(type)); + break; + } + + /* + * Figure out where to put the handler. + * This is O(N^2), but we want to preserve the order, and N is + * generally small. + */ + for (p = &sc->sc_ih[irq]; (c = *p) != NULL; p = &c->ih_next) + ; + + /* + * Poke the real handler in now. + */ + ih->ih_fun = ih_fun; + ih->ih_arg = ih_arg; + ih->ih_count = 0; + ih->ih_next = NULL; + ih->ih_irq = irq; + ih->ih_what = ih_what; + ih->ih_mask = 1 << cross_int_map[irq + 1]; + ih->ih_status = sc->sc_status; + ih->ih_isr.isr_intr = crossintr; + ih->ih_isr.isr_arg = ih; + ih->ih_isr.isr_ipl = 6; + ih->ih_isr.isr_mapped_ipl = level; + *p = ih; + add_isr(&ih->ih_isr); + + sc->sc_imask |= 1 << cross_int_map[irq + 1]; + CROSS_ENABLE_INTS(sc->sc_zargs.va, sc->sc_imask); + + return ih; } void -cross_disestablish_intr(handler) - void *handler; +cross_intr_disestablish(ic, arg) + void *ic; + void *arg; { - struct crossintr_desc **cid = handler; + struct intrhand *ih = arg; + struct cross_softc *sc = (struct cross_softc *)ic; + int irq = ih->ih_irq; + struct intrhand **p, *q; + + if (irq > ICU_LEN) + panic("cross_intr_establish: bogus irq"); + + sc->sc_imask &= ~ih->ih_mask; + CROSS_ENABLE_INTS (sc->sc_zargs.va, sc->sc_imask); + remove_isr(&ih->ih_isr); - remove_isr(&(*cid)->cid_isr); - crossp->cd_imask &= ~(*cid)->cid_mask; - FREE(*cid, M_DEVBUF); - *cid = 0; - CROSS_ENABLE_INTS (crossp->cd_zargs.va, crossp->cd_imask); + /* + * Remove the handler from the chain. + * This is O(n^2), too. + */ + for (p = &sc->sc_ih[irq]; (q = *p) != NULL && q != ih; p = &q->ih_next) + ; + if (q) + *p = q->ih_next; + else + panic("cross_intr_disestablish: handler not registered"); + free(ih, M_DEVBUF); + + if (sc->sc_intrsharetype[irq] == NULL) + sc->sc_intrsharetype[irq] = IST_NONE; +} + +/* Swap bytes in a short word. */ +static u_short +swap(u_short x) +{ + __asm("rolw #8,%0" : "=r" (x) : "0" (x)); + return x; } |