summaryrefslogtreecommitdiff
path: root/sys/dev/isa/pcmcia_pcic.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/isa/pcmcia_pcic.c')
-rw-r--r--sys/dev/isa/pcmcia_pcic.c553
1 files changed, 440 insertions, 113 deletions
diff --git a/sys/dev/isa/pcmcia_pcic.c b/sys/dev/isa/pcmcia_pcic.c
index 8e8f16f1594..dfcf2ea4a81 100644
--- a/sys/dev/isa/pcmcia_pcic.c
+++ b/sys/dev/isa/pcmcia_pcic.c
@@ -1,3 +1,32 @@
+/* $Id: pcmcia_pcic.c,v 1.5 1996/04/29 14:16:53 hvozda Exp $ */
+/*
+ * Copyright (c) 1995, 1996 John T. Kohl
+ * 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, 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. 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.
+ *
+ */
/*
* Device Driver for Intel 82365 based pcmcia slots
*
@@ -20,13 +49,16 @@
#include <sys/device.h>
#include <sys/proc.h>
#include <sys/user.h>
+#include <sys/cpu.h>
#include <machine/pio.h>
#include <dev/isa/isavar.h>
#include <dev/ic/i82365reg.h>
-#include <dev/pcmcia/pcmciabus.h>
+#include <dev/pcmcia/pcmciavar.h>
+#include <dev/pcmcia/pcmciareg.h>
+
#ifdef IBM_WD
#define PCIC_DEBUG 0xf
#endif
@@ -36,13 +68,13 @@
#define PCDINTR 0x04
#define PCDSERV 0x08
#define PCDRW 0x10
+#define PCDCONF 0x20
int pcic_debug = PCIC_DEBUG;
#define DEBUG(a) (pcic_debug & (a))
#else
#define DEBUG(a) (0)
#endif
-
/*
* pcic_softc: per line info and status
*/
@@ -69,20 +101,26 @@ struct slot {
struct pcmcia_link *link;
struct pcic_softc *chip;
};
+
struct pcic_softc {
- struct device sc_dev;
+ struct device sc_dev;
+ bus_chipset_tag_t sc_bc;
+ struct pcmcia_adapter sc_adapter;
void *sc_ih;
int sc_polltimo;
int sc_pcic_irq;
- u_short pcic_base; /* base port for each board */
- u_char slot_id;
- u_char chip_inf;
- struct slot slot[2];
-} pcic_softc[4];
+ bus_io_handle_t sc_ioh;
+ bus_mem_handle_t sc_memh;
+ u_short pcic_base; /* base port for each board */
+ u_char chip_inf;
+ struct slot slot[4]; /* treat up to 4 as on the same pcic */
+};
+#define pcic_parent(sc) ((struct pcicmaster_softc *)(sc)->sc_dev.dv_parent)
static int pcic_map_io __P((struct pcmcia_link *, u_int, u_int, int));
-static int pcic_map_mem __P((struct pcmcia_link *, caddr_t,
+static int pcic_map_mem __P((struct pcmcia_link *, bus_chipset_tag_t,
+ bus_mem_handle_t,
u_int, u_int, int));
static int pcic_map_intr __P((struct pcmcia_link *, int, int));
static int pcic_service __P((struct pcmcia_link *, int, void *, int));
@@ -94,15 +132,53 @@ static struct pcmcia_funcs pcic_funcs = {
pcic_service
};
-int pcicprobe __P((struct device *, void *, void *));
-void pcicattach __P((struct device *, struct device *, void *));
+int pcic_probe __P((struct device *, void *, void *));
+void pcic_attach __P((struct device *, struct device *, void *));
+int pcic_print __P((void *, char *));
+
+int pcicmaster_probe __P((struct device *, void *, void *));
+void pcicmaster_attach __P((struct device *, struct device *, void *));
+int pcicmaster_print __P((void *, char *));
extern struct pcmciabus_link pcmcia_isa_link;
-struct cfdriver pciccd = {
- NULL, "pcic", pcicprobe, pcicattach, DV_DULL, sizeof(struct pcic_softc)
+struct cfattach pcic_ca = {
+ sizeof(struct pcic_softc), pcic_probe, pcic_attach,
};
+struct cfdriver pcic_cd = {
+ NULL, "pcic", DV_DULL
+};
+
+struct pcicmaster_softc {
+ struct device sc_dev;
+ bus_chipset_tag_t sc_bc;
+ bus_io_handle_t sc_ioh;
+ struct pcic_softc *sc_ctlrs[2];
+ char sc_slavestate[2];
+#define SLAVE_NOTPRESENT 0
+#define SLAVE_FOUND 1
+#define SLAVE_CONFIGURED 2
+};
+
+struct cfattach pcicmaster_ca = {
+ sizeof(struct pcicmaster_softc), pcicmaster_probe, pcicmaster_attach,
+};
+
+struct cfdriver pcicmaster_cd = {
+ NULL, "pcicmaster", DV_DULL, 1
+};
+
+struct pcic_attach_args {
+ int pia_ctlr; /* pcic ctlr number */
+ bus_chipset_tag_t pia_bc; /* bus chipset tag */
+ bus_io_handle_t pia_ioh; /* base i/o address */
+ int pia_iosize; /* span of ports used */
+ int pia_irq; /* interrupt request */
+ int pia_drq; /* DMA request */
+ int pia_maddr; /* physical i/o mem addr */
+ u_int pia_msize; /* size of i/o memory */
+};
static u_char pcic_rd __P((struct slot *, int));
static void pcic_wr __P((struct slot *, int, int));
@@ -114,12 +190,13 @@ pcic_rd(slot, reg)
int reg;
{
u_char res;
+ bus_chipset_tag_t bc = slot->chip->sc_bc;
+ bus_io_handle_t ioh = slot->chip->sc_ioh;
if (DEBUG(PCDRW))
- printf("pcic_rd(%x [%x %x]) = ", reg, slot->reg_off,
- slot->chip->pcic_base);
- outb(slot->chip->pcic_base, slot->reg_off + reg);
+ printf("pcic_rd(%x [%x %x]) = ", reg, slot->reg_off, ioh);
+ bus_io_write_1(bc, ioh, 0, slot->reg_off + reg);
delay(1);
- res = inb(slot->chip->pcic_base + 1);
+ res = bus_io_read_1(bc, ioh, 1);
if (DEBUG(PCDRW))
printf("%x\n", res);
return res;
@@ -130,15 +207,17 @@ pcic_wr(slot, reg, val)
struct slot *slot;
int reg, val;
{
- outb(slot->chip->pcic_base, slot->reg_off + reg);
+ bus_chipset_tag_t bc = slot->chip->sc_bc;
+ bus_io_handle_t ioh = slot->chip->sc_ioh;
+ bus_io_write_1(bc, ioh, 0, slot->reg_off + reg);
delay(1);
- outb(slot->chip->pcic_base + 1, val);
+ bus_io_write_1(bc, ioh, 1, val);
if (DEBUG(PCDRW)) {
int res;
delay(1);
- outb(slot->chip->pcic_base, slot->reg_off + reg);
+ bus_io_write_1(bc, ioh, 0, slot->reg_off + reg);
delay(1);
- res = inb(slot->chip->pcic_base + 1);
+ res = bus_io_read_1(bc, ioh, 1);
printf("pcic_wr(%x %x) = %x\n", reg, val, res);
}
}
@@ -154,121 +233,214 @@ pcic_wait(slot, i)
}
int
-pcicprobe(parent, self, aux)
+pcic_probe(parent, self, aux)
struct device *parent;
void *self;
void *aux;
{
- struct pcic_softc *pcic = (void *) self;
- struct isa_attach_args *ia = aux;
- struct cfdata *cf = pcic->sc_dev.dv_cfdata;
- u_int chip_inf = 0;
- int i;
+ struct pcic_softc *pcic = self;
+ struct pcicmaster_softc *pcicm = (struct pcicmaster_softc *) parent;
+ struct pcic_attach_args *pia = aux;
+ bus_mem_handle_t memh;
+ u_int chip_inf = 0, ochip_inf = 0;
+ int first = 1;
+ int i, j, maxslot;
- pcic->pcic_base = ia->ia_iobase;
- pcic->slot_id = 0; /* XXX */
bzero(pcic->slot, sizeof(pcic->slot));
- pcic->slot[0].chip = pcic;
- pcic->slot[0].reg_off = (pcic->slot_id & 1) * 0x80;
- pcic->slot[1].chip = pcic;
- pcic->slot[1].reg_off = ((pcic->slot_id & 1) * 0x80) + 0x40;
- chip_inf = pcic_rd(&pcic->slot[0], PCIC_ID_REV);
- switch (chip_inf) {
- case PCIC_INTEL0:
+
+ if (DEBUG(PCDCONF)) {
+ printf("pcic_probe controller %d unit %d\n", pia->pia_ctlr,
+ pcic->sc_dev.dv_unit);
+ delay(2000000);
+ }
+ if (pcicm->sc_slavestate[pia->pia_ctlr] != SLAVE_FOUND)
+ return 0;
+ if (pcic->sc_dev.dv_cfdata->cf_loc[1] == -1 ||
+ pcic->sc_dev.dv_cfdata->cf_loc[2] == 0)
+ return 0;
+
+ /*
+ * select register offsets based on which controller we are.
+ * 2 pcic controllers (w/ 2 slots each) possible at each
+ * IO port location, for a total of 8 possible PCMCIA slots.
+ *
+ * for VLSI controllers, we probe up to 4 slots for the same chip type,
+ * and handle them on one controller. This is slightly
+ * cheating (two separate pcic's are required for 4 slots, according
+ * to the i82365 spec).
+ *
+ * For other controllers, we only take up to 2 slots.
+ */
+ pcic->sc_ioh = pia->pia_ioh;
+ pcic->sc_bc = pia->pia_bc;
+ pcic->sc_adapter.nslots = 0;
+ maxslot = 2;
+ for (i = j = 0; i < maxslot; i++) {
+ pcic->slot[j].reg_off = 0x80 * pia->pia_ctlr + 0x40 * i;
+ pcic->slot[j].chip = pcic;
+
+ chip_inf = pcic_rd(&pcic->slot[j], PCIC_ID_REV);
+ if (DEBUG(PCDCONF)) {
+ printf("pcic_probe read info %x\n", chip_inf);
+ delay(2000000);
+ }
+ if (!first && ochip_inf != chip_inf)
+ continue; /* don't attach, it's different */
+ ochip_inf = chip_inf;
+ switch (chip_inf) {
+ case PCIC_INTEL0:
pcic->chip_inf = PCMICA_CHIP_82365_0;
goto ok;
- case PCIC_INTEL1:
+ case PCIC_INTEL1:
pcic->chip_inf = PCMICA_CHIP_82365_1;
goto ok;
- case PCIC_IBM1:
+ case PCIC_IBM1:
pcic->chip_inf = PCMICA_CHIP_IBM_1;
goto ok;
- case PCIC_IBM2:
+ case PCIC_146FC6:
+ pcic->chip_inf = PCMICA_CHIP_146FC6;
+ maxslot = 4;
+ goto ok;
+ case PCIC_146FC7:
+ pcic->chip_inf = PCMICA_CHIP_146FC7;
+ maxslot = 4;
+ goto ok;
+ case PCIC_IBM2:
pcic->chip_inf = PCMICA_CHIP_IBM_2;
-ok:
- ia->ia_msize = 0;
- ia->ia_iosize = 2;
- pcmcia_register(pcic, &pcmcia_isa_link, &pcic_funcs,
- pcic->slot_id);
+ ok:
+ if (first) {
+ pcic->sc_adapter.adapter_softc = (void *)pcic;
+ pcic->sc_adapter.chip_link = &pcic_funcs;
+ pcic->sc_adapter.bus_link = &pcmcia_isa_link;
+ pcicm->sc_ctlrs[pia->pia_ctlr] = pcic;
+ pcicm->sc_slavestate[pia->pia_ctlr] = SLAVE_CONFIGURED;
+ first = 0;
+ }
+ pcic->sc_adapter.nslots++;
+ j++;
+ default:
+ if (DEBUG(PCDCONF)) {
+ printf("found ID %x at pcic%d position\n",
+ chip_inf & 0xff, pcic->sc_dev.dv_unit);
+ }
+ continue;
+ }
+ }
+ if (pcic->sc_adapter.nslots != 0) {
+ pcic->sc_memh = memh;
return 1;
- default:
- printf("found ID %x at pcic position\n", chip_inf & 0xff);
- break;
}
- /* reset mappings .... */
- pcic_wr(&pcic->slot[0], PCIC_POWER, pcic->slot[0].pow=PCIC_DISRST);
- pcic_wr(&pcic->slot[1], PCIC_POWER, pcic->slot[1].pow=PCIC_DISRST);
-
- delay(1000);
-
- for (i = PCIC_INT_GEN; i < 0x40; i++) {
- pcic_wr(&pcic->slot[0], i, 0);
- pcic_wr(&pcic->slot[1], i, 0);
+ if (DEBUG(PCDCONF)) {
+ printf("pcic_probe failed\n");
+ delay(2000000);
}
- delay(10000);
+ bus_mem_unmap(pia->pia_bc, memh, pia->pia_msize);
return 0;
}
int
pcic_intr __P((void *));
+int
+pcic_print(aux, name)
+ void *aux;
+ char *name;
+{
+ if (name != NULL)
+ printf("%s: pcmciabus ", name);
+ return UNCONF;
+}
void
-pcicattach(parent, self, aux)
+pcic_attach(parent, self, aux)
struct device *parent, *self;
void *aux;
{
struct pcic_softc *pcic = (void *) self;
- struct isa_attach_args *ia = aux;
+ struct pcic_attach_args *pia = aux;
+ struct pcmciabus_attach_args pba;
struct slot *slot;
int i;
static char *pcic_names[] = {
"Intel 82365sl Rev. 0",
"Intel 82365sl Rev. 1",
"IBM 82365sl clone Rev. 1",
- "IBM 82365sl clone Rev. 2"};
- printf(": %s slots %d-%d (%x %x)\n", pcic_names[pcic->chip_inf -
- PCMICA_CHIP_82365_0], pcic->slot_id * 2, pcic->slot_id * 2 + 1,
- &pcic->slot[0], &pcic->slot[1]);
+ "IBM 82365sl clone Rev. 2",
+ "VL82146 (82365sl clone) Rev. 6",
+ "VL82146 (82365sl clone) Rev. 7" };
+ if (DEBUG(PCDCONF)) {
+ printf("pcic_attach found\n");
+ delay(2000000);
+ }
+ pia->pia_irq = self->dv_cfdata->cf_loc[0];
+ pia->pia_maddr = self->dv_cfdata->cf_loc[1];
+ pia->pia_msize = self->dv_cfdata->cf_loc[2];
+
+ printf(": %s slots %d-%d iomem %x-%x",
+ pcic_names[pcic->chip_inf - PCMICA_CHIP_82365_0],
+ pcic->sc_dev.dv_unit * 2,
+ pcic->sc_dev.dv_unit * 2 + pcic->sc_adapter.nslots - 1,
+ pia->pia_maddr, pia->pia_maddr + pia->pia_msize - 1);
+ if (pia->pia_irq != IRQUNK)
+ printf(" irq %d\n", pia->pia_irq);
+ else
+ printf("\n");
+ if (DEBUG(PCDCONF))
+ delay(2000000);
+
+#ifdef PCMCIA_ISA_DEBUG
+ printf("pcic %p slots %p,%p\nisaaddr %p ports %x size %d irq %d drq %d maddr %x msize %x\n",
+ pcic, &pcic->slot[0], &pcic->slot[1],
+ pia, pia->pia_ioh, pia->pia_iosize,
+ pia->pia_irq, pia->pia_drq, pia->pia_maddr, pia->pia_msize);
+ if (DEBUG(PCDCONF))
+ delay(2000000);
+#endif
+
/* enable interrupts on events */
- if (ia->ia_irq != IRQUNK)
- pcic->sc_pcic_irq = ia->ia_irq;
+ if (pia->pia_irq != IRQUNK)
+ pcic->sc_pcic_irq = pia->pia_irq;
else
pcic->sc_pcic_irq = 0;
- for (i = 0; i < 2; i++) {
- slot = &pcic->slot[i];
- slot->irq = pcic->sc_pcic_irq | PCIC_INTR_ENA;
- pcic_wr(slot, PCIC_STAT_INT,
- (pcic->sc_pcic_irq << 4) |PCIC_CDTCH | PCIC_STCH);
- pcic_wr(&pcic->slot[i], PCIC_INT_GEN, slot->irq);
- (void) pcic_rd(&pcic->slot[i], PCIC_STAT_CHG);
+ for (i = 0; i < pcic->sc_adapter.nslots; i++) {
+ slot = &pcic->slot[i];
+ /*
+ * Arrange for card status change interrupts
+ * to be steered to specified IRQ.
+ * Treat all cards as I/O cards for the moment so we get
+ * sensible card change interrupt codes (besides, we don't
+ * support memory cards :)
+ */
+ pcic_wr(slot, PCIC_STAT_INT,
+ (pcic->sc_pcic_irq << 4) |
+ PCIC_CDTCH | PCIC_IOCARD);
+ slot->irq = pcic_rd(slot, PCIC_INT_GEN) & ~PCIC_INTR_ENA;
+ pcic_wr(slot, PCIC_INT_GEN, slot->irq);
+ (void) pcic_rd(slot, PCIC_STAT_CHG);
}
- if (ia->ia_irq == IRQUNK) {
+ if (pia->pia_irq == IRQUNK) {
pcic->sc_polltimo = hz/2;
timeout((void (*)(void *))pcic_intr, pcic, pcic->sc_polltimo);
} else {
- pcic->sc_ih = isa_intr_establish(ia->ia_irq, IST_EDGE,
- IPL_NET, pcic_intr, pcic, pcic->sc_dev.dv_xname);
+ pcic->sc_ih = isa_intr_establish(pia->pia_bc,
+ pia->pia_irq, IST_EDGE,
+ IPL_PCMCIA, pcic_intr, pcic, pcic->sc_dev.dv_xname);
pcic->sc_polltimo = 0;
}
-}
-
-#ifdef DDB
-int pcic_intr_test(slot)
-struct slot *slot;
-{
- printf("CSC interrupt state: %x\n", pcic_rd(slot, PCIC_STAT_INT));
- printf("General interrupt state: %x\n", pcic_rd(slot, PCIC_INT_GEN));
-}
-
-int pcic_intr_set(slot)
-struct slot *slot;
-{
- pcic_wr(slot, PCIC_INT_GEN, pcic_rd(slot, PCIC_INT_GEN)|PCIC_INTR_ENA);
- pcic_intr_test(slot);
-}
+ /*
+ * Probe the pcmciabus at this controller.
+ */
+ pba.pba_bc = pia->pia_bc;
+ pba.pba_maddr = pia->pia_maddr;
+ pba.pba_msize = pia->pia_msize;
+ pba.pba_aux = &pcic->sc_adapter;
+#ifdef PCMCIA_DEBUG
+ printf("config_found(%p, %p, %p)\n",
+ self, &pba, pcic_print);
#endif
+ config_found(self, (void *)&pba, pcic_print);
+}
int
pcic_intr(arg)
@@ -278,14 +450,17 @@ void *arg;
u_char statchg, intgen;
register int i;
+#ifdef PCMCIA_PCIC_DEBUG
if (pcic->sc_polltimo == 0)
printf("%s: interrupt:", pcic->sc_dev.dv_xname);
- for (i = 0; i < 2; i++) {
+#endif
+ for (i = 0; i < pcic->sc_adapter.nslots; i++) {
struct pcmcia_link *link = pcic->slot[i].link;
statchg = pcic_rd(&pcic->slot[i], PCIC_STAT_CHG);
if (statchg == 0)
continue;
intgen = pcic_rd(&pcic->slot[i], PCIC_INT_GEN);
+#ifdef PCMCIA_PCIC_DEBUG
if (intgen & PCIC_IOCARD) {
printf("%s: slot %d iocard status %s%s\n",
pcic->sc_dev.dv_xname, i,
@@ -295,6 +470,7 @@ void *arg;
printf("%s: slot %d memcard status %x\n",
pcic->sc_dev.dv_xname, i, statchg);
}
+#endif
if ((statchg & PCIC_CDTCH) &&
(link->flags & PCMCIA_SLOT_OPEN) == 0) {
#if 0
@@ -330,7 +506,10 @@ pcic_map_io(link, start, len, flags)
int flags;
{
struct pcic_softc *sc = link->adapter->adapter_softc;
- struct slot *slot = &sc->slot[link->slot & 1];
+ struct slot *slot;
+ if (link->slot >= sc->sc_adapter.nslots)
+ return ENXIO;
+ slot = &sc->slot[link->slot];
len--;
if (DEBUG(PCDIO)) {
@@ -403,10 +582,8 @@ pcic_map_io(link, start, len, flags)
delay(1000);
return 0;
} else {
- u_int stop;
int window;
int winid;
- int ioflags;
if (flags & PCMCIA_LAST_WIN) {
window = MAX_IOSECTION - 1;
} else if (flags & PCMCIA_FIRST_WIN) {
@@ -440,26 +617,33 @@ pcic_map_io(link, start, len, flags)
slot->io_addr[window] = start;
slot->io_len[window] = len;
+ return 0;
}
-
}
+
static int
-pcic_map_mem(link, haddr, start, len, flags)
+pcic_map_mem(link, bc, ioh, start, len, flags)
struct pcmcia_link *link;
- caddr_t haddr;
+ bus_chipset_tag_t bc;
+ bus_mem_handle_t ioh;
u_int start, len;
int flags;
{
- struct pcic_softc *sc = link->adapter->adapter_softc;
- struct slot *slot = &sc->slot[link->slot & 1];
vm_offset_t physaddr;
+ struct pcic_softc *sc = link->adapter->adapter_softc;
+ struct slot *slot;
+ caddr_t haddr = ioh; /* XXX */
+ if (link->slot >= sc->sc_adapter.nslots)
+ return ENXIO;
+
+ slot = &sc->slot[link->slot];
if (flags & PCMCIA_PHYSICAL_ADDR)
physaddr = (vm_offset_t) haddr;
else
physaddr = pmap_extract(pmap_kernel(), (vm_offset_t) haddr);
if (DEBUG(PCDMEM))
- printf("pcic_map_mem %x %x %x %x %x\n", haddr, physaddr,
+ printf("pcic_map_mem %p %lx %x %x %x\n", haddr, physaddr,
start, len, flags);
(u_long) physaddr >>= 12;
@@ -491,7 +675,7 @@ pcic_map_mem(link, haddr, start, len, flags)
offs = (start - (u_long) physaddr) & 0x3fff;
if (DEBUG(PCDMEM))
- printf("mapmem 2:%x %x %x\n", offs, physaddr + offs,
+ printf("mapmem 2:%x %lx %x\n", offs, physaddr + offs,
start);
stop = (u_long) physaddr + len;
@@ -503,7 +687,7 @@ pcic_map_mem(link, haddr, start, len, flags)
(u_long) physaddr & 0xff);
pcic_wr(slot, winid | PCIC_START | PCIC_ADDR_HIGH,
(((u_long) physaddr >> 8) & 0x3f) |
- /* PCIC_ZEROWS|/* */
+ /* PCIC_ZEROWS| */
((flags & PCMCIA_MAP_16) ? PCIC_DATA16 : 0));
pcic_wr(slot, winid | PCIC_END | PCIC_ADDR_LOW,
@@ -527,8 +711,6 @@ pcic_map_mem(link, haddr, start, len, flags)
delay(1000);
return 0;
} else {
- u_int offs;
- u_int stop;
int window;
int winid;
@@ -571,20 +753,24 @@ pcic_map_mem(link, haddr, start, len, flags)
return 0;
}
}
+
static int
pcic_map_intr(link, irq, flags)
struct pcmcia_link *link;
int irq, flags;
{
struct pcic_softc *sc = link->adapter->adapter_softc;
- struct slot *slot = &sc->slot[link->slot & 1];
+ struct slot *slot;
+ if (link->slot >= sc->sc_adapter.nslots)
+ return ENXIO;
+
+ slot = &sc->slot[link->slot];
if (DEBUG(PCDINTR))
printf("pcic_map_intr %x %x\n", irq, flags);
if (flags & PCMCIA_UNMAP) {
- slot->irq &= ~PCIC_INT_MASK;
- slot->irq |= sc->sc_pcic_irq | PCIC_INTR_ENA;
+ slot->irq &= ~(PCIC_INT_MASK|PCIC_INTR_ENA);
pcic_wr(slot, PCIC_INT_GEN, slot->irq);
}
else {
@@ -592,8 +778,8 @@ pcic_map_intr(link, irq, flags)
return EINVAL;
if(irq==2)
irq=9;
- slot->irq = (slot->irq & PCIC_INT_FLAGMASK) |
- irq | PCIC_INTR_ENA;
+ slot->irq &= ~(PCIC_INTR_ENA|PCIC_INT_MASK);
+ slot->irq |= irq | PCIC_CARDRESET; /* reset is inverted */
pcic_wr(slot, PCIC_INT_GEN, slot->irq);
}
return 0;
@@ -608,7 +794,11 @@ pcic_service(link, opcode, arg, flags)
int flags;
{
struct pcic_softc *sc = link->adapter->adapter_softc;
- struct slot *slot = &sc->slot[link->slot & 1];
+ struct slot *slot;
+ if (link->slot >= sc->sc_adapter.nslots)
+ return ENXIO;
+
+ slot = &sc->slot[link->slot];
slot->link = link; /* save it for later :) */
switch (opcode) {
@@ -656,7 +846,10 @@ pcic_service(link, opcode, arg, flags)
if (DEBUG(PCDSERV))
printf("pcic_service(reset)\n");
- slot->irq |= flags ? PCIC_IOCARD : 0;
+ if (flags)
+ slot->irq |= PCIC_IOCARD;
+ else
+ slot->irq &= ~PCIC_IOCARD; /* XXX? */
pcic_wr(slot, PCIC_POWER, slot->pow &= ~PCIC_DISRST);
slot->irq &= ~PCIC_CARDRESET;
pcic_wr(slot, PCIC_INT_GEN, slot->irq);
@@ -683,7 +876,8 @@ pcic_service(link, opcode, arg, flags)
printf("pcic_service(power): ");
if (flags & PCMCIA_POWER_ON) {
int nv = (PCIC_DISRST|PCIC_OUTENA);
- pcic_wr(slot, PCIC_INT_GEN, slot->irq = 0);
+ pcic_wr(slot, PCIC_INT_GEN,
+ slot->irq = PCIC_IOCARD);
if(flags & PCMCIA_POWER_3V)
nv |= PCIC_VCC3V;
if(flags & PCMCIA_POWER_5V)
@@ -731,3 +925,136 @@ pcic_service(link, opcode, arg, flags)
return EINVAL;
}
}
+
+/*
+ * Handle I/O space mapping for children. Thin layer.
+ */
+int
+pcicmaster_probe(parent, self, aux)
+ struct device *parent;
+ void *self;
+ void *aux;
+{
+ struct pcicmaster_softc *pcicm = self;
+ struct isa_attach_args *ia = aux;
+ struct cfdata *cf = pcicm->sc_dev.dv_cfdata;
+ bus_chipset_tag_t bc;
+ bus_io_handle_t ioh;
+
+ u_int chip_inf = 0;
+ int i, j;
+ int rval = 0;
+ struct pcic_softc pcic; /* faked up for probing only */
+
+ if (DEBUG(PCDCONF)) {
+ printf("pcicmaster_probe\n");
+ delay(2000000);
+ }
+ bc = ia->ia_bc;
+ if (bus_io_map(bc, ia->ia_iobase, PCIC_NPORTS, &ioh))
+ return (0);
+ /*
+ * Probe the slots for each of the possible child controllers,
+ * and if any are there we return a positive indication.
+ */
+ pcic.sc_ioh = ioh;
+ for (i = 0; i < 2; i++) {
+ bzero(pcic.slot, sizeof(pcic.slot));
+ pcic.slot[0].chip = &pcic;
+ pcic.slot[0].reg_off = i * 0x80;
+ chip_inf = pcic_rd(&pcic.slot[0], PCIC_ID_REV);
+ switch (chip_inf) {
+ case PCIC_INTEL0:
+ case PCIC_INTEL1:
+ case PCIC_IBM1:
+ case PCIC_146FC6:
+ case PCIC_146FC7:
+ case PCIC_IBM2:
+ if (DEBUG(PCDCONF)) {
+ printf("pcicmaster_probe found, cf=%p\n", cf);
+ delay(2000000);
+ }
+ pcicm->sc_slavestate[i] = SLAVE_FOUND;
+ rval++;
+ break;
+ default:
+ pcicm->sc_slavestate[i] = SLAVE_NOTPRESENT;
+ if (DEBUG(PCDCONF)) {
+ printf("found ID %x at slave %d\n",
+ chip_inf & 0xff, i);
+ }
+ break;
+ }
+ if (pcicm->sc_slavestate[i] != SLAVE_FOUND) {
+ /* reset mappings .... */
+ pcic_wr(&pcic.slot[0], PCIC_POWER,
+ pcic.slot[0].pow=PCIC_DISRST);
+ delay(1000);
+ for (j = PCIC_INT_GEN; j < 0x40; j++) {
+ pcic_wr(&pcic.slot[0], j, 0);
+ }
+ delay(10000);
+ }
+ }
+ if (rval) {
+ ia->ia_iosize = 2;
+ pcicm->sc_bc = bc;
+ pcicm->sc_ioh = ioh;
+ } else
+ bus_io_unmap(bc, ioh, PCIC_NPORTS);
+ return rval;
+}
+
+void
+pcicmaster_attach(parent, self, aux)
+ struct device *parent, *self;
+ void *aux;
+{
+ struct pcicmaster_softc *pcicm = (void *) self;
+ struct isa_attach_args *ia = aux;
+ struct pcic_attach_args pia;
+ int i;
+ printf("\n");
+ if (DEBUG(PCDCONF)) {
+ printf("pcicmaster_attach\n");
+ delay(2000000);
+ }
+#ifdef PCMCIA_ISA_DEBUG
+ printf("pcicm %p isaaddr %p ports %x size %d irq %d drq %d maddr %x msize %x\n",
+ pcicm, ia, ia->ia_iobase, ia->ia_iosize,
+ ia->ia_irq, ia->ia_drq, ia->ia_maddr, ia->ia_msize);
+ if (DEBUG(PCDCONF))
+ delay(2000000);
+#endif
+ /* attach up to two PCICs at this I/O address */
+ for (i = 0; i < 2; i++) {
+ if (pcicm->sc_slavestate[i] == SLAVE_FOUND) {
+ pia.pia_ctlr = i;
+ /*
+ * share the I/O space and memory mapping space.
+ */
+ pia.pia_bc = pcicm->sc_bc;
+ pia.pia_ioh = pcicm->sc_ioh;
+ pia.pia_iosize = ia->ia_iosize;
+ pia.pia_drq = ia->ia_drq;
+#if 0
+ pia.pia_irq = ia->ia_irq;
+ pia.pia_irq = cf->cf_loc[0]; /* irq from master attach */
+ pia.pia_maddr = ia->ia_maddr + (ia->ia_msize / 2) * i;
+ pia.pia_msize = ia->ia_msize / 2;
+#endif
+
+ config_found(self, &pia, pcicmaster_print);
+ }
+ }
+}
+
+int
+pcicmaster_print(aux, name)
+ void *aux;
+ char *name;
+{
+ if (name != NULL)
+ printf("%s: master controller ", name);
+ return UNCONF;
+}