diff options
Diffstat (limited to 'sys/dev/ic')
-rw-r--r-- | sys/dev/ic/i82365.c | 227 | ||||
-rw-r--r-- | sys/dev/ic/i82365var.h | 22 |
2 files changed, 224 insertions, 25 deletions
diff --git a/sys/dev/ic/i82365.c b/sys/dev/ic/i82365.c index 159614cbd63..be36072bc36 100644 --- a/sys/dev/ic/i82365.c +++ b/sys/dev/ic/i82365.c @@ -1,4 +1,4 @@ -/* $OpenBSD: i82365.c,v 1.8 1999/07/26 05:43:15 deraadt Exp $ */ +/* $OpenBSD: i82365.c,v 1.9 1999/08/08 01:07:02 niklas Exp $ */ /* $NetBSD: i82365.c,v 1.10 1998/06/09 07:36:55 thorpej Exp $ */ /* @@ -35,7 +35,9 @@ #include <sys/systm.h> #include <sys/device.h> #include <sys/extent.h> +#include <sys/kernel.h> #include <sys/malloc.h> +#include <sys/kthread.h> #include <vm/vm.h> @@ -89,12 +91,18 @@ int pcic_print __P((void *arg, const char *pnp)); int pcic_intr_socket __P((struct pcic_handle *)); void pcic_attach_card __P((struct pcic_handle *)); -void pcic_detach_card __P((struct pcic_handle *)); +void pcic_detach_card __P((struct pcic_handle *, int)); +void pcic_deactivate_card __P((struct pcic_handle *)); void pcic_chip_do_mem_map __P((struct pcic_handle *, int)); void pcic_chip_do_io_map __P((struct pcic_handle *, int)); -static void pcic_wait_ready __P((struct pcic_handle *)); +void pcic_create_event_thread __P((void *)); +void pcic_event_thread __P((void *)); + +void pcic_queue_event __P((struct pcic_handle *, int)); + +void pcic_wait_ready __P((struct pcic_handle *)); struct cfdriver pcic_cd = { NULL, "pcic", DV_DULL @@ -209,6 +217,7 @@ pcic_attach(sc) } else { sc->handle[0].flags = 0; } + sc->handle[0].laststate = PCIC_LASTSTATE_EMPTY; DPRINTF((" 0x%02x", reg)); @@ -220,6 +229,7 @@ pcic_attach(sc) } else { sc->handle[1].flags = 0; } + sc->handle[1].laststate = PCIC_LASTSTATE_EMPTY; DPRINTF((" 0x%02x", reg)); @@ -239,6 +249,7 @@ pcic_attach(sc) } else { sc->handle[2].flags = 0; } + sc->handle[2].laststate = PCIC_LASTSTATE_EMPTY; DPRINTF((" 0x%02x", reg)); @@ -251,6 +262,7 @@ pcic_attach(sc) } else { sc->handle[3].flags = 0; } + sc->handle[3].laststate = PCIC_LASTSTATE_EMPTY; DPRINTF((" 0x%02x\n", reg)); } else { @@ -271,6 +283,7 @@ pcic_attach(sc) * boot time. */ if (sc->handle[i].flags & PCIC_FLAG_SOCKETP) { + SIMPLEQ_INIT(&sc->handle[i].events); pcic_write(&sc->handle[i], PCIC_CSC_INTR, 0); pcic_read(&sc->handle[i], PCIC_CSC); } @@ -341,11 +354,139 @@ pcic_attach_socket(h) } void +pcic_create_event_thread(arg) + void *arg; +{ + struct pcic_handle *h = arg; + const char *cs; + + switch (h->sock) { + case C0SA: + cs = "0,0"; + break; + case C0SB: + cs = "0,1"; + break; + case C1SA: + cs = "1,0"; + break; + case C1SB: + cs = "1,1"; + break; + default: + panic("pcic_create_event_thread: unknown pcic socket"); + } + + if (kthread_create(pcic_event_thread, h, &h->event_thread, + "%s,%s", h->sc->dev.dv_xname, cs)) { + printf("%s: unable to create event thread for sock 0x%02x\n", + h->sc->dev.dv_xname, h->sock); + panic("pcic_create_event_thread"); + } +} + +void +pcic_event_thread(arg) + void *arg; +{ + struct pcic_handle *h = arg; + struct pcic_event *pe; + int s; + + while (h->shutdown == 0) { + s = splhigh(); + if ((pe = SIMPLEQ_FIRST(&h->events)) == NULL) { + splx(s); + (void) tsleep(&h->events, PWAIT, "pcicev", 0); + continue; + } else { + splx(s); + /* sleep .25s to be enqueued chatterling interrupts */ + (void) tsleep((caddr_t)pcic_event_thread, PWAIT, "pcicss", hz/4); + } + s = splhigh(); + SIMPLEQ_REMOVE_HEAD(&h->events, pe, pe_q); + splx(s); + + switch (pe->pe_type) { + case PCIC_EVENT_INSERTION: + s = splhigh(); + while (1) { + struct pcic_event *pe1, *pe2; + + if ((pe1 = SIMPLEQ_FIRST(&h->events)) == NULL) + break; + if (pe1->pe_type != PCIC_EVENT_REMOVAL) + break; + if ((pe2 = SIMPLEQ_NEXT(pe1, pe_q)) == NULL) + break; + if (pe2->pe_type == PCIC_EVENT_INSERTION) { + SIMPLEQ_REMOVE_HEAD(&h->events, pe1, pe_q); + free(pe1, M_TEMP); + SIMPLEQ_REMOVE_HEAD(&h->events, pe2, pe_q); + free(pe2, M_TEMP); + } + } + splx(s); + + DPRINTF(("%s: insertion event\n", h->sc->dev.dv_xname)); + pcic_attach_card(h); + break; + + case PCIC_EVENT_REMOVAL: + s = splhigh(); + while (1) { + struct pcic_event *pe1, *pe2; + + if ((pe1 = SIMPLEQ_FIRST(&h->events)) == NULL) + break; + if (pe1->pe_type != PCIC_EVENT_INSERTION) + break; + if ((pe2 = SIMPLEQ_NEXT(pe1, pe_q)) == NULL) + break; + if (pe2->pe_type == PCIC_EVENT_REMOVAL) { + SIMPLEQ_REMOVE_HEAD(&h->events, pe1, pe_q); + free(pe1, M_TEMP); + SIMPLEQ_REMOVE_HEAD(&h->events, pe2, pe_q); + free(pe2, M_TEMP); + } + } + splx(s); + + DPRINTF(("%s: removal event\n", h->sc->dev.dv_xname)); + pcic_detach_card(h, DETACH_FORCE); + break; + + default: + panic("pcic_event_thread: unknown event %d", + pe->pe_type); + } + free(pe, M_TEMP); + } + + h->event_thread = NULL; + + /* In case parent is waiting for us to exit. */ + wakeup(h->sc); + + kthread_exit(0); +} + +void pcic_init_socket(h) struct pcic_handle *h; { int reg; + /* + * queue creation of a kernel thread to handle insert/removal events. + */ +#ifdef DIAGNOSTIC + if (h->event_thread != NULL) + panic("pcic_attach_socket: event thread"); +#endif + kthread_create_deferred(pcic_create_event_thread, h); + /* set up the card to interrupt on card detect */ pcic_write(h, PCIC_CSC_INTR, (h->sc->irq << PCIC_CSC_INTR_IRQ_SHIFT) | @@ -370,8 +511,11 @@ pcic_init_socket(h) reg = pcic_read(h, PCIC_IF_STATUS); if ((reg & PCIC_IF_STATUS_CARDDETECT_MASK) == - PCIC_IF_STATUS_CARDDETECT_PRESENT) + PCIC_IF_STATUS_CARDDETECT_PRESENT) { pcic_attach_card(h); + h->laststate = PCIC_LASTSTATE_PRESENT; + } else + h->laststate = PCIC_LASTSTATE_EMPTY; } int @@ -508,18 +652,27 @@ pcic_intr_socket(h) DPRINTF(("%s: %02x CD %x\n", h->sc->dev.dv_xname, h->sock, statreg)); - /* - * XXX This should probably schedule something to happen - * after the interrupt handler completes - */ - if ((statreg & PCIC_IF_STATUS_CARDDETECT_MASK) == PCIC_IF_STATUS_CARDDETECT_PRESENT) { - if (!(h->flags & PCIC_FLAG_CARDP)) - pcic_attach_card(h); + if (h->laststate != PCIC_LASTSTATE_PRESENT) { + DPRINTF(("%s: enqueing INSERTION event\n", + h->sc->dev.dv_xname)); + pcic_queue_event(h, PCIC_EVENT_INSERTION); + } + h->laststate = PCIC_LASTSTATE_PRESENT; } else { - if (h->flags & PCIC_FLAG_CARDP) - pcic_detach_card(h); + if (h->laststate == PCIC_LASTSTATE_PRESENT) { + /* Deactivate the card now. */ + DPRINTF(("%s: deactivating card\n", + h->sc->dev.dv_xname)); + pcic_deactivate_card(h); + + DPRINTF(("%s: enqueing REMOVAL event\n", + h->sc->dev.dv_xname)); + pcic_queue_event(h, PCIC_EVENT_REMOVAL); + } + h->laststate = ((statreg & PCIC_IF_STATUS_CARDDETECT_MASK) == 0) + ? PCIC_LASTSTATE_EMPTY : PCIC_LASTSTATE_HALF; } } if (cscreg & PCIC_CSC_READY) { @@ -536,6 +689,25 @@ pcic_intr_socket(h) } void +pcic_queue_event(h, event) + struct pcic_handle *h; + int event; +{ + struct pcic_event *pe; + int s; + + pe = malloc(sizeof(*pe), M_TEMP, M_NOWAIT); + if (pe == NULL) + panic("pcic_queue_event: can't allocate event"); + + pe->pe_type = event; + s = splhigh(); + SIMPLEQ_INSERT_TAIL(&h->events, pe, pe_q); + splx(s); + wakeup(&h->events); +} + +void pcic_attach_card(h) struct pcic_handle *h; { @@ -550,26 +722,33 @@ pcic_attach_card(h) } void -pcic_detach_card(h) +pcic_detach_card(h, flags) struct pcic_handle *h; + int flags; /* DETACH_* */ { - if (!(h->flags & PCIC_FLAG_CARDP)) - panic("pcic_attach_card: already detached"); - h->flags &= ~PCIC_FLAG_CARDP; + if (h->flags & PCIC_FLAG_CARDP) { + h->flags &= ~PCIC_FLAG_CARDP; - /* call the MI attach function */ + /* call the MI detach function */ + pcmcia_card_detach(h->pcmcia, flags); + } else { + DPRINTF(("pcic_detach_card: already detached")); + } +} - pcmcia_card_detach(h->pcmcia); +void +pcic_deactivate_card(h) + struct pcic_handle *h; +{ - /* disable card detect resume and configuration reset */ + /* call the MI deactivate function */ + pcmcia_card_deactivate(h->pcmcia); /* power down the socket */ - pcic_write(h, PCIC_PWRCTL, 0); - /* reset the card */ - + /* reset the socket */ pcic_write(h, PCIC_INTR, 0); } @@ -1077,7 +1256,7 @@ pcic_chip_io_unmap(pch, window) h->ioalloc &= ~(1 << window); } -static void +void pcic_wait_ready(h) struct pcic_handle *h; { diff --git a/sys/dev/ic/i82365var.h b/sys/dev/ic/i82365var.h index 952b8ca50e1..81af7b771f2 100644 --- a/sys/dev/ic/i82365var.h +++ b/sys/dev/ic/i82365var.h @@ -1,4 +1,4 @@ -/* $OpenBSD: i82365var.h,v 1.4 1999/07/26 05:43:15 deraadt Exp $ */ +/* $OpenBSD: i82365var.h,v 1.5 1999/08/08 01:07:02 niklas Exp $ */ /* $NetBSD: i82365var.h,v 1.4 1998/05/23 18:32:29 matt Exp $ */ /* @@ -37,11 +37,23 @@ #include <dev/ic/i82365reg.h> +struct proc; + +struct pcic_event { + SIMPLEQ_ENTRY(pcic_event) pe_q; + int pe_type; +}; + +/* pe_type */ +#define PCIC_EVENT_INSERTION 0 +#define PCIC_EVENT_REMOVAL 1 + struct pcic_handle { struct pcic_softc *sc; int vendor; int sock; int flags; + int laststate; int memalloc; struct { bus_addr_t addr; @@ -57,11 +69,19 @@ struct pcic_handle { } io[PCIC_IO_WINS]; int ih_irq; struct device *pcmcia; + + int shutdown; + struct proc *event_thread; + SIMPLEQ_HEAD(, pcic_event) events; }; #define PCIC_FLAG_SOCKETP 0x0001 #define PCIC_FLAG_CARDP 0x0002 +#define PCIC_LASTSTATE_PRESENT 0x0002 +#define PCIC_LASTSTATE_HALF 0x0001 +#define PCIC_LASTSTATE_EMPTY 0x0000 + #define C0SA PCIC_CHIP0_BASE+PCIC_SOCKETA_INDEX #define C0SB PCIC_CHIP0_BASE+PCIC_SOCKETB_INDEX #define C1SA PCIC_CHIP1_BASE+PCIC_SOCKETA_INDEX |