diff options
Diffstat (limited to 'sys/dev/ic/adw.c')
-rw-r--r-- | sys/dev/ic/adw.c | 634 |
1 files changed, 446 insertions, 188 deletions
diff --git a/sys/dev/ic/adw.c b/sys/dev/ic/adw.c index bd5c15860c3..e5f5272fca7 100644 --- a/sys/dev/ic/adw.c +++ b/sys/dev/ic/adw.c @@ -1,10 +1,9 @@ -/* $OpenBSD: adw.c,v 1.2 1999/08/04 23:27:48 niklas Exp $ */ -/* $NetBSD: adw.c,v 1.3 1998/10/10 00:28:33 thorpej Exp $ */ +/* $NetBSD: adw.c,v 1.13 2000/02/03 20:29:15 dante Exp $ */ /* * Generic driver for the Advanced Systems Inc. SCSI controllers * - * Copyright (c) 1998 The NetBSD Foundation, Inc. + * Copyright (c) 1998, 1999, 2000 The NetBSD Foundation, Inc. * All rights reserved. * * Author: Baldassare Dante Profeta <dante@mclink.it> @@ -64,7 +63,7 @@ #include <dev/ic/adw.h> #ifndef DDB -#define Debugger() panic("should call debugger here (adv.c)") +#define Debugger() panic("should call debugger here (adw.c)") #endif /* ! DDB */ /******************************************************************************/ @@ -73,24 +72,26 @@ static void adw_enqueue __P((ADW_SOFTC *, struct scsi_xfer *, int)); static struct scsi_xfer *adw_dequeue __P((ADW_SOFTC *)); -static int adw_alloc_ccbs __P((ADW_SOFTC *)); +static int adw_alloc_controls __P((ADW_SOFTC *)); +static int adw_alloc_carriers __P((ADW_SOFTC *)); +static int adw_create_carriers __P((ADW_SOFTC *)); +static int adw_init_carrier __P((ADW_SOFTC *, ADW_CARRIER *)); static int adw_create_ccbs __P((ADW_SOFTC *, ADW_CCB *, int)); static void adw_free_ccb __P((ADW_SOFTC *, ADW_CCB *)); static void adw_reset_ccb __P((ADW_CCB *)); static int adw_init_ccb __P((ADW_SOFTC *, ADW_CCB *)); static ADW_CCB *adw_get_ccb __P((ADW_SOFTC *, int)); -static void adw_queue_ccb __P((ADW_SOFTC *, ADW_CCB *)); -static void adw_start_ccbs __P((ADW_SOFTC *)); +static int adw_queue_ccb __P((ADW_SOFTC *, ADW_CCB *, int)); static int adw_scsi_cmd __P((struct scsi_xfer *)); static int adw_build_req __P((struct scsi_xfer *, ADW_CCB *)); -static void adw_build_sglist __P((ADW_CCB *, ADW_SCSI_REQ_Q *)); +static void adw_build_sglist __P((ADW_CCB *, ADW_SCSI_REQ_Q *, ADW_SG_BLOCK *)); static void adwminphys __P((struct buf *)); -static void adw_wide_isr_callback __P((ADW_SOFTC *, ADW_SCSI_REQ_Q *)); +static void adw_isr_callback __P((ADW_SOFTC *, ADW_SCSI_REQ_Q *)); +static void adw_async_callback __P((ADW_SOFTC *, u_int8_t)); static int adw_poll __P((ADW_SOFTC *, struct scsi_xfer *, int)); static void adw_timeout __P((void *)); -static void adw_watchdog __P((void *)); /******************************************************************************/ @@ -100,16 +101,6 @@ struct cfdriver adw_cd = { NULL, "adw", DV_DULL }; - -struct scsi_adapter adw_switch = -{ - adw_scsi_cmd, /* called to start/enqueue a SCSI command */ - adwminphys, /* to limit the transfer to max device can do */ - 0, - 0, -}; - - /* the below structure is so we have a default dev struct for out link struct */ struct scsi_device adw_dev = { @@ -125,7 +116,7 @@ struct scsi_device adw_dev = /******************************************************************************/ -/* scsi_xfer queue routines */ +/* scsi_xfer queue routines */ /******************************************************************************/ /* @@ -169,21 +160,20 @@ adw_dequeue(sc) return (xs); } - /******************************************************************************/ -/* Control Blocks routines */ +/* Control Blocks routines */ /******************************************************************************/ static int -adw_alloc_ccbs(sc) +adw_alloc_controls(sc) ADW_SOFTC *sc; { bus_dma_segment_t seg; int error, rseg; /* - * Allocate the control blocks. + * Allocate the control structure. */ if ((error = bus_dmamem_alloc(sc->sc_dmat, sizeof(struct adw_control), NBPG, 0, &seg, 1, &rseg, BUS_DMA_NOWAIT)) != 0) { @@ -198,6 +188,7 @@ adw_alloc_ccbs(sc) sc->sc_dev.dv_xname, error); return (error); } + /* * Create and load the DMA map used for the control blocks. */ @@ -215,11 +206,164 @@ adw_alloc_ccbs(sc) sc->sc_dev.dv_xname, error); return (error); } + + return (0); +} + + +static int +adw_alloc_carriers(sc) + ADW_SOFTC *sc; +{ + bus_dma_segment_t seg; + int error, rseg; + + /* + * Allocate the control structure. + */ + sc->sc_control->carriers = malloc(ADW_CARRIER_SIZE * ADW_MAX_CARRIER, + M_DEVBUF, M_WAITOK); + if(!sc->sc_control->carriers) { + printf("%s: malloc() failed in allocating carrier structures," + " error = %d\n", sc->sc_dev.dv_xname, error); + return (error); + } + + if ((error = bus_dmamem_alloc(sc->sc_dmat, + ADW_CARRIER_SIZE * ADW_MAX_CARRIER, + NBPG, 0, &seg, 1, &rseg, BUS_DMA_NOWAIT)) != 0) { + printf("%s: unable to allocate carrier structures," + " error = %d\n", sc->sc_dev.dv_xname, error); + return (error); + } + if ((error = bus_dmamem_map(sc->sc_dmat, &seg, rseg, + ADW_CARRIER_SIZE * ADW_MAX_CARRIER, + (caddr_t *) &sc->sc_control->carriers, + BUS_DMA_NOWAIT | BUS_DMA_COHERENT)) != 0) { + printf("%s: unable to map carrier structures," + " error = %d\n", sc->sc_dev.dv_xname, error); + return (error); + } + + /* + * Create and load the DMA map used for the control blocks. + */ + if ((error = bus_dmamap_create(sc->sc_dmat, + ADW_CARRIER_SIZE * ADW_MAX_CARRIER, 1, + ADW_CARRIER_SIZE * ADW_MAX_CARRIER, 0, BUS_DMA_NOWAIT, + &sc->sc_dmamap_carrier)) != 0) { + printf("%s: unable to create carriers DMA map," + " error = %d\n", sc->sc_dev.dv_xname, error); + return (error); + } + if ((error = bus_dmamap_load(sc->sc_dmat, + sc->sc_dmamap_carrier, sc->sc_control->carriers, + ADW_CARRIER_SIZE * ADW_MAX_CARRIER, NULL, + BUS_DMA_NOWAIT)) != 0) { + printf("%s: unable to load carriers DMA map," + " error = %d\n", sc->sc_dev.dv_xname, error); + return (error); + } + + error = bus_dmamap_create(sc->sc_dmat, ADW_CARRIER_SIZE* ADW_MAX_CARRIER, + 1, ADW_CARRIER_SIZE * ADW_MAX_CARRIER, + 0, BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, + &sc->sc_control->dmamap_xfer); + if (error) { + printf("%s: unable to create Carrier DMA map, error = %d\n", + sc->sc_dev.dv_xname, error); + return (error); + } + return (0); } /* + * Create a set of Carriers and add them to the free list. Called once + * by adw_init(). We return the number of Carriers successfully created. + */ +static int +adw_create_carriers(sc) + ADW_SOFTC *sc; +{ + ADW_CARRIER *carr; + u_int32_t carr_next = NULL; + int i, error; + + for(i=0; i < ADW_MAX_CARRIER; i++) { + carr = (ADW_CARRIER *)(((u_int8_t *)sc->sc_control->carriers) + + (ADW_CARRIER_SIZE * i)); + if ((error = adw_init_carrier(sc, carr)) != 0) { + printf("%s: unable to initialize carrier, error = %d\n", + sc->sc_dev.dv_xname, error); + return (i); + } + carr->next_vpa = carr_next; + carr_next = carr->carr_pa; +carr->id = i; + } + sc->carr_freelist = carr; + return (i); +} + + +static int +adw_init_carrier(sc, carr) + ADW_SOFTC *sc; + ADW_CARRIER *carr; +{ + u_int32_t carr_pa; + int /*error, */hashnum; + + /* + * Create the DMA map for all of the Carriers. + */ +/* error = bus_dmamap_create(sc->sc_dmat, ADW_CARRIER_SIZE, + 1, ADW_CARRIER_SIZE, + 0, BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, + &carr->dmamap_xfer); + if (error) { + printf("%s: unable to create Carrier DMA map, error = %d\n", + sc->sc_dev.dv_xname, error); + return (error); + } +*/ + /* + * put in the phystokv hash table + * Never gets taken out. + */ + carr_pa = ADW_CARRIER_ADDR(sc, carr); + carr->carr_pa = carr_pa; + hashnum = CARRIER_HASH(carr_pa); + carr->nexthash = sc->sc_carrhash[hashnum]; + sc->sc_carrhash[hashnum] = carr; + + return(0); +} + + +/* + * Given a physical address, find the Carrier that it corresponds to. + */ +ADW_CARRIER * +adw_carrier_phys_kv(sc, carr_phys) + ADW_SOFTC *sc; + u_int32_t carr_phys; +{ + int hashnum = CARRIER_HASH(carr_phys); + ADW_CARRIER *carr = sc->sc_carrhash[hashnum]; + + while (carr) { + if (carr->carr_pa == carr_phys) + break; + carr = carr->nexthash; + } + return (carr); +} + + +/* * Create a set of ccbs and add them to the free list. Called once * by adw_init(). We return the number of CCBs successfully created. */ @@ -232,7 +376,6 @@ adw_create_ccbs(sc, ccbstore, count) ADW_CCB *ccb; int i, error; - bzero(ccbstore, sizeof(ADW_CCB) * count); for (i = 0; i < count; i++) { ccb = &ccbstore[i]; if ((error = adw_init_ccb(sc, ccb)) != 0) { @@ -287,7 +430,7 @@ adw_init_ccb(sc, ccb) ADW_SOFTC *sc; ADW_CCB *ccb; { - int error; + int hashnum, error; /* * Create the DMA map for this CCB. @@ -297,10 +440,20 @@ adw_init_ccb(sc, ccb) ADW_MAX_SG_LIST, (ADW_MAX_SG_LIST - 1) * PAGE_SIZE, 0, BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, &ccb->dmamap_xfer); if (error) { - printf("%s: unable to create DMA map, error = %d\n", + printf("%s: unable to create CCB DMA map, error = %d\n", sc->sc_dev.dv_xname, error); return (error); } + + /* + * put in the phystokv hash table + * Never gets taken out. + */ + ccb->hashkey = sc->sc_dmamap_control->dm_segs[0].ds_addr + + ADW_CCB_OFF(ccb); + hashnum = CCB_HASH(ccb->hashkey); + ccb->nexthash = sc->sc_ccbhash[hashnum]; + sc->sc_ccbhash[hashnum] = ccb; adw_reset_ccb(ccb); return (0); } @@ -346,46 +499,68 @@ out: /* - * Queue a CCB to be sent to the controller, and send it if possible. + * Given a physical address, find the ccb that it corresponds to. */ -static void -adw_queue_ccb(sc, ccb) - ADW_SOFTC *sc; - ADW_CCB *ccb; +ADW_CCB * +adw_ccb_phys_kv(sc, ccb_phys) + ADW_SOFTC *sc; + u_int32_t ccb_phys; { + int hashnum = CCB_HASH(ccb_phys); + ADW_CCB *ccb = sc->sc_ccbhash[hashnum]; - TAILQ_INSERT_TAIL(&sc->sc_waiting_ccb, ccb, chain); - - adw_start_ccbs(sc); + while (ccb) { + if (ccb->hashkey == ccb_phys) + break; + ccb = ccb->nexthash; + } + return (ccb); } -static void -adw_start_ccbs(sc) +/* + * Queue a CCB to be sent to the controller, and send it if possible. + */ +static int +adw_queue_ccb(sc, ccb, retry) ADW_SOFTC *sc; -{ ADW_CCB *ccb; + int retry; +{ + int errcode; + + if(!retry) + TAILQ_INSERT_TAIL(&sc->sc_waiting_ccb, ccb, chain); while ((ccb = sc->sc_waiting_ccb.tqh_first) != NULL) { - if (ccb->flags & CCB_WATCHDOG) - untimeout(adw_watchdog, ccb); - if (AdvExeScsiQueue(sc, &ccb->scsiq) == ADW_BUSY) { - ccb->flags |= CCB_WATCHDOG; - timeout(adw_watchdog, ccb, - (ADW_WATCH_TIMEOUT * hz) / 1000); + errcode = AdvExeScsiQueue(sc, &ccb->scsiq); + switch(errcode) { + case ADW_SUCCESS: break; + + case ADW_BUSY: + printf("ADW_BUSY\n"); + return(ADW_BUSY); + + case ADW_ERROR: + printf("ADW_ERROR\n"); + TAILQ_REMOVE(&sc->sc_waiting_ccb, ccb, chain); + return(ADW_ERROR); } + TAILQ_REMOVE(&sc->sc_waiting_ccb, ccb, chain); if ((ccb->xs->flags & SCSI_POLL) == 0) timeout(adw_timeout, ccb, (ccb->timeout * hz) / 1000); } + + return(errcode); } /******************************************************************************/ -/* SCSI layer interfacing routines */ +/* SCSI layer interfacing routines */ /******************************************************************************/ @@ -409,7 +584,10 @@ adw_init(sc) } else { AdvResetChip(sc->sc_iot, sc->sc_ioh); - warn_code = AdvInitFromEEP(sc); + warn_code = (sc->chip_type == ADV_CHIP_ASC3550)? + AdvInitFrom3550EEP(sc) : + AdvInitFrom38C0800EEP(sc); + if (warn_code & ASC_WARN_EEPROM_CHKSUM) printf("%s: Bad checksum found. " "Setting default values\n", @@ -418,16 +596,10 @@ adw_init(sc) printf("%s: Bad bus termination setting." "Using automatic termination.\n", sc->sc_dev.dv_xname); - - /* - * Reset the SCSI Bus if the EEPROM indicates that SCSI Bus - * Resets should be performed. - */ - if (sc->bios_ctrl & BIOS_CTRL_RESET_SCSI_BUS) - AdvResetSCSIBus(sc); } - sc->isr_callback = (ulong) adw_wide_isr_callback; + sc->isr_callback = (ADW_CALLBACK) adw_isr_callback; + sc->async_callback = (ADW_CALLBACK) adw_async_callback; return (0); } @@ -440,10 +612,61 @@ adw_attach(sc) int i, error; + TAILQ_INIT(&sc->sc_free_ccb); + TAILQ_INIT(&sc->sc_waiting_ccb); + LIST_INIT(&sc->sc_queue); + + + /* + * Allocate the Control Blocks. + */ + error = adw_alloc_controls(sc); + if (error) + return; /* (error) */ ; + + bzero(sc->sc_control, sizeof(struct adw_control)); + + /* + * Create and initialize the Control Blocks. + */ + i = adw_create_ccbs(sc, sc->sc_control->ccbs, ADW_MAX_CCB); + if (i == 0) { + printf("%s: unable to create Control Blocks\n", + sc->sc_dev.dv_xname); + return; /* (ENOMEM) */ ; + } else if (i != ADW_MAX_CCB) { + printf("%s: WARNING: only %d of %d Control Blocks" + " created\n", + sc->sc_dev.dv_xname, i, ADW_MAX_CCB); + } + + /* + * Create and initialize the Carriers. + */ + error = adw_alloc_carriers(sc); + if (error) + return; /* (error) */ ; + + bzero(sc->sc_control->carriers, ADW_CARRIER_SIZE * ADW_MAX_CARRIER); + + i = adw_create_carriers(sc); + if (i == 0) { + printf("%s: unable to create Carriers\n", + sc->sc_dev.dv_xname); + return; /* (ENOMEM) */ ; + } else if (i != ADW_MAX_CARRIER) { + printf("%s: WARNING: only %d of %d Carriers created\n", + sc->sc_dev.dv_xname, i, ADW_MAX_CARRIER); + } + + /* * Initialize the ASC3550. */ - switch (AdvInitAsc3550Driver(sc)) { + error = (sc->chip_type == ADV_CHIP_ASC3550)? + AdvInitAsc3550Driver(sc) : + AdvInitAsc38C0800Driver(sc); + switch (error) { case ASC_IERR_MCODE_CHKSUM: panic("%s: Microcode checksum error", sc->sc_dev.dv_xname); @@ -464,45 +687,34 @@ adw_attach(sc) " one of the connectors", sc->sc_dev.dv_xname); break; + + case ASC_IERR_NO_CARRIER: + panic("%s: no carrier", + sc->sc_dev.dv_xname); + break; + + case ASC_WARN_BUSRESET_ERROR: + printf("%s: WARNING: Bus Reset Error\n", + sc->sc_dev.dv_xname); + break; } + /* + * Fill in the adapter. + */ + sc->sc_adapter.scsi_cmd = adw_scsi_cmd; + sc->sc_adapter.scsi_minphys = adwminphys; /* * fill in the prototype scsi_link. */ sc->sc_link.adapter_softc = sc; sc->sc_link.adapter_target = sc->chip_scsi_id; - sc->sc_link.adapter = &adw_switch; + sc->sc_link.adapter = &sc->sc_adapter; sc->sc_link.device = &adw_dev; sc->sc_link.openings = 4; - sc->sc_link.adapter_buswidth = ADW_MAX_TID; - - - TAILQ_INIT(&sc->sc_free_ccb); - TAILQ_INIT(&sc->sc_waiting_ccb); - LIST_INIT(&sc->sc_queue); - - - /* - * Allocate the Control Blocks. - */ - error = adw_alloc_ccbs(sc); - if (error) - return; /* (error) */ ; + sc->sc_link.adapter_buswidth = ADW_MAX_TID+1; - /* - * Create and initialize the Control Blocks. - */ - i = adw_create_ccbs(sc, sc->sc_control->ccbs, ADW_MAX_CCB); - if (i == 0) { - printf("%s: unable to create control blocks\n", - sc->sc_dev.dv_xname); - return; /* (ENOMEM) */ ; - } else if (i != ADW_MAX_CCB) { - printf("%s: WARNING: only %d of %d control blocks" - " created\n", - sc->sc_dev.dv_xname, i, ADW_MAX_CCB); - } config_found(&sc->sc_dev, &sc->sc_link, scsiprint); } @@ -529,7 +741,7 @@ adw_scsi_cmd(xs) struct scsi_link *sc_link = xs->sc_link; ADW_SOFTC *sc = sc_link->adapter_softc; ADW_CCB *ccb; - int s, fromqueue = 1, dontqueue = 0; + int s, fromqueue = 1, dontqueue = 0, retry = 0; s = splbio(); /* protect the queue */ @@ -597,10 +809,21 @@ adw_scsi_cmd(xs) ccb->timeout = xs->timeout; if (adw_build_req(xs, ccb)) { +retryagain: s = splbio(); - adw_queue_ccb(sc, ccb); + retry = adw_queue_ccb(sc, ccb, retry); splx(s); + switch(retry) { + case ADW_BUSY: + goto retryagain; + + case ADW_ERROR: + xs->error = XS_DRIVER_STUFFUP; + return (COMPLETE); + + } + /* * Usually return SUCCESSFULLY QUEUED */ @@ -638,10 +861,10 @@ adw_build_req(xs, ccb) bzero(scsiqp, sizeof(ADW_SCSI_REQ_Q)); /* - * Set the ADW_SCSI_REQ_Q 'ccb_ptr' to point to the CCB structure. + * Set the ADW_SCSI_REQ_Q 'ccb_ptr' to point to the + * physical CCB structure. */ - scsiqp->ccb_ptr = (ulong) ccb; - + scsiqp->ccb_ptr = ccb->hashkey; /* * Build the ADW_SCSI_REQ_Q request. @@ -655,10 +878,12 @@ adw_build_req(xs, ccb) scsiqp->target_id = sc_link->target; scsiqp->target_lun = sc_link->lun; - scsiqp->vsense_addr = (ulong) & ccb->scsi_sense; + scsiqp->vsense_addr = & ccb->scsi_sense; scsiqp->sense_addr = sc->sc_dmamap_control->dm_segs[0].ds_addr + - ADW_CCB_OFF(ccb) + offsetof(struct adw_ccb, scsi_sense); - scsiqp->sense_len = sizeof(struct scsi_sense_data); + ADW_CCB_OFF(ccb) + offsetof(struct adw_ccb, scsi_sense); +/* scsiqp->sense_addr = ccb->hashkey + + offsetof(struct adw_ccb, scsi_sense); +*/ scsiqp->sense_len = sizeof(struct scsi_sense_data); /* * Build ADW_SCSI_REQ_Q for a scatter-gather buffer command. @@ -671,15 +896,15 @@ adw_build_req(xs, ccb) if (xs->flags & SCSI_DATA_UIO) { error = bus_dmamap_load_uio(dmat, ccb->dmamap_xfer, (struct uio *) xs->data, - (xs->flags & SCSI_NOSLEEP) ? BUS_DMA_NOWAIT : - BUS_DMA_WAITOK); + (xs->flags & SCSI_NOSLEEP) ? + BUS_DMA_NOWAIT : BUS_DMA_WAITOK); } else -#endif /* TFS */ +#endif /* TFS */ { error = bus_dmamap_load(dmat, ccb->dmamap_xfer, xs->data, xs->datalen, NULL, - (xs->flags & SCSI_NOSLEEP) ? BUS_DMA_NOWAIT : - BUS_DMA_WAITOK); + (xs->flags & SCSI_NOSLEEP) ? + BUS_DMA_NOWAIT : BUS_DMA_WAITOK); } if (error) { @@ -698,19 +923,17 @@ adw_build_req(xs, ccb) return (0); } bus_dmamap_sync(dmat, ccb->dmamap_xfer, - (xs->flags & SCSI_DATA_IN) ? BUS_DMASYNC_PREREAD : - BUS_DMASYNC_PREWRITE); + (xs->flags & SCSI_DATA_IN) ? + BUS_DMASYNC_PREREAD : BUS_DMASYNC_PREWRITE); /* * Build scatter-gather list. */ scsiqp->data_cnt = xs->datalen; - scsiqp->vdata_addr = (ulong) xs->data; + scsiqp->vdata_addr = xs->data; scsiqp->data_addr = ccb->dmamap_xfer->dm_segs[0].ds_addr; - scsiqp->sg_list_ptr = &ccb->sg_block[0]; - bzero(scsiqp->sg_list_ptr, - sizeof(ADW_SG_BLOCK) * ADW_NUM_SG_BLOCK); - adw_build_sglist(ccb, scsiqp); + bzero(ccb->sg_block, sizeof(ADW_SG_BLOCK) * ADW_NUM_SG_BLOCK); + adw_build_sglist(ccb, scsiqp, ccb->sg_block); } else { /* * No data xfer, use non S/G values. @@ -718,7 +941,6 @@ adw_build_req(xs, ccb) scsiqp->data_cnt = 0; scsiqp->vdata_addr = 0; scsiqp->data_addr = 0; - scsiqp->sg_list_ptr = NULL; } return (1); @@ -729,23 +951,21 @@ adw_build_req(xs, ccb) * Build scatter-gather list for Wide Boards. */ static void -adw_build_sglist(ccb, scsiqp) +adw_build_sglist(ccb, scsiqp, sg_block) ADW_CCB *ccb; ADW_SCSI_REQ_Q *scsiqp; + ADW_SG_BLOCK *sg_block; { - struct scsi_xfer *xs = ccb->xs; - ADW_SOFTC *sc = xs->sc_link->adapter_softc; - ADW_SG_BLOCK *sg_block = scsiqp->sg_list_ptr; - ulong sg_block_next_addr; /* block and its next */ - ulong sg_block_physical_addr; - int sg_block_index, i; /* how many SG entries */ + u_long sg_block_next_addr; /* block and its next */ + u_int32_t sg_block_physical_addr; + int i; /* how many SG entries */ bus_dma_segment_t *sg_list = &ccb->dmamap_xfer->dm_segs[0]; int sg_elem_cnt = ccb->dmamap_xfer->dm_nsegs; - sg_block_next_addr = (ulong) sg_block; /* allow math operation */ - sg_block_physical_addr = sc->sc_dmamap_control->dm_segs[0].ds_addr + - ADW_CCB_OFF(ccb) + offsetof(struct adw_ccb, sg_block[0]); + sg_block_next_addr = (u_long) sg_block; /* allow math operation */ + sg_block_physical_addr = ccb->hashkey + + offsetof(struct adw_ccb, sg_block[0]); scsiqp->sg_real_addr = sg_block_physical_addr; /* @@ -753,17 +973,14 @@ adw_build_sglist(ccb, scsiqp) * then split the request into multiple sg-list blocks. */ - sg_block_index = 0; do { - sg_block->first_entry_no = sg_block_index; for (i = 0; i < NO_OF_SG_PER_BLOCK; i++) { sg_block->sg_list[i].sg_addr = sg_list->ds_addr; sg_block->sg_list[i].sg_count = sg_list->ds_len; if (--sg_elem_cnt == 0) { /* last entry, get out */ - scsiqp->sg_entry_cnt = sg_block_index + i + 1; - sg_block->last_entry_no = sg_block_index + i; + sg_block->sg_cnt = i + i; sg_block->sg_ptr = NULL; /* next link = NULL */ return; } @@ -772,12 +989,10 @@ adw_build_sglist(ccb, scsiqp) sg_block_next_addr += sizeof(ADW_SG_BLOCK); sg_block_physical_addr += sizeof(ADW_SG_BLOCK); - sg_block_index += NO_OF_SG_PER_BLOCK; - sg_block->sg_ptr = (ADW_SG_BLOCK *) sg_block_physical_addr; - sg_block->last_entry_no = sg_block_index - 1; + sg_block->sg_cnt = NO_OF_SG_PER_BLOCK; + sg_block->sg_ptr = sg_block_physical_addr; sg_block = (ADW_SG_BLOCK *) sg_block_next_addr; /* virt. addr */ - } - while (1); + } while (1); } @@ -789,18 +1004,18 @@ adw_intr(arg) struct scsi_xfer *xs; - AdvISR(sc); - - /* - * If there are queue entries in the software queue, try to - * run the first one. We should be more or less guaranteed - * to succeed, since we just freed a CCB. - * - * NOTE: adw_scsi_cmd() relies on our calling it with - * the first entry in the queue. - */ - if ((xs = sc->sc_queue.lh_first) != NULL) - (void) adw_scsi_cmd(xs); + if(AdvISR(sc) != ADW_FALSE) { + /* + * If there are queue entries in the software queue, try to + * run the first one. We should be more or less guaranteed + * to succeed, since we just freed a CCB. + * + * NOTE: adw_scsi_cmd() relies on our calling it with + * the first entry in the queue. + */ + if ((xs = sc->sc_queue.lh_first) != NULL) + (void) adw_scsi_cmd(xs); + } return (1); } @@ -847,78 +1062,76 @@ adw_timeout(arg) * If it has been through before, then a previous abort has failed, * don't try abort again, reset the bus instead. */ - if (ccb->flags & CCB_ABORT) { - /* abort timed out */ - printf(" AGAIN. Resetting Bus\n"); - /* Lets try resetting the bus! */ + if (ccb->flags & CCB_ABORTED) { + /* + * Abort Timed Out + * Lets try resetting the bus! + */ + printf(" AGAIN. Resetting SCSI Bus\n"); + ccb->flags &= ~CCB_ABORTED; + /* AdvResetSCSIBus() will call sbreset_callback() */ AdvResetSCSIBus(sc); - ccb->timeout = ADW_ABORT_TIMEOUT; - adw_queue_ccb(sc, ccb); } else { - /* abort the operation that has timed out */ + /* + * Abort the operation that has timed out + */ printf("\n"); - ADW_ABORT_CCB(sc, ccb); xs->error = XS_TIMEOUT; - ccb->timeout = ADW_ABORT_TIMEOUT; - ccb->flags |= CCB_ABORT; - adw_queue_ccb(sc, ccb); + ccb->flags |= CCB_ABORTING; + /* ADW_ABORT_CCB() will implicitly call isr_callback() */ + ADW_ABORT_CCB(sc, ccb); } splx(s); } -static void -adw_watchdog(arg) - void *arg; -{ - ADW_CCB *ccb = arg; - struct scsi_xfer *xs = ccb->xs; - struct scsi_link *sc_link = xs->sc_link; - ADW_SOFTC *sc = sc_link->adapter_softc; - int s; - - s = splbio(); - - ccb->flags &= ~CCB_WATCHDOG; - adw_start_ccbs(sc); - - splx(s); -} - - /******************************************************************************/ -/* NARROW and WIDE boards Interrupt callbacks */ +/* WIDE boards Interrupt callbacks */ /******************************************************************************/ /* - * adw_wide_isr_callback() - Second Level Interrupt Handler called by AdvISR() + * adw__isr_callback() - Second Level Interrupt Handler called by AdvISR() * * Interrupt callback function for the Wide SCSI Adv Library. */ static void -adw_wide_isr_callback(sc, scsiq) +adw_isr_callback(sc, scsiq) ADW_SOFTC *sc; ADW_SCSI_REQ_Q *scsiq; { bus_dma_tag_t dmat = sc->sc_dmat; - ADW_CCB *ccb = (ADW_CCB *) scsiq->ccb_ptr; - struct scsi_xfer *xs = ccb->xs; + ADW_CCB *ccb; + struct scsi_xfer *xs; struct scsi_sense_data *s1, *s2; - //int underrun = ASC_FALSE; +// int s; + ccb = adw_ccb_phys_kv(sc, scsiq->ccb_ptr); + untimeout(adw_timeout, ccb); +/* if(ccb->flags & CCB_ABORTING) { + printf("Retrying request\n"); + ccb->flags &= ~CCB_ABORTING; + ccb->flags |= CCB_ABORTED; + s = splbio(); + adw_queue_ccb(sc, ccb); + splx(s); + return; + } +*/ + xs = ccb->xs; + /* * If we were a data transfer, unload the map that described * the data buffer. */ if (xs->datalen) { bus_dmamap_sync(dmat, ccb->dmamap_xfer, - (xs->flags & SCSI_DATA_IN) ? BUS_DMASYNC_POSTREAD : - BUS_DMASYNC_POSTWRITE); + (xs->flags & SCSI_DATA_IN) ? + BUS_DMASYNC_POSTREAD : BUS_DMASYNC_POSTWRITE); bus_dmamap_unload(dmat, ccb->dmamap_xfer); } if ((ccb->flags & CCB_ALLOC) == 0) { @@ -949,31 +1162,38 @@ adw_wide_isr_callback(sc, scsiq) xs->error = XS_DRIVER_STUFFUP; break; } - /* - * If there was an underrun without any other error, - * set DID_ERROR to indicate the underrun error. - * - * Note: There is no way yet to indicate the number - * of underrun bytes. - */ - /* - * if (xs->error == XS_NOERROR && underrun == ASC_TRUE) { - * scp->result = HOST_BYTE(DID_UNDERRUN); } - */ break; + break; case QD_WITH_ERROR: switch (scsiq->host_status) { case QHSTA_NO_ERROR: - if (scsiq->scsi_status == SS_CHK_CONDITION) { + switch(scsiq->scsi_status) { + case SS_CHK_CONDITION: + case SS_CMD_TERMINATED: s1 = &ccb->scsi_sense; s2 = &xs->sense; *s2 = *s1; xs->error = XS_SENSE; - } else { + break; + case SS_TARGET_BUSY: + case SS_RSERV_CONFLICT: + case SS_QUEUE_FULL: xs->error = XS_DRIVER_STUFFUP; + break; + case SS_CONDITION_MET: + case SS_INTERMID: + case SS_INTERMID_COND_MET: + xs->error = XS_DRIVER_STUFFUP; + break; + case SS_GOOD: + break; } break; + case QHSTA_M_SEL_TIMEOUT: + xs->error = XS_DRIVER_STUFFUP; + break; + default: /* Some other QHSTA error occurred. */ xs->error = XS_DRIVER_STUFFUP; @@ -982,13 +1202,51 @@ adw_wide_isr_callback(sc, scsiq) break; case QD_ABORTED_BY_HOST: + xs->error = XS_DRIVER_STUFFUP; + break; + default: xs->error = XS_DRIVER_STUFFUP; break; } - adw_free_ccb(sc, ccb); xs->flags |= ITSDONE; scsi_done(xs); } + + +/* + * adv_async_callback() - Adv Library asynchronous event callback function. + */ +static void +adw_async_callback(sc, code) + ADW_SOFTC *sc; + u_int8_t code; +{ + switch (code) { + case ADV_ASYNC_SCSI_BUS_RESET_DET: + /* + * The firmware detected a SCSI Bus reset. + */ + break; + + case ADV_ASYNC_RDMA_FAILURE: + /* + * Handle RDMA failure by resetting the SCSI Bus and + * possibly the chip if it is unresponsive. Log the error + * with a unique code. + */ + AdvResetSCSIBus(sc); + break; + + case ADV_HOST_SCSI_BUS_RESET: + /* + * Host generated SCSI bus reset occurred. + */ + break; + + default: + break; + } +} |