summaryrefslogtreecommitdiff
path: root/sys/arch/i386/isa
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/i386/isa')
-rw-r--r--sys/arch/i386/isa/ahc_isa.c462
1 files changed, 462 insertions, 0 deletions
diff --git a/sys/arch/i386/isa/ahc_isa.c b/sys/arch/i386/isa/ahc_isa.c
new file mode 100644
index 00000000000..3efe4ec3827
--- /dev/null
+++ b/sys/arch/i386/isa/ahc_isa.c
@@ -0,0 +1,462 @@
+/* $OpenBSD: ahc_isa.c,v 1.1 1996/10/04 02:51:20 deraadt Exp $ */
+/* $NetBSD: ahc_isa.c,v 1.1 1996/08/05 21:14:29 soda Exp $ */
+
+/*
+ * Product specific probe and attach routines for:
+ * 284X VLbus SCSI controllers
+ *
+ * Copyright (c) 1996 Jason R. Thorpe.
+ * All rights reserved.
+ *
+ * Copyright (c) 1995, 1996 Christopher G. Demetriou.
+ * All rights reserved.
+ *
+ * Copyright (c) 1994, 1995, 1996 Justin T. Gibbs.
+ * All rights reserved.
+ *
+ * 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 immediately at the beginning of the file, without modification,
+ * 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 Christopher G. Demetriou
+ * for the NetBSD Project.
+ * 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 AND CONTRIBUTORS ``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 OR CONTRIBUTORS 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.
+ */
+
+/*
+ * This front-end driver is really sort of a hack. The AHA-284X likes
+ * to masquerade as an EISA device. However, on VLbus machines with
+ * no EISA signature in the BIOS, the EISA bus will never be scanned.
+ * This is intended to catch the 284X controllers on those systems
+ * by looking in "EISA i/o space" for 284X controllers.
+ *
+ * This relies heavily on i/o port accounting. We also just use the
+ * EISA macros for everything ... it's a real waste to redefine them.
+ *
+ * Note: there isn't any #ifdef for FreeBSD in this file, since the
+ * FreeBSD EISA driver handles all cases of the 284X.
+ *
+ * -- Jason R. Thorpe <thorpej@NetBSD.ORG>
+ * July 12, 1996
+ *
+ * TODO: some code could be shared with ahc_eisa.c, but it would probably
+ * be a logistical mightmare to even try.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/kernel.h>
+#include <sys/device.h>
+#include <sys/queue.h>
+#include <sys/malloc.h>
+
+#include <machine/bus.h>
+#include <machine/intr.h>
+
+#include <scsi/scsi_all.h>
+#include <scsi/scsiconf.h>
+
+#include <dev/isa/isavar.h>
+
+#include <dev/eisa/eisareg.h>
+#include <dev/eisa/eisavar.h>
+#include <dev/eisa/eisadevs.h>
+
+#include <dev/ic/aic7xxxreg.h>
+#include <dev/ic/aic7xxxvar.h>
+
+/* IO port address setting range as EISA slot number */
+#define AHC_ISA_MIN_SLOT 0x1 /* from iobase = 0x1c00 */
+#define AHC_ISA_MAX_SLOT 0xe /* to iobase = 0xec00 */
+
+#define AHC_ISA_SLOT_OFFSET 0xc00 /* offset from EISA IO space */
+#define AHC_ISA_IOSIZE 0x100
+
+/*
+ * I/O port offsets
+ */
+#define INTDEF 0x5cul /* Interrupt Definition Register */
+#define AHC_ISA_VID (EISA_SLOTOFF_VID - AHC_ISA_SLOT_OFFSET)
+#define AHC_ISA_PID (EISA_SLOTOFF_PID - AHC_ISA_SLOT_OFFSET)
+#define AHC_ISA_PRIMING AHC_ISA_VID /* enable vendor/product ID */
+
+/*
+ * AHC_ISA_PRIMING register values (write)
+ */
+#define AHC_ISA_PRIMING_VID(index) (AHC_ISA_VID + (index))
+#define AHC_ISA_PRIMING_PID(index) (AHC_ISA_PID + (index))
+
+int ahc_isa_irq __P((bus_chipset_tag_t, bus_io_handle_t));
+int ahc_isa_idstring __P((bus_chipset_tag_t, bus_io_handle_t, char *));
+int ahc_isa_match __P((struct isa_attach_args *, bus_io_addr_t));
+
+int ahc_isa_probe __P((struct device *, void *, void *));
+void ahc_isa_attach __P((struct device *, struct device *, void *));
+
+struct cfattach ahc_isa_ca = {
+ sizeof(struct ahc_data), ahc_isa_probe, ahc_isa_attach
+};
+
+/*
+ * This keeps track of which slots are to be checked next if the
+ * iobase locator is a wildcard. A simple static variable isn't enough,
+ * since it's conceivable that a system might have more than one ISA
+ * bus.
+ *
+ * The "bus" member is the unit number of the parent ISA bus, e.g. "0"
+ * for "isa0".
+ */
+struct ahc_isa_slot {
+ LIST_ENTRY(ahc_isa_slot) link;
+ int bus;
+ int slot;
+};
+static LIST_HEAD(, ahc_isa_slot) ahc_isa_all_slots;
+static int ahc_isa_slot_initialized;
+
+/*
+ * Return irq setting of the board, otherwise -1.
+ */
+int
+ahc_isa_irq(bc, ioh)
+ bus_chipset_tag_t bc;
+ bus_io_handle_t ioh;
+{
+ int irq;
+ u_char intdef;
+
+ ahc_reset("ahc_isa", bc, ioh);
+ intdef = bus_io_read_1(bc, ioh, INTDEF);
+ switch (irq = (intdef & 0xf)) {
+ case 9:
+ case 10:
+ case 11:
+ case 12:
+ case 14:
+ case 15:
+ break;
+ default:
+ printf("ahc_isa_irq: illegal irq setting %d\n", intdef);
+ return -1;
+ }
+
+ /* Note that we are going and return (to probe) */
+ return irq;
+}
+
+int
+ahc_isa_idstring(bc, ioh, idstring)
+ bus_chipset_tag_t bc;
+ bus_io_handle_t ioh;
+ char *idstring;
+{
+ u_int8_t vid[EISA_NVIDREGS], pid[EISA_NPIDREGS];
+ int i;
+
+ /* Get the vendor ID bytes */
+ for (i = 0; i < EISA_NVIDREGS; i++) {
+ bus_io_write_1(bc, ioh, AHC_ISA_PRIMING,
+ AHC_ISA_PRIMING_VID(i));
+ vid[i] = bus_io_read_1(bc, ioh, AHC_ISA_VID + i);
+ }
+
+ /* Check for device existence */
+ if (EISA_VENDID_NODEV(vid)) {
+#if 0
+ printf("ahc_isa_idstring: no device at 0x%lx\n",
+ ioh); /* XXX knows about ioh guts */
+ printf("\t(0x%x, 0x%x)\n", vid[0], vid[1]);
+#endif
+ return (0);
+ }
+
+ /* And check that the firmware didn't biff something badly */
+ if (EISA_VENDID_IDDELAY(vid)) {
+ printf("ahc_isa_idstring: BIOS biffed it at 0x%lx\n",
+ ioh); /* XXX knows about ioh guts */
+ return (0);
+ }
+
+ /* Get the product ID bytes */
+ for (i = 0; i < EISA_NPIDREGS; i++) {
+ bus_io_write_1(bc, ioh, AHC_ISA_PRIMING,
+ AHC_ISA_PRIMING_PID(i));
+ pid[i] = bus_io_read_1(bc, ioh, AHC_ISA_PID + i);
+ }
+
+ /* Create the ID string from the vendor and product IDs */
+ idstring[0] = EISA_VENDID_0(vid);
+ idstring[1] = EISA_VENDID_1(vid);
+ idstring[2] = EISA_VENDID_2(vid);
+ idstring[3] = EISA_PRODID_0(pid);
+ idstring[4] = EISA_PRODID_1(pid);
+ idstring[5] = EISA_PRODID_2(pid);
+ idstring[6] = EISA_PRODID_3(pid);
+ idstring[7] = '\0'; /* sanity */
+
+ return (1);
+}
+
+int
+ahc_isa_match(ia, iobase)
+ struct isa_attach_args *ia;
+ bus_io_addr_t iobase;
+{
+ bus_chipset_tag_t bc = ia->ia_bc;
+ bus_io_handle_t ioh;
+ int irq;
+ char idstring[EISA_IDSTRINGLEN];
+
+ /*
+ * Get a mapping for the while slot-specific address
+ * space. If we can't, assume nothing's there, but
+ * warn about it.
+ */
+ if (bus_io_map(bc, iobase, AHC_ISA_IOSIZE, &ioh)) {
+#if 0
+ /*
+ * Don't print anything out here, since this could
+ * be common on machines configured to look for
+ * ahc_eisa and ahc_isa.
+ */
+ printf("ahc_isa_match: can't map I/O space for 0x%x\n",
+ iobase);
+#endif
+ return (0);
+ }
+
+ if (!ahc_isa_idstring(bc, ioh, idstring))
+ irq = -1; /* cannot get the ID string */
+ else if (strcmp(idstring, "ADP7756") &&
+ strcmp(idstring, "ADP7757"))
+ irq = -1; /* unknown ID strings */
+ else
+ irq = ahc_isa_irq(bc, ioh);
+
+ bus_io_unmap(bc, ioh, AHC_ISA_IOSIZE);
+
+ if (irq < 0)
+ return (0);
+
+ if (ia->ia_irq != IRQUNK &&
+ ia->ia_irq != irq) {
+ printf("ahc_isa_match: irq mismatch (kernel %d, card %d)\n",
+ ia->ia_irq, irq);
+ return (0);
+ }
+
+ /* We have a match */
+ ia->ia_iobase = iobase;
+ ia->ia_irq = irq;
+ ia->ia_iosize = AHC_ISA_IOSIZE;
+ ia->ia_msize = 0;
+ return (1);
+}
+
+/*
+ * Check the slots looking for a board we recognise
+ * If we find one, note it's address (slot) and call
+ * the actual probe routine to check it out.
+ */
+int
+ahc_isa_probe(parent, match, aux)
+ struct device *parent;
+ void *match, *aux;
+{
+ struct isa_attach_args *ia = aux;
+ struct ahc_isa_slot *as;
+
+ if (ahc_isa_slot_initialized == 0) {
+ LIST_INIT(&ahc_isa_all_slots);
+ ahc_isa_slot_initialized = 1;
+ }
+
+ if (ia->ia_iobase != IOBASEUNK)
+ return (ahc_isa_match(ia, ia->ia_iobase));
+
+ /*
+ * Find this bus's state. If we don't yet have a slot
+ * marker, allocate and initialize one.
+ */
+ for (as = ahc_isa_all_slots.lh_first; as != NULL;
+ as = as->link.le_next)
+ if (as->bus == parent->dv_unit)
+ goto found_slot_marker;
+
+ /*
+ * Don't have one, so make one.
+ */
+ as = (struct ahc_isa_slot *)
+ malloc(sizeof(struct ahc_isa_slot), M_DEVBUF, M_NOWAIT);
+ if (as == NULL)
+ panic("ahc_isa_probe: can't allocate slot marker");
+
+ as->bus = parent->dv_unit;
+ as->slot = AHC_ISA_MIN_SLOT;
+ LIST_INSERT_HEAD(&ahc_isa_all_slots, as, link);
+
+ found_slot_marker:
+
+ for (; as->slot <= AHC_ISA_MAX_SLOT; as->slot++) {
+ if (ahc_isa_match(ia, EISA_SLOT_ADDR(as->slot) +
+ AHC_ISA_SLOT_OFFSET)) {
+ as->slot++; /* next slot to search */
+ return (1);
+ }
+ }
+
+ /* No matching cards were found. */
+ return (0);
+}
+
+void
+ahc_isa_attach(parent, self, aux)
+ struct device *parent, *self;
+ void *aux;
+{
+ ahc_type type;
+ struct ahc_data *ahc = (void *)self;
+ struct isa_attach_args *ia = aux;
+ bus_chipset_tag_t bc = ia->ia_bc;
+ bus_io_handle_t ioh;
+ int irq;
+ char idstring[EISA_IDSTRINGLEN];
+ const char *model;
+
+ if (bus_io_map(bc, ia->ia_iobase, ia->ia_iosize, &ioh))
+ panic("ahc_isa_attach: could not map slot I/O addresses");
+ if (!ahc_isa_idstring(bc, ioh, idstring))
+ panic("ahc_isa_attach: could not read ID string");
+ if ((irq = ahc_isa_irq(bc, ioh)) < 0)
+ panic("ahc_isa_attach: ahc_isa_irq failed!");
+
+ if (strcmp(idstring, "ADP7756") == 0) {
+ model = EISA_PRODUCT_ADP7756;
+ type = AHC_284;
+ } else if (strcmp(idstring, "ADP7757") == 0) {
+ model = EISA_PRODUCT_ADP7757;
+ type = AHC_284;
+ } else {
+ panic("ahc_isa_attach: Unknown device type %s\n", idstring);
+ }
+ printf(": %s\n", model);
+
+ ahc_construct(ahc, bc, ioh, type, AHC_FNONE);
+
+ /*
+ * Tell the user what type of interrupts we're using.
+ * usefull for debugging irq problems
+ */
+ printf( "%s: Using %s Interrupts\n", ahc_name(ahc),
+ ahc->pause & IRQMS ? "Level Sensitive" : "Edge Triggered");
+
+ /*
+ * Now that we know we own the resources we need, do the
+ * card initialization.
+ *
+ * First, the aic7770 card specific setup.
+ */
+
+ /* XXX
+ * On AHA-284x,
+ * all values are automagically intialized at
+ * POST for these cards, so we can always rely
+ * on the Scratch Ram values. However, we should
+ * read the SEEPROM here (Dan has the code to do
+ * it) so we can say what kind of translation the
+ * BIOS is using. Printing out the geometry could
+ * save a lot of users the grief of failed installs.
+ */
+
+ /*
+ * See if we have a Rev E or higher aic7770. Anything below a
+ * Rev E will have a R/O autoflush disable configuration bit.
+ * It's still not clear exactly what is differenent about the Rev E.
+ * We think it allows 8 bit entries in the QOUTFIFO to support
+ * "paging" SCBs so you can have more than 4 commands active at
+ * once.
+ */
+ {
+ char *id_string;
+ u_char sblkctl;
+ u_char sblkctl_orig;
+
+ sblkctl_orig = AHC_INB(ahc, SBLKCTL);
+ sblkctl = sblkctl_orig ^ AUTOFLUSHDIS;
+ AHC_OUTB(ahc, SBLKCTL, sblkctl);
+ sblkctl = AHC_INB(ahc, SBLKCTL);
+ if(sblkctl != sblkctl_orig)
+ {
+ id_string = "aic7770 >= Rev E, ";
+ /*
+ * Ensure autoflush is enabled
+ */
+ sblkctl &= ~AUTOFLUSHDIS;
+ AHC_OUTB(ahc, SBLKCTL, sblkctl);
+
+ /* Allow paging on this adapter */
+ ahc->flags |= AHC_PAGESCBS;
+ }
+ else
+ id_string = "aic7770 <= Rev C, ";
+
+ printf("%s: %s", ahc_name(ahc), id_string);
+ }
+
+ /* Setup the FIFO threshold and the bus off time */
+ {
+ u_char hostconf = AHC_INB(ahc, HOSTCONF);
+ AHC_OUTB(ahc, BUSSPD, hostconf & DFTHRSH);
+ AHC_OUTB(ahc, BUSTIME, (hostconf << 2) & BOFF);
+ }
+
+ /*
+ * Generic aic7xxx initialization.
+ */
+ if(ahc_init(ahc)){
+ ahc_free(ahc);
+ return;
+ }
+
+ /*
+ * Enable the board's BUS drivers
+ */
+ AHC_OUTB(ahc, BCTL, ENABLE);
+
+ /*
+ * The IRQMS bit enables level sensitive interrupts only allow
+ * IRQ sharing if its set.
+ */
+ ahc->sc_ih = isa_intr_establish(ia->ia_ic, irq,
+ ahc->pause & IRQMS ? IST_LEVEL : IST_EDGE, IPL_BIO, ahc_intr,
+ ahc, ahc->sc_dev.dv_xname);
+ if (ahc->sc_ih == NULL) {
+ printf("%s: couldn't establish interrupt\n",
+ ahc->sc_dev.dv_xname);
+ ahc_free(ahc);
+ return;
+ }
+
+ /* Attach sub-devices - always succeeds */
+ ahc_attach(ahc);
+}