diff options
-rw-r--r-- | sys/dev/ic/aac.c | 3646 | ||||
-rw-r--r-- | sys/dev/ic/aac_tables.h | 4 | ||||
-rw-r--r-- | sys/dev/ic/aacreg.h | 1426 | ||||
-rw-r--r-- | sys/dev/ic/aacvar.h | 448 | ||||
-rw-r--r-- | sys/dev/pci/aac_pci.c | 39 |
5 files changed, 3814 insertions, 1749 deletions
diff --git a/sys/dev/ic/aac.c b/sys/dev/ic/aac.c index 3373f80911e..2ba71c4b898 100644 --- a/sys/dev/ic/aac.c +++ b/sys/dev/ic/aac.c @@ -1,9 +1,12 @@ -/* $OpenBSD: aac.c,v 1.24 2005/09/15 05:33:39 krw Exp $ */ +/* $OpenBSD: aac.c,v 1.25 2005/11/18 05:39:10 nate Exp $ */ /*- * Copyright (c) 2000 Michael Smith + * Copyright (c) 2001 Scott Long * Copyright (c) 2000 BSDi + * Copyright (c) 2001 Adaptec, Inc. * Copyright (c) 2000 Niklas Hallqvist + * Copyright (c) 2004 Nathan Binkert * All rights reserved. * * Redistribution and use in source and binary forms, with or without @@ -47,7 +50,9 @@ #include <sys/buf.h> #include <sys/device.h> #include <sys/kernel.h> +#include <sys/kthread.h> #include <sys/malloc.h> +#include <sys/rwlock.h> #include <machine/bus.h> @@ -71,43 +76,51 @@ #define AAC_BIGSECS 63 /* mapping 255*63 */ #define AAC_SECS32 0x1f /* round capacity */ -void aac_bio_complete(struct aac_ccb *); -void aac_complete(void *, int); +struct scsi_xfer; + void aac_copy_internal_data(struct scsi_xfer *, u_int8_t *, size_t); -struct scsi_xfer *aac_dequeue(struct aac_softc *); -int aac_dequeue_fib(struct aac_softc *, int, u_int32_t *, - struct aac_fib **); char *aac_describe_code(struct aac_code_lookup *, u_int32_t); void aac_describe_controller(struct aac_softc *); void aac_enqueue(struct aac_softc *, struct scsi_xfer *, int); -void aac_enqueue_ccb(struct aac_softc *, struct aac_ccb *); -int aac_enqueue_fib(struct aac_softc *, int, struct aac_ccb *); +int aac_enqueue_fib(struct aac_softc *, int, struct aac_command *); +int aac_dequeue_fib(struct aac_softc *, int, u_int32_t *, + struct aac_fib **); +int aac_enqueue_response(struct aac_softc *sc, int queue, + struct aac_fib *fib); + void aac_eval_mapping(u_int32_t, int *, int *, int *); -int aac_exec_ccb(struct aac_ccb *); -void aac_free_ccb(struct aac_softc *, struct aac_ccb *); -struct aac_ccb *aac_get_ccb(struct aac_softc *, int); -#if 0 -void aac_handle_aif(struct aac_softc *, struct aac_aif_command *); -#endif +void aac_print_printf(struct aac_softc *); void aac_host_command(struct aac_softc *); void aac_host_response(struct aac_softc *); int aac_init(struct aac_softc *); +int aac_check_firmware(struct aac_softc *); int aac_internal_cache_cmd(struct scsi_xfer *); -int aac_map_command(struct aac_ccb *); -#ifdef AAC_DEBUG -void aac_print_fib(struct aac_softc *, struct aac_fib *, const char *); -#endif + +/* Command Processing */ +void aac_timeout(struct aac_softc *); +void aac_command_timeout(struct aac_command *); +int aac_map_command(struct aac_command *); +void aac_complete(void *); +int aac_bio_command(struct aac_softc *, struct aac_command **); +void aac_bio_complete(struct aac_command *); +int aac_wait_command(struct aac_command *, int); +void aac_create_thread(void *); +void aac_command_thread(void *); + +/* Command Buffer Management */ +void aac_map_command_sg(void *, bus_dma_segment_t *, int, int); +int aac_alloc_commands(struct aac_softc *); +void aac_free_commands(struct aac_softc *); +void aac_unmap_command(struct aac_command *); + int aac_raw_scsi_cmd(struct scsi_xfer *); int aac_scsi_cmd(struct scsi_xfer *); -int aac_start(struct aac_ccb *); -void aac_start_ccbs(struct aac_softc *); +void aac_startio(struct aac_softc *); void aac_startup(struct aac_softc *); +void aac_add_container(struct aac_softc *, struct aac_mntinforesp *, int); +void aac_shutdown(void *); int aac_sync_command(struct aac_softc *, u_int32_t, u_int32_t, u_int32_t, u_int32_t, u_int32_t, u_int32_t *); -int aac_sync_fib(struct aac_softc *, u_int32_t, u_int32_t, void *, - u_int16_t, void *, u_int16_t *); -void aac_timeout(void *); -void aac_unmap_command(struct aac_ccb *); void aac_watchdog(void *); struct cfdriver aac_cd = { @@ -126,15 +139,25 @@ struct scsi_device aac_dev = { NULL, NULL, NULL, NULL }; -/* i960Rx interface */ -int aac_rx_get_fwstatus(struct aac_softc *); -void aac_rx_qnotify(struct aac_softc *, int); -int aac_rx_get_istatus(struct aac_softc *); -void aac_rx_clear_istatus(struct aac_softc *, int); -void aac_rx_set_mailbox(struct aac_softc *, u_int32_t, u_int32_t, - u_int32_t, u_int32_t, u_int32_t); -int aac_rx_get_mailboxstatus(struct aac_softc *); -void aac_rx_set_interrupts(struct aac_softc *, int); +/* Falcon/PPC interface */ +int aac_fa_get_fwstatus(struct aac_softc *); +void aac_fa_qnotify(struct aac_softc *, int); +int aac_fa_get_istatus(struct aac_softc *); +void aac_fa_clear_istatus(struct aac_softc *, int); +void aac_fa_set_mailbox(struct aac_softc *, u_int32_t, u_int32_t, u_int32_t, + u_int32_t, u_int32_t); +int aac_fa_get_mailbox(struct aac_softc *, int); +void aac_fa_set_interrupts(struct aac_softc *, int); + +struct aac_interface aac_fa_interface = { + aac_fa_get_fwstatus, + aac_fa_qnotify, + aac_fa_get_istatus, + aac_fa_clear_istatus, + aac_fa_set_mailbox, + aac_fa_get_mailbox, + aac_fa_set_interrupts +}; /* StrongARM interface */ int aac_sa_get_fwstatus(struct aac_softc *); @@ -143,914 +166,921 @@ int aac_sa_get_istatus(struct aac_softc *); void aac_sa_clear_istatus(struct aac_softc *, int); void aac_sa_set_mailbox(struct aac_softc *, u_int32_t, u_int32_t, u_int32_t, u_int32_t, u_int32_t); -int aac_sa_get_mailboxstatus(struct aac_softc *); +int aac_sa_get_mailbox(struct aac_softc *, int); void aac_sa_set_interrupts(struct aac_softc *, int); -struct aac_interface aac_rx_interface = { - aac_rx_get_fwstatus, - aac_rx_qnotify, - aac_rx_get_istatus, - aac_rx_clear_istatus, - aac_rx_set_mailbox, - aac_rx_get_mailboxstatus, - aac_rx_set_interrupts -}; - struct aac_interface aac_sa_interface = { aac_sa_get_fwstatus, aac_sa_qnotify, aac_sa_get_istatus, aac_sa_clear_istatus, aac_sa_set_mailbox, - aac_sa_get_mailboxstatus, + aac_sa_get_mailbox, aac_sa_set_interrupts }; +/* i960Rx interface */ +int aac_rx_get_fwstatus(struct aac_softc *); +void aac_rx_qnotify(struct aac_softc *, int); +int aac_rx_get_istatus(struct aac_softc *); +void aac_rx_clear_istatus(struct aac_softc *, int); +void aac_rx_set_mailbox(struct aac_softc *, u_int32_t, u_int32_t, + u_int32_t, u_int32_t, u_int32_t); +int aac_rx_get_mailbox(struct aac_softc *, int); +void aac_rx_set_interrupts(struct aac_softc *, int); + +struct aac_interface aac_rx_interface = { + aac_rx_get_fwstatus, + aac_rx_qnotify, + aac_rx_get_istatus, + aac_rx_clear_istatus, + aac_rx_set_mailbox, + aac_rx_get_mailbox, + aac_rx_set_interrupts +}; + #ifdef AAC_DEBUG int aac_debug = AAC_DEBUG; #endif int -aac_attach(sc) - struct aac_softc *sc; +aac_attach(struct aac_softc *sc) { - int i, error; - bus_dma_segment_t seg; - int nsegs; - struct aac_ccb *ccb; + int error; - TAILQ_INIT(&sc->sc_free_ccb); - TAILQ_INIT(&sc->sc_ccbq); - TAILQ_INIT(&sc->sc_completed); - LIST_INIT(&sc->sc_queue); + /* + * Initialise per-controller queues. + */ + aac_initq_free(sc); + aac_initq_ready(sc); + aac_initq_busy(sc); + aac_initq_bio(sc); /* disable interrupts before we enable anything */ AAC_MASK_INTERRUPTS(sc); /* mark controller as suspended until we get ourselves organised */ - sc->sc_state |= AAC_STATE_SUSPEND; + sc->aac_state |= AAC_STATE_SUSPEND; /* - * Initialise the adapter. + * Check that the firmware on the card is supported. */ - error = aac_init(sc); + error = aac_check_firmware(sc); if (error) return (error); - /* - * Print a little information about the controller. + /* + * Initialize locks */ - aac_describe_controller(sc); + AAC_LOCK_INIT(&sc->aac_sync_lock, "AAC sync FIB lock"); + AAC_LOCK_INIT(&sc->aac_aifq_lock, "AAC AIF lock"); + AAC_LOCK_INIT(&sc->aac_io_lock, "AAC I/O lock"); + AAC_LOCK_INIT(&sc->aac_container_lock, "AAC container lock"); + TAILQ_INIT(&sc->aac_container_tqh); - /* Initialize the ccbs */ - for (i = 0; i < AAC_ADAP_NORM_CMD_ENTRIES; i++) { - ccb = &sc->sc_ccbs[i]; - error = bus_dmamap_create(sc->sc_dmat, - (AAC_MAXSGENTRIES - 1) << PGSHIFT, AAC_MAXSGENTRIES, - (AAC_MAXSGENTRIES - 1) << PGSHIFT, 0, - BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, &ccb->ac_dmamap_xfer); - if (error) { - printf("%s: cannot create ccb dmamap (%d)", - sc->sc_dev.dv_xname, error); - /* XXX cleanup */ - return (1); - } - - /* allocate the FIB cluster in DMAable memory and load it */ - if (bus_dmamem_alloc(sc->sc_dmat, sizeof *ccb->ac_fib, 1, 0, - &seg, 1, &nsegs, BUS_DMA_NOWAIT)) { - printf("%s: can't allocate FIB structure\n", - sc->sc_dev.dv_xname); - /* XXX cleanup */ - return (1); - } - ccb->ac_fibphys = seg.ds_addr; - if (bus_dmamem_map(sc->sc_dmat, &seg, nsegs, - sizeof *ccb->ac_fib, (caddr_t *)&ccb->ac_fib, 0)) { - printf("%s: can't map FIB structure\n", - sc->sc_dev.dv_xname); - /* XXX cleanup */ - return (1); - } - - TAILQ_INSERT_TAIL(&sc->sc_free_ccb, &sc->sc_ccbs[i], - ac_chain); - } + /* Initialize the local AIF queue pointers */ + sc->aac_aifq_head = sc->aac_aifq_tail = AAC_AIFQ_LENGTH; - /* Fill in the prototype scsi_link. */ - sc->sc_link.adapter_softc = sc; - sc->sc_link.adapter = &aac_switch; - sc->sc_link.device = &aac_dev; /* - * XXX Theoretically this should be AAC_ADAP_NORM_CMD_ENTRIES but - * XXX in some configurations this can cause "not queued" errors. - * XXX A quarter of that number has been reported to be safe. + * Initialise the adapter. */ - sc->sc_link.openings = AAC_ADAP_NORM_CMD_ENTRIES / 4; - sc->sc_link.adapter_buswidth = AAC_MAX_CONTAINERS; - sc->sc_link.adapter_target = AAC_MAX_CONTAINERS; - - config_found(&sc->sc_dev, &sc->sc_link, scsiprint); - - return (0); -} - -/* - * Look up a text description of a numeric error code and return a pointer to - * same. - */ -char * -aac_describe_code(table, code) - struct aac_code_lookup *table; - u_int32_t code; -{ - int i; + error = aac_init(sc); + if (error) + return (error); - for (i = 0; table[i].string != NULL; i++) - if (table[i].code == code) - return (table[i].string); - return (table[i + 1].string); -} + /* Fill in the prototype scsi_link. */ + sc->aac_link.adapter_softc = sc; + sc->aac_link.adapter = &aac_switch; + sc->aac_link.device = &aac_dev; + sc->aac_link.openings = (sc->total_fibs - 8) / + (sc->aac_container_count ? sc->aac_container_count : 1); + sc->aac_link.adapter_buswidth = AAC_MAX_CONTAINERS; + sc->aac_link.adapter_target = AAC_MAX_CONTAINERS; -void -aac_describe_controller(sc) - struct aac_softc *sc; -{ - u_int8_t buf[AAC_FIB_DATASIZE]; /* XXX a bit big for the stack */ - u_int16_t bufsize; - struct aac_adapter_info *info; - u_int8_t arg; + config_found(&sc->aac_dev, &sc->aac_link, scsiprint); - arg = 0; - if (aac_sync_fib(sc, RequestAdapterInfo, 0, &arg, sizeof arg, &buf, - &bufsize)) { - printf("%s: RequestAdapterInfo failed\n", sc->sc_dev.dv_xname); - return; - } - if (bufsize != sizeof *info) { - printf("%s: " - "RequestAdapterInfo returned wrong data size (%d != %d)\n", - sc->sc_dev.dv_xname, bufsize, sizeof *info); - return; - } - info = (struct aac_adapter_info *)&buf[0]; + /* Create the AIF thread */ + sc->aifthread = 0; + sc->aifflags = 0; + kthread_create_deferred(aac_create_thread, sc); - printf("%s: %s %dMHz, %dMB, %s (%d) Kernel %d.%d-%d\n", - sc->sc_dev.dv_xname, - aac_describe_code(aac_cpu_variant, info->CpuVariant), - info->ClockSpeed, info->TotalMem / (1024 * 1024), - aac_describe_code(aac_battery_platform, info->batteryPlatform), - info->batteryPlatform, info->KernelRevision.external.comp.major, - info->KernelRevision.external.comp.minor, - info->KernelRevision.external.comp.dash); +#if 0 + /* Register the shutdown method to only be called post-dump */ + sc->aac_sdh = shutdownhook_establish(aac_shutdown, (void *)sc); +#endif - /* save the kernel revision structure for later use */ - sc->sc_revision = info->KernelRevision; + return (0); } -int -aac_init(sc) - struct aac_softc *sc; +void +aac_create_thread(void *arg) { - bus_dma_segment_t seg; - int nsegs; - int i, error; - int state = 0; - struct aac_adapter_init *ip; - u_int32_t code; - u_int8_t *qaddr; - - /* - * First wait for the adapter to come ready. - */ - for (i = 0; i < AAC_BOOT_TIMEOUT * 1000; i++) { - code = AAC_GET_FWSTATUS(sc); - if (code & AAC_SELF_TEST_FAILED) { - printf("%s: FATAL: selftest failed\n", - sc->sc_dev.dv_xname); - return (ENXIO); - } - if (code & AAC_KERNEL_PANIC) { - printf("%s: FATAL: controller kernel panic\n", - sc->sc_dev.dv_xname); - return (ENXIO); - } - if (code & AAC_UP_AND_RUNNING) - break; - DELAY(1000); - } - if (i == AAC_BOOT_TIMEOUT * 1000) { - printf("%s: FATAL: controller not coming ready, status %x\n", - sc->sc_dev.dv_xname, code); - return (ENXIO); - } - - if (bus_dmamem_alloc(sc->sc_dmat, sizeof *sc->sc_common, 1, 0, &seg, 1, - &nsegs, BUS_DMA_NOWAIT)) { - printf("%s: can't allocate common structure\n", - sc->sc_dev.dv_xname); - return (ENOMEM); - } - state++; - sc->sc_common_busaddr = seg.ds_addr; - if (bus_dmamem_map(sc->sc_dmat, &seg, nsegs, sizeof *sc->sc_common, - (caddr_t *)&sc->sc_common, 0)) { - printf("%s: can't map common structure\n", - sc->sc_dev.dv_xname); - error = ENOMEM; - goto bail_out; - } - state++; - bzero(sc->sc_common, sizeof *sc->sc_common); - - /* - * Fill in the init structure. This tells the adapter about - * the physical location * of various important shared data - * structures. - */ - ip = &sc->sc_common->ac_init; - ip->InitStructRevision = AAC_INIT_STRUCT_REVISION; - - ip->AdapterFibsPhysicalAddress = - sc->sc_common_busaddr + offsetof(struct aac_common, ac_fibs); - ip->AdapterFibsVirtualAddress = &sc->sc_common->ac_fibs[0]; - ip->AdapterFibsSize = AAC_ADAPTER_FIBS * sizeof(struct aac_fib); - ip->AdapterFibAlign = sizeof(struct aac_fib); - - ip->PrintfBufferAddress = - sc->sc_common_busaddr + offsetof(struct aac_common, ac_printf); - ip->PrintfBufferSize = AAC_PRINTF_BUFSIZE; - - ip->HostPhysMemPages = 0; /* not used? */ - ip->HostElapsedSeconds = 0; /* reset later if invalid */ - - /* - * Initialise FIB queues. Note that it appears that the - * layout of the indexes and the segmentation of the entries - * is mandated by the adapter, which is only told about the - * base of the queue index fields. - * - * The initial values of the indices are assumed to inform the - * adapter of the sizes of the respective queues. - * - * The Linux driver uses a much more complex scheme whereby - * several header * records are kept for each queue. We use a - * couple of generic list manipulation functions which - * 'know' the size of each list by virtue of a table. - */ - qaddr = &sc->sc_common->ac_qbuf[0] + AAC_QUEUE_ALIGN; - qaddr -= (u_int32_t)qaddr % AAC_QUEUE_ALIGN; /* XXX not portable */ - sc->sc_queues = (struct aac_queue_table *)qaddr; - ip->CommHeaderAddress = sc->sc_common_busaddr + - ((char *)sc->sc_queues - (char *)sc->sc_common); - bzero(sc->sc_queues, sizeof(struct aac_queue_table)); - - sc->sc_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] = - AAC_HOST_NORM_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] = - AAC_HOST_NORM_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] = - AAC_HOST_HIGH_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] = - AAC_HOST_HIGH_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] = - AAC_ADAP_NORM_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] = - AAC_ADAP_NORM_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] = - AAC_ADAP_HIGH_CMD_ENTRIES; - sc->sc_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] = - AAC_ADAP_HIGH_CMD_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX] = - AAC_HOST_NORM_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX] = - AAC_HOST_NORM_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX] = - AAC_HOST_HIGH_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX] = - AAC_HOST_HIGH_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX] = - AAC_ADAP_NORM_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX] = - AAC_ADAP_NORM_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX] = - AAC_ADAP_HIGH_RESP_ENTRIES; - sc->sc_queues-> - qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX] = - AAC_ADAP_HIGH_RESP_ENTRIES; - sc->sc_qentries[AAC_HOST_NORM_CMD_QUEUE] = - &sc->sc_queues->qt_HostNormCmdQueue[0]; - sc->sc_qentries[AAC_HOST_HIGH_CMD_QUEUE] = - &sc->sc_queues->qt_HostHighCmdQueue[0]; - sc->sc_qentries[AAC_ADAP_NORM_CMD_QUEUE] = - &sc->sc_queues->qt_AdapNormCmdQueue[0]; - sc->sc_qentries[AAC_ADAP_HIGH_CMD_QUEUE] = - &sc->sc_queues->qt_AdapHighCmdQueue[0]; - sc->sc_qentries[AAC_HOST_NORM_RESP_QUEUE] = - &sc->sc_queues->qt_HostNormRespQueue[0]; - sc->sc_qentries[AAC_HOST_HIGH_RESP_QUEUE] = - &sc->sc_queues->qt_HostHighRespQueue[0]; - sc->sc_qentries[AAC_ADAP_NORM_RESP_QUEUE] = - &sc->sc_queues->qt_AdapNormRespQueue[0]; - sc->sc_qentries[AAC_ADAP_HIGH_RESP_QUEUE] = - &sc->sc_queues->qt_AdapHighRespQueue[0]; + struct aac_softc *sc = arg; - /* - * Do controller-type-specific initialisation - */ - switch (sc->sc_hwif) { - case AAC_HWIF_I960RX: - AAC_SETREG4(sc, AAC_RX_ODBR, ~0); - break; + if (kthread_create(aac_command_thread, sc, &sc->aifthread, "%s", + sc->aac_dev.dv_xname)) { + /* TODO disable aac */ + printf("%s: failed to create kernel thread, disabled", + sc->aac_dev.dv_xname); } + AAC_DPRINTF(AAC_D_MISC, ("%s: aac_create_thread\n", + sc->aac_dev.dv_xname)); - /* - * Give the init structure to the controller. - */ - if (aac_sync_command(sc, AAC_MONKER_INITSTRUCT, - sc->sc_common_busaddr + offsetof(struct aac_common, ac_init), 0, 0, - 0, NULL)) { - printf("%s: error establishing init structure\n", - sc->sc_dev.dv_xname); - error = EIO; - goto bail_out; - } - - aac_startup(sc); - - return (0); - - bail_out: - if (state > 1) - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->sc_common, - sizeof *sc->sc_common); - if (state > 0) - bus_dmamem_free(sc->sc_dmat, &seg, 1); - return (error); } /* * Probe for containers, create disks. */ void -aac_startup (sc) - struct aac_softc *sc; +aac_startup(struct aac_softc *sc) { - struct aac_mntinfo mi; - struct aac_mntinforesponse mir; - u_int16_t rsize; - int i, drv_cyls, drv_hds, drv_secs; + struct aac_fib *fib; + struct aac_mntinfo *mi; + struct aac_mntinforesp *mir = NULL; + int count = 0, i = 0; + + aac_alloc_sync_fib(sc, &fib, 0); + mi = (struct aac_mntinfo *)&fib->data[0]; + + AAC_DPRINTF(AAC_D_MISC, ("%s: aac startup\n", sc->aac_dev.dv_xname)); + + sc->aac_container_count = 0; /* loop over possible containers */ - mi.Command = VM_NameServe; - mi.MntType = FT_FILESYS; - for (i = 0; i < AAC_MAX_CONTAINERS; i++) { + do { /* request information on this container */ - mi.MntCount = i; - if (aac_sync_fib(sc, ContainerCommand, 0, &mi, sizeof mi, &mir, - &rsize)) { - printf("%s: error probing container %d", - sc->sc_dev.dv_xname, i); - continue; - } - /* check response size */ - if (rsize != sizeof mir) { - printf("%s: container info response wrong size " - "(%d should be %d)", - sc->sc_dev.dv_xname, rsize, sizeof mir); + bzero(mi, sizeof(struct aac_mntinfo)); + mi->Command = VM_NameServe; + mi->MntType = FT_FILESYS; + mi->MntCount = i; + if (aac_sync_fib(sc, ContainerCommand, 0, fib, + sizeof(struct aac_mntinfo))) { + printf("%s: error probing container %d\n", + sc->aac_dev.dv_xname, i); continue; } + mir = (struct aac_mntinforesp *)&fib->data[0]; + /* XXX Need to check if count changed */ + count = mir->MntRespCount; + +#if 0 + aac_add_container(sc, mir, 0); +#else /* * Check container volume type for validity. Note - * that many of the possible types * may never show - * up. + * that many of the possible types may never show up. */ - if (mir.Status == ST_OK && - mir.MntTable[0].VolType != CT_NONE) { + if (mir->Status == ST_OK && + mir->MntTable[0].VolType != CT_NONE) { + int drv_cyls, drv_hds, drv_secs; + AAC_DPRINTF(AAC_D_MISC, - ("%d: id %x name '%.16s' size %u type %d", i, - mir.MntTable[0].ObjectId, - mir.MntTable[0].FileSystemName, - mir.MntTable[0].Capacity, - mir.MntTable[0].VolType)); + ("%s: %d: id %x name '%.16s' size %u type %d\n", + sc->aac_dev.dv_xname, i, + mir->MntTable[0].ObjectId, + mir->MntTable[0].FileSystemName, + mir->MntTable[0].Capacity, + mir->MntTable[0].VolType)); - sc->sc_hdr[i].hd_present = 1; - sc->sc_hdr[i].hd_size = mir.MntTable[0].Capacity; + sc->aac_container_count++; + sc->aac_hdr[i].hd_present = 1; + sc->aac_hdr[i].hd_size = mir->MntTable[0].Capacity; /* * Evaluate mapping (sectors per head, heads per cyl) */ - sc->sc_hdr[i].hd_size &= ~AAC_SECS32; - aac_eval_mapping(sc->sc_hdr[i].hd_size, &drv_cyls, - &drv_hds, &drv_secs); - sc->sc_hdr[i].hd_heads = drv_hds; - sc->sc_hdr[i].hd_secs = drv_secs; + sc->aac_hdr[i].hd_size &= ~AAC_SECS32; + aac_eval_mapping(sc->aac_hdr[i].hd_size, &drv_cyls, + &drv_hds, &drv_secs); + sc->aac_hdr[i].hd_heads = drv_hds; + sc->aac_hdr[i].hd_secs = drv_secs; /* Round the size */ - sc->sc_hdr[i].hd_size = drv_cyls * drv_hds * drv_secs; + sc->aac_hdr[i].hd_size = drv_cyls * drv_hds * drv_secs; - sc->sc_hdr[i].hd_devtype = mir.MntTable[0].VolType; + sc->aac_hdr[i].hd_devtype = mir->MntTable[0].VolType; /* XXX Save the name too for use in IDENTIFY later */ } - } +#endif + + i++; + } while ((i < count) && (i < AAC_MAX_CONTAINERS)); + + aac_release_sync_fib(sc); + +#if 0 + /* poke the bus to actually attach the child devices */ + if (bus_generic_attach(sc->aac_dev)) + printf("%s: bus_generic_attach failed\n", + sc->aac_dev.dv_xname); +#endif + /* mark the controller up */ - sc->sc_state &= ~AAC_STATE_SUSPEND; + sc->aac_state &= ~AAC_STATE_SUSPEND; /* enable interrupts now */ AAC_UNMASK_INTERRUPTS(sc); } +#if 0 +/* + * Create a device to respresent a new container + */ void -aac_eval_mapping(size, cyls, heads, secs) - u_int32_t size; - int *cyls, *heads, *secs; +aac_add_container(struct aac_softc *sc, struct aac_mntinforesp *mir, int f) { - *cyls = size / AAC_HEADS / AAC_SECS; - if (*cyls < AAC_MAXCYLS) { - *heads = AAC_HEADS; - *secs = AAC_SECS; - } else { - /* Too high for 64 * 32 */ - *cyls = size / AAC_MEDHEADS / AAC_MEDSECS; - if (*cyls < AAC_MAXCYLS) { - *heads = AAC_MEDHEADS; - *secs = AAC_MEDSECS; - } else { - /* Too high for 127 * 63 */ - *cyls = size / AAC_BIGHEADS / AAC_BIGSECS; - *heads = AAC_BIGHEADS; - *secs = AAC_BIGSECS; - } + struct aac_container *co; + device_t child; + + /* + * Check container volume type for validity. Note that many of + * the possible types may never show up. + */ + if ((mir->Status == ST_OK) && (mir->MntTable[0].VolType != CT_NONE)) { + co = (struct aac_container *)malloc(sizeof *co, M_DEVBUF, + M_NOWAIT); + if (co == NULL) + panic("Out of memory?!\n"); + bzero(co, sizeof *co); + AAC_DPRINTF(AAC_D_MISC, + ("%s: id %x name '%.16s' size %u type %d\n", + sc->aac_dev.dv_xname, + mir->MntTable[0].ObjectId, + mir->MntTable[0].FileSystemName, + mir->MntTable[0].Capacity, + mir->MntTable[0].VolType); + + if ((child = device_add_child(sc->aac_dev, "aacd", -1)) == NULL) + printf("%s: device_add_child failed\n", + sc->aac_dev.dv_xname); + else + device_set_ivars(child, co); + device_set_desc(child, aac_describe_code(aac_container_types, + mir->MntTable[0].VolType)); + co->co_disk = child; + co->co_found = f; + bcopy(&mir->MntTable[0], &co->co_mntobj, + sizeof(struct aac_mntobj)); + AAC_LOCK_ACQUIRE(&sc->aac_container_lock); + TAILQ_INSERT_TAIL(&sc->aac_container_tqh, co, co_link); + AAC_LOCK_RELEASE(&sc->aac_container_lock); } } +#endif -int -aac_raw_scsi_cmd(xs) - struct scsi_xfer *xs; +#if 0 +/* + * Free all of the resources associated with (sc) + * + * Should not be called if the controller is active. + */ +void +aac_free(struct aac_softc *sc) { - AAC_DPRINTF(AAC_D_CMD, ("aac_raw_scsi_cmd ")); - /* XXX Not yet implemented */ - xs->error = XS_DRIVER_STUFFUP; - return (COMPLETE); + debug_called(1); + + /* remove the control device */ + if (sc->aac_dev_t != NULL) + destroy_dev(sc->aac_dev_t); + + /* throw away any FIB buffers, discard the FIB DMA tag */ + aac_free_commands(sc); + if (sc->aac_fib_dmat) + bus_dma_tag_destroy(sc->aac_fib_dmat); + + free(sc->aac_commands, M_AACBUF); + + /* destroy the common area */ + if (sc->aac_common) { + bus_dmamap_unload(sc->aac_common_dmat, sc->aac_common_dmamap); + bus_dmamem_free(sc->aac_common_dmat, sc->aac_common, + sc->aac_common_dmamap); + } + if (sc->aac_common_dmat) + bus_dma_tag_destroy(sc->aac_common_dmat); + + /* disconnect the interrupt handler */ + if (sc->aac_intr) + bus_teardown_intr(sc->aac_dev, sc->aac_irq, sc->aac_intr); + if (sc->aac_irq != NULL) + bus_release_resource(sc->aac_dev, SYS_RES_IRQ, sc->aac_irq_rid, + sc->aac_irq); + + /* destroy data-transfer DMA tag */ + if (sc->aac_buffer_dmat) + bus_dma_tag_destroy(sc->aac_buffer_dmat); + + /* destroy the parent DMA tag */ + if (sc->aac_parent_dmat) + bus_dma_tag_destroy(sc->aac_parent_dmat); + + /* release the register window mapping */ + if (sc->aac_regs_resource != NULL) + bus_release_resource(sc->aac_dev, SYS_RES_MEMORY, + sc->aac_regs_rid, sc->aac_regs_resource); } +/* + * Disconnect from the controller completely, in preparation for unload. + */ int -aac_scsi_cmd(xs) - struct scsi_xfer *xs; +aac_detach(device_t dev) { - struct scsi_link *link = xs->sc_link; - struct aac_softc *sc = link->adapter_softc; - u_int8_t target = link->target; - struct aac_ccb *ccb; - u_int32_t blockno, blockcnt; - struct scsi_rw *rw; - struct scsi_rw_big *rwb; - aac_lock_t lock; - int retval = SUCCESSFULLY_QUEUED; + struct aac_softc *sc; + struct aac_container *co; + struct aac_sim *sim; + int error; - AAC_DPRINTF(AAC_D_CMD, ("aac_scsi_cmd ")); + debug_called(1); - xs->error = XS_NOERROR; + sc = device_get_softc(dev); - if (target >= AAC_MAX_CONTAINERS || !sc->sc_hdr[target].hd_present || - link->lun != 0) { - /* - * XXX Should be XS_SENSE but that would require setting up a - * faked sense too. - */ - xs->error = XS_DRIVER_STUFFUP; - xs->flags |= ITSDONE; - scsi_done(xs); - return (COMPLETE); + if (sc->aac_state & AAC_STATE_OPEN) + return(EBUSY); + + /* Remove the child containers */ + while ((co = TAILQ_FIRST(&sc->aac_container_tqh)) != NULL) { + error = device_delete_child(dev, co->co_disk); + if (error) + return (error); + TAILQ_REMOVE(&sc->aac_container_tqh, co, co_link); + free(co, M_AACBUF); } - lock = AAC_LOCK(sc); + /* Remove the CAM SIMs */ + while ((sim = TAILQ_FIRST(&sc->aac_sim_tqh)) != NULL) { + TAILQ_REMOVE(&sc->aac_sim_tqh, sim, sim_link); + error = device_delete_child(dev, sim->sim_dev); + if (error) + return (error); + free(sim, M_AACBUF); + } - /* Don't double enqueue if we came from aac_chain. */ - if (xs != LIST_FIRST(&sc->sc_queue)) - aac_enqueue(sc, xs, 0); + if (sc->aifflags & AAC_AIFFLAGS_RUNNING) { + sc->aifflags |= AAC_AIFFLAGS_EXIT; + wakeup(sc->aifthread); + tsleep(sc->aac_dev, PUSER | PCATCH, "aacdch", 30 * hz); + } - while ((xs = aac_dequeue(sc))) { - xs->error = XS_NOERROR; - ccb = NULL; - link = xs->sc_link; - target = link->target; - - switch (xs->cmd->opcode) { - case TEST_UNIT_READY: - case REQUEST_SENSE: - case INQUIRY: - case MODE_SENSE: - case START_STOP: - case READ_CAPACITY: -#if 0 - case VERIFY: -#endif - if (!aac_internal_cache_cmd(xs)) { - AAC_UNLOCK(sc, lock); - return (TRY_AGAIN_LATER); - } - xs->flags |= ITSDONE; - scsi_done(xs); - goto ready; + if (sc->aifflags & AAC_AIFFLAGS_RUNNING) + panic("Cannot shutdown AIF thread\n"); - case PREVENT_ALLOW: - AAC_DPRINTF(AAC_D_CMD, ("PREVENT/ALLOW ")); - /* XXX Not yet implemented */ - xs->error = XS_NOERROR; - xs->flags |= ITSDONE; - scsi_done(xs); - goto ready; + if ((error = aac_shutdown(dev))) + return(error); - case SYNCHRONIZE_CACHE: - AAC_DPRINTF(AAC_D_CMD, ("SYNCHRONIZE_CACHE ")); - /* XXX Not yet implemented */ - xs->error = XS_NOERROR; - xs->flags |= ITSDONE; - scsi_done(xs); - goto ready; + EVENTHANDLER_DEREGISTER(shutdown_final, sc->eh); - default: - AAC_DPRINTF(AAC_D_CMD, - ("unknown opc %d ", xs->cmd->opcode)); - /* XXX Not yet implemented */ - xs->error = XS_DRIVER_STUFFUP; - xs->flags |= ITSDONE; - scsi_done(xs); - goto ready; + aac_free(sc); - case READ_COMMAND: - case READ_BIG: - case WRITE_COMMAND: - case WRITE_BIG: - AAC_DPRINTF(AAC_D_CMD, - ("rw opc %d ", xs->cmd->opcode)); - - if (xs->cmd->opcode != SYNCHRONIZE_CACHE) { - /* A read or write operation. */ - if (xs->cmdlen == 6) { - rw = (struct scsi_rw *)xs->cmd; - blockno = _3btol(rw->addr) & - (SRW_TOPADDR << 16 | 0xffff); - blockcnt = - rw->length ? rw->length : 0x100; - } else { - rwb = (struct scsi_rw_big *)xs->cmd; - blockno = _4btol(rwb->addr); - blockcnt = _2btol(rwb->length); - } - if (blockno >= sc->sc_hdr[target].hd_size || - blockno + blockcnt > - sc->sc_hdr[target].hd_size) { - printf( - "%s: out of bounds %u-%u >= %u\n", - sc->sc_dev.dv_xname, blockno, - blockcnt, - sc->sc_hdr[target].hd_size); - /* - * XXX Should be XS_SENSE but that - * would require setting up a faked - * sense too. - */ - xs->error = XS_DRIVER_STUFFUP; - xs->flags |= ITSDONE; - scsi_done(xs); - goto ready; - } - } + return(0); +} + +/* + * Bring the controller down to a dormant state and detach all child devices. + * + * This function is called before detach or system shutdown. + * + * Note that we can assume that the bioq on the controller is empty, as we won't + * allow shutdown if any device is open. + */ +int +aac_shutdown(device_t dev) +{ + struct aac_softc *sc; + struct aac_fib *fib; + struct aac_close_command *cc; - ccb = aac_get_ccb(sc, xs->flags); + debug_called(1); - /* - * We are out of commands, try again in a little while. - */ - if (ccb == NULL) { - xs->error = XS_DRIVER_STUFFUP; - AAC_UNLOCK(sc, lock); - return (TRY_AGAIN_LATER); - } + sc = device_get_softc(dev); - ccb->ac_blockno = blockno; - ccb->ac_blockcnt = blockcnt; - ccb->ac_xs = xs; - ccb->ac_timeout = xs->timeout; - - if (xs->cmd->opcode != SYNCHRONIZE_CACHE && - aac_map_command(ccb)) { - aac_free_ccb(sc, ccb); - xs->error = XS_DRIVER_STUFFUP; - xs->flags |= ITSDONE; - scsi_done(xs); - goto ready; - } + sc->aac_state |= AAC_STATE_SUSPEND; - aac_enqueue_ccb(sc, ccb); - /* XXX what if enqueue did not start a transfer? */ - if (xs->flags & SCSI_POLL) { -#if 0 - if (!aac_wait(sc, ccb, ccb->ac_timeout)) { - AAC_UNLOCK(sc, lock); - printf("%s: command timed out\n", - sc->sc_dev.dv_xname); - xs->error = XS_TIMEOUT; - return (TRY_AGAIN_LATER); - } - xs->flags |= ITSDONE; - scsi_done(xs); -#endif - } - } + /* + * Send a Container shutdown followed by a HostShutdown FIB to the + * controller to convince it that we don't want to talk to it anymore. + * We've been closed and all I/O completed already + */ + device_printf(sc->aac_dev, "shutting down controller..."); + + aac_alloc_sync_fib(sc, &fib, AAC_SYNC_LOCK_FORCE); + cc = (struct aac_close_command *)&fib->data[0]; - ready: + bzero(cc, sizeof(struct aac_close_command)); + cc->Command = VM_CloseAll; + cc->ContainerId = 0xffffffff; + if (aac_sync_fib(sc, ContainerCommand, 0, fib, + sizeof(struct aac_close_command))) + printf("FAILED.\n"); + else + printf("done\n"); + else { + fib->data[0] = 0; /* - * Don't process the queue if we are polling. + * XXX Issuing this command to the controller makes it + * shut down but also keeps it from coming back up + * without a reset of the PCI bus. This is not + * desirable if you are just unloading the driver + * module with the intent to reload it later. */ - if (xs->flags & SCSI_POLL) { - retval = COMPLETE; - break; + if (aac_sync_fib(sc, FsaHostShutdown, AAC_FIBSTATE_SHUTDOWN, + fib, 1)) { + printf("FAILED.\n"); + } else { + printf("done.\n"); } } - AAC_UNLOCK(sc, lock); - return (retval); + AAC_MASK_INTERRUPTS(sc); + + return(0); } -void -aac_copy_internal_data(xs, data, size) - struct scsi_xfer *xs; - u_int8_t *data; - size_t size; +/* + * Bring the controller to a quiescent state, ready for system suspend. + */ +int +aac_suspend(device_t dev) { - size_t copy_cnt; + struct aac_softc *sc; - AAC_DPRINTF(AAC_D_MISC, ("aac_copy_internal_data ")); + debug_called(1); - if (!xs->datalen) - printf("uio move not yet supported\n"); - else { - copy_cnt = MIN(size, xs->datalen); - bcopy(data, xs->data, copy_cnt); - } + sc = device_get_softc(dev); + + sc->aac_state |= AAC_STATE_SUSPEND; + + AAC_MASK_INTERRUPTS(sc); + return(0); } -/* Emulated SCSI operation on cache device */ +/* + * Bring the controller back to a state ready for operation. + */ int -aac_internal_cache_cmd(xs) - struct scsi_xfer *xs; +aac_resume(device_t dev) { - struct scsi_link *link = xs->sc_link; - struct aac_softc *sc = link->adapter_softc; - struct scsi_inquiry_data inq; - struct scsi_sense_data sd; - struct scsi_read_cap_data rcd; - u_int8_t target = link->target; + struct aac_softc *sc; - AAC_DPRINTF(AAC_D_CMD, ("aac_internal_cache_cmd ")); + debug_called(1); - switch (xs->cmd->opcode) { - case TEST_UNIT_READY: - case START_STOP: -#if 0 - case VERIFY: + sc = device_get_softc(dev); + + sc->aac_state &= ~AAC_STATE_SUSPEND; + AAC_UNMASK_INTERRUPTS(sc); + return(0); +} #endif - AAC_DPRINTF(AAC_D_CMD, ("opc %d tgt %d ", xs->cmd->opcode, - target)); - break; - case REQUEST_SENSE: - AAC_DPRINTF(AAC_D_CMD, ("REQUEST SENSE tgt %d ", target)); - bzero(&sd, sizeof sd); - sd.error_code = 0x70; - sd.segment = 0; - sd.flags = SKEY_NO_SENSE; - aac_enc32(sd.info, 0); - sd.extra_len = 0; - aac_copy_internal_data(xs, (u_int8_t *)&sd, sizeof sd); - break; +/* + * Take an interrupt. + */ +int +aac_intr(void *arg) +{ + struct aac_softc *sc = arg; + u_int16_t reason; - case INQUIRY: - AAC_DPRINTF(AAC_D_CMD, ("INQUIRY tgt %d devtype %x ", target, - sc->sc_hdr[target].hd_devtype)); - bzero(&inq, sizeof inq); - /* XXX How do we detect removable/CD-ROM devices? */ - inq.device = T_DIRECT; - inq.dev_qual2 = 0; - inq.version = 2; - inq.response_format = 2; - inq.additional_length = 32; - strlcpy(inq.vendor, "Adaptec", sizeof inq.vendor); - snprintf(inq.product, sizeof inq.product, "Container #%02d", - target); - strlcpy(inq.revision, " ", sizeof inq.revision); - aac_copy_internal_data(xs, (u_int8_t *)&inq, sizeof inq); - break; - case READ_CAPACITY: - AAC_DPRINTF(AAC_D_CMD, ("READ CAPACITY tgt %d ", target)); - bzero(&rcd, sizeof rcd); - _lto4b(sc->sc_hdr[target].hd_size - 1, rcd.addr); - _lto4b(AAC_BLOCK_SIZE, rcd.length); - aac_copy_internal_data(xs, (u_int8_t *)&rcd, sizeof rcd); - break; + /* + * Read the status register directly. This is faster than taking the + * driver lock and reading the queues directly. It also saves having + * to turn parts of the driver lock into a spin mutex, which would be + * ugly. + */ + reason = AAC_GET_ISTATUS(sc); + AAC_CLEAR_ISTATUS(sc, reason); + (void)AAC_GET_ISTATUS(sc); - default: - AAC_DPRINTF(AAC_D_CMD, ("unsupported scsi command %#x tgt %d ", - xs->cmd->opcode, target)); - xs->error = XS_DRIVER_STUFFUP; + if (reason == 0) return (0); + + AAC_DPRINTF(AAC_D_INTR, ("%s: intr: sc=%p: reason=%#x\n", + sc->aac_dev.dv_xname, sc, reason)); + + /* controller wants to talk to us */ + if (reason & (AAC_DB_PRINTF | AAC_DB_COMMAND_READY | + AAC_DB_RESPONSE_READY)) { + + if (reason & AAC_DB_RESPONSE_READY) { + /* handle completion processing */ + if (sc->aifflags & AAC_AIFFLAGS_RUNNING) { + sc->aifflags |= AAC_AIFFLAGS_COMPLETE; + } else { + AAC_LOCK_ACQUIRE(&sc->aac_io_lock); + aac_complete(sc); + AAC_LOCK_RELEASE(&sc->aac_io_lock); + } + } + + + /* + * XXX Make sure that we don't get fooled by strange messages + * that start with a NULL. + */ + if (reason & AAC_DB_PRINTF) + if (sc->aac_common->ac_printf[0] == 0) + sc->aac_common->ac_printf[0] = 32; + + /* + * This might miss doing the actual wakeup. However, the + * msleep that this is waking up has a timeout, so it will + * wake up eventually. AIFs and printfs are low enough + * priority that they can handle hanging out for a few seconds + * if needed. + */ + if (sc->aifthread) + wakeup(sc->aifthread); + } - xs->error = XS_NOERROR; return (1); } /* - * Take an interrupt. + * Command Processing */ -int -aac_intr(arg) - void *arg; + +/* + * Start as much queued I/O as possible on the controller + */ +void +aac_startio(struct aac_softc *sc) { - struct aac_softc *sc = arg; - u_int16_t reason; - int claimed = 0; + struct aac_command *cm; - AAC_DPRINTF(AAC_D_INTR, ("aac_intr(%p) ", sc)); + AAC_DPRINTF(AAC_D_CMD, ("%s: start command", sc->aac_dev.dv_xname)); - reason = AAC_GET_ISTATUS(sc); - AAC_DPRINTF(AAC_D_INTR, ("istatus 0x%04x ", reason)); - - /* controller wants to talk to the log? XXX should we defer this? */ - if (reason & AAC_DB_PRINTF) { - if (sc->sc_common->ac_printf[0]) { - printf("%s: ** %.*s", sc->sc_dev.dv_xname, - AAC_PRINTF_BUFSIZE, sc->sc_common->ac_printf); - sc->sc_common->ac_printf[0] = 0; - } - AAC_CLEAR_ISTATUS(sc, AAC_DB_PRINTF); - AAC_QNOTIFY(sc, AAC_DB_PRINTF); - claimed = 1; + if (sc->flags & AAC_QUEUE_FRZN) { + AAC_DPRINTF(AAC_D_CMD, (": queue frozen")); + return; } - /* Controller has a message for us? */ - if (reason & AAC_DB_COMMAND_READY) { - aac_host_command(sc); - AAC_CLEAR_ISTATUS(sc, AAC_DB_COMMAND_READY); - claimed = 1; - } - - /* Controller has a response for us? */ - if (reason & AAC_DB_RESPONSE_READY) { - aac_host_response(sc); - AAC_CLEAR_ISTATUS(sc, AAC_DB_RESPONSE_READY); - claimed = 1; + AAC_DPRINTF(AAC_D_CMD, ("\n")); + + for (;;) { + /* + * Try to get a command that's been put off for lack of + * resources + */ + cm = aac_dequeue_ready(sc); + + /* + * Try to build a command off the bio queue (ignore error + * return) + */ + if (cm == NULL) { + AAC_DPRINTF(AAC_D_CMD, ("\n")); + aac_bio_command(sc, &cm); + AAC_DPRINTF(AAC_D_CMD, ("%s: start done bio", + sc->aac_dev.dv_xname)); + } + + /* nothing to do? */ + if (cm == NULL) + break; + + /* + * Try to give the command to the controller. Any error is + * catastrophic since it means that bus_dmamap_load() failed. + */ + if (aac_map_command(cm) != 0) + panic("aac: error mapping command %p\n", cm); + + AAC_DPRINTF(AAC_D_CMD, ("\n%s: another command", + sc->aac_dev.dv_xname)); } - /* - * Spurious interrupts that we don't use - reset the mask and clear - * the interrupts. - */ - if (reason & (AAC_DB_SYNC_COMMAND | AAC_DB_COMMAND_NOT_FULL | - AAC_DB_RESPONSE_NOT_FULL)) { - AAC_UNMASK_INTERRUPTS(sc); - AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND | - AAC_DB_COMMAND_NOT_FULL | AAC_DB_RESPONSE_NOT_FULL); - claimed = 1; + AAC_DPRINTF(AAC_D_CMD, ("\n")); +} + +/* + * Deliver a command to the controller; allocate controller resources at the + * last moment when possible. + */ +int +aac_map_command(struct aac_command *cm) +{ + struct aac_softc *sc = cm->cm_sc; + int error = 0; + + AAC_DPRINTF(AAC_D_CMD, (": map command")); + + /* don't map more than once */ + if (cm->cm_flags & AAC_CMD_MAPPED) + panic("aac: command %p already mapped", cm); + + if (cm->cm_datalen != 0) { + error = bus_dmamap_load(sc->aac_dmat, cm->cm_datamap, + cm->cm_data, cm->cm_datalen, NULL, + BUS_DMA_NOWAIT); + if (error) + return (error); + + aac_map_command_sg(cm, cm->cm_datamap->dm_segs, + cm->cm_datamap->dm_nsegs, 0); + } else { + aac_map_command_sg(cm, NULL, 0, 0); } - return (claimed); + return (error); } /* * Handle notification of one or more FIBs coming from the controller. */ void -aac_host_command(struct aac_softc *sc) +aac_command_thread(void *arg) { + struct aac_softc *sc = arg; struct aac_fib *fib; u_int32_t fib_size; + int size, retval; + + AAC_DPRINTF(AAC_D_THREAD, ("%s: aac_command_thread: starting\n", + sc->aac_dev.dv_xname)); + AAC_LOCK_ACQUIRE(&sc->aac_io_lock); + sc->aifflags = AAC_AIFFLAGS_RUNNING; + + while ((sc->aifflags & AAC_AIFFLAGS_EXIT) == 0) { + + AAC_DPRINTF(AAC_D_THREAD, + ("%s: aac_command_thread: aifflags=%#x\n", + sc->aac_dev.dv_xname, sc->aifflags)); + retval = 0; + + if ((sc->aifflags & AAC_AIFFLAGS_PENDING) == 0) { + AAC_DPRINTF(AAC_D_THREAD, + ("%s: command thread sleeping\n", + sc->aac_dev.dv_xname)); + AAC_LOCK_RELEASE(&sc->aac_io_lock); + retval = tsleep(sc->aifthread, PRIBIO, "aifthd", + AAC_PERIODIC_INTERVAL * hz); + AAC_LOCK_ACQUIRE(&sc->aac_io_lock); + } - for (;;) { - if (aac_dequeue_fib(sc, AAC_HOST_NORM_CMD_QUEUE, &fib_size, - &fib)) - break; /* nothing to do */ + if ((sc->aifflags & AAC_AIFFLAGS_COMPLETE) != 0) { + aac_complete(sc); + sc->aifflags &= ~AAC_AIFFLAGS_COMPLETE; + } - switch(fib->Header.Command) { - case AifRequest: -#if 0 - aac_handle_aif(sc, - (struct aac_aif_command *)&fib->data[0]); -#endif + /* + * While we're here, check to see if any commands are stuck. + * This is pretty low-priority, so it's ok if it doesn't + * always fire. + */ + if (retval == EWOULDBLOCK) + aac_timeout(sc); - break; - default: - printf("%s: unknown command from controller\n", - sc->sc_dev.dv_xname); + /* Check the hardware printf message buffer */ + if (sc->aac_common->ac_printf[0] != 0) + aac_print_printf(sc); + + /* Also check to see if the adapter has a command for us. */ + while (aac_dequeue_fib(sc, AAC_HOST_NORM_CMD_QUEUE, + &fib_size, &fib) == 0) { + AAC_PRINT_FIB(sc, fib); - break; - } + + switch (fib->Header.Command) { + case AifRequest: + //aac_handle_aif(sc, fib); + break; + default: + printf("%s: unknown command from controller\n", + sc->aac_dev.dv_xname); + break; + } - /* XXX reply to FIBs requesting responses ?? */ - /* XXX how do we return these FIBs to the controller? */ + if ((fib->Header.XferState == 0) || + (fib->Header.StructType != AAC_FIBTYPE_TFIB)) + break; + + /* Return the AIF to the controller. */ + if (fib->Header.XferState & AAC_FIBSTATE_FROMADAP) { + fib->Header.XferState |= AAC_FIBSTATE_DONEHOST; + *(AAC_FSAStatus*)fib->data = ST_OK; + + /* XXX Compute the Size field? */ + size = fib->Header.Size; + if (size > sizeof(struct aac_fib)) { + size = sizeof(struct aac_fib); + fib->Header.Size = size; + } + + /* + * Since we did not generate this command, it + * cannot go through the normal + * enqueue->startio chain. + */ + aac_enqueue_response(sc, + AAC_ADAP_NORM_RESP_QUEUE, + fib); + } + } } + sc->aifflags &= ~AAC_AIFFLAGS_RUNNING; + AAC_LOCK_RELEASE(&sc->aac_io_lock); + +#if 0 + /* + * if we ever implement detach, we should have detach tsleep + * to wait for this thread to finish + */ + wakeup(sc->aac_dev); +#endif + + AAC_DPRINTF(AAC_D_THREAD, ("%s: aac_command_thread: exiting\n", + sc->aac_dev.dv_xname)); + kthread_exit(0); } /* - * Handle notification of one or more FIBs completed by the controller + * Process completed commands. */ void -aac_host_response(struct aac_softc *sc) +aac_complete(void *context) { - struct aac_ccb *ccb; + struct aac_softc *sc = (struct aac_softc *)context; + struct aac_command *cm; struct aac_fib *fib; u_int32_t fib_size; + AAC_DPRINTF(AAC_D_CMD, ("%s: complete", sc->aac_dev.dv_xname)); + + /* pull completed commands off the queue */ for (;;) { /* look for completed FIBs on our queue */ if (aac_dequeue_fib(sc, AAC_HOST_NORM_RESP_QUEUE, &fib_size, - &fib)) + &fib)) break; /* nothing to do */ - - /* get the command, unmap and queue for later processing */ - ccb = (struct aac_ccb *)fib->Header.SenderData; - if (ccb == NULL) { + + /* get the command, unmap and hand off for processing */ + cm = sc->aac_commands + fib->Header.SenderData; + if (cm == NULL) { AAC_PRINT_FIB(sc, fib); + break; + } + + aac_remove_busy(cm); + aac_unmap_command(cm); + cm->cm_flags |= AAC_CMD_COMPLETED; + + /* is there a completion handler? */ + if (cm->cm_complete != NULL) { + cm->cm_complete(cm); } else { - timeout_del(&ccb->ac_xs->stimeout); - aac_unmap_command(ccb); /* XXX defer? */ - aac_enqueue_completed(ccb); + /* assume that someone is sleeping on this command */ + wakeup(cm); } } - /* handle completion processing */ - aac_complete(sc, 0); + AAC_DPRINTF(AAC_D_CMD, ("\n")); + /* see if we can start some more I/O */ + sc->flags &= ~AAC_QUEUE_FRZN; + aac_startio(sc); } /* - * Process completed commands. + * Get a bio and build a command to go with it. */ -void -aac_complete(void *context, int pending) +int +aac_bio_command(struct aac_softc *sc, struct aac_command **cmp) { - struct aac_softc *sc = (struct aac_softc *)context; - struct aac_ccb *ccb; + struct aac_command *cm; + struct aac_fib *fib; + struct scsi_xfer *xs; + u_int8_t opcode = 0; - /* pull completed commands off the queue */ - for (;;) { - ccb = aac_dequeue_completed(sc); - if (ccb == NULL) - return; - ccb->ac_flags |= AAC_ACF_COMPLETED; + AAC_DPRINTF(AAC_D_CMD, ("%s: bio command", sc->aac_dev.dv_xname)); -#if 0 - /* is there a completion handler? */ - if (ccb->ac_complete != NULL) { - ccb->ac_complete(ccb); + /* get the resources we will need */ + if ((cm = aac_dequeue_bio(sc)) == NULL) + goto fail; + xs = cm->cm_private; + + /* build the FIB */ + fib = cm->cm_fib; + fib->Header.Size = sizeof(struct aac_fib_header); + fib->Header.XferState = + AAC_FIBSTATE_HOSTOWNED | + AAC_FIBSTATE_INITIALISED | + AAC_FIBSTATE_EMPTY | + AAC_FIBSTATE_FROMHOST | + AAC_FIBSTATE_REXPECTED | + AAC_FIBSTATE_NORM | + AAC_FIBSTATE_ASYNC | + AAC_FIBSTATE_FAST_RESPONSE; + + switch(xs->cmd->opcode) { + case READ_COMMAND: + case READ_BIG: + opcode = READ_COMMAND; + break; + case WRITE_COMMAND: + case WRITE_BIG: + opcode = WRITE_COMMAND; + break; + default: + panic("%s: invalid opcode %#x\n", sc->aac_dev.dv_xname, + xs->cmd->opcode); + } + + /* build the read/write request */ + if ((sc->flags & AAC_FLAGS_SG_64BIT) == 0) { + fib->Header.Command = ContainerCommand; + if (opcode == READ_COMMAND) { + struct aac_blockread *br; + br = (struct aac_blockread *)&fib->data[0]; + br->Command = VM_CtBlockRead; + br->ContainerId = xs->sc_link->target; + br->BlockNumber = cm->cm_blkno; + br->ByteCount = cm->cm_bcount * AAC_BLOCK_SIZE; + fib->Header.Size += sizeof(struct aac_blockread); + cm->cm_sgtable = &br->SgMap; + cm->cm_flags |= AAC_CMD_DATAIN; } else { - /* assume that someone is sleeping on this command */ - wakeup(ccb); + struct aac_blockwrite *bw; + bw = (struct aac_blockwrite *)&fib->data[0]; + bw->Command = VM_CtBlockWrite; + bw->ContainerId = xs->sc_link->target; + bw->BlockNumber = cm->cm_blkno; + bw->ByteCount = cm->cm_bcount * AAC_BLOCK_SIZE; + bw->Stable = CUNSTABLE; + fib->Header.Size += sizeof(struct aac_blockwrite); + cm->cm_flags |= AAC_CMD_DATAOUT; + cm->cm_sgtable = &bw->SgMap; + } + } else { + fib->Header.Command = ContainerCommand64; + if (opcode == READ_COMMAND) { + struct aac_blockread64 *br; + br = (struct aac_blockread64 *)&fib->data[0]; + br->Command = VM_CtHostRead64; + br->ContainerId = xs->sc_link->target; + br->BlockNumber = cm->cm_blkno; + br->SectorCount = cm->cm_bcount; + br->Pad = 0; + br->Flags = 0; + fib->Header.Size += sizeof(struct aac_blockread64); + cm->cm_flags |= AAC_CMD_DATAOUT; + (struct aac_sg_table64 *)cm->cm_sgtable = &br->SgMap64; + } else { + struct aac_blockwrite64 *bw; + bw = (struct aac_blockwrite64 *)&fib->data[0]; + bw->Command = VM_CtHostWrite64; + bw->ContainerId = xs->sc_link->target; + bw->BlockNumber = cm->cm_blkno; + bw->SectorCount = cm->cm_bcount; + bw->Pad = 0; + bw->Flags = 0; + fib->Header.Size += sizeof(struct aac_blockwrite64); + cm->cm_flags |= AAC_CMD_DATAIN; + (struct aac_sg_table64 *)cm->cm_sgtable = &bw->SgMap64; } -#else - aac_bio_complete(ccb); -#endif } + + *cmp = cm; + AAC_DPRINTF(AAC_D_CMD, ("\n")); + return(0); + +fail: + AAC_DPRINTF(AAC_D_CMD, ("\n")); + return(ENOMEM); } /* * Handle a bio-instigated command that has been completed. */ void -aac_bio_complete(struct aac_ccb *ccb) +aac_bio_complete(struct aac_command *cm) { - struct scsi_xfer *xs = ccb->ac_xs; - struct aac_softc *sc = xs->sc_link->adapter_softc; - struct buf *bp = xs->bp; struct aac_blockread_response *brr; struct aac_blockwrite_response *bwr; + struct scsi_xfer *xs = (struct scsi_xfer *)cm->cm_private; + struct buf *bp = xs->bp; AAC_FSAStatus status; + int s; + + AAC_DPRINTF(AAC_D_CMD, + ("%s: bio complete\n", cm->cm_sc->aac_dev.dv_xname)); + s = splbio(); + aac_release_command(cm); if (bp == NULL) - goto done; + goto exit; /* fetch relevant status and then release the command */ if (bp->b_flags & B_READ) { - brr = (struct aac_blockread_response *)&ccb->ac_fib->data[0]; + brr = (struct aac_blockread_response *)&cm->cm_fib->data[0]; status = brr->Status; } else { - bwr = (struct aac_blockwrite_response *)&ccb->ac_fib->data[0]; + bwr = (struct aac_blockwrite_response *)&cm->cm_fib->data[0]; status = bwr->Status; } - aac_free_ccb(sc, ccb); /* fix up the bio based on status */ if (status == ST_OK) { @@ -1058,32 +1088,634 @@ aac_bio_complete(struct aac_ccb *ccb) } else { bp->b_error = EIO; bp->b_flags |= B_ERROR; - - /* XXX be more verbose? */ - printf("%s: I/O error %d (%s)\n", sc->sc_dev.dv_xname, - status, AAC_COMMAND_STATUS(status)); + + /* pass an error string out to the disk layer */ + aac_describe_code(aac_command_status_table, status); } -done: + exit: + xs->error = XS_NOERROR; + xs->resid = 0; + xs->flags |= ITSDONE; scsi_done(xs); + splx(s); +} + +/* + * Submit a command to the controller, return when it completes. + * XXX This is very dangerous! If the card has gone out to lunch, we could + * be stuck here forever. At the same time, signals are not caught + * because there is a risk that a signal could wakeup the tsleep before + * the card has a chance to complete the command. The passed in timeout + * is ignored for the same reason. Since there is no way to cancel a + * command in progress, we should probably create a 'dead' queue where + * commands go that have been interrupted/timed-out/etc, that keeps them + * out of the free pool. That way, if the card is just slow, it won't + * spam the memory of a command that has been recycled. + */ +int +aac_wait_command(struct aac_command *cm, int timeout) +{ + struct aac_softc *sc = cm->cm_sc; + int error = 0; + + AAC_DPRINTF(AAC_D_CMD, (": wait for command")); + + /* Put the command on the ready queue and get things going */ + cm->cm_queue = AAC_ADAP_NORM_CMD_QUEUE; + aac_enqueue_ready(cm); + AAC_DPRINTF(AAC_D_CMD, ("\n")); + aac_startio(sc); + while (!(cm->cm_flags & AAC_CMD_COMPLETED) && (error != EWOULDBLOCK)) { + AAC_DPRINTF(AAC_D_MISC, ("%s: sleeping until command done\n", + sc->aac_dev.dv_xname)); + AAC_LOCK_RELEASE(&sc->aac_io_lock); + error = tsleep(cm, PRIBIO, "aacwait", timeout); + AAC_LOCK_ACQUIRE(&sc->aac_io_lock); + } + return (error); +} + +/* + *Command Buffer Management + */ + +/* + * Allocate a command. + */ +int +aac_alloc_command(struct aac_softc *sc, struct aac_command **cmp) +{ + struct aac_command *cm; + + AAC_DPRINTF(AAC_D_CMD, (": allocate command")); + if ((cm = aac_dequeue_free(sc)) == NULL) { + AAC_DPRINTF(AAC_D_CMD, (" failed")); + return (EBUSY); + } + + *cmp = cm; + return(0); +} + +/* + * Release a command back to the freelist. + */ +void +aac_release_command(struct aac_command *cm) +{ + AAC_DPRINTF(AAC_D_CMD, (": release command")); + + /* (re)initialise the command/FIB */ + cm->cm_sgtable = NULL; + cm->cm_flags = 0; + cm->cm_complete = NULL; + cm->cm_private = NULL; + cm->cm_fib->Header.XferState = AAC_FIBSTATE_EMPTY; + cm->cm_fib->Header.StructType = AAC_FIBTYPE_TFIB; + cm->cm_fib->Header.Flags = 0; + cm->cm_fib->Header.SenderSize = sizeof(struct aac_fib); + + /* + * These are duplicated in aac_start to cover the case where an + * intermediate stage may have destroyed them. They're left + * initialised here for debugging purposes only. + */ + cm->cm_fib->Header.ReceiverFibAddress = (u_int32_t)cm->cm_fibphys; + cm->cm_fib->Header.SenderData = 0; + + aac_enqueue_free(cm); +} + +/* + * Allocate and initialise commands/FIBs for this adapter. + */ +int +aac_alloc_commands(struct aac_softc *sc) +{ + struct aac_command *cm; + struct aac_fibmap *fm; + int i, error; + + if (sc->total_fibs + AAC_FIB_COUNT > sc->aac_max_fibs) + return (ENOMEM); + + fm = malloc(sizeof(struct aac_fibmap), M_DEVBUF, M_NOWAIT); + if (fm == NULL) + goto exit; + bzero(fm, sizeof(struct aac_fibmap)); + + /* allocate the FIBs in DMAable memory and load them */ + if (bus_dmamem_alloc(sc->aac_dmat, AAC_FIBMAP_SIZE, PAGE_SIZE, 0, + &fm->aac_seg, 1, &fm->aac_nsegs, BUS_DMA_NOWAIT)) { + printf("%s: can't alloc FIBs\n", sc->aac_dev.dv_xname); + error = ENOBUFS; + goto exit_alloc; + } + + if (bus_dmamem_map(sc->aac_dmat, &fm->aac_seg, 1, + AAC_FIBMAP_SIZE, (caddr_t *)&fm->aac_fibs, BUS_DMA_NOWAIT)) { + printf("%s: can't map FIB structure\n", sc->aac_dev.dv_xname); + error = ENOBUFS; + goto exit_map; + } + + if (bus_dmamap_create(sc->aac_dmat, AAC_FIBMAP_SIZE, 1, + AAC_FIBMAP_SIZE, 0, BUS_DMA_NOWAIT, &fm->aac_fibmap)) { + printf("%s: can't create dma map\n", sc->aac_dev.dv_xname); + error = ENOBUFS; + goto exit_create; + } + + if (bus_dmamap_load(sc->aac_dmat, fm->aac_fibmap, fm->aac_fibs, + AAC_FIBMAP_SIZE, NULL, BUS_DMA_NOWAIT)) { + printf("%s: can't load dma map\n", sc->aac_dev.dv_xname); + error = ENOBUFS; + goto exit_load; + } + + /* initialise constant fields in the command structure */ + AAC_LOCK_ACQUIRE(&sc->aac_io_lock); + bzero(fm->aac_fibs, AAC_FIB_COUNT * sizeof(struct aac_fib)); + for (i = 0; i < AAC_FIB_COUNT; i++) { + cm = sc->aac_commands + sc->total_fibs; + fm->aac_commands = cm; + cm->cm_sc = sc; + cm->cm_fib = fm->aac_fibs + i; + cm->cm_fibphys = fm->aac_fibmap->dm_segs[0].ds_addr + + (i * sizeof(struct aac_fib)); + cm->cm_index = sc->total_fibs; + + if (bus_dmamap_create(sc->aac_dmat, MAXBSIZE, AAC_MAXSGENTRIES, + MAXBSIZE, 0, BUS_DMA_NOWAIT, &cm->cm_datamap)) { + break; + } + aac_release_command(cm); + sc->total_fibs++; + } + + if (i > 0) { + TAILQ_INSERT_TAIL(&sc->aac_fibmap_tqh, fm, fm_link); + AAC_DPRINTF(AAC_D_MISC, ("%s: total_fibs= %d\n", + sc->aac_dev.dv_xname, + sc->total_fibs)); + AAC_LOCK_RELEASE(&sc->aac_io_lock); + return (0); + } + + exit_load: + bus_dmamap_destroy(sc->aac_dmat, fm->aac_fibmap); + exit_create: + bus_dmamem_unmap(sc->aac_dmat, (caddr_t)fm->aac_fibs, AAC_FIBMAP_SIZE); + exit_map: + bus_dmamem_free(sc->aac_dmat, &fm->aac_seg, fm->aac_nsegs); + exit_alloc: + free(fm, M_DEVBUF); + exit: + AAC_LOCK_RELEASE(&sc->aac_io_lock); + return (error); +} + +/* + * Free FIBs owned by this adapter. + */ +void +aac_free_commands(struct aac_softc *sc) +{ + struct aac_fibmap *fm; + struct aac_command *cm; + int i; + + while ((fm = TAILQ_FIRST(&sc->aac_fibmap_tqh)) != NULL) { + + TAILQ_REMOVE(&sc->aac_fibmap_tqh, fm, fm_link); + + /* + * We check against total_fibs to handle partially + * allocated blocks. + */ + for (i = 0; i < AAC_FIB_COUNT && sc->total_fibs--; i++) { + cm = fm->aac_commands + i; + bus_dmamap_destroy(sc->aac_dmat, cm->cm_datamap); + } + + bus_dmamap_unload(sc->aac_dmat, fm->aac_fibmap); + bus_dmamap_destroy(sc->aac_dmat, fm->aac_fibmap); + bus_dmamem_unmap(sc->aac_dmat, (caddr_t)fm->aac_fibs, + AAC_FIBMAP_SIZE); + bus_dmamem_free(sc->aac_dmat, &fm->aac_seg, fm->aac_nsegs); + free(fm, M_DEVBUF); + } +} + + +/* + * Command-mapping helper function - populate this command's s/g table. + */ +void +aac_map_command_sg(void *arg, bus_dma_segment_t *segs, int nseg, int error) +{ + struct aac_command *cm = arg; + struct aac_softc *sc = cm->cm_sc; + struct aac_fib *fib = cm->cm_fib; + int i; + + /* copy into the FIB */ + if (cm->cm_sgtable != NULL) { + if ((cm->cm_sc->flags & AAC_FLAGS_SG_64BIT) == 0) { + struct aac_sg_table *sg = cm->cm_sgtable; + sg->SgCount = nseg; + for (i = 0; i < nseg; i++) { + sg->SgEntry[i].SgAddress = segs[i].ds_addr; + sg->SgEntry[i].SgByteCount = segs[i].ds_len; + } + /* update the FIB size for the s/g count */ + fib->Header.Size += nseg * sizeof(struct aac_sg_entry); + } else { + struct aac_sg_table64 *sg; + sg = (struct aac_sg_table64 *)cm->cm_sgtable; + sg->SgCount = nseg; + for (i = 0; i < nseg; i++) { + sg->SgEntry64[i].SgAddress = segs[i].ds_addr; + sg->SgEntry64[i].SgByteCount = segs[i].ds_len; + } + /* update the FIB size for the s/g count */ + fib->Header.Size += nseg*sizeof(struct aac_sg_entry64); + } + } + + /* Fix up the address values in the FIB. Use the command array index + * instead of a pointer since these fields are only 32 bits. Shift + * the SenderFibAddress over to make room for the fast response bit. + */ + cm->cm_fib->Header.SenderFibAddress = (cm->cm_index << 1); + cm->cm_fib->Header.ReceiverFibAddress = cm->cm_fibphys; + + /* save a pointer to the command for speedy reverse-lookup */ + cm->cm_fib->Header.SenderData = cm->cm_index; + + if (cm->cm_flags & AAC_CMD_DATAIN) + bus_dmamap_sync(sc->aac_dmat, cm->cm_datamap, 0, + cm->cm_datamap->dm_mapsize, + BUS_DMASYNC_PREREAD); + if (cm->cm_flags & AAC_CMD_DATAOUT) + bus_dmamap_sync(sc->aac_dmat, cm->cm_datamap, 0, + cm->cm_datamap->dm_mapsize, + BUS_DMASYNC_PREWRITE); + cm->cm_flags |= AAC_CMD_MAPPED; + + /* put the FIB on the outbound queue */ + if (aac_enqueue_fib(sc, cm->cm_queue, cm) == EBUSY) { + aac_remove_busy(cm); + aac_unmap_command(cm); + aac_requeue_ready(cm); + } +} + +/* + * Unmap a command from controller-visible space. + */ +void +aac_unmap_command(struct aac_command *cm) +{ + struct aac_softc *sc = cm->cm_sc; + + if (!(cm->cm_flags & AAC_CMD_MAPPED)) + return; + + if (cm->cm_datalen != 0) { + if (cm->cm_flags & AAC_CMD_DATAIN) + bus_dmamap_sync(sc->aac_dmat, cm->cm_datamap, 0, + cm->cm_datamap->dm_mapsize, + BUS_DMASYNC_POSTREAD); + if (cm->cm_flags & AAC_CMD_DATAOUT) + bus_dmamap_sync(sc->aac_dmat, cm->cm_datamap, 0, + cm->cm_datamap->dm_mapsize, + BUS_DMASYNC_POSTWRITE); + + bus_dmamap_unload(sc->aac_dmat, cm->cm_datamap); + } + cm->cm_flags &= ~AAC_CMD_MAPPED; +} + +/* + * Hardware Interface + */ + +/* + * Initialise the adapter. + */ +int +aac_check_firmware(struct aac_softc *sc) +{ + u_int32_t major, minor, options; + + /* + * Retrieve the firmware version numbers. Dell PERC2/QC cards with + * firmware version 1.x are not compatible with this driver. + */ + if (sc->flags & AAC_FLAGS_PERC2QC) { + if (aac_sync_command(sc, AAC_MONKER_GETKERNVER, 0, 0, 0, 0, + NULL)) { + printf("%s: Error reading firmware version\n", + sc->aac_dev.dv_xname); + return (EIO); + } + + /* These numbers are stored as ASCII! */ + major = (AAC_GET_MAILBOX(sc, 1) & 0xff) - 0x30; + minor = (AAC_GET_MAILBOX(sc, 2) & 0xff) - 0x30; + if (major == 1) { + printf("%s: Firmware version %d.%d is not supported\n", + sc->aac_dev.dv_xname, major, minor); + return (EINVAL); + } + } + + /* + * Retrieve the capabilities/supported options word so we know what + * work-arounds to enable. + */ + if (aac_sync_command(sc, AAC_MONKER_GETINFO, 0, 0, 0, 0, NULL)) { + printf("%s: RequestAdapterInfo failed\n", + sc->aac_dev.dv_xname); + return (EIO); + } + options = AAC_GET_MAILBOX(sc, 1); + sc->supported_options = options; + + if ((options & AAC_SUPPORTED_4GB_WINDOW) != 0 && + (sc->flags & AAC_FLAGS_NO4GB) == 0) + sc->flags |= AAC_FLAGS_4GB_WINDOW; + if (options & AAC_SUPPORTED_NONDASD) + sc->flags |= AAC_FLAGS_ENABLE_CAM; + if ((options & AAC_SUPPORTED_SGMAP_HOST64) != 0 + && (sizeof(bus_addr_t) > 4)) { + printf("%s: Enabling 64-bit address support\n", + sc->aac_dev.dv_xname); + sc->flags |= AAC_FLAGS_SG_64BIT; + } + + /* Check for broken hardware that does a lower number of commands */ + if ((sc->flags & AAC_FLAGS_256FIBS) == 0) + sc->aac_max_fibs = AAC_MAX_FIBS; + else + sc->aac_max_fibs = 256; + + return (0); +} + +int +aac_init(struct aac_softc *sc) +{ + bus_dma_segment_t seg; + int nsegs; + int i, error; + int state = 0; + struct aac_adapter_init *ip; + time_t then; + u_int32_t code, qoffset; + + /* + * First wait for the adapter to come ready. + */ + then = time_second; + for (i = 0; i < AAC_BOOT_TIMEOUT * 1000; i++) { + code = AAC_GET_FWSTATUS(sc); + if (code & AAC_SELF_TEST_FAILED) { + printf("%s: FATAL: selftest failed\n", + sc->aac_dev.dv_xname); + return (ENXIO); + } + if (code & AAC_KERNEL_PANIC) { + printf("%s: FATAL: controller kernel panic\n", + sc->aac_dev.dv_xname); + return (ENXIO); + } + if (code & AAC_UP_AND_RUNNING) + break; + DELAY(1000); + } + if (i == AAC_BOOT_TIMEOUT * 1000) { + printf("%s: FATAL: controller not coming ready, status %x\n", + sc->aac_dev.dv_xname, code); + return (ENXIO); + } + + /* + * Work around a bug in the 2120 and 2200 that cannot DMA commands + * below address 8192 in physical memory. + * XXX If the padding is not needed, can it be put to use instead + * of ignored? + */ + if (bus_dmamem_alloc(sc->aac_dmat, AAC_COMMON_ALLOCSIZE, PAGE_SIZE, 0, + &seg, 1, &nsegs, BUS_DMA_NOWAIT)) { + printf("%s: can't allocate common structure\n", + sc->aac_dev.dv_xname); + return (ENOMEM); + } + state++; + + if (bus_dmamem_map(sc->aac_dmat, &seg, nsegs, AAC_COMMON_ALLOCSIZE, + (caddr_t *)&sc->aac_common, BUS_DMA_NOWAIT)) { + printf("%s: can't map common structure\n", + sc->aac_dev.dv_xname); + error = ENOMEM; + goto bail_out; + } + state++; + + if (bus_dmamap_create(sc->aac_dmat, AAC_COMMON_ALLOCSIZE, 1, + AAC_COMMON_ALLOCSIZE, 0, BUS_DMA_NOWAIT, &sc->aac_common_map)) { + printf("%s: can't create dma map\n", sc->aac_dev.dv_xname); + error = ENOBUFS; + goto bail_out; + } + state++; + + if (bus_dmamap_load(sc->aac_dmat, sc->aac_common_map, sc->aac_common, + AAC_COMMON_ALLOCSIZE, NULL, BUS_DMA_NOWAIT)) { + printf("%s: can't load dma map\n", sc->aac_dev.dv_xname); + error = ENOBUFS; + goto bail_out; + } + state++; + + sc->aac_common_busaddr = sc->aac_common_map->dm_segs[0].ds_addr; + + if (sc->aac_common_busaddr < 8192) { + (uint8_t *)sc->aac_common += 8192; + sc->aac_common_busaddr += 8192; + } + bzero(sc->aac_common, sizeof *sc->aac_common); + + /* Allocate some FIBs and associated command structs */ + TAILQ_INIT(&sc->aac_fibmap_tqh); + sc->aac_commands = malloc(AAC_MAX_FIBS * sizeof(struct aac_command), + M_DEVBUF, M_WAITOK); + bzero(sc->aac_commands, AAC_MAX_FIBS * sizeof(struct aac_command)); + while (sc->total_fibs < AAC_MAX_FIBS) { + if (aac_alloc_commands(sc) != 0) + break; + } + if (sc->total_fibs == 0) + goto out; + + /* + * Fill in the init structure. This tells the adapter about the + * physical location of various important shared data structures. + */ + ip = &sc->aac_common->ac_init; + ip->InitStructRevision = AAC_INIT_STRUCT_REVISION; + ip->MiniPortRevision = AAC_INIT_STRUCT_MINIPORT_REVISION; + + ip->AdapterFibsPhysicalAddress = sc->aac_common_busaddr + + offsetof(struct aac_common, ac_fibs); + ip->AdapterFibsVirtualAddress = 0; + ip->AdapterFibsSize = AAC_ADAPTER_FIBS * sizeof(struct aac_fib); + ip->AdapterFibAlign = sizeof(struct aac_fib); + + ip->PrintfBufferAddress = sc->aac_common_busaddr + + offsetof(struct aac_common, ac_printf); + ip->PrintfBufferSize = AAC_PRINTF_BUFSIZE; + + /* + * The adapter assumes that pages are 4K in size, except on some + * broken firmware versions that do the page->byte conversion twice, + * therefore 'assuming' that this value is in 16MB units (2^24). + * Round up since the granularity is so high. + */ + ip->HostPhysMemPages = ctob(physmem) / AAC_PAGE_SIZE; + if (sc->flags & AAC_FLAGS_BROKEN_MEMMAP) { + ip->HostPhysMemPages = + (ip->HostPhysMemPages + AAC_PAGE_SIZE) / AAC_PAGE_SIZE; + } + ip->HostElapsedSeconds = time_second; /* reset later if invalid */ + + /* + * Initialise FIB queues. Note that it appears that the layout of the + * indexes and the segmentation of the entries may be mandated by the + * adapter, which is only told about the base of the queue index fields. + * + * The initial values of the indices are assumed to inform the adapter + * of the sizes of the respective queues, and theoretically it could + * work out the entire layout of the queue structures from this. We + * take the easy route and just lay this area out like everyone else + * does. + * + * The Linux driver uses a much more complex scheme whereby several + * header records are kept for each queue. We use a couple of generic + * list manipulation functions which 'know' the size of each list by + * virtue of a table. + */ + qoffset = offsetof(struct aac_common, ac_qbuf) + AAC_QUEUE_ALIGN; + qoffset &= ~(AAC_QUEUE_ALIGN - 1); + sc->aac_queues = + (struct aac_queue_table *)((caddr_t)sc->aac_common + qoffset); + ip->CommHeaderAddress = sc->aac_common_busaddr + qoffset; + + sc->aac_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] = + AAC_HOST_NORM_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] = + AAC_HOST_NORM_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] = + AAC_HOST_HIGH_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] = + AAC_HOST_HIGH_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] = + AAC_ADAP_NORM_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] = + AAC_ADAP_NORM_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] = + AAC_ADAP_HIGH_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] = + AAC_ADAP_HIGH_CMD_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX]= + AAC_HOST_NORM_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX]= + AAC_HOST_NORM_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX]= + AAC_HOST_HIGH_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX]= + AAC_HOST_HIGH_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX]= + AAC_ADAP_NORM_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX]= + AAC_ADAP_NORM_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX]= + AAC_ADAP_HIGH_RESP_ENTRIES; + sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX]= + AAC_ADAP_HIGH_RESP_ENTRIES; + sc->aac_qentries[AAC_HOST_NORM_CMD_QUEUE] = + &sc->aac_queues->qt_HostNormCmdQueue[0]; + sc->aac_qentries[AAC_HOST_HIGH_CMD_QUEUE] = + &sc->aac_queues->qt_HostHighCmdQueue[0]; + sc->aac_qentries[AAC_ADAP_NORM_CMD_QUEUE] = + &sc->aac_queues->qt_AdapNormCmdQueue[0]; + sc->aac_qentries[AAC_ADAP_HIGH_CMD_QUEUE] = + &sc->aac_queues->qt_AdapHighCmdQueue[0]; + sc->aac_qentries[AAC_HOST_NORM_RESP_QUEUE] = + &sc->aac_queues->qt_HostNormRespQueue[0]; + sc->aac_qentries[AAC_HOST_HIGH_RESP_QUEUE] = + &sc->aac_queues->qt_HostHighRespQueue[0]; + sc->aac_qentries[AAC_ADAP_NORM_RESP_QUEUE] = + &sc->aac_queues->qt_AdapNormRespQueue[0]; + sc->aac_qentries[AAC_ADAP_HIGH_RESP_QUEUE] = + &sc->aac_queues->qt_AdapHighRespQueue[0]; + + /* + * Do controller-type-specific initialisation + */ + switch (sc->aac_hwif) { + case AAC_HWIF_I960RX: + AAC_SETREG4(sc, AAC_RX_ODBR, ~0); + break; + } + + /* + * Give the init structure to the controller. + */ + if (aac_sync_command(sc, AAC_MONKER_INITSTRUCT, + sc->aac_common_busaddr + + offsetof(struct aac_common, ac_init), 0, 0, 0, + NULL)) { + printf("%s: error establishing init structure\n", + sc->aac_dev.dv_xname); + error = EIO; + goto bail_out; + } + + aac_describe_controller(sc); + aac_startup(sc); + + return (0); + + bail_out: + if (state > 3) + bus_dmamap_unload(sc->aac_dmat, sc->aac_common_map); + if (state > 2) + bus_dmamap_destroy(sc->aac_dmat, sc->aac_common_map); + if (state > 1) + bus_dmamem_unmap(sc->aac_dmat, (caddr_t)sc->aac_common, + sizeof *sc->aac_common); + if (state > 0) + bus_dmamem_free(sc->aac_dmat, &seg, 1); + + out: + return (error); } /* * Send a synchronous command to the controller and wait for a result. */ int -aac_sync_command(sc, command, arg0, arg1, arg2, arg3, sp) - struct aac_softc *sc; - u_int32_t command; - u_int32_t arg0; - u_int32_t arg1; - u_int32_t arg2; - u_int32_t arg3; - u_int32_t *sp; +aac_sync_command(struct aac_softc *sc, u_int32_t command, u_int32_t arg0, + u_int32_t arg1, u_int32_t arg2, u_int32_t arg3, u_int32_t *sp) { +// time_t then; int i; u_int32_t status; - aac_lock_t lock = AAC_LOCK(sc); + u_int16_t reason; /* populate the mailbox */ AAC_SET_MAILBOX(sc, command, arg0, arg1, arg2, arg3); @@ -1093,188 +1725,485 @@ aac_sync_command(sc, command, arg0, arg1, arg2, arg3, sp) /* then set it to signal the adapter */ AAC_QNOTIFY(sc, AAC_DB_SYNC_COMMAND); + +#if 0 + /* spin waiting for the command to complete */ + then = time_second; + do { + if (time_second > (then + AAC_IMMEDIATE_TIMEOUT)) { + AAC_DPRINTF(AAC_D_MISC, ("timed out")); + return(EIO); + } + } while (!(AAC_GET_ISTATUS(sc) & AAC_DB_SYNC_COMMAND)); +#else DELAY(AAC_SYNC_DELAY); /* spin waiting for the command to complete */ for (i = 0; i < AAC_IMMEDIATE_TIMEOUT * 1000; i++) { - if (AAC_GET_ISTATUS(sc) & AAC_DB_SYNC_COMMAND) + reason = AAC_GET_ISTATUS(sc); + if (reason & AAC_DB_SYNC_COMMAND) + break; + reason = AAC_GET_ISTATUS(sc); + if (reason & AAC_DB_SYNC_COMMAND) + break; + reason = AAC_GET_ISTATUS(sc); + if (reason & AAC_DB_SYNC_COMMAND) break; DELAY(1000); } if (i == AAC_IMMEDIATE_TIMEOUT * 1000) { - AAC_UNLOCK(sc, lock); + printf("aac_sync_command: failed, reason=%#x\n", reason); return (EIO); } +#endif /* clear the completion flag */ AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND); /* get the command status */ - status = AAC_GET_MAILBOXSTATUS(sc); - AAC_UNLOCK(sc, lock); + status = AAC_GET_MAILBOX(sc, 0); + if (sp != NULL) *sp = status; - return (0); /* check command return status? */ + + return(0); +} + +/* + * Grab the sync fib area. + */ +int +aac_alloc_sync_fib(struct aac_softc *sc, struct aac_fib **fib, int flags) +{ + + /* + * If the force flag is set, the system is shutting down, or in + * trouble. Ignore the mutex. + */ + if (!(flags & AAC_SYNC_LOCK_FORCE)) + AAC_LOCK_ACQUIRE(&sc->aac_sync_lock); + + *fib = &sc->aac_common->ac_sync_fib; + + return (1); +} + +/* + * Release the sync fib area. + */ +void +aac_release_sync_fib(struct aac_softc *sc) +{ + AAC_LOCK_RELEASE(&sc->aac_sync_lock); } /* * Send a synchronous FIB to the controller and wait for a result. */ int -aac_sync_fib(sc, command, xferstate, data, datasize, result, resultsize) - struct aac_softc *sc; - u_int32_t command; - u_int32_t xferstate; - void *data; - u_int16_t datasize; - void *result; - u_int16_t *resultsize; +aac_sync_fib(struct aac_softc *sc, u_int32_t command, u_int32_t xferstate, + struct aac_fib *fib, u_int16_t datasize) { - struct aac_fib *fib = &sc->sc_common->ac_sync_fib; - if (datasize > AAC_FIB_DATASIZE) - return (EINVAL); + if (datasize > AAC_FIB_DATASIZE) { + printf("aac_sync_fib 1: datasize=%d AAC_FIB_DATASIZE\n", + datasize, AAC_FIB_DATASIZE); + return(EINVAL); + } /* * Set up the sync FIB */ fib->Header.XferState = AAC_FIBSTATE_HOSTOWNED | - AAC_FIBSTATE_INITIALISED | AAC_FIBSTATE_EMPTY; + AAC_FIBSTATE_INITIALISED | + AAC_FIBSTATE_EMPTY; fib->Header.XferState |= xferstate; fib->Header.Command = command; fib->Header.StructType = AAC_FIBTYPE_TFIB; - fib->Header.Size = sizeof fib + datasize; - fib->Header.SenderSize = sizeof *fib; - fib->Header.SenderFibAddress = (u_int32_t)fib; - fib->Header.ReceiverFibAddress = - sc->sc_common_busaddr + offsetof(struct aac_common, ac_sync_fib); + fib->Header.Size = sizeof(struct aac_fib) + datasize; + fib->Header.SenderSize = sizeof(struct aac_fib); + fib->Header.SenderFibAddress = 0; /* Not needed */ + fib->Header.ReceiverFibAddress = sc->aac_common_busaddr + + offsetof(struct aac_common, + ac_sync_fib); /* - * Copy in data. + * Give the FIB to the controller, wait for a response. */ - if (data != NULL) { - bcopy(data, fib->data, datasize); - fib->Header.XferState |= - AAC_FIBSTATE_FROMHOST | AAC_FIBSTATE_NORM; + if (aac_sync_command(sc, AAC_MONKER_SYNCFIB, + fib->Header.ReceiverFibAddress, 0, 0, 0, NULL)) { + AAC_DPRINTF(AAC_D_IO, ("%s: aac_sync_fib: IO error\n", + sc->aac_dev.dv_xname)); + printf("aac_sync_fib 2\n"); + return(EIO); + } + + return (0); +} + +/***************************************************************************** + * Adapter-space FIB queue manipulation + * + * Note that the queue implementation here is a little funky; neither the PI or + * CI will ever be zero. This behaviour is a controller feature. + */ +static struct { + int size; + int notify; +} aac_qinfo[] = { + { AAC_HOST_NORM_CMD_ENTRIES, AAC_DB_COMMAND_NOT_FULL }, + { AAC_HOST_HIGH_CMD_ENTRIES, 0 }, + { AAC_ADAP_NORM_CMD_ENTRIES, AAC_DB_COMMAND_READY }, + { AAC_ADAP_HIGH_CMD_ENTRIES, 0 }, + { AAC_HOST_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_NOT_FULL }, + { AAC_HOST_HIGH_RESP_ENTRIES, 0 }, + { AAC_ADAP_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_READY }, + { AAC_ADAP_HIGH_RESP_ENTRIES, 0 } +}; + +/* + * Atomically insert an entry into the nominated queue, returns 0 on success + * or EBUSY if the queue is full. + * + * Note: it would be more efficient to defer notifying the controller in + * the case where we may be inserting several entries in rapid + * succession, but implementing this usefully may be difficult + * (it would involve a separate queue/notify interface). + */ +int +aac_enqueue_fib(struct aac_softc *sc, int queue, struct aac_command *cm) +{ + u_int32_t pi, ci; + int error; + u_int32_t fib_size; + u_int32_t fib_addr; + + fib_size = cm->cm_fib->Header.Size; + fib_addr = cm->cm_fib->Header.ReceiverFibAddress; + + /* get the producer/consumer indices */ + pi = sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]; + ci = sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]; + + /* wrap the queue? */ + if (pi >= aac_qinfo[queue].size) + pi = 0; + + /* check for queue full */ + if ((pi + 1) == ci) { + error = EBUSY; + goto out; } + /* populate queue entry */ + (sc->aac_qentries[queue] + pi)->aq_fib_size = fib_size; + (sc->aac_qentries[queue] + pi)->aq_fib_addr = fib_addr; + + /* update producer index */ + sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX] = pi + 1; + /* - * Give the FIB to the controller, wait for a response. + * To avoid a race with its completion interrupt, place this command on + * the busy queue prior to advertising it to the controller. */ - if (aac_sync_command(sc, AAC_MONKER_SYNCFIB, - fib->Header.ReceiverFibAddress, 0, 0, 0, NULL)) { - return (EIO); + aac_enqueue_busy(cm); + + /* notify the adapter if we know how */ + if (aac_qinfo[queue].notify != 0) + AAC_QNOTIFY(sc, aac_qinfo[queue].notify); + + error = 0; + +out: + return (error); +} + +/* + * Atomically remove one entry from the nominated queue, returns 0 on success + * or ENOENT if the queue is empty. + */ +int +aac_dequeue_fib(struct aac_softc *sc, int queue, u_int32_t *fib_size, + struct aac_fib **fib_addr) +{ + u_int32_t pi, ci; + u_int32_t fib_index; + int notify; + int error; + + /* get the producer/consumer indices */ + pi = sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]; + ci = sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]; + + /* check for queue empty */ + if (ci == pi) { + error = ENOENT; + goto out; } - /* - * Copy out the result - */ - if (result != NULL) { - *resultsize = fib->Header.Size - sizeof fib->Header; - bcopy(fib->data, result, *resultsize); + /* wrap the pi so the following test works */ + if (pi >= aac_qinfo[queue].size) + pi = 0; + + notify = 0; + if (ci == pi + 1) + notify++; + + /* wrap the queue? */ + if (ci >= aac_qinfo[queue].size) + ci = 0; + + /* fetch the entry */ + *fib_size = (sc->aac_qentries[queue] + ci)->aq_fib_size; + + switch (queue) { + case AAC_HOST_NORM_CMD_QUEUE: + case AAC_HOST_HIGH_CMD_QUEUE: + /* + * The aq_fib_addr is only 32 bits wide so it can't be counted + * on to hold an address. For AIF's, the adapter assumes + * that it's giving us an address into the array of AIF fibs. + * Therefore, we have to convert it to an index. + */ + fib_index = (sc->aac_qentries[queue] + ci)->aq_fib_addr / + sizeof(struct aac_fib); + *fib_addr = &sc->aac_common->ac_fibs[fib_index]; + break; + + case AAC_HOST_NORM_RESP_QUEUE: + case AAC_HOST_HIGH_RESP_QUEUE: + { + struct aac_command *cm; + + /* + * As above, an index is used instead of an actual address. + * Gotta shift the index to account for the fast response + * bit. No other correction is needed since this value was + * originally provided by the driver via the SenderFibAddress + * field. + */ + fib_index = (sc->aac_qentries[queue] + ci)->aq_fib_addr; + cm = sc->aac_commands + (fib_index >> 1); + *fib_addr = cm->cm_fib; + + /* + * Is this a fast response? If it is, update the fib fields in + * local memory since the whole fib isn't DMA'd back up. + */ + if (fib_index & 0x01) { + (*fib_addr)->Header.XferState |= AAC_FIBSTATE_DONEADAP; + *((u_int32_t*)((*fib_addr)->data)) = AAC_ERROR_NORMAL; + } + break; } - return (0); + default: + panic("Invalid queue in aac_dequeue_fib()"); + break; + } + + + /* update consumer index */ + sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX] = ci + 1; + + /* if we have made the queue un-full, notify the adapter */ + if (notify && (aac_qinfo[queue].notify != 0)) + AAC_QNOTIFY(sc, aac_qinfo[queue].notify); + error = 0; + +out: + return (error); +} + +/* + * Put our response to an Adapter Initialed Fib on the response queue + */ +int +aac_enqueue_response(struct aac_softc *sc, int queue, struct aac_fib *fib) +{ + u_int32_t pi, ci; + int error; + u_int32_t fib_size; + u_int32_t fib_addr; + + /* Tell the adapter where the FIB is */ + fib_size = fib->Header.Size; + fib_addr = fib->Header.SenderFibAddress; + fib->Header.ReceiverFibAddress = fib_addr; + + /* get the producer/consumer indices */ + pi = sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]; + ci = sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]; + + /* wrap the queue? */ + if (pi >= aac_qinfo[queue].size) + pi = 0; + + /* check for queue full */ + if ((pi + 1) == ci) { + error = EBUSY; + goto out; + } + + /* populate queue entry */ + (sc->aac_qentries[queue] + pi)->aq_fib_size = fib_size; + (sc->aac_qentries[queue] + pi)->aq_fib_addr = fib_addr; + + /* update producer index */ + sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX] = pi + 1; + + /* notify the adapter if we know how */ + if (aac_qinfo[queue].notify != 0) + AAC_QNOTIFY(sc, aac_qinfo[queue].notify); + + error = 0; + +out: + return(error); } void -aacminphys(bp) - struct buf *bp; +aac_command_timeout(struct aac_command *cm) { -#if 0 - u_int8_t *buf = bp->b_data; - paddr_t pa; - long off; -#endif + struct aac_softc *sc = cm->cm_sc; - AAC_DPRINTF(AAC_D_MISC, ("aacminphys(0x%x) ", bp)); + printf("%s: COMMAND %p (flags=%#x) TIMEOUT AFTER %d SECONDS\n", + sc->aac_dev.dv_xname, cm, cm->cm_flags, + (int)(time_second - cm->cm_timestamp)); - minphys(bp); + if (cm->cm_flags & AAC_CMD_TIMEDOUT) + return; + + cm->cm_flags |= AAC_CMD_TIMEDOUT; + + AAC_PRINT_FIB(sc, cm->cm_fib); + + if (cm->cm_flags & AAC_ON_AACQ_BIO) { + struct scsi_xfer *xs = cm->cm_private; + int s = splbio(); + xs->error = XS_DRIVER_STUFFUP; + xs->flags |= ITSDONE; + scsi_done(xs); + splx(s); + + aac_remove_bio(cm); + aac_unmap_command(cm); + } } +void +aac_timeout(struct aac_softc *sc) +{ + struct aac_command *cm; + time_t deadline; + + /* + * Traverse the busy command list and timeout any commands + * that are past their deadline. + */ + deadline = time_second - AAC_CMD_TIMEOUT; + TAILQ_FOREACH(cm, &sc->aac_busy, cm_link) { + if (cm->cm_timestamp < deadline) + aac_command_timeout(cm); + } +} + +/* + * Interface Function Vectors + */ + /* * Read the current firmware status word. */ int -aac_sa_get_fwstatus(sc) - struct aac_softc *sc; +aac_sa_get_fwstatus(struct aac_softc *sc) { return (AAC_GETREG4(sc, AAC_SA_FWSTATUS)); } int -aac_rx_get_fwstatus(sc) - struct aac_softc *sc; +aac_rx_get_fwstatus(struct aac_softc *sc) { return (AAC_GETREG4(sc, AAC_RX_FWSTATUS)); } +int +aac_fa_get_fwstatus(struct aac_softc *sc) +{ + return (AAC_GETREG4(sc, AAC_FA_FWSTATUS)); +} + /* * Notify the controller of a change in a given queue */ void -aac_sa_qnotify(sc, qbit) - struct aac_softc *sc; - int qbit; +aac_sa_qnotify(struct aac_softc *sc, int qbit) { AAC_SETREG2(sc, AAC_SA_DOORBELL1_SET, qbit); } void -aac_rx_qnotify(sc, qbit) - struct aac_softc *sc; - int qbit; +aac_rx_qnotify(struct aac_softc *sc, int qbit) { AAC_SETREG4(sc, AAC_RX_IDBR, qbit); } +void +aac_fa_qnotify(struct aac_softc *sc, int qbit) +{ + AAC_SETREG2(sc, AAC_FA_DOORBELL1, qbit); + AAC_FA_HACK(sc); +} + /* * Get the interrupt reason bits */ int -aac_sa_get_istatus(sc) - struct aac_softc *sc; +aac_sa_get_istatus(struct aac_softc *sc) { return (AAC_GETREG2(sc, AAC_SA_DOORBELL0)); } int -aac_rx_get_istatus(sc) - struct aac_softc *sc; +aac_rx_get_istatus(struct aac_softc *sc) { return (AAC_GETREG4(sc, AAC_RX_ODBR)); } +int +aac_fa_get_istatus(struct aac_softc *sc) +{ + return (AAC_GETREG2(sc, AAC_FA_DOORBELL0)); +} + /* * Clear some interrupt reason bits */ void -aac_sa_clear_istatus(sc, mask) - struct aac_softc *sc; - int mask; +aac_sa_clear_istatus(struct aac_softc *sc, int mask) { AAC_SETREG2(sc, AAC_SA_DOORBELL0_CLEAR, mask); } void -aac_rx_clear_istatus(sc, mask) - struct aac_softc *sc; - int mask; +aac_rx_clear_istatus(struct aac_softc *sc, int mask) { AAC_SETREG4(sc, AAC_RX_ODBR, mask); } +void +aac_fa_clear_istatus(struct aac_softc *sc, int mask) +{ + AAC_SETREG2(sc, AAC_FA_DOORBELL0_CLEAR, mask); + AAC_FA_HACK(sc); +} + /* * Populate the mailbox and set the command word */ void -aac_sa_set_mailbox(sc, command, arg0, arg1, arg2, arg3) - struct aac_softc *sc; - u_int32_t command; - u_int32_t arg0; - u_int32_t arg1; - u_int32_t arg2; - u_int32_t arg3; +aac_sa_set_mailbox(struct aac_softc *sc, u_int32_t command, u_int32_t arg0, + u_int32_t arg1, u_int32_t arg2, u_int32_t arg3) { AAC_SETREG4(sc, AAC_SA_MAILBOX, command); AAC_SETREG4(sc, AAC_SA_MAILBOX + 4, arg0); @@ -1284,13 +2213,8 @@ aac_sa_set_mailbox(sc, command, arg0, arg1, arg2, arg3) } void -aac_rx_set_mailbox(sc, command, arg0, arg1, arg2, arg3) - struct aac_softc *sc; - u_int32_t command; - u_int32_t arg0; - u_int32_t arg1; - u_int32_t arg2; - u_int32_t arg3; +aac_rx_set_mailbox(struct aac_softc *sc, u_int32_t command, u_int32_t arg0, + u_int32_t arg1, u_int32_t arg2, u_int32_t arg3) { AAC_SETREG4(sc, AAC_RX_MAILBOX, command); AAC_SETREG4(sc, AAC_RX_MAILBOX + 4, arg0); @@ -1299,31 +2223,52 @@ aac_rx_set_mailbox(sc, command, arg0, arg1, arg2, arg3) AAC_SETREG4(sc, AAC_RX_MAILBOX + 16, arg3); } +void +aac_fa_set_mailbox(struct aac_softc *sc, u_int32_t command, u_int32_t arg0, + u_int32_t arg1, u_int32_t arg2, u_int32_t arg3) +{ + AAC_SETREG4(sc, AAC_FA_MAILBOX, command); + AAC_FA_HACK(sc); + AAC_SETREG4(sc, AAC_FA_MAILBOX + 4, arg0); + AAC_FA_HACK(sc); + AAC_SETREG4(sc, AAC_FA_MAILBOX + 8, arg1); + AAC_FA_HACK(sc); + AAC_SETREG4(sc, AAC_FA_MAILBOX + 12, arg2); + AAC_FA_HACK(sc); + AAC_SETREG4(sc, AAC_FA_MAILBOX + 16, arg3); + AAC_FA_HACK(sc); +} + /* * Fetch the immediate command status word */ int -aac_sa_get_mailboxstatus(sc) - struct aac_softc *sc; +aac_sa_get_mailbox(struct aac_softc *sc, int mb) { - return (AAC_GETREG4(sc, AAC_SA_MAILBOX)); + return (AAC_GETREG4(sc, AAC_SA_MAILBOX + (mb * 4))); } int -aac_rx_get_mailboxstatus(sc) - struct aac_softc *sc; +aac_rx_get_mailbox(struct aac_softc *sc, int mb) { - return (AAC_GETREG4(sc, AAC_RX_MAILBOX)); + return (AAC_GETREG4(sc, AAC_RX_MAILBOX + (mb * 4))); +} + +int +aac_fa_get_mailbox(struct aac_softc *sc, int mb) +{ + return (AAC_GETREG4(sc, AAC_FA_MAILBOX + (mb * 4))); } /* * Set/clear interrupt masks */ void -aac_sa_set_interrupts(sc, enable) - struct aac_softc *sc; - int enable; +aac_sa_set_interrupts(struct aac_softc *sc, int enable) { + AAC_DPRINTF(AAC_D_INTR, ("%s: %sable interrupts\n", + sc->aac_dev.dv_xname, enable ? "en" : "dis")); + if (enable) AAC_SETREG2((sc), AAC_SA_MASK0_CLEAR, AAC_DB_INTERRUPTS); else @@ -1331,506 +2276,431 @@ aac_sa_set_interrupts(sc, enable) } void -aac_rx_set_interrupts(sc, enable) - struct aac_softc *sc; - int enable; +aac_rx_set_interrupts(struct aac_softc *sc, int enable) { + AAC_DPRINTF(AAC_D_INTR, ("%s: %sable interrupts", + sc->aac_dev.dv_xname, enable ? "en" : "dis")); + if (enable) AAC_SETREG4(sc, AAC_RX_OIMR, ~AAC_DB_INTERRUPTS); else AAC_SETREG4(sc, AAC_RX_OIMR, ~0); } -struct aac_ccb * -aac_get_ccb(sc, flags) - struct aac_softc *sc; - int flags; -{ - struct aac_ccb *ccb; - aac_lock_t lock; - - AAC_DPRINTF(AAC_D_QUEUE, ("aac_get_ccb(%p, 0x%x) ", sc, flags)); - - lock = AAC_LOCK(sc); - - for (;;) { - ccb = TAILQ_FIRST(&sc->sc_free_ccb); - if (ccb != NULL) - break; - if (flags & SCSI_NOSLEEP) - goto bail_out; - tsleep(&sc->sc_free_ccb, PRIBIO, "aac_ccb", 0); - } - - TAILQ_REMOVE(&sc->sc_free_ccb, ccb, ac_chain); - - /* initialise the command/FIB */ - ccb->ac_sgtable = NULL; - ccb->ac_flags = 0; - ccb->ac_fib->Header.XferState = AAC_FIBSTATE_EMPTY; - ccb->ac_fib->Header.StructType = AAC_FIBTYPE_TFIB; - ccb->ac_fib->Header.Flags = 0; - ccb->ac_fib->Header.SenderSize = sizeof(struct aac_fib); - - /* - * These are duplicated in aac_start to cover the case where an - * intermediate stage may have destroyed them. They're left - * initialised here for debugging purposes only. - */ - ccb->ac_fib->Header.SenderFibAddress = (u_int32_t)ccb->ac_fib; - ccb->ac_fib->Header.ReceiverFibAddress = ccb->ac_fibphys; - - bail_out: - AAC_UNLOCK(sc, lock); - return (ccb); -} - void -aac_free_ccb(sc, ccb) - struct aac_softc *sc; - struct aac_ccb *ccb; +aac_fa_set_interrupts(struct aac_softc *sc, int enable) { - aac_lock_t lock; - - AAC_DPRINTF(AAC_D_QUEUE, ("aac_free_ccb(%p, %p) ", sc, ccb)); + AAC_DPRINTF(AAC_D_INTR, ("%s: %sable interrupts", + sc->aac_dev.dv_xname, enable ? "en" : "dis")); - lock = AAC_LOCK(sc); - - TAILQ_INSERT_HEAD(&sc->sc_free_ccb, ccb, ac_chain); - - /* If the free list was empty, wake up potential waiters. */ - if (TAILQ_NEXT(ccb, ac_chain) == NULL) - wakeup(&sc->sc_free_ccb); - - AAC_UNLOCK(sc, lock); + if (enable) { + AAC_SETREG2((sc), AAC_FA_MASK0_CLEAR, AAC_DB_INTERRUPTS); + AAC_FA_HACK(sc); + } else { + AAC_SETREG2((sc), AAC_FA_MASK0, ~0); + AAC_FA_HACK(sc); + } } void -aac_enqueue_ccb(sc, ccb) - struct aac_softc *sc; - struct aac_ccb *ccb; +aac_eval_mapping(size, cyls, heads, secs) + u_int32_t size; + int *cyls, *heads, *secs; { - AAC_DPRINTF(AAC_D_QUEUE, ("aac_enqueue_ccb(%p, %p) ", sc, ccb)); - - timeout_set(&ccb->ac_xs->stimeout, aac_timeout, ccb); - TAILQ_INSERT_TAIL(&sc->sc_ccbq, ccb, ac_chain); - aac_start_ccbs(sc); + *cyls = size / AAC_HEADS / AAC_SECS; + if (*cyls < AAC_MAXCYLS) { + *heads = AAC_HEADS; + *secs = AAC_SECS; + } else { + /* Too high for 64 * 32 */ + *cyls = size / AAC_MEDHEADS / AAC_MEDSECS; + if (*cyls < AAC_MAXCYLS) { + *heads = AAC_MEDHEADS; + *secs = AAC_MEDSECS; + } else { + /* Too high for 127 * 63 */ + *cyls = size / AAC_BIGHEADS / AAC_BIGSECS; + *heads = AAC_BIGHEADS; + *secs = AAC_BIGSECS; + } + } } void -aac_start_ccbs(sc) - struct aac_softc *sc; +aac_copy_internal_data(struct scsi_xfer *xs, u_int8_t *data, size_t size) { - struct aac_ccb *ccb; - struct scsi_xfer *xs; - - AAC_DPRINTF(AAC_D_QUEUE, ("aac_start_ccbs(%p) ", sc)); - - while ((ccb = TAILQ_FIRST(&sc->sc_ccbq)) != NULL) { - - xs = ccb->ac_xs; - if (ccb->ac_flags & AAC_ACF_WATCHDOG) - timeout_del(&xs->stimeout); + struct aac_softc *sc = xs->sc_link->adapter_softc; + size_t copy_cnt; - if (aac_exec_ccb(ccb) == 0) { - ccb->ac_flags |= AAC_ACF_WATCHDOG; - timeout_set(&ccb->ac_xs->stimeout, aac_watchdog, ccb); - timeout_add(&xs->stimeout, - (AAC_WATCH_TIMEOUT * hz) / 1000); - break; - } - TAILQ_REMOVE(&sc->sc_ccbq, ccb, ac_chain); + AAC_DPRINTF(AAC_D_MISC, ("%s: aac_copy_internal_data\n", + sc->aac_dev.dv_xname)); - if ((xs->flags & SCSI_POLL) == 0) { - timeout_set(&ccb->ac_xs->stimeout, aac_timeout, ccb); - timeout_add(&xs->stimeout, - (ccb->ac_timeout * hz) / 1000); - } + if (!xs->datalen) + printf("%s: uio move not yet supported\n", + sc->aac_dev.dv_xname); + else { + copy_cnt = MIN(size, xs->datalen); + bcopy(data, xs->data, copy_cnt); } } +/* Emulated SCSI operation on cache device */ int -aac_exec_ccb(ccb) - struct aac_ccb *ccb; +aac_internal_cache_cmd(struct scsi_xfer *xs) { - struct scsi_xfer *xs = ccb->ac_xs; struct scsi_link *link = xs->sc_link; + struct aac_softc *sc = link->adapter_softc; + struct scsi_inquiry_data inq; + struct scsi_sense_data sd; + struct scsi_read_cap_data rcd; u_int8_t target = link->target; - int i; - struct aac_fib *fib; - struct aac_blockread *br; - struct aac_blockwrite *bw; - bus_dmamap_t xfer; - AAC_DPRINTF(AAC_D_CMD, ("aac_exec_ccb(%p, %p) ", xs, ccb)); - - /* build the FIB */ - fib = ccb->ac_fib; - fib->Header.XferState = AAC_FIBSTATE_HOSTOWNED | - AAC_FIBSTATE_INITIALISED | AAC_FIBSTATE_FROMHOST | - AAC_FIBSTATE_REXPECTED | AAC_FIBSTATE_NORM; - fib->Header.Command = ContainerCommand; - fib->Header.Size = sizeof(struct aac_fib_header); + AAC_DPRINTF(AAC_D_CMD, ("aac_internal_cache_cmd: ", + sc->aac_dev.dv_xname)); switch (xs->cmd->opcode) { - case PREVENT_ALLOW: - case SYNCHRONIZE_CACHE: - if (xs->cmd->opcode == PREVENT_ALLOW) { - /* XXX PREVENT_ALLOW support goes here */ - } else { - AAC_DPRINTF(AAC_D_CMD, - ("SYNCHRONIZE CACHE tgt %d ", target)); - } + case TEST_UNIT_READY: + case START_STOP: +#if 0 + case VERIFY: +#endif + AAC_DPRINTF(AAC_D_CMD, ("opc %#x tgt %d ", xs->cmd->opcode, + target)); break; - case WRITE_COMMAND: - case WRITE_BIG: - bw = (struct aac_blockwrite *)&fib->data[0]; - bw->Command = VM_CtBlockWrite; - bw->ContainerId = target; - bw->BlockNumber = ccb->ac_blockno; - bw->ByteCount = ccb->ac_blockcnt * DEV_BSIZE; - bw->Stable = CUNSTABLE; /* XXX what's appropriate here? */ - fib->Header.Size += sizeof(struct aac_blockwrite); - ccb->ac_sgtable = &bw->SgMap; + case REQUEST_SENSE: + AAC_DPRINTF(AAC_D_CMD, ("REQUEST SENSE tgt %d ", target)); + bzero(&sd, sizeof sd); + sd.error_code = 0x70; + sd.segment = 0; + sd.flags = SKEY_NO_SENSE; + aac_enc32(sd.info, 0); + sd.extra_len = 0; + aac_copy_internal_data(xs, (u_int8_t *)&sd, sizeof sd); break; - case READ_COMMAND: - case READ_BIG: - br = (struct aac_blockread *)&fib->data[0]; - br->Command = VM_CtBlockRead; - br->ContainerId = target; - br->BlockNumber = ccb->ac_blockno; - br->ByteCount = ccb->ac_blockcnt * DEV_BSIZE; - fib->Header.Size += sizeof(struct aac_blockread); - ccb->ac_sgtable = &br->SgMap; + case INQUIRY: + AAC_DPRINTF(AAC_D_CMD, ("INQUIRY tgt %d devtype %x ", target, + sc->aac_hdr[target].hd_devtype)); + bzero(&inq, sizeof inq); + /* XXX How do we detect removable/CD-ROM devices? */ + inq.device = T_DIRECT; + inq.dev_qual2 = 0; + inq.version = 2; + inq.response_format = 2; + inq.additional_length = 32; + strlcpy(inq.vendor, "Adaptec", sizeof inq.vendor); + snprintf(inq.product, sizeof inq.product, "Container #%02d", + target); + strlcpy(inq.revision, " ", sizeof inq.revision); + aac_copy_internal_data(xs, (u_int8_t *)&inq, sizeof inq); break; - } - if (xs->cmd->opcode != PREVENT_ALLOW && - xs->cmd->opcode != SYNCHRONIZE_CACHE) { - xfer = ccb->ac_dmamap_xfer; - ccb->ac_sgtable->SgCount = xfer->dm_nsegs; - for (i = 0; i < xfer->dm_nsegs; i++) { - ccb->ac_sgtable->SgEntry[i].SgAddress = - xfer->dm_segs[i].ds_addr; - ccb->ac_sgtable->SgEntry[i].SgByteCount = - xfer->dm_segs[i].ds_len; - AAC_DPRINTF(AAC_D_IO, - ("#%d va %p pa %p len %x\n", i, buf, - xfer->dm_segs[i].ds_addr, - xfer->dm_segs[i].ds_len)); - } + case READ_CAPACITY: + AAC_DPRINTF(AAC_D_CMD, ("READ CAPACITY tgt %d ", target)); + bzero(&rcd, sizeof rcd); + _lto4b(sc->aac_hdr[target].hd_size - 1, rcd.addr); + _lto4b(AAC_BLOCK_SIZE, rcd.length); + aac_copy_internal_data(xs, (u_int8_t *)&rcd, sizeof rcd); + break; - /* update the FIB size for the s/g count */ - fib->Header.Size += xfer->dm_nsegs * - sizeof(struct aac_sg_entry); + default: + AAC_DPRINTF(AAC_D_CMD, ("\n")); + printf("aac_internal_cache_cmd got bad opcode: %#x\n", + xs->cmd->opcode); + xs->error = XS_DRIVER_STUFFUP; + return (0); } - if (aac_start(ccb) == 0) { - xs->error = XS_NOERROR; - xs->resid = 0; - return (1); - } - return (0); + xs->error = XS_NOERROR; + return (1); } -/******************************************************************************** - * Deliver a command to the controller; allocate controller resources at the - * last moment when possible. - */ -int -aac_start(struct aac_ccb *ccb) +void +aacminphys(struct buf *bp) { - struct aac_softc *sc = ccb->ac_xs->sc_link->adapter_softc; - #if 0 - /* get the command mapped */ - aac_map_command(ccb); + u_int8_t *buf = bp->b_data; + paddr_t pa; + long off; #endif - /* fix up the address values */ - ccb->ac_fib->Header.SenderFibAddress = (u_int32_t)ccb->ac_fib; - ccb->ac_fib->Header.ReceiverFibAddress = ccb->ac_fibphys; - - /* save a pointer to the command for speedy reverse-lookup */ - ccb->ac_fib->Header.SenderData = (u_int32_t)ccb; /* XXX ack, sizing */ + AAC_DPRINTF(AAC_D_MISC, ("aacminphys(0x%x)\n", bp)); - /* put the FIB on the outbound queue */ - if (aac_enqueue_fib(sc, AAC_ADAP_NORM_CMD_QUEUE, ccb)) - return (EBUSY); +#if 0 /* As this is way more than MAXPHYS it's really not necessary. */ + if (bp->b_bcount > ((AAC_MAXOFFSETS - 1) * PAGE_SIZE)) + bp->b_bcount = ((AAC_MAXOFFSETS - 1) * PAGE_SIZE); +#endif - return (0); +#if 0 + for (off = PAGE_SIZE, pa = vtophys(buf); off < bp->b_bcount; + off += PAGE_SIZE) + if (pa + off != vtophys(buf + off)) { + bp->b_bcount = off; + break; + } +#endif + minphys(bp); } -/* - * Map a command into controller-visible space. - */ int -aac_map_command(struct aac_ccb *ccb) +aac_raw_scsi_cmd(struct scsi_xfer *xs) { - struct scsi_xfer *xs = ccb->ac_xs; +#ifdef AAC_DEBUG struct aac_softc *sc = xs->sc_link->adapter_softc; - int error; - -#if 0 - /* don't map more than once */ - if (ccb->ac_flags & AAC_CMD_MAPPED) - return; #endif + AAC_DPRINTF(AAC_D_CMD, ("%s: aac_raw_scsi_cmd\n", + sc->aac_dev.dv_xname)); - if (xs->datalen != 0) { - error = bus_dmamap_load(sc->sc_dmat, ccb->ac_dmamap_xfer, - xs->data, xs->datalen, NULL, - (xs->flags & SCSI_NOSLEEP) ? BUS_DMA_NOWAIT : - BUS_DMA_WAITOK); - if (error) { - printf("%s: aac_scsi_cmd: ", sc->sc_dev.dv_xname); - if (error == EFBIG) - printf("more than %d dma segs\n", - AAC_MAXSGENTRIES); - else - printf("error %d loading dma map\n", error); - return (error); - } - - bus_dmamap_sync(sc->sc_dmat, ccb->ac_dmamap_xfer, 0, - ccb->ac_dmamap_xfer->dm_mapsize, - (xs->flags & SCSI_DATA_IN) ? BUS_DMASYNC_PREREAD : - BUS_DMASYNC_PREWRITE); - } - -#if 0 - ccb->ac_flags |= AAC_CMD_MAPPED; -#endif - return (0); + /* XXX Not yet implemented */ + xs->error = XS_DRIVER_STUFFUP; + return (COMPLETE); } -/* - * Unmap a command from controller-visible space. - */ -void -aac_unmap_command(struct aac_ccb *ccb) +int +aac_scsi_cmd(struct scsi_xfer *xs) { - struct scsi_xfer *xs = ccb->ac_xs; - struct aac_softc *sc = xs->sc_link->adapter_softc; - -#if 0 - if (!(ccb->ac_flags & AAC_CMD_MAPPED)) - return; -#endif + struct scsi_link *link = xs->sc_link; + struct aac_softc *sc = link->adapter_softc; + u_int8_t target = link->target; + struct aac_command *cm; + u_int32_t blockno, blockcnt; + struct scsi_rw *rw; + struct scsi_rw_big *rwb; + int retval = SUCCESSFULLY_QUEUED; + int s = splbio(); - if (xs->datalen != 0) { - bus_dmamap_sync(sc->sc_dmat, ccb->ac_dmamap_xfer, 0, - ccb->ac_dmamap_xfer->dm_mapsize, - (xs->flags & SCSI_DATA_IN) ? BUS_DMASYNC_POSTREAD : - BUS_DMASYNC_POSTWRITE); + xs->error = XS_NOERROR; - bus_dmamap_unload(sc->sc_dmat, ccb->ac_dmamap_xfer); + if (target >= AAC_MAX_CONTAINERS || !sc->aac_hdr[target].hd_present || + link->lun != 0) { + /* + * XXX Should be XS_SENSE but that would require setting up a + * faked sense too. + */ + xs->error = XS_DRIVER_STUFFUP; + xs->flags |= ITSDONE; + scsi_done(xs); + splx(s); + return (COMPLETE); } + + AAC_DPRINTF(AAC_D_CMD, ("%s: aac_scsi_cmd: ", sc->aac_dev.dv_xname)); + + xs->error = XS_NOERROR; + cm = NULL; + link = xs->sc_link; + target = link->target; + + switch (xs->cmd->opcode) { + case TEST_UNIT_READY: + case REQUEST_SENSE: + case INQUIRY: + case START_STOP: + case READ_CAPACITY: #if 0 - ccb->ac_flags &= ~AAC_CMD_MAPPED; + case VERIFY: #endif -} + if (!aac_internal_cache_cmd(xs)) { + splx(s); + return (TRY_AGAIN_LATER); + } + xs->flags |= ITSDONE; + scsi_done(xs); + goto ready; -void -aac_timeout(arg) - void *arg; -{ - struct aac_ccb *ccb = arg; - struct scsi_link *link = ccb->ac_xs->sc_link; - struct aac_softc *sc = link->adapter_softc; - aac_lock_t lock; + case PREVENT_ALLOW: + AAC_DPRINTF(AAC_D_CMD, ("PREVENT/ALLOW ")); + /* XXX Not yet implemented */ + xs->error = XS_NOERROR; + xs->flags |= ITSDONE; + scsi_done(xs); + goto ready; - sc_print_addr(link); - printf("timed out\n"); + case SYNCHRONIZE_CACHE: + AAC_DPRINTF(AAC_D_CMD, ("SYNCHRONIZE_CACHE ")); + /* XXX Not yet implemented */ + xs->error = XS_NOERROR; + xs->flags |= ITSDONE; + scsi_done(xs); + goto ready; - /* XXX Test for multiple timeouts */ + default: + AAC_DPRINTF(AAC_D_CMD, ("unknown opc %#x ", xs->cmd->opcode)); + /* XXX Not yet implemented */ + xs->error = XS_DRIVER_STUFFUP; + xs->flags |= ITSDONE; + scsi_done(xs); + goto ready; - ccb->ac_xs->error = XS_TIMEOUT; - lock = AAC_LOCK(sc); - aac_enqueue_ccb(sc, ccb); - AAC_UNLOCK(sc, lock); -} + case READ_COMMAND: + case READ_BIG: + case WRITE_COMMAND: + case WRITE_BIG: + AAC_DPRINTF(AAC_D_CMD, ("rw opc %#x ", xs->cmd->opcode)); + + /* A read or write operation. */ + if (xs->cmdlen == 6) { + rw = (struct scsi_rw *)xs->cmd; + blockno = _3btol(rw->addr) & + (SRW_TOPADDR << 16 | 0xffff); + blockcnt = rw->length ? rw->length : 0x100; + } else { + rwb = (struct scsi_rw_big *)xs->cmd; + blockno = _4btol(rwb->addr); + blockcnt = _2btol(rwb->length); + } -void -aac_watchdog(arg) - void *arg; -{ - struct aac_ccb *ccb = arg; - struct scsi_link *link = ccb->ac_xs->sc_link; - struct aac_softc *sc = link->adapter_softc; - aac_lock_t lock; + AAC_DPRINTF(AAC_D_CMD, ("blkno=%d bcount=%d ", + xs->cmd->opcode, blockno, blockcnt)); - lock = AAC_LOCK(sc); - ccb->ac_flags &= ~AAC_ACF_WATCHDOG; - aac_start_ccbs(sc); - AAC_UNLOCK(sc, lock); -} -/* - * Insert a command into the driver queue, either at the front or at the tail. - * It's ok to overload the freelist link as these structures are never on - * the freelist at this time. - */ -void -aac_enqueue(sc, xs, infront) - struct aac_softc *sc; - struct scsi_xfer *xs; - int infront; -{ - if (infront || LIST_FIRST(&sc->sc_queue) == NULL) { - if (LIST_FIRST(&sc->sc_queue) == NULL) - sc->sc_queuelast = xs; - LIST_INSERT_HEAD(&sc->sc_queue, xs, free_list); - return; - } - LIST_INSERT_AFTER(sc->sc_queuelast, xs, free_list); - sc->sc_queuelast = xs; -} + if (blockno >= sc->aac_hdr[target].hd_size || + blockno + blockcnt > sc->aac_hdr[target].hd_size) { + AAC_DPRINTF(AAC_D_CMD, ("\n")); + printf("%s: out of bounds %u-%u >= %u\n", + sc->aac_dev.dv_xname, blockno, + blockcnt, sc->aac_hdr[target].hd_size); + /* + * XXX Should be XS_SENSE but that + * would require setting up a faked + * sense too. + */ + xs->error = XS_DRIVER_STUFFUP; + xs->flags |= ITSDONE; + scsi_done(xs); + goto ready; + } -/* - * Pull a command off the front of the driver queue. - */ -struct scsi_xfer * -aac_dequeue(sc) - struct aac_softc *sc; -{ - struct scsi_xfer *xs; + if (aac_alloc_command(sc, &cm)) { + AAC_DPRINTF(AAC_D_CMD, + (": out of commands, try later\n")); + /* + * We are out of commands, try again + * in a little while. + */ + xs->error = XS_DRIVER_STUFFUP; + splx(s); + return (TRY_AGAIN_LATER); + } - xs = LIST_FIRST(&sc->sc_queue); - if (xs == NULL) - return (NULL); - LIST_REMOVE(xs, free_list); + /* fill out the command */ + cm->cm_data = (void *)xs->data; + cm->cm_datalen = xs->datalen; + cm->cm_complete = aac_bio_complete; + cm->cm_private = xs; + cm->cm_timestamp = time_second; + cm->cm_queue = AAC_ADAP_NORM_CMD_QUEUE; + cm->cm_blkno = blockno; + cm->cm_bcount = blockcnt; + + AAC_DPRINTF(AAC_D_CMD, ("\n")); + aac_enqueue_bio(cm); + aac_startio(sc); + + /* XXX what if enqueue did not start a transfer? */ + if (xs->flags & SCSI_POLL) { + if (!aac_wait_command(cm, xs->timeout)) + { + printf("%s: command timed out\n", + sc->aac_dev.dv_xname); + xs->error = XS_TIMEOUT; + splx(s); + return (TRY_AGAIN_LATER); + } + xs->flags |= ITSDONE; + scsi_done(xs); + } + } - if (LIST_FIRST(&sc->sc_queue) == NULL) - sc->sc_queuelast = NULL; + ready: + /* + * Don't process the queue if we are polling. + */ + if (xs->flags & SCSI_POLL) + retval = COMPLETE; - return (xs); + splx(s); + AAC_DPRINTF(AAC_D_CMD, ("%s: scsi_cmd complete\n", + sc->aac_dev.dv_xname)); + return (retval); } -/******************************************************************************** - * Adapter-space FIB queue manipulation - * - * Note that the queue implementation here is a little funky; neither the PI or - * CI will ever be zero. This behaviour is a controller feature. +/* + * Debugging and Diagnostics */ -static struct { - int size; - int notify; -} aac_qinfo[] = { - { AAC_HOST_NORM_CMD_ENTRIES, AAC_DB_COMMAND_NOT_FULL }, - { AAC_HOST_HIGH_CMD_ENTRIES, 0 }, - { AAC_ADAP_NORM_CMD_ENTRIES, AAC_DB_COMMAND_READY }, - { AAC_ADAP_HIGH_CMD_ENTRIES, 0 }, - { AAC_HOST_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_NOT_FULL }, - { AAC_HOST_HIGH_RESP_ENTRIES, 0 }, - { AAC_ADAP_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_READY }, - { AAC_ADAP_HIGH_RESP_ENTRIES, 0 } -}; /* - * Atomically insert an entry into the nominated queue, returns 0 on success - * or EBUSY if the queue is full. - * - * XXX Note that it would be more efficient to defer notifying the controller - * in the case where we may be inserting several entries in rapid succession, - * but implementing this usefully is difficult. + * Print some information about the controller. */ -int -aac_enqueue_fib(struct aac_softc *sc, int queue, struct aac_ccb *ccb) +void +aac_describe_controller(struct aac_softc *sc) { - u_int32_t pi, ci; - int error; - aac_lock_t lock; - u_int32_t fib_size; - u_int32_t fib_addr; - - fib_size = ccb->ac_fib->Header.Size; - fib_addr = ccb->ac_fib->Header.ReceiverFibAddress; - - lock = AAC_LOCK(sc); - - /* get the producer/consumer indices */ - pi = sc->sc_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]; - ci = sc->sc_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]; + struct aac_fib *fib; + struct aac_adapter_info *info; - /* wrap the queue? */ - if (pi >= aac_qinfo[queue].size) - pi = 0; + aac_alloc_sync_fib(sc, &fib, 0); - /* check for queue full */ - if ((pi + 1) == ci) { - error = EBUSY; - goto out; + fib->data[0] = 0; + if (aac_sync_fib(sc, RequestAdapterInfo, 0, fib, 1)) { + printf("%s: RequestAdapterInfo failed 2\n", + sc->aac_dev.dv_xname); + aac_release_sync_fib(sc); + return; } + info = (struct aac_adapter_info *)&fib->data[0]; - /* populate queue entry */ - (sc->sc_qentries[queue] + pi)->aq_fib_size = fib_size; - (sc->sc_qentries[queue] + pi)->aq_fib_addr = fib_addr; + printf("%s: %s %dMHz, %dMB cache memory, %s\n", sc->aac_dev.dv_xname, + aac_describe_code(aac_cpu_variant, info->CpuVariant), + info->ClockSpeed, info->BufferMem / (1024 * 1024), + aac_describe_code(aac_battery_platform, info->batteryPlatform)); - /* update producer index */ - sc->sc_queues->qt_qindex[queue][AAC_PRODUCER_INDEX] = pi + 1; + /* save the kernel revision structure for later use */ + sc->aac_revision = info->KernelRevision; + printf("%s: Kernel %d.%d-%d, Build %d, S/N %6X\n", + sc->aac_dev.dv_xname, + info->KernelRevision.external.comp.major, + info->KernelRevision.external.comp.minor, + info->KernelRevision.external.comp.dash, + info->KernelRevision.buildNumber, + (u_int32_t)(info->SerialNumber & 0xffffff)); - /* notify the adapter if we know how */ - if (aac_qinfo[queue].notify != 0) - AAC_QNOTIFY(sc, aac_qinfo[queue].notify); + aac_release_sync_fib(sc); - error = 0; - -out: - AAC_UNLOCK(sc, lock); - return (error); +#if 0 + if (1 || bootverbose) { + device_printf(sc->aac_dev, "Supported Options=%b\n", + sc->supported_options, + "\20" + "\1SNAPSHOT" + "\2CLUSTERS" + "\3WCACHE" + "\4DATA64" + "\5HOSTTIME" + "\6RAID50" + "\7WINDOW4GB" + "\10SCSIUPGD" + "\11SOFTERR" + "\12NORECOND" + "\13SGMAP64" + "\14ALARM" + "\15NONDASD"); + } +#endif } /* - * Atomically remove one entry from the nominated queue, returns 0 on success - * or ENOENT if the queue is empty. + * Look up a text description of a numeric error code and return a pointer to + * same. */ -int -aac_dequeue_fib(struct aac_softc *sc, int queue, u_int32_t *fib_size, - struct aac_fib **fib_addr) +char * +aac_describe_code(struct aac_code_lookup *table, u_int32_t code) { - u_int32_t pi, ci; - int notify; - int error; - aac_lock_t lock; - - lock = AAC_LOCK(sc); - - /* get the producer/consumer indices */ - pi = sc->sc_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]; - ci = sc->sc_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]; - - /* check for queue empty */ - if (ci == pi) { - error = ENOENT; - goto out; - } - - notify = 0; - if (ci == pi + 1) - notify++; - - /* wrap the queue? */ - if (ci >= aac_qinfo[queue].size) - ci = 0; - - /* fetch the entry */ - *fib_size = (sc->sc_qentries[queue] + ci)->aq_fib_size; - *fib_addr = - (struct aac_fib *)(sc->sc_qentries[queue] + ci)->aq_fib_addr; - - /* update consumer index */ - sc->sc_queues->qt_qindex[queue][AAC_CONSUMER_INDEX] = ci + 1; - - /* if we have made the queue un-full, notify the adapter */ - if (notify && (aac_qinfo[queue].notify != 0)) - AAC_QNOTIFY(sc, aac_qinfo[queue].notify); - error = 0; + int i; -out: - AAC_UNLOCK(sc, lock); - return (error); + for (i = 0; table[i].string != NULL; i++) + if (table[i].code == code) + return(table[i].string); + return(table[i + 1].string); } #ifdef AAC_DEBUG @@ -1904,4 +2774,286 @@ aac_print_fib(struct aac_softc *sc, struct aac_fib *fib, const char *caller) break; } } + +/* + * Describe an AIF we have received. + */ +void +aac_print_aif(struct aac_softc *sc, struct aac_aif_command *aif) +{ + printf("%s: print_aif: ", sc->aac_dev.dv_xname); + + switch(aif->command) { + case AifCmdEventNotify: + printf("EventNotify(%d)\n", aif->seqNumber); + + switch(aif->data.EN.type) { + case AifEnGeneric: + /* Generic notification */ + printf("\t(Generic) %.*s\n", + (int)sizeof(aif->data.EN.data.EG), + aif->data.EN.data.EG.text); + break; + case AifEnTaskComplete: + /* Task has completed */ + printf("\t(TaskComplete)\n"); + break; + case AifEnConfigChange: + /* Adapter configuration change occurred */ + printf("\t(ConfigChange)\n"); + break; + case AifEnContainerChange: + /* Adapter specific container configuration change */ + printf("\t(ContainerChange) container %d,%d\n", + aif->data.EN.data.ECC.container[0], + aif->data.EN.data.ECC.container[1]); + break; + case AifEnDeviceFailure: + /* SCSI device failed */ + printf("\t(DeviceFailure) handle %d\n", + aif->data.EN.data.EDF.deviceHandle); + break; + case AifEnMirrorFailover: + /* Mirror failover started */ + printf("\t(MirrorFailover) container %d failed, " + "migrating from slice %d to %d\n", + aif->data.EN.data.EMF.container, + aif->data.EN.data.EMF.failedSlice, + aif->data.EN.data.EMF.creatingSlice); + break; + case AifEnContainerEvent: + /* Significant container event */ + printf("\t(ContainerEvent) container %d event %d\n", + aif->data.EN.data.ECE.container, + aif->data.EN.data.ECE.eventType); + break; + case AifEnFileSystemChange: + /* File system changed */ + printf("\t(FileSystemChange)\n"); + break; + case AifEnConfigPause: + /* Container pause event */ + printf("\t(ConfigPause)\n"); + break; + case AifEnConfigResume: + /* Container resume event */ + printf("\t(ConfigResume)\n"); + break; + case AifEnFailoverChange: + /* Failover space assignment changed */ + printf("\t(FailoverChange)\n"); + break; + case AifEnRAID5RebuildDone: + /* RAID5 rebuild finished */ + printf("\t(RAID5RebuildDone)\n"); + break; + case AifEnEnclosureManagement: + /* Enclosure management event */ + printf("\t(EnclosureManagement) EMPID %d unit %d " + "event %d\n", + aif->data.EN.data.EEE.empID, + aif->data.EN.data.EEE.unitID, + aif->data.EN.data.EEE.eventType); + break; + case AifEnBatteryEvent: + /* Significant NV battery event */ + printf("\t(BatteryEvent) %d (state was %d, is %d\n", + aif->data.EN.data.EBE.transition_type, + aif->data.EN.data.EBE.current_state, + aif->data.EN.data.EBE.prior_state); + break; + case AifEnAddContainer: + /* A new container was created. */ + printf("\t(AddContainer)\n"); + break; + case AifEnDeleteContainer: + /* A container was deleted. */ + printf("\t(DeleteContainer)\n"); + break; + case AifEnBatteryNeedsRecond: + /* The battery needs reconditioning */ + printf("\t(BatteryNeedsRecond)\n"); + break; + case AifEnClusterEvent: + /* Some cluster event */ + printf("\t(ClusterEvent) event %d\n", + aif->data.EN.data.ECLE.eventType); + break; + case AifEnDiskSetEvent: + /* A disk set event occured. */ + printf("(DiskSetEvent) event %d " + "diskset %lld creator %lld\n", + aif->data.EN.data.EDS.eventType, + aif->data.EN.data.EDS.DsNum, + aif->data.EN.data.EDS.CreatorId); + break; + case AifDenMorphComplete: + /* A morph operation completed */ + printf("\t(MorphComplete)\n"); + break; + case AifDenVolumeExtendComplete: + /* A volume expand operation completed */ + printf("\t(VolumeExtendComplete)\n"); + break; + default: + printf("\t(%d)\n", aif->data.EN.type); + break; + } + break; + case AifCmdJobProgress: + { + char *status; + switch(aif->data.PR[0].status) { + case AifJobStsSuccess: + status = "success"; break; + case AifJobStsFinished: + status = "finished"; break; + case AifJobStsAborted: + status = "aborted"; break; + case AifJobStsFailed: + status = "failed"; break; + case AifJobStsSuspended: + status = "suspended"; break; + case AifJobStsRunning: + status = "running"; break; + default: + status = "unknown status"; break; + } + + printf("JobProgress (%d) - %s (%d, %d)\n", + aif->seqNumber, status, + aif->data.PR[0].currentTick, + aif->data.PR[0].finalTick); + + switch(aif->data.PR[0].jd.type) { + case AifJobScsiZero: + /* SCSI dev clear operation */ + printf("\t(ScsiZero) handle %d\n", + aif->data.PR[0].jd.client.scsi_dh); + break; + case AifJobScsiVerify: + /* SCSI device Verify operation NO REPAIR */ + printf("\t(ScsiVerify) handle %d\n", + aif->data.PR[0].jd.client.scsi_dh); + break; + case AifJobScsiExercise: + /* SCSI device Exercise operation */ + printf("\t(ScsiExercise) handle %d\n", + aif->data.PR[0].jd.client.scsi_dh); + break; + case AifJobScsiVerifyRepair: + /* SCSI device Verify operation WITH repair */ + printf("\t(ScsiVerifyRepair) handle %d\n", + aif->data.PR[0].jd.client.scsi_dh); + break; + case AifJobCtrZero: + /* Container clear operation */ + printf("\t(ConatainerZero) container %d\n", + aif->data.PR[0].jd.client.container.src); + break; + case AifJobCtrCopy: + /* Container copy operation */ + printf("\t(ConatainerCopy) container %d to %d\n", + aif->data.PR[0].jd.client.container.src, + aif->data.PR[0].jd.client.container.dst); + break; + case AifJobCtrCreateMirror: + /* Container Create Mirror operation */ + printf("\t(ConatainerCreateMirror) container %d\n", + aif->data.PR[0].jd.client.container.src); + /* XXX two containers? */ + break; + case AifJobCtrMergeMirror: + /* Container Merge Mirror operation */ + printf("\t(ConatainerMergeMirror) container %d\n", + aif->data.PR[0].jd.client.container.src); + /* XXX two containers? */ + break; + case AifJobCtrScrubMirror: + /* Container Scrub Mirror operation */ + printf("\t(ConatainerScrubMirror) container %d\n", + aif->data.PR[0].jd.client.container.src); + break; + case AifJobCtrRebuildRaid5: + /* Container Rebuild Raid5 operation */ + printf("\t(ConatainerRebuildRaid5) container %d\n", + aif->data.PR[0].jd.client.container.src); + break; + case AifJobCtrScrubRaid5: + /* Container Scrub Raid5 operation */ + printf("\t(ConatainerScrubRaid5) container %d\n", + aif->data.PR[0].jd.client.container.src); + break; + case AifJobCtrMorph: + /* Container morph operation */ + printf("\t(ConatainerMorph) container %d\n", + aif->data.PR[0].jd.client.container.src); + /* XXX two containers? */ + break; + case AifJobCtrPartCopy: + /* Container Partition copy operation */ + printf("\t(ConatainerPartCopy) container %d to %d\n", + aif->data.PR[0].jd.client.container.src, + aif->data.PR[0].jd.client.container.dst); + break; + case AifJobCtrRebuildMirror: + /* Container Rebuild Mirror operation */ + printf("\t(ConatainerRebuildMirror) container %d\n", + aif->data.PR[0].jd.client.container.src); + break; + case AifJobCtrCrazyCache: + /* crazy cache */ + printf("\t(ConatainerCrazyCache) container %d\n", + aif->data.PR[0].jd.client.container.src); + /* XXX two containers? */ + break; + case AifJobFsCreate: + /* File System Create operation */ + printf("\t(FsCreate)\n"); + break; + case AifJobFsVerify: + /* File System Verify operation */ + printf("\t(FsVerivy)\n"); + break; + case AifJobFsExtend: + /* File System Extend operation */ + printf("\t(FsExtend)\n"); + break; + case AifJobApiFormatNTFS: + /* Format a drive to NTFS */ + printf("\t(FormatNTFS)\n"); + break; + case AifJobApiFormatFAT: + /* Format a drive to FAT */ + printf("\t(FormatFAT)\n"); + break; + case AifJobApiUpdateSnapshot: + /* update the read/write half of a snapshot */ + printf("\t(UpdateSnapshot)\n"); + break; + case AifJobApiFormatFAT32: + /* Format a drive to FAT32 */ + printf("\t(FormatFAT32)\n"); + break; + case AifJobCtlContinuousCtrVerify: + /* Adapter operation */ + printf("\t(ContinuousCtrVerify)\n"); + break; + default: + printf("\t(%d)\n", aif->data.PR[0].jd.type); + break; + } + break; + } + case AifCmdAPIReport: + printf("APIReport (%d)\n", aif->seqNumber); + break; + case AifCmdDriverNotify: + printf("DriverNotify (%d)\n", aif->seqNumber); + break; + default: + printf("AIF %d (%d)\n", aif->command, aif->seqNumber); + break; + } +} #endif diff --git a/sys/dev/ic/aac_tables.h b/sys/dev/ic/aac_tables.h index c0fedc3ca60..3ba773425fd 100644 --- a/sys/dev/ic/aac_tables.h +++ b/sys/dev/ic/aac_tables.h @@ -1,4 +1,4 @@ -/* $OpenBSD: aac_tables.h,v 1.2 2004/05/21 20:34:52 marco Exp $ */ +/* $OpenBSD: aac_tables.h,v 1.3 2005/11/18 05:39:10 nate Exp $ */ /*- * Copyright (c) 2000 Michael Smith @@ -83,7 +83,7 @@ static struct aac_code_lookup aac_cpu_variant[] = { { "i960HX", CPUI960_HX }, { "i960RX", CPUI960_RX }, { "StrongARM SA110", CPUARM_SA110 }, - { "PowerPC 603e", CPUPPC_603e }, + { "MPC824x", CPUMPC_824x }, { "Unknown StrongARM", CPUARM_xxx }, { "Unknown PowerPC", CPUPPC_xxx }, { "Intel GC80302 IOP", CPUI960_302}, diff --git a/sys/dev/ic/aacreg.h b/sys/dev/ic/aacreg.h index 1a66fa1a28a..4557e1485e4 100644 --- a/sys/dev/ic/aacreg.h +++ b/sys/dev/ic/aacreg.h @@ -1,9 +1,10 @@ -/* $OpenBSD: aacreg.h,v 1.6 2004/05/21 20:34:52 marco Exp $ */ +/* $OpenBSD: aacreg.h,v 1.7 2005/11/18 05:39:10 nate Exp $ */ /*- * Copyright (c) 2000 Michael Smith - * Copyright (c) 2000 Scott Long + * Copyright (c) 2000-2001 Scott Long * Copyright (c) 2000 BSDi + * Copyright (c) 2001 Adaptec, Inc. * Copyright (c) 2000 Niklas Hallqvist * All rights reserved. * @@ -28,7 +29,7 @@ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * - * $FreeBSD: /c/ncvs/src/sys/dev/aac/aacreg.h,v 1.1 2000/09/13 03:20:34 msmith Exp $ + * $FreeBSD$ */ /* @@ -55,27 +56,37 @@ * We establish 4 command queues and matching response queues. Queues must * be 16-byte aligned, and are sized as follows: */ -#define AAC_HOST_NORM_CMD_ENTRIES 8 /* cmd adapter->host, normal pri */ -#define AAC_HOST_HIGH_CMD_ENTRIES 4 /* cmd adapter->host, high pri */ -#define AAC_ADAP_NORM_CMD_ENTRIES 512 /* cmd host->adapter, normal pri */ -#define AAC_ADAP_HIGH_CMD_ENTRIES 4 /* cmd host->adapter, high pri */ -#define AAC_HOST_NORM_RESP_ENTRIES 512 /* resp, adapter->host, normal pri */ -#define AAC_HOST_HIGH_RESP_ENTRIES 4 /* resp, adapter->host, high pri */ -#define AAC_ADAP_NORM_RESP_ENTRIES 8 /* resp, host->adapter, normal pri */ -#define AAC_ADAP_HIGH_RESP_ENTRIES 4 /* resp, host->adapter, high pri */ - -#define AAC_TOTALQ_LENGTH \ - (AAC_HOST_HIGH_CMD_ENTRIES + AAC_HOST_NORM_CMD_ENTRIES + \ - AAC_ADAP_HIGH_CMD_ENTRIES + AAC_ADAP_NORM_CMD_ENTRIES + \ - AAC_HOST_HIGH_RESP_ENTRIES + AAC_HOST_NORM_RESP_ENTRIES + \ - AAC_ADAP_HIGH_RESP_ENTRIES + AAC_ADAP_NORM_RESP_ENTRIES) - +#define AAC_HOST_NORM_CMD_ENTRIES 8 /* command adapter->host, + * normal priority */ +#define AAC_HOST_HIGH_CMD_ENTRIES 4 /* command adapter->host, + * high priority */ +#define AAC_ADAP_NORM_CMD_ENTRIES 512 /* command host->adapter, + * normal priority */ +#define AAC_ADAP_HIGH_CMD_ENTRIES 4 /* command host->adapter, + * high priority */ +#define AAC_HOST_NORM_RESP_ENTRIES 512 /* response, adapter->host, + * normal priority */ +#define AAC_HOST_HIGH_RESP_ENTRIES 4 /* response, adapter->host, + * high priority */ +#define AAC_ADAP_NORM_RESP_ENTRIES 8 /* response, host->adapter, + * normal priority */ +#define AAC_ADAP_HIGH_RESP_ENTRIES 4 /* response, host->adapter, + * high priority */ + +#define AAC_TOTALQ_LENGTH (AAC_HOST_HIGH_CMD_ENTRIES + \ + AAC_HOST_NORM_CMD_ENTRIES + \ + AAC_ADAP_HIGH_CMD_ENTRIES + \ + AAC_ADAP_NORM_CMD_ENTRIES + \ + AAC_HOST_HIGH_RESP_ENTRIES + \ + AAC_HOST_NORM_RESP_ENTRIES + \ + AAC_ADAP_HIGH_RESP_ENTRIES + \ + AAC_ADAP_NORM_RESP_ENTRIES) #define AAC_QUEUE_COUNT 8 #define AAC_QUEUE_ALIGN 16 struct aac_queue_entry { - u_int32_t aq_fib_size; /* FIB size in bytes */ - u_int32_t aq_fib_addr; /* receiver-space address of the FIB */ + u_int32_t aq_fib_size; /* FIB size in bytes */ + u_int32_t aq_fib_addr; /* receiver-space address of the FIB */ } __attribute__ ((__packed__)); #define AAC_PRODUCER_INDEX 0 @@ -87,48 +98,200 @@ struct aac_queue_entry { */ struct aac_queue_table { /* queue consumer/producer indexes (layout mandated by adapter) */ - u_int32_t qt_qindex[AAC_QUEUE_COUNT][2]; + u_int32_t qt_qindex[AAC_QUEUE_COUNT][2]; /* queue entry structures (layout mandated by adapter) */ - struct aac_queue_entry qt_HostNormCmdQueue[AAC_HOST_NORM_CMD_ENTRIES]; - struct aac_queue_entry qt_HostHighCmdQueue[AAC_HOST_HIGH_CMD_ENTRIES]; - struct aac_queue_entry qt_AdapNormCmdQueue[AAC_ADAP_NORM_CMD_ENTRIES]; - struct aac_queue_entry qt_AdapHighCmdQueue[AAC_ADAP_HIGH_CMD_ENTRIES]; - struct aac_queue_entry - qt_HostNormRespQueue[AAC_HOST_NORM_RESP_ENTRIES]; - struct aac_queue_entry - qt_HostHighRespQueue[AAC_HOST_HIGH_RESP_ENTRIES]; - struct aac_queue_entry - qt_AdapNormRespQueue[AAC_ADAP_NORM_RESP_ENTRIES]; - struct aac_queue_entry - qt_AdapHighRespQueue[AAC_ADAP_HIGH_RESP_ENTRIES]; + struct aac_queue_entry qt_HostNormCmdQueue [AAC_HOST_NORM_CMD_ENTRIES]; + struct aac_queue_entry qt_HostHighCmdQueue [AAC_HOST_HIGH_CMD_ENTRIES]; + struct aac_queue_entry qt_AdapNormCmdQueue [AAC_ADAP_NORM_CMD_ENTRIES]; + struct aac_queue_entry qt_AdapHighCmdQueue [AAC_ADAP_HIGH_CMD_ENTRIES]; + struct aac_queue_entry qt_HostNormRespQueue[AAC_HOST_NORM_RESP_ENTRIES]; + struct aac_queue_entry qt_HostHighRespQueue[AAC_HOST_HIGH_RESP_ENTRIES]; + struct aac_queue_entry qt_AdapNormRespQueue[AAC_ADAP_NORM_RESP_ENTRIES]; + struct aac_queue_entry qt_AdapHighRespQueue[AAC_ADAP_HIGH_RESP_ENTRIES]; +} __attribute__ ((__packed__)); + +/* + * Queue names + * + * Note that we base these at 0 in order to use them as array indices. Adaptec + * used base 1 for some unknown reason, and sorted them in a different order. + */ +#define AAC_HOST_NORM_CMD_QUEUE 0 +#define AAC_HOST_HIGH_CMD_QUEUE 1 +#define AAC_ADAP_NORM_CMD_QUEUE 2 +#define AAC_ADAP_HIGH_CMD_QUEUE 3 +#define AAC_HOST_NORM_RESP_QUEUE 4 +#define AAC_HOST_HIGH_RESP_QUEUE 5 +#define AAC_ADAP_NORM_RESP_QUEUE 6 +#define AAC_ADAP_HIGH_RESP_QUEUE 7 + +/* + * List structure used to chain FIBs (used by the adapter - we hang FIBs off + * our private command structure and don't touch these) + */ +struct aac_fib_list_entry { + u_int32_t Flink; + u_int32_t Blink; } __attribute__ ((__packed__)); /* + * FIB (FSA Interface Block?); this is the datastructure passed between the host + * and adapter. + */ +struct aac_fib_header { + u_int32_t XferState; + u_int16_t Command; + u_int8_t StructType; + u_int8_t Flags; + u_int16_t Size; + u_int16_t SenderSize; + u_int32_t SenderFibAddress; + u_int32_t ReceiverFibAddress; + u_int32_t SenderData; + union { + struct { + u_int32_t ReceiverTimeStart; + u_int32_t ReceiverTimeDone; + } _s; + struct aac_fib_list_entry FibLinks; + } _u; +} __attribute__ ((__packed__)); + +#define AAC_FIB_DATASIZE (512 - sizeof(struct aac_fib_header)) + +struct aac_fib { + struct aac_fib_header Header; + u_int8_t data[AAC_FIB_DATASIZE]; +} __attribute__ ((__packed__)); + +/* + * FIB commands + */ +typedef enum { + TestCommandResponse = 1, + TestAdapterCommand = 2, + + /* lowlevel and comm commands */ + LastTestCommand = 100, + ReinitHostNormCommandQueue = 101, + ReinitHostHighCommandQueue = 102, + ReinitHostHighRespQueue = 103, + ReinitHostNormRespQueue = 104, + ReinitAdapNormCommandQueue = 105, + ReinitAdapHighCommandQueue = 107, + ReinitAdapHighRespQueue = 108, + ReinitAdapNormRespQueue = 109, + InterfaceShutdown = 110, + DmaCommandFib = 120, + StartProfile = 121, + TermProfile = 122, + SpeedTest = 123, + TakeABreakPt = 124, + RequestPerfData = 125, + SetInterruptDefTimer= 126, + SetInterruptDefCount= 127, + GetInterruptDefStatus= 128, + LastCommCommand = 129, + + /* filesystem commands */ + NuFileSystem = 300, + UFS = 301, + HostFileSystem = 302, + LastFileSystemCommand = 303, + + /* Container Commands */ + ContainerCommand = 500, + ContainerCommand64 = 501, + + /* Cluster Commands */ + ClusterCommand = 550, + + /* Scsi Port commands (scsi passthrough) */ + ScsiPortCommand = 600, + + /* misc house keeping and generic adapter initiated commands */ + AifRequest = 700, + CheckRevision = 701, + FsaHostShutdown = 702, + RequestAdapterInfo = 703, + IsAdapterPaused = 704, + SendHostTime = 705, + LastMiscCommand = 706 +} AAC_FibCommands; + +/* + * FIB types + */ +#define AAC_FIBTYPE_TFIB 1 +#define AAC_FIBTYPE_TQE 2 +#define AAC_FIBTYPE_TCTPERF 3 + +/* + * FIB transfer state + */ +#define AAC_FIBSTATE_HOSTOWNED (1<<0) /* owned by the host */ +#define AAC_FIBSTATE_ADAPTEROWNED (1<<1) /* owned by the adapter */ +#define AAC_FIBSTATE_INITIALISED (1<<2) /* initialised */ +#define AAC_FIBSTATE_EMPTY (1<<3) /* empty */ +#define AAC_FIBSTATE_FROMPOOL (1<<4) /* allocated from pool */ +#define AAC_FIBSTATE_FROMHOST (1<<5) /* sent from the host */ +#define AAC_FIBSTATE_FROMADAP (1<<6) /* sent from the adapter */ +#define AAC_FIBSTATE_REXPECTED (1<<7) /* response is expected */ +#define AAC_FIBSTATE_RNOTEXPECTED (1<<8) /* response is not expected */ +#define AAC_FIBSTATE_DONEADAP (1<<9) /* processed by the adapter */ +#define AAC_FIBSTATE_DONEHOST (1<<10) /* processed by the host */ +#define AAC_FIBSTATE_HIGH (1<<11) /* high priority */ +#define AAC_FIBSTATE_NORM (1<<12) /* normal priority */ +#define AAC_FIBSTATE_ASYNC (1<<13) +#define AAC_FIBSTATE_ASYNCIO (1<<13) /* to be removed */ +#define AAC_FIBSTATE_PAGEFILEIO (1<<14) /* to be removed */ +#define AAC_FIBSTATE_SHUTDOWN (1<<15) +#define AAC_FIBSTATE_LAZYWRITE (1<<16) /* to be removed */ +#define AAC_FIBSTATE_ADAPMICROFIB (1<<17) +#define AAC_FIBSTATE_BIOSFIB (1<<18) +#define AAC_FIBSTATE_FAST_RESPONSE (1<<19) /* fast response capable */ +#define AAC_FIBSTATE_APIFIB (1<<20) + +/* + * FIB error values + */ +#define AAC_ERROR_NORMAL 0x00 +#define AAC_ERROR_PENDING 0x01 +#define AAC_ERROR_FATAL 0x02 +#define AAC_ERROR_INVALID_QUEUE 0x03 +#define AAC_ERROR_NOENTRIES 0x04 +#define AAC_ERROR_SENDFAILED 0x05 +#define AAC_ERROR_INVALID_QUEUE_PRIORITY 0x06 +#define AAC_ERROR_FIB_ALLOCATION_FAILED 0x07 +#define AAC_ERROR_FIB_DEALLOCATION_FAILED 0x08 + +/* * Adapter Init Structure: this is passed to the adapter with the * AAC_MONKER_INITSTRUCT command to point it at our control structures. */ struct aac_adapter_init { - u_int32_t InitStructRevision; -#define AAC_INIT_STRUCT_REVISION 3 - u_int32_t MiniPortRevision; - u_int32_t FilesystemRevision; - u_int32_t CommHeaderAddress; - u_int32_t FastIoCommAreaAddress; - u_int32_t AdapterFibsPhysicalAddress; - void *AdapterFibsVirtualAddress; - u_int32_t AdapterFibsSize; - u_int32_t AdapterFibAlign; - u_int32_t PrintfBufferAddress; - u_int32_t PrintfBufferSize; - u_int32_t HostPhysMemPages; - u_int32_t HostElapsedSeconds; -} __packed; + u_int32_t InitStructRevision; +#define AAC_INIT_STRUCT_REVISION 3 + u_int32_t MiniPortRevision; +#define AAC_INIT_STRUCT_MINIPORT_REVISION 1 + u_int32_t FilesystemRevision; + u_int32_t CommHeaderAddress; + u_int32_t FastIoCommAreaAddress; + u_int32_t AdapterFibsPhysicalAddress; + u_int32_t AdapterFibsVirtualAddress; + u_int32_t AdapterFibsSize; + u_int32_t AdapterFibAlign; + u_int32_t PrintfBufferAddress; + u_int32_t PrintfBufferSize; +#define AAC_PAGE_SIZE 4096 + u_int32_t HostPhysMemPages; + u_int32_t HostElapsedSeconds; +} __attribute__ ((__packed__)); /* * Shared data types */ - /* * Container types */ @@ -146,7 +309,8 @@ typedef enum { CT_RAID10, /* stripe of mirror */ CT_RAID00, /* stripe of stripe */ CT_VOLUME_OF_MIRRORS, /* volume of mirror */ - CT_PSEUDO_RAID3 /* really raid4 */ + CT_PSEUDO_RAID3, /* really raid4 */ + CT_RAID50, /* stripe of raid5 */ } AAC_FSAVolType; /* @@ -161,9 +325,10 @@ typedef enum { FT_SOCK, /* socket */ FT_FIFO, /* fifo */ FT_FILESYS, /* ADAPTEC's "FSA"(tm) filesystem */ - FT_DRIVE, /* phys disk - addressable in scsi by bus/target/lun */ + FT_DRIVE, /* physical disk - addressable in scsi by b/t/l */ FT_SLICE, /* virtual disk - raw volume - slice */ - FT_PARTITION, /* FSA part, inside slice, container building block */ + FT_PARTITION, /* FSA partition - carved out of a slice - building + * block for containers */ FT_VOLUME, /* Container - Volume Set */ FT_STRIPE, /* Container - Stripe Set */ FT_MIRROR, /* Container - Mirror Set */ @@ -175,53 +340,72 @@ typedef enum { * Host-side scatter/gather list for 32-bit commands. */ struct aac_sg_entry { - u_int32_t SgAddress; - u_int32_t SgByteCount; + u_int32_t SgAddress; + u_int32_t SgByteCount; +} __attribute__ ((__packed__)); + +struct aac_sg_entry64 { + u_int64_t SgAddress; + u_int32_t SgByteCount; } __attribute__ ((__packed__)); struct aac_sg_table { - u_int32_t SgCount; - struct aac_sg_entry SgEntry[0]; + u_int32_t SgCount; + struct aac_sg_entry SgEntry[0]; } __attribute__ ((__packed__)); /* * Host-side scatter/gather list for 64-bit commands. */ struct aac_sg_table64 { - u_int8_t SgCount; - u_int8_t SgSectorsPerPage; - u_int16_t SgByteOffset; - u_int64_t SgEntry[0]; + u_int32_t SgCount; + struct aac_sg_entry64 SgEntry64[0]; } __attribute__ ((__packed__)); /* * Container creation data */ struct aac_container_creation { - u_int8_t ViaBuildNumber; - u_int8_t MicroSecond; - u_int8_t Via; /* 1 = FSU, 2 = API, etc. */ - u_int8_t YearsSince1900; - u_int32_t Month:4; /* 1-12 */ - u_int32_t Day:6; /* 1-32 */ - u_int32_t Hour:6; /* 0-23 */ - u_int32_t Minute:6; /* 0-59 */ - u_int32_t Second:6; /* 0-59 */ - u_int64_t ViaAdapterSerialNumber; + u_int8_t ViaBuildNumber; + u_int8_t MicroSecond; + u_int8_t Via; /* 1 = FSU, 2 = API, etc. */ + u_int8_t YearsSince1900; + u_int32_t Month:4; /* 1-12 */ + u_int32_t Day:6; /* 1-32 */ + u_int32_t Hour:6; /* 0-23 */ + u_int32_t Minute:6; /* 0-59 */ + u_int32_t Second:6; /* 0-59 */ + u_int64_t ViaAdapterSerialNumber; } __attribute__ ((__packed__)); +/* + * Revision number handling + */ + +typedef enum { + RevApplication = 1, + RevDkiCli, + RevNetService, + RevApi, + RevFileSysDriver, + RevMiniportDriver, + RevAdapterSW, + RevMonitor, + RevRemoteApi +} RevComponent; + struct FsaRevision { union { - struct { - u_int8_t dash; - u_int8_t type; - u_int8_t minor; - u_int8_t major; - } comp; - u_int32_t ul; + struct { + u_int8_t dash; + u_int8_t type; + u_int8_t minor; + u_int8_t major; + } comp; + u_int32_t ul; } external; - u_int32_t buildNumber; -} __packed; + u_int32_t buildNumber; +} __attribute__ ((__packed__)); /* * Adapter Information @@ -246,7 +430,7 @@ typedef enum { CPUI960_RX, CPUARM_SA110, CPUARM_xxx, - CPUPPC_603e, + CPUMPC_824x, CPUPPC_xxx, CPUI960_302, CPUSUBTYPE__last @@ -291,7 +475,8 @@ typedef enum { /* * XXX the aac-2622 with no battery present reports PLATFORM_BAT_OPT_PRESENT */ -typedef enum { +typedef enum +{ PLATFORM_BAT_REQ_PRESENT = 1, /* BATTERY REQUIRED AND PRESENT */ PLATFORM_BAT_REQ_NOTPRESENT, /* BATTERY REQUIRED AND NOT PRESENT */ PLATFORM_BAT_OPT_PRESENT, /* BATTERY OPTIONAL AND PRESENT */ @@ -300,27 +485,50 @@ typedef enum { } AAC_BatteryPlatform; /* + * options supported by this board + * there has to be a one to one mapping of these defines and the ones in + * fsaapi.h, search for FSA_SUPPORT_SNAPSHOT + */ +#define AAC_SUPPORTED_SNAPSHOT 0x01 +#define AAC_SUPPORTED_CLUSTERS 0x02 +#define AAC_SUPPORTED_WRITE_CACHE 0x04 +#define AAC_SUPPORTED_64BIT_DATA 0x08 +#define AAC_SUPPORTED_HOST_TIME_FIB 0x10 +#define AAC_SUPPORTED_RAID50 0x20 +#define AAC_SUPPORTED_4GB_WINDOW 0x40 +#define AAC_SUPPORTED_SCSI_UPGRADEABLE 0x80 +#define AAC_SUPPORTED_SOFT_ERR_REPORT 0x100 +#define AAC_SUPPORTED_NOT_RECONDITION 0x200 +#define AAC_SUPPORTED_SGMAP_HOST64 0x400 +#define AAC_SUPPORTED_ALARM 0x800 +#define AAC_SUPPORTED_NONDASD 0x1000 + +/* * Structure used to respond to a RequestAdapterInfo fib. */ struct aac_adapter_info { - AAC_Platform PlatformBase; /* adapter type */ - AAC_CpuType CpuArchitecture; /* adapter CPU type */ - AAC_CpuSubType CpuVariant; /* adapter CPU subtype */ - u_int32_t ClockSpeed; /* adapter CPU clockspeed */ - u_int32_t ExecutionMem; /* adapter Execution Memory size */ - u_int32_t BufferMem; /* adapter Data Memory */ - u_int32_t TotalMem; /* adapter Total Memory */ - struct FsaRevision KernelRevision; /* adapter Kernel SW Revision */ - struct FsaRevision MonitorRevision; /* adapter Monitor/Diag SW Rev */ - struct FsaRevision HardwareRevision; /* TDB */ - struct FsaRevision BIOSRevision; /* adapter BIOS Revision */ - u_int32_t ClusteringEnabled; - u_int32_t ClusterChannelMask; - u_int64_t SerialNumber; - AAC_BatteryPlatform batteryPlatform; - u_int32_t SupportedOptions; /* supported features of this ctrlr */ - AAC_OemFlavor OemVariant; -} __packed; + AAC_Platform PlatformBase; /* adapter type */ + AAC_CpuType CpuArchitecture; /* adapter CPU type */ + AAC_CpuSubType CpuVariant; /* adapter CPU subtype */ + u_int32_t ClockSpeed; /* adapter CPU clockspeed */ + u_int32_t ExecutionMem; /* adapter Execution Memory + * size */ + u_int32_t BufferMem; /* adapter Data Memory */ + u_int32_t TotalMem; /* adapter Total Memory */ + struct FsaRevision KernelRevision; /* adapter Kernel Software + * Revision */ + struct FsaRevision MonitorRevision; /* adapter Monitor/Diagnostic + * Software Revision */ + struct FsaRevision HardwareRevision;/* TBD */ + struct FsaRevision BIOSRevision; /* adapter BIOS Revision */ + u_int32_t ClusteringEnabled; + u_int32_t ClusterChannelMask; + u_int64_t SerialNumber; + AAC_BatteryPlatform batteryPlatform; + u_int32_t SupportedOptions; /* supported features of this + * controller */ + AAC_OemFlavor OemVariant; +} __attribute__ ((__packed__)); /* * Monitor/Kernel interface. @@ -331,104 +539,588 @@ struct aac_adapter_info { */ #define AAC_MONKER_INITSTRUCT 0x05 #define AAC_MONKER_SYNCFIB 0x0c +#define AAC_MONKER_GETKERNVER 0x11 +#define AAC_MONKER_GETINFO 0x19 + +/* + * Adapter Status Register + * + * Phase Staus mailbox is 32bits: + * <31:16> = Phase Status + * <15:0> = Phase + * + * The adapter reports its present state through the phase. Only + * a single phase should be ever be set. Each phase can have multiple + * phase status bits to provide more detailed information about the + * state of the adapter. + */ +#define AAC_SELF_TEST_FAILED 0x00000004 +#define AAC_UP_AND_RUNNING 0x00000080 +#define AAC_KERNEL_PANIC 0x00000100 + +/* + * Data types relating to control and monitoring of the NVRAM/WriteCache + * subsystem. + */ + +#define AAC_NFILESYS 24 /* maximum number of filesystems */ + +/* + * NVRAM/Write Cache subsystem states + */ +typedef enum { + NVSTATUS_DISABLED = 0, /* present, clean, not being used */ + NVSTATUS_ENABLED, /* present, possibly dirty, ready for use */ + NVSTATUS_ERROR, /* present, dirty, contains dirty data */ + NVSTATUS_BATTERY, /* present, bad or low battery, may contain + * dirty data */ + NVSTATUS_UNKNOWN /* for bad/missing device */ +} AAC_NVSTATUS; + +/* + * NVRAM/Write Cache subsystem battery component states + * + */ +typedef enum { + NVBATTSTATUS_NONE = 0, /* battery has no power or is not present */ + NVBATTSTATUS_LOW, /* battery is low on power */ + NVBATTSTATUS_OK, /* battery is okay - normal operation possible + * only in this state */ + NVBATTSTATUS_RECONDITIONING /* no battery present - reconditioning + * in process */ +} AAC_NVBATTSTATUS; + +/* + * Battery transition type + */ +typedef enum { + NVBATT_TRANSITION_NONE = 0, /* battery now has no power or is not + * present */ + NVBATT_TRANSITION_LOW, /* battery is now low on power */ + NVBATT_TRANSITION_OK /* battery is now okay - normal + * operation possible only in this + * state */ +} AAC_NVBATT_TRANSITION; + +/* + * NVRAM Info structure returned for NVRAM_GetInfo call + */ +struct aac_nvramdevinfo { + u_int32_t NV_Enabled; /* write caching enabled */ + u_int32_t NV_Error; /* device in error state */ + u_int32_t NV_NDirty; /* count of dirty NVRAM buffers */ + u_int32_t NV_NActive; /* count of NVRAM buffers being + * written */ +} __attribute__ ((__packed__)); + +struct aac_nvraminfo { + AAC_NVSTATUS NV_Status; /* nvram subsystem status */ + AAC_NVBATTSTATUS NV_BattStatus; /* battery status */ + u_int32_t NV_Size; /* size of WriteCache NVRAM in + * bytes */ + u_int32_t NV_BufSize; /* size of NVRAM buffers in + * bytes */ + u_int32_t NV_NBufs; /* number of NVRAM buffers */ + u_int32_t NV_NDirty; /* Num dirty NVRAM buffers */ + u_int32_t NV_NClean; /* Num clean NVRAM buffers */ + u_int32_t NV_NActive; /* Num NVRAM buffers being + * written */ + u_int32_t NV_NBrokered; /* Num brokered NVRAM buffers */ + struct aac_nvramdevinfo NV_DevInfo[AAC_NFILESYS]; /* per device + * info */ + u_int32_t NV_BattNeedsReconditioning; /* boolean */ + u_int32_t NV_TotalSize; /* size of all non-volatile + * memories in bytes */ +} __attribute__ ((__packed__)); + +/* + * Data types relating to adapter-initiated FIBs + * + * Based on types and structures in <aifstruc.h> + */ + +/* + * Progress Reports + */ +typedef enum { + AifJobStsSuccess = 1, + AifJobStsFinished, + AifJobStsAborted, + AifJobStsFailed, + AifJobStsLastReportMarker = 100, /* All prior mean last report */ + AifJobStsSuspended, + AifJobStsRunning +} AAC_AifJobStatus; + +typedef enum { + AifJobScsiMin = 1, /* Minimum value for Scsi operation */ + AifJobScsiZero, /* SCSI device clear operation */ + AifJobScsiVerify, /* SCSI device Verify operation NO + * REPAIR */ + AifJobScsiExercise, /* SCSI device Exercise operation */ + AifJobScsiVerifyRepair, /* SCSI device Verify operation WITH + * repair */ + AifJobScsiMax = 99, /* Max Scsi value */ + AifJobCtrMin, /* Min Ctr op value */ + AifJobCtrZero, /* Container clear operation */ + AifJobCtrCopy, /* Container copy operation */ + AifJobCtrCreateMirror, /* Container Create Mirror operation */ + AifJobCtrMergeMirror, /* Container Merge Mirror operation */ + AifJobCtrScrubMirror, /* Container Scrub Mirror operation */ + AifJobCtrRebuildRaid5, /* Container Rebuild Raid5 operation */ + AifJobCtrScrubRaid5, /* Container Scrub Raid5 operation */ + AifJobCtrMorph, /* Container morph operation */ + AifJobCtrPartCopy, /* Container Partition copy operation */ + AifJobCtrRebuildMirror, /* Container Rebuild Mirror operation */ + AifJobCtrCrazyCache, /* crazy cache */ + AifJobCtrMax = 199, /* Max Ctr type operation */ + AifJobFsMin, /* Min Fs type operation */ + AifJobFsCreate, /* File System Create operation */ + AifJobFsVerify, /* File System Verify operation */ + AifJobFsExtend, /* File System Extend operation */ + AifJobFsMax = 299, /* Max Fs type operation */ + AifJobApiFormatNTFS, /* Format a drive to NTFS */ + AifJobApiFormatFAT, /* Format a drive to FAT */ + AifJobApiUpdateSnapshot, /* update the read/write half of a + * snapshot */ + AifJobApiFormatFAT32, /* Format a drive to FAT32 */ + AifJobApiMax = 399, /* Max API type operation */ + AifJobCtlContinuousCtrVerify, /* Adapter operation */ + AifJobCtlMax = 499 /* Max Adapter type operation */ +} AAC_AifJobType; + +struct aac_AifContainers { + u_int32_t src; /* from/master */ + u_int32_t dst; /* to/slave */ +} __attribute__ ((__packed__)); + +union aac_AifJobClient { + struct aac_AifContainers container; /* For Container and + * filesystem progress + * ops; */ + int32_t scsi_dh; /* For SCSI progress + * ops */ +}; + +struct aac_AifJobDesc { + u_int32_t jobID; /* DO NOT FILL IN! Will be + * filled in by AIF */ + AAC_AifJobType type; /* Operation that is being + * performed */ + union aac_AifJobClient client; /* Details */ +} __attribute__ ((__packed__)); + +struct aac_AifJobProgressReport { + struct aac_AifJobDesc jd; + AAC_AifJobStatus status; + u_int32_t finalTick; + u_int32_t currentTick; + u_int32_t jobSpecificData1; + u_int32_t jobSpecificData2; +} __attribute__ ((__packed__)); + +/* + * Event Notification + */ +typedef enum { + /* General application notifies start here */ + AifEnGeneric = 1, /* Generic notification */ + AifEnTaskComplete, /* Task has completed */ + AifEnConfigChange, /* Adapter config change occurred */ + AifEnContainerChange, /* Adapter specific container + * configuration change */ + AifEnDeviceFailure, /* SCSI device failed */ + AifEnMirrorFailover, /* Mirror failover started */ + AifEnContainerEvent, /* Significant container event */ + AifEnFileSystemChange, /* File system changed */ + AifEnConfigPause, /* Container pause event */ + AifEnConfigResume, /* Container resume event */ + AifEnFailoverChange, /* Failover space assignment changed */ + AifEnRAID5RebuildDone, /* RAID5 rebuild finished */ + AifEnEnclosureManagement, /* Enclosure management event */ + AifEnBatteryEvent, /* Significant NV battery event */ + AifEnAddContainer, /* A new container was created. */ + AifEnDeleteContainer, /* A container was deleted. */ + AifEnSMARTEvent, /* SMART Event */ + AifEnBatteryNeedsRecond, /* The battery needs reconditioning */ + AifEnClusterEvent, /* Some cluster event */ + AifEnDiskSetEvent, /* A disk set event occured. */ + AifDriverNotifyStart=199, /* Notifies for host driver go here */ + /* Host driver notifications start here */ + AifDenMorphComplete, /* A morph operation completed */ + AifDenVolumeExtendComplete /* Volume expand operation completed */ +} AAC_AifEventNotifyType; + +struct aac_AifEnsGeneric { + char text[132]; /* Generic text */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsDeviceFailure { + u_int32_t deviceHandle; /* SCSI device handle */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsMirrorFailover { + u_int32_t container; /* Container with failed element */ + u_int32_t failedSlice; /* Old slice which failed */ + u_int32_t creatingSlice; /* New slice used for auto-create */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsContainerChange { + u_int32_t container[2]; /* container that changed, -1 if no + * container */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsContainerEvent { + u_int32_t container; /* container number */ + u_int32_t eventType; /* event type */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsEnclosureEvent { + u_int32_t empID; /* enclosure management proc number */ + u_int32_t unitID; /* unitId, fan id, power supply id, + * slot id, tempsensor id. */ + u_int32_t eventType; /* event type */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsBatteryEvent { + AAC_NVBATT_TRANSITION transition_type; /* eg from low to ok */ + AAC_NVBATTSTATUS current_state; /* current batt state */ + AAC_NVBATTSTATUS prior_state; /* prev batt state */ +} __attribute__ ((__packed__)); + +struct aac_AifEnsDiskSetEvent { + u_int32_t eventType; + u_int64_t DsNum; + u_int64_t CreatorId; +} __attribute__ ((__packed__)); + +typedef enum { + CLUSTER_NULL_EVENT = 0, + CLUSTER_PARTNER_NAME_EVENT, /* change in partner hostname or + * adaptername from NULL to non-NULL */ + /* (partner's agent may be up) */ + CLUSTER_PARTNER_NULL_NAME_EVENT /* change in partner hostname or + * adaptername from non-null to NULL */ + /* (partner has rebooted) */ +} AAC_ClusterAifEvent; + +struct aac_AifEnsClusterEvent { + AAC_ClusterAifEvent eventType; +} __attribute__ ((__packed__)); + +struct aac_AifEventNotify { + AAC_AifEventNotifyType type; + union { + struct aac_AifEnsGeneric EG; + struct aac_AifEnsDeviceFailure EDF; + struct aac_AifEnsMirrorFailover EMF; + struct aac_AifEnsContainerChange ECC; + struct aac_AifEnsContainerEvent ECE; + struct aac_AifEnsEnclosureEvent EEE; + struct aac_AifEnsBatteryEvent EBE; + struct aac_AifEnsDiskSetEvent EDS; +/* struct aac_AifEnsSMARTEvent ES;*/ + struct aac_AifEnsClusterEvent ECLE; + } data; +} __attribute__ ((__packed__)); + +/* + * Adapter Initiated FIB command structures. Start with the adapter + * initiated FIBs that really come from the adapter, and get responded + * to by the host. + */ +#define AAC_AIF_REPORT_MAX_SIZE 64 + +typedef enum { + AifCmdEventNotify = 1, /* Notify of event */ + AifCmdJobProgress, /* Progress report */ + AifCmdAPIReport, /* Report from other user of API */ + AifCmdDriverNotify, /* Notify host driver of event */ + AifReqJobList = 100, /* Gets back complete job list */ + AifReqJobsForCtr, /* Gets back jobs for specific container */ + AifReqJobsForScsi, /* Gets back jobs for specific SCSI device */ + AifReqJobReport, /* Gets back a specific job report or list */ + AifReqTerminateJob, /* Terminates job */ + AifReqSuspendJob, /* Suspends a job */ + AifReqResumeJob, /* Resumes a job */ + AifReqSendAPIReport, /* API generic report requests */ + AifReqAPIJobStart, /* Start a job from the API */ + AifReqAPIJobUpdate, /* Update a job report from the API */ + AifReqAPIJobFinish /* Finish a job from the API */ +} AAC_AifCommand; + +struct aac_aif_command { + AAC_AifCommand command; /* Tell host what type of + * notify this is */ + u_int32_t seqNumber; /* To allow ordering of + * reports (if necessary) */ + union { + struct aac_AifEventNotify EN; /* Event notify */ + struct aac_AifJobProgressReport PR[1]; /* Progress report */ + u_int8_t AR[AAC_AIF_REPORT_MAX_SIZE]; + u_int8_t data[AAC_FIB_DATASIZE - 8]; + } data; +} __attribute__ ((__packed__)); + +/* + * Filesystem commands/data + * + * The adapter has a very complex filesystem interface, most of which we ignore. + * (And which seems not to be implemented, anyway.) + */ + +/* + * FSA commands + * (not used?) + */ +typedef enum { + Null = 0, + GetAttributes, + SetAttributes, + Lookup, + ReadLink, + Read, + Write, + Create, + MakeDirectory, + SymbolicLink, + MakeNode, + Removex, + RemoveDirectory, + Rename, + Link, + ReadDirectory, + ReadDirectoryPlus, + FileSystemStatus, + FileSystemInfo, + PathConfigure, + Commit, + Mount, + UnMount, + Newfs, + FsCheck, + FsSync, + SimReadWrite, + SetFileSystemStatus, + BlockRead, + BlockWrite, + NvramIoctl, + FsSyncWait, + ClearArchiveBit, + SetAcl, + GetAcl, + AssignAcl, + FaultInsertion, + CrazyCache +} AAC_FSACommand; /* * Command status values */ typedef enum { - ST_OK = 0, - ST_PERM = 1, - ST_NOENT = 2, - ST_IO = 5, - ST_NXIO = 6, - ST_E2BIG = 7, - ST_ACCES = 13, - ST_EXIST = 17, - ST_XDEV = 18, - ST_NODEV = 19, - ST_NOTDIR = 20, - ST_ISDIR = 21, - ST_INVAL = 22, - ST_FBIG = 27, - ST_NOSPC = 28, - ST_ROFS = 30, - ST_MLINK = 31, - ST_WOULDBLOCK = 35, - ST_NAMETOOLONG = 63, - ST_NOTEMPTY = 66, - ST_DQUOT = 69, - ST_STALE = 70, - ST_REMOTE = 71, - ST_BADHANDLE = 10001, - ST_NOT_SYNC = 10002, - ST_BAD_COOKIE = 10003, - ST_NOTSUPP = 10004, - ST_TOOSMALL = 10005, - ST_SERVERFAULT = 10006, - ST_BADTYPE = 10007, - ST_JUKEBOX = 10008, - ST_NOTMOUNTED = 10009, - ST_MAINTMODE = 10010, - ST_STALEACL = 10011 + ST_OK = 0, + ST_PERM = 1, + ST_NOENT = 2, + ST_IO = 5, + ST_NXIO = 6, + ST_E2BIG = 7, + ST_ACCES = 13, + ST_EXIST = 17, + ST_XDEV = 18, + ST_NODEV = 19, + ST_NOTDIR = 20, + ST_ISDIR = 21, + ST_INVAL = 22, + ST_FBIG = 27, + ST_NOSPC = 28, + ST_ROFS = 30, + ST_MLINK = 31, + ST_WOULDBLOCK = 35, + ST_NAMETOOLONG = 63, + ST_NOTEMPTY = 66, + ST_DQUOT = 69, + ST_STALE = 70, + ST_REMOTE = 71, + ST_BADHANDLE = 10001, + ST_NOT_SYNC = 10002, + ST_BAD_COOKIE = 10003, + ST_NOTSUPP = 10004, + ST_TOOSMALL = 10005, + ST_SERVERFAULT = 10006, + ST_BADTYPE = 10007, + ST_JUKEBOX = 10008, + ST_NOTMOUNTED = 10009, + ST_MAINTMODE = 10010, + ST_STALEACL = 10011 } AAC_FSAStatus; /* * Volume manager commands */ typedef enum _VM_COMMANDS { - VM_Null = 0, - VM_NameServe, - VM_ContainerConfig, - VM_Ioctl, - VM_FilesystemIoctl, - VM_CloseAll, - VM_CtBlockRead, - VM_CtBlockWrite, - VM_SliceBlockRead, /* raw access to configured "storage objects" */ - VM_SliceBlockWrite, - VM_DriveBlockRead, /* raw access to physical devices */ - VM_DriveBlockWrite, - VM_EnclosureMgt, /* enclosure management */ - VM_Unused, /* used to be diskset management */ - VM_CtBlockVerify, - VM_CtPerf, /* performance test */ - VM_CtBlockRead64, - VM_CtBlockWrite64, - VM_CtBlockVerify64, + VM_Null = 0, + VM_NameServe, + VM_ContainerConfig, + VM_Ioctl, + VM_FilesystemIoctl, + VM_CloseAll, + VM_CtBlockRead, + VM_CtBlockWrite, + VM_SliceBlockRead, /* raw access to configured storage objects */ + VM_SliceBlockWrite, + VM_DriveBlockRead, /* raw access to physical devices */ + VM_DriveBlockWrite, + VM_EnclosureMgt, /* enclosure management */ + VM_Unused, /* used to be diskset management */ + VM_CtBlockVerify, + VM_CtPerf, /* performance test */ + VM_CtBlockRead64, + VM_CtBlockWrite64, + VM_CtBlockVerify64, + VM_CtHostRead64, + VM_CtHostWrite64, } AAC_VMCommand; /* - * "Mountable object" + * "mountable object" */ struct aac_mntobj { - u_int32_t ObjectId; - char FileSystemName[16]; - struct aac_container_creation CreateInfo; - u_int32_t Capacity; - AAC_FSAVolType VolType; - AAC_FType ObjType; - u_int32_t ContentState; -#define AAC_FSCS_READONLY 0x0002 /* XXX need more information than this */ + u_int32_t ObjectId; + char FileSystemName[16]; + struct aac_container_creation CreateInfo; + u_int32_t Capacity; + u_int32_t VolType; + u_int32_t ObjType; + u_int32_t ContentState; +#define FSCS_READONLY 0x0002 /* XXX need more information + * than this */ union { - u_int32_t pad[8]; + u_int32_t pad[8]; } ObjExtension; - u_int32_t AlterEgoId; + u_int32_t AlterEgoId; } __attribute__ ((__packed__)); struct aac_mntinfo { - AAC_VMCommand Command; - AAC_FType MntType; - u_int32_t MntCount; + u_int32_t Command; + u_int32_t MntType; + u_int32_t MntCount; } __attribute__ ((__packed__)); -struct aac_mntinforesponse { - AAC_FSAStatus Status; - AAC_FType MntType; - u_int32_t MntRespCount; - struct aac_mntobj MntTable[1]; +struct aac_mntinforesp { + u_int32_t Status; + u_int32_t MntType; + u_int32_t MntRespCount; + struct aac_mntobj MntTable[1]; } __attribute__ ((__packed__)); /* + * Container shutdown command. + */ +struct aac_closecommand { + u_int32_t Command; + u_int32_t ContainerId; +} __attribute__ ((__packed__)); + +/* + * Container Config Command + */ +#define CT_GET_SCSI_METHOD 64 +struct aac_ctcfg { + u_int32_t Command; + u_int32_t cmd; + u_int32_t param; +} __attribute__ ((__packed__)); + +struct aac_ctcfg_resp { + u_int32_t Status; + u_int32_t resp; + u_int32_t param; +} __attribute__ ((__packed__)); + +/* + * 'Ioctl' commads + */ +#define AAC_SCSI_MAX_PORTS 10 +#define AAC_BUS_NO_EXIST 0 +#define AAC_BUS_VALID 1 +#define AAC_BUS_FAULTED 2 +#define AAC_BUS_DISABLED 3 +#define GetBusInfo 0x9 + +struct aac_getbusinf { + u_int32_t ProbeComplete; + u_int32_t BusCount; + u_int32_t TargetsPerBus; + u_int8_t InitiatorBusId[AAC_SCSI_MAX_PORTS]; + u_int8_t BusValid[AAC_SCSI_MAX_PORTS]; +} __attribute__ ((__packed__)); + +struct aac_vmioctl { + u_int32_t Command; + u_int32_t ObjType; + u_int32_t MethId; + u_int32_t ObjId; + u_int32_t IoctlCmd; + u_int32_t IoctlBuf[1]; /* Placeholder? */ +} __attribute__ ((__packed__)); + +struct aac_vmi_businf_resp { + u_int32_t Status; + u_int32_t ObjType; + u_int32_t MethId; + u_int32_t ObjId; + u_int32_t IoctlCmd; + struct aac_getbusinf BusInf; +} __attribute__ ((__packed__)); + +#define AAC_BTL_TO_HANDLE(b, t, l) \ + (((b & 0x3f) << 7) | ((l & 0x7) << 4) | (t & 0xf)) +#define GetDeviceProbeInfo 0x5 + +struct aac_vmi_devinfo_resp { + u_int32_t Status; + u_int32_t ObjType; + u_int32_t MethId; + u_int32_t ObjId; + u_int32_t IoctlCmd; + u_int8_t VendorId[8]; + u_int8_t ProductId[16]; + u_int8_t ProductRev[4]; + u_int32_t Inquiry7; + u_int32_t align1; + u_int32_t Inquiry0; + u_int32_t align2; + u_int32_t Inquiry1; + u_int32_t align3; + u_int32_t reserved[2]; + u_int8_t VendorSpecific[20]; + u_int32_t Smart:1; + u_int32_t AAC_Managed:1; + u_int32_t align4; + u_int32_t reserved2:6; + u_int32_t Bus; + u_int32_t Target; + u_int32_t Lun; + u_int32_t ultraEnable:1, + disconnectEnable:1, + fast20EnabledW:1, + scamDevice:1, + scamTolerant:1, + setForSync:1, + setForWide:1, + syncDevice:1, + wideDevice:1, + reserved1:7, + ScsiRate:8, + ScsiOffset:8; +}; /* Do not pack */ + +#define ResetBus 0x16 +struct aac_resetbus { + u_int32_t BusNumber; +}; + +/* * Write 'stability' options. */ typedef enum { @@ -453,68 +1145,199 @@ typedef enum { */ struct aac_blockread { - AAC_VMCommand Command; /* not FSACommand! */ - u_int32_t ContainerId; - u_int32_t BlockNumber; - u_int32_t ByteCount; - struct aac_sg_table SgMap; /* variable size */ + u_int32_t Command; /* not FSACommand! */ + u_int32_t ContainerId; + u_int32_t BlockNumber; + u_int32_t ByteCount; + struct aac_sg_table SgMap; /* variable size */ +} __attribute__ ((__packed__)); + +struct aac_blockread64 { + u_int32_t Command; + u_int16_t ContainerId; + u_int16_t SectorCount; + u_int32_t BlockNumber; + u_int16_t Pad; + u_int16_t Flags; + struct aac_sg_table64 SgMap64; } __attribute__ ((__packed__)); struct aac_blockread_response { - AAC_FSAStatus Status; - u_int32_t ByteCount; + u_int32_t Status; + u_int32_t ByteCount; } __attribute__ ((__packed__)); struct aac_blockwrite { - AAC_VMCommand Command; /* not FSACommand! */ - u_int32_t ContainerId; - u_int32_t BlockNumber; - u_int32_t ByteCount; - AAC_CacheLevel Stable; - struct aac_sg_table SgMap; /* variable size */ + u_int32_t Command; /* not FSACommand! */ + u_int32_t ContainerId; + u_int32_t BlockNumber; + u_int32_t ByteCount; + u_int32_t Stable; + struct aac_sg_table SgMap; /* variable size */ +} __attribute__ ((__packed__)); + +struct aac_blockwrite64 { + u_int32_t Command; /* not FSACommand! */ + u_int16_t ContainerId; + u_int16_t SectorCount; + u_int32_t BlockNumber; + u_int16_t Pad; + u_int16_t Flags; + struct aac_sg_table64 SgMap64; /* variable size */ } __attribute__ ((__packed__)); struct aac_blockwrite_response { - AAC_FSAStatus Status; - u_int32_t ByteCount; - AAC_CommitLevel Committed; + u_int32_t Status; + u_int32_t ByteCount; + u_int32_t Committed; } __attribute__ ((__packed__)); /* + * Container shutdown command. + */ +struct aac_close_command { + u_int32_t Command; + u_int32_t ContainerId; +}; + +/* + * SCSI Passthrough structures + */ +struct aac_srb32 { + u_int32_t function; + u_int32_t bus; + u_int32_t target; + u_int32_t lun; + u_int32_t timeout; + u_int32_t flags; + u_int32_t data_len; + u_int32_t retry_limit; + u_int32_t cdb_len; + u_int8_t cdb[16]; + struct aac_sg_table sg_map32; +}; + +enum { + AAC_SRB_FUNC_EXECUTE_SCSI = 0x00, + AAC_SRB_FUNC_CLAIM_DEVICE, + AAC_SRB_FUNC_IO_CONTROL, + AAC_SRB_FUNC_RECEIVE_EVENT, + AAC_SRB_FUNC_RELEASE_QUEUE, + AAC_SRB_FUNC_ATTACH_DEVICE, + AAC_SRB_FUNC_RELEASE_DEVICE, + AAC_SRB_FUNC_SHUTDOWN, + AAC_SRB_FUNC_FLUSH, + AAC_SRB_FUNC_ABORT_COMMAND = 0x10, + AAC_SRB_FUNC_RELEASE_RECOVERY, + AAC_SRB_FUNC_RESET_BUS, + AAC_SRB_FUNC_RESET_DEVICE, + AAC_SRB_FUNC_TERMINATE_IO, + AAC_SRB_FUNC_FLUSH_QUEUE, + AAC_SRB_FUNC_REMOVE_DEVICE, + AAC_SRB_FUNC_DOMAIN_VALIDATION +}; + +#define AAC_SRB_FLAGS_NO_DATA_XFER 0x0000 +#define AAC_SRB_FLAGS_DISABLE_DISCONNECT 0x0004 +#define AAC_SRB_FLAGS_DISABLE_SYNC_TRANSFER 0x0008 +#define AAC_SRB_FLAGS_BYPASS_FROZEN_QUEUE 0x0010 +#define AAC_SRB_FLAGS_DISABLE_AUTOSENSE 0x0020 +#define AAC_SRB_FLAGS_DATA_IN 0x0040 +#define AAC_SRB_FLAGS_DATA_OUT 0x0080 +#define AAC_SRB_FLAGS_UNSPECIFIED_DIRECTION \ + (AAC_SRB_FLAGS_DATA_IN | AAC_SRB_FLAGS_DATA_OUT) + +#define AAC_HOST_SENSE_DATA_MAX 30 + +struct aac_srb_response { + u_int32_t fib_status; + u_int32_t srb_status; + u_int32_t scsi_status; + u_int32_t data_len; + u_int32_t sense_len; + u_int8_t sense[AAC_HOST_SENSE_DATA_MAX]; +}; + +enum { + AAC_SRB_STS_PENDING = 0x00, + AAC_SRB_STS_SUCCESS, + AAC_SRB_STS_ABORTED, + AAC_SRB_STS_ABORT_FAILED, + AAC_SRB_STS_ERROR, + AAC_SRB_STS_BUSY, + AAC_SRB_STS_INVALID_REQUEST, + AAC_SRB_STS_INVALID_PATH_ID, + AAC_SRB_STS_NO_DEVICE, + AAC_SRB_STS_TIMEOUT, + AAC_SRB_STS_SELECTION_TIMEOUT, + AAC_SRB_STS_COMMAND_TIMEOUT, + AAC_SRB_STS_MESSAGE_REJECTED = 0x0D, + AAC_SRB_STS_BUS_RESET, + AAC_SRB_STS_PARITY_ERROR, + AAC_SRB_STS_REQUEST_SENSE_FAILED, + AAC_SRB_STS_NO_HBA, + AAC_SRB_STS_DATA_OVERRUN, + AAC_SRB_STS_UNEXPECTED_BUS_FREE, + AAC_SRB_STS_PHASE_SEQUENCE_FAILURE, + AAC_SRB_STS_BAD_SRB_BLOCK_LENGTH, + AAC_SRB_STS_REQUEST_FLUSHED, + AAC_SRB_STS_INVALID_LUN = 0x20, + AAC_SRB_STS_INVALID_TARGET_ID, + AAC_SRB_STS_BAD_FUNCTION, + AAC_SRB_STS_ERROR_RECOVERY +}; + +/* + * Register set for adapters based on the Falcon bridge and PPC core + */ + +#define AAC_FA_DOORBELL0_CLEAR 0x00 +#define AAC_FA_DOORBELL1_CLEAR 0x02 +#define AAC_FA_DOORBELL0 0x04 +#define AAC_FA_DOORBELL1 0x06 +#define AAC_FA_MASK0_CLEAR 0x08 +#define AAC_FA_MASK1_CLEAR 0x0a +#define AAC_FA_MASK0 0x0c +#define AAC_FA_MASK1 0x0e +#define AAC_FA_MAILBOX 0x10 +#define AAC_FA_FWSTATUS 0x2c /* Mailbox 7 */ +#define AAC_FA_INTSRC 0x900 + +#define AAC_FA_HACK(sc) (void)AAC_GETREG4(sc, AAC_FA_INTSRC) + +/* * Register definitions for the Adaptec AAC-364 'Jalapeno I/II' adapters, based * on the SA110 'StrongArm'. */ -#define AAC_REGSIZE 0x100 +#define AAC_REGSIZE 0x100 -/* doorbell 0 (adapter->host) */ -#define AAC_SA_DOORBELL0_CLEAR 0x98 -#define AAC_SA_DOORBELL0_SET 0x9c -#define AAC_SA_DOORBELL0 0x9c -#define AAC_SA_MASK0_CLEAR 0xa0 -#define AAC_SA_MASK0_SET 0xa4 +#define AAC_SA_DOORBELL0_CLEAR 0x98 /* doorbell 0 (adapter->host) */ +#define AAC_SA_DOORBELL0_SET 0x9c +#define AAC_SA_DOORBELL0 0x9c +#define AAC_SA_MASK0_CLEAR 0xa0 +#define AAC_SA_MASK0_SET 0xa4 -/* doorbell 1 (host->adapter) */ -#define AAC_SA_DOORBELL1_CLEAR 0x9a -#define AAC_SA_DOORBELL1_SET 0x9e -#define AAC_SA_MASK1_CLEAR 0xa2 -#define AAC_SA_MASK1_SET 0xa6 +#define AAC_SA_DOORBELL1_CLEAR 0x9a /* doorbell 1 (host->adapter) */ +#define AAC_SA_DOORBELL1_SET 0x9e +#define AAC_SA_DOORBELL1 0x9e +#define AAC_SA_MASK1_CLEAR 0xa2 +#define AAC_SA_MASK1_SET 0xa6 -/* mailbox (20 bytes) */ -#define AAC_SA_MAILBOX 0xa8 -#define AAC_SA_FWSTATUS 0xc4 +#define AAC_SA_MAILBOX 0xa8 /* mailbox (20 bytes) */ +#define AAC_SA_FWSTATUS 0xc4 /* - * Register definitions for the Adaptec 'Pablano' adapters, based on the - * i960Rx, and other related adapters. + * Register definitions for the Adaptec 'Pablano' adapters, based on the i960Rx, + * and other related adapters. */ -#define AAC_RX_IDBR 0x20 /* inbound doorbell */ -#define AAC_RX_IISR 0x24 /* inbound interrupt status */ -#define AAC_RX_IIMR 0x28 /* inbound interrupt mask */ -#define AAC_RX_ODBR 0x2c /* outbound doorbell */ -#define AAC_RX_OISR 0x30 /* outbound interrupt status */ -#define AAC_RX_OIMR 0x34 /* outbound interrupt mask */ +#define AAC_RX_IDBR 0x20 /* inbound doorbell register */ +#define AAC_RX_IISR 0x24 /* inbound interrupt status register */ +#define AAC_RX_IIMR 0x28 /* inbound interrupt mask register */ +#define AAC_RX_ODBR 0x2c /* outbound doorbell register */ +#define AAC_RX_OISR 0x30 /* outbound interrupt status register */ +#define AAC_RX_OIMR 0x34 /* outbound interrupt mask register */ #define AAC_RX_MAILBOX 0x50 /* mailbox (20 bytes) */ #define AAC_RX_FWSTATUS 0x6c @@ -539,182 +1362,13 @@ struct aac_blockwrite_response { * DOORBELL0 and setting it in DOORBELL1. * (ODBR and IDBR respectively for the i960Rx adapters) */ -#define AAC_DB_PRINTF (1<<5) - -/* - * Mask containing the interrupt bits we care about. We don't anticipate - * (or want) interrupts not in this mask. - */ -#define AAC_DB_INTERRUPTS \ - (AAC_DB_COMMAND_READY | AAC_DB_RESPONSE_READY | AAC_DB_PRINTF) - -/* - * Queue names - * - * Note that we base these at 0 in order to use them as array indices. Adaptec - * used base 1 for some unknown reason, and sorted them in a different order. - */ -#define AAC_HOST_NORM_CMD_QUEUE 0 -#define AAC_HOST_HIGH_CMD_QUEUE 1 -#define AAC_ADAP_NORM_CMD_QUEUE 2 -#define AAC_ADAP_HIGH_CMD_QUEUE 3 -#define AAC_HOST_NORM_RESP_QUEUE 4 -#define AAC_HOST_HIGH_RESP_QUEUE 5 -#define AAC_ADAP_NORM_RESP_QUEUE 6 -#define AAC_ADAP_HIGH_RESP_QUEUE 7 - -/* - * List structure used to chain FIBs (used by the adapter - we hang FIBs off - * our private command structure and don't touch these) - */ -struct aac_fib_list_entry { - struct fib_list_entry *Flink; - struct fib_list_entry *Blink; -} __packed; - -/* - * FIB (FSA Interface Block?); this is the datastructure passed between the - * host and adapter. - */ -struct aac_fib_header { - u_int32_t XferState; - u_int16_t Command; - u_int8_t StructType; - u_int8_t Flags; - u_int16_t Size; - u_int16_t SenderSize; - u_int32_t SenderFibAddress; - u_int32_t ReceiverFibAddress; - u_int32_t SenderData; - union { - struct { - u_int32_t ReceiverTimeStart; - u_int32_t ReceiverTimeDone; - } _s; - struct aac_fib_list_entry FibLinks; - } _u; -} __packed; - -#define AAC_FIB_DATASIZE (512 - sizeof(struct aac_fib_header)) - -struct aac_fib { - struct aac_fib_header Header; - u_int8_t data[AAC_FIB_DATASIZE]; -} __packed; - -/* - * FIB commands - */ -typedef enum { - TestCommandResponse = 1, - TestAdapterCommand = 2, - - /* lowlevel and comm commands */ - LastTestCommand = 100, - ReinitHostNormCommandQueue = 101, - ReinitHostHighCommandQueue = 102, - ReinitHostHighRespQueue = 103, - ReinitHostNormRespQueue = 104, - ReinitAdapNormCommandQueue = 105, - ReinitAdapHighCommandQueue = 107, - ReinitAdapHighRespQueue = 108, - ReinitAdapNormRespQueue = 109, - InterfaceShutdown = 110, - DmaCommandFib = 120, - StartProfile = 121, - TermProfile = 122, - SpeedTest = 123, - TakeABreakPt = 124, - RequestPerfData = 125, - SetInterruptDefTimer= 126, - SetInterruptDefCount= 127, - GetInterruptDefStatus= 128, - LastCommCommand = 129, - - /* filesystem commands */ - NuFileSystem = 300, - UFS = 301, - HostFileSystem = 302, - LastFileSystemCommand = 303, - - /* Container Commands */ - ContainerCommand = 500, - ContainerCommand64 = 501, - - /* Cluster Commands */ - ClusterCommand = 550, - - /* Scsi Port commands (scsi passthrough) */ - ScsiPortCommand = 600, - - /* misc house keeping and generic adapter initiated commands */ - AifRequest = 700, - CheckRevision = 701, - FsaHostShutdown = 702, - RequestAdapterInfo = 703, - IsAdapterPaused = 704, - SendHostTime = 705, - LastMiscCommand = 706 -} AAC_FibCommands; +#define AAC_DB_PRINTF (1<<5) /* adapter requests host printf */ +#define AAC_PRINTF_DONE (1<<5) /* Host completed printf processing */ /* - * FIB types - */ -#define AAC_FIBTYPE_TFIB 1 -#define AAC_FIBTYPE_TQE 2 -#define AAC_FIBTYPE_TCTPERF 3 - -/* - * FIB transfer state + * Mask containing the interrupt bits we care about. We don't anticipate (or + * want) interrupts not in this mask. */ -#define AAC_FIBSTATE_HOSTOWNED (1<<0) /* owned by the host */ -#define AAC_FIBSTATE_ADAPTEROWNED (1<<1) /* owned by the adapter */ -#define AAC_FIBSTATE_INITIALISED (1<<2) /* initialised */ -#define AAC_FIBSTATE_EMPTY (1<<3) /* empty */ -#define AAC_FIBSTATE_FROMPOOL (1<<4) /* allocated from pool */ -#define AAC_FIBSTATE_FROMHOST (1<<5) /* sent from the host */ -#define AAC_FIBSTATE_FROMADAP (1<<6) /* sent from the adapter */ -#define AAC_FIBSTATE_REXPECTED (1<<7) /* response is expected */ -#define AAC_FIBSTATE_RNOTEXPECTED (1<<8) /* response is not expected */ -#define AAC_FIBSTATE_DONEADAP (1<<9) /* processed by the adapter */ -#define AAC_FIBSTATE_DONEHOST (1<<10) /* processed by the host */ -#define AAC_FIBSTATE_HIGH (1<<11) /* high priority */ -#define AAC_FIBSTATE_NORM (1<<12) /* normal priority */ -#define AAC_FIBSTATE_ASYNC (1<<13) -#define AAC_FIBSTATE_ASYNCIO (1<<13) /* to be removed */ -#define AAC_FIBSTATE_PAGEFILEIO (1<<14) /* to be removed */ -#define AAC_FIBSTATE_SHUTDOWN (1<<15) -#define AAC_FIBSTATE_LAZYWRITE (1<<16) /* to be removed */ -#define AAC_FIBSTATE_ADAPMICROFIB (1<<17) -#define AAC_FIBSTATE_BIOSFIB (1<<18) -#define AAC_FIBSTATE_FAST_RESPONSE (1<<19) /* fast response capable */ -#define AAC_FIBSTATE_APIFIB (1<<20) - -/* - * FIB error values - */ -#define AAC_ERROR_NORMAL 0x00 -#define AAC_ERROR_PENDING 0x01 -#define AAC_ERROR_FATAL 0x02 -#define AAC_ERROR_INVALID_QUEUE 0x03 -#define AAC_ERROR_NOENTRIES 0x04 -#define AAC_ERROR_SENDFAILED 0x05 -#define AAC_ERROR_INVALID_QUEUE_PRIORITY 0x06 -#define AAC_ERROR_FIB_ALLOCATION_FAILED 0x07 -#define AAC_ERROR_FIB_DEALLOCATION_FAILED 0x08 - -/* - * Adapter Status Register - * - * Phase Staus mailbox is 32bits: - * <31:16> = Phase Status - * <15:0> = Phase - * - * The adapter reports its present state through the phase. Only - * a single phase should be ever be set. Each phase can have multiple - * phase status bits to provide more detailed information about the - * state of the adapter. - */ -#define AAC_SELF_TEST_FAILED 0x00000004 -#define AAC_UP_AND_RUNNING 0x00000080 -#define AAC_KERNEL_PANIC 0x00000100 +#define AAC_DB_INTERRUPTS (AAC_DB_COMMAND_READY | \ + AAC_DB_RESPONSE_READY | \ + AAC_DB_PRINTF) diff --git a/sys/dev/ic/aacvar.h b/sys/dev/ic/aacvar.h index e654d7595d6..2be60aba46a 100644 --- a/sys/dev/ic/aacvar.h +++ b/sys/dev/ic/aacvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: aacvar.h,v 1.4 2003/10/21 18:58:48 jmc Exp $ */ +/* $OpenBSD: aacvar.h,v 1.5 2005/11/18 05:39:10 nate Exp $ */ /*- * Copyright (c) 2000 Michael Smith @@ -38,17 +38,29 @@ * - Niklas Hallqvist */ +/* compatability */ +#define time_second (mono_time.tv_sec) + /* Debugging */ +// #define AAC_DEBUG 0x0 + #ifdef AAC_DEBUG #define AAC_DPRINTF(mask, args) if (aac_debug & (mask)) printf args -#define AAC_D_INTR 0x01 -#define AAC_D_MISC 0x02 -#define AAC_D_CMD 0x04 -#define AAC_D_QUEUE 0x08 -#define AAC_D_IO 0x10 +#define AAC_D_INTR 0x001 +#define AAC_D_MISC 0x002 +#define AAC_D_CMD 0x004 +#define AAC_D_QUEUE 0x008 +#define AAC_D_IO 0x010 +#define AAC_D_IOCTL 0x020 +#define AAC_D_LOCK 0x040 +#define AAC_D_THREAD 0x080 +#define AAC_D_FIB 0x100 extern int aac_debug; -#define AAC_PRINT_FIB(sc, fib) aac_print_fib((sc), (fib), __func__) +#define AAC_PRINT_FIB(sc, fib) do { \ + if (aac_debug & AAC_D_FIB) \ + aac_print_fib((sc), (fib), __func__); \ +} while (0) #else #define AAC_DPRINTF(mask, args) #define AAC_PRINT_FIB(sc, fib) @@ -67,6 +79,20 @@ struct aac_softc; #define AAC_ADAPTER_FIBS 8 /* + * FIBs are allocated in page-size chunks and can grow up to the 512 + * limit imposed by the hardware. + */ +#define AAC_FIB_COUNT (PAGE_SIZE/sizeof(struct aac_fib)) +#define AAC_MAX_FIBS 512 +#define AAC_FIBMAP_SIZE (PAGE_SIZE) + +/* + * The controller reports status events in AIFs. We hang on to a number of + * these in order to pass them out to user-space management tools. + */ +#define AAC_AIFQ_LENGTH 64 + +/* * Firmware messages are passed in the printf buffer. */ #define AAC_PRINTF_BUFSIZE 256 @@ -78,14 +104,25 @@ struct aac_softc; #define AAC_BOOT_TIMEOUT (3 * 60) /* - * Wait this long for a lost interrupt to get detected. + * Timeout for immediate commands. */ -#define AAC_WATCH_TIMEOUT 10000 /* 10000 * 1ms = 10s */ +#define AAC_IMMEDIATE_TIMEOUT 30 /* - * Timeout for immediate commands. + * Timeout for normal commands */ -#define AAC_IMMEDIATE_TIMEOUT 30 +#define AAC_CMD_TIMEOUT 30 /* seconds */ + +/* + * Rate at which we periodically check for timed out commands and kick the + * controller. + */ +#define AAC_PERIODIC_INTERVAL 20 /* seconds */ + +/* + * Wait this long for a lost interrupt to get detected. + */ +#define AAC_WATCH_TIMEOUT 10000 /* 10000 * 1ms = 10s */ /* * Delay 20ms after the qnotify in sync operations. Experimentally deduced. @@ -97,6 +134,7 @@ struct aac_softc; * ourselves to a reasonable maximum and ensure alignment. */ #define AAC_MAXSGENTRIES 64 /* max S/G entries, limit 65535 */ + /* * We gather a number of adapter-visible items into a single structure. * @@ -126,6 +164,7 @@ struct aac_common { /* fib for synchronous commands */ struct aac_fib ac_sync_fib; }; +#define AAC_COMMON_ALLOCSIZE (8192 + sizeof(struct aac_common)) /* * Interface operations @@ -137,42 +176,63 @@ struct aac_interface { void (*aif_set_istatus)(struct aac_softc *, int); void (*aif_set_mailbox)(struct aac_softc *, u_int32_t, u_int32_t, u_int32_t, u_int32_t, u_int32_t); - int (*aif_get_mailboxstatus)(struct aac_softc *); + int (*aif_get_mailbox)(struct aac_softc *, int mb); void (*aif_set_interrupts)(struct aac_softc *, int); }; -extern struct aac_interface aac_rx_interface; +extern struct aac_interface aac_fa_interface; extern struct aac_interface aac_sa_interface; +extern struct aac_interface aac_rx_interface; -#define AAC_GET_FWSTATUS(sc) ((sc)->sc_if.aif_get_fwstatus(sc)) +#define AAC_GET_FWSTATUS(sc) ((sc)->aac_if.aif_get_fwstatus(sc)) #define AAC_QNOTIFY(sc, qbit) \ - ((sc)->sc_if.aif_qnotify((sc), (qbit))) -#define AAC_GET_ISTATUS(sc) ((sc)->sc_if.aif_get_istatus(sc)) + ((sc)->aac_if.aif_qnotify((sc), (qbit))) +#define AAC_GET_ISTATUS(sc) ((sc)->aac_if.aif_get_istatus(sc)) #define AAC_CLEAR_ISTATUS(sc, mask) \ - ((sc)->sc_if.aif_set_istatus((sc), (mask))) + ((sc)->aac_if.aif_set_istatus((sc), (mask))) #define AAC_SET_MAILBOX(sc, command, arg0, arg1, arg2, arg3) \ do { \ - ((sc)->sc_if.aif_set_mailbox((sc), (command), (arg0), \ + ((sc)->aac_if.aif_set_mailbox((sc), (command), (arg0), \ (arg1), (arg2), (arg3))); \ } while(0) -#define AAC_GET_MAILBOXSTATUS(sc) \ - ((sc)->sc_if.aif_get_mailboxstatus(sc)) +#define AAC_GET_MAILBOX(sc, mb) \ + ((sc)->aac_if.aif_get_mailbox(sc, (mb))) #define AAC_MASK_INTERRUPTS(sc) \ - ((sc)->sc_if.aif_set_interrupts((sc), 0)) + ((sc)->aac_if.aif_set_interrupts((sc), 0)) #define AAC_UNMASK_INTERRUPTS(sc) \ - ((sc)->sc_if.aif_set_interrupts((sc), 1)) + ((sc)->aac_if.aif_set_interrupts((sc), 1)) #define AAC_SETREG4(sc, reg, val) \ - bus_space_write_4((sc)->sc_memt, (sc)->sc_memh, (reg), (val)) + bus_space_write_4((sc)->aac_memt, (sc)->aac_memh, (reg), (val)) #define AAC_GETREG4(sc, reg) \ - bus_space_read_4((sc)->sc_memt, (sc)->sc_memh, (reg)) + bus_space_read_4((sc)->aac_memt, (sc)->aac_memh, (reg)) #define AAC_SETREG2(sc, reg, val) \ - bus_space_write_2((sc)->sc_memt, (sc)->sc_memh, (reg), (val)) + bus_space_write_2((sc)->aac_memt, (sc)->aac_memh, (reg), (val)) #define AAC_GETREG2(sc, reg) \ - bus_space_read_2((sc)->sc_memt, (sc)->sc_memh, (reg)) + bus_space_read_2((sc)->aac_memt, (sc)->aac_memh, (reg)) #define AAC_SETREG1(sc, reg, val) \ - bus_space_write_1((sc)->sc_memt, (sc)->sc_memh, (reg), (val)) + bus_space_write_1((sc)->aac_memt, (sc)->aac_memh, (reg), (val)) #define AAC_GETREG1(sc, reg) \ - bus_space_read_1((sc)->sc_memt, (sc)->sc_memh, (reg)) + bus_space_read_1((sc)->aac_memt, (sc)->aac_memh, (reg)) + +/* Define the OS version specific locks */ +typedef struct rwlock aac_lock_t; +#define AAC_LOCK_INIT(l, s) do { \ + rw_init((l)); \ + AAC_DPRINTF(AAC_D_LOCK, ("%s: init lock @%s: %d\n", \ + sc->aac_dev.dv_xname, __FUNCTION__, __LINE__)); \ +} while (0) + +#define AAC_LOCK_ACQUIRE(l) do { \ + AAC_DPRINTF(AAC_D_LOCK, ("%s: lock @%s: %d\n", \ + sc->aac_dev.dv_xname, __FUNCTION__, __LINE__)); \ + rw_enter_write((l)); \ +} while (0) + +#define AAC_LOCK_RELEASE(l) do { \ + rw_exit_write((l)); \ + AAC_DPRINTF(AAC_D_LOCK, ("%s: unlock @%s: %d\n", \ + sc->aac_dev.dv_xname, __FUNCTION__, __LINE__)); \ +} while (0) /* * Per-container data structure @@ -180,69 +240,136 @@ extern struct aac_interface aac_sa_interface; struct aac_container { struct aac_mntobj co_mntobj; - struct device co_disk; + int co_found; + TAILQ_ENTRY(aac_container) co_link; }; /* * A command contol block, one for each corresponding command index of the * controller. */ -struct aac_ccb { - TAILQ_ENTRY(aac_ccb) ac_chain; - struct scsi_xfer *ac_xs; - struct aac_fib *ac_fib; /* FIB associated with this command */ - bus_addr_t ac_fibphys; /* bus address of the FIB */ - bus_dmamap_t ac_dmamap_xfer; - struct aac_sg_table *ac_sgtable;/* pointer to s/g table in command */ - int ac_timeout; - u_int32_t ac_blockno; - u_int32_t ac_blockcnt; - u_int8_t ac_flags; -#define AAC_ACF_WATCHDOG 0x1 -#define AAC_ACF_COMPLETED 0x2 +struct aac_command +{ + TAILQ_ENTRY(aac_command) cm_link; /* list linkage */ + + struct aac_softc *cm_sc; /* controller that owns us */ + + struct aac_fib *cm_fib; /* FIB for this command */ + bus_addr_t cm_fibphys; /* bus address of the FIB */ + void *cm_data; + size_t cm_datalen; + bus_dmamap_t cm_datamap; + struct aac_sg_table *cm_sgtable; /* pointer to s/g table */ + + u_int cm_flags; +#define AAC_CMD_MAPPED (1<<0) /* command has had its data mapped */ +#define AAC_CMD_DATAIN (1<<1) /* command involves data moving + * from controller to host */ +#define AAC_CMD_DATAOUT (1<<2) /* command involves data moving + * from host to controller */ +#define AAC_CMD_COMPLETED (1<<3) /* command has been completed */ +#define AAC_CMD_TIMEDOUT (1<<4) /* command taken too long */ +#define AAC_ON_AACQ_FREE (1<<5) +#define AAC_ON_AACQ_READY (1<<6) +#define AAC_ON_AACQ_BUSY (1<<7) +#define AAC_ON_AACQ_BIO (1<<8) +#define AAC_ON_AACQ_MASK ((1<<5)|(1<<6)|(1<<7)|(1<<8)) +#define AAC_QUEUE_FRZN (1<<9) /* Freeze the processing of + * commands on the queue. */ +#define AAC_ACF_WATCHDOG (1<<10) + + void (*cm_complete)(struct aac_command *); + void *cm_private; + u_int32_t cm_blkno; + u_int32_t cm_bcount; + time_t cm_timestamp; /* command creation time */ + int cm_queue; + int cm_index; +}; + +struct aac_fibmap { + TAILQ_ENTRY(aac_fibmap) fm_link; /* list linkage */ + struct aac_fib *aac_fibs; + bus_dmamap_t aac_fibmap; + bus_dma_segment_t aac_seg; + int aac_nsegs; + struct aac_command *aac_commands; +}; + +/* + * Command queue statistics + */ +#define AACQ_FREE 0 +#define AACQ_BIO 1 +#define AACQ_READY 2 +#define AACQ_BUSY 3 +#define AACQ_COUNT 4 /* total number of queues */ + +struct aac_qstat { + u_int32_t q_length; + u_int32_t q_max; }; /* * Per-controller structure. */ -struct aac_softc { - struct device sc_dev; - void *sc_ih; - struct scsi_link sc_link; /* Virtual SCSI bus for cache devs */ +struct aac_softc +{ + struct device aac_dev; + void *aac_ih; + struct scsi_link aac_link; /* Virtual SCSI bus for cache devs */ - bus_space_tag_t sc_memt; - bus_space_handle_t sc_memh; - bus_dma_tag_t sc_dmat; /* parent DMA tag */ + bus_space_tag_t aac_memt; + bus_space_handle_t aac_memh; + bus_dma_tag_t aac_dmat; /* parent DMA tag */ /* controller features, limits and status */ - int sc_state; + int aac_state; #define AAC_STATE_SUSPEND (1<<0) #define AAC_STATE_OPEN (1<<1) #define AAC_STATE_INTERRUPTS_ON (1<<2) #define AAC_STATE_AIF_SLEEPER (1<<3) - struct FsaRevision sc_revision; + struct FsaRevision aac_revision; - int sc_hwif; /* controller hardware interface */ + int aac_hwif; /* controller hardware interface */ #define AAC_HWIF_I960RX 0 #define AAC_HWIF_STRONGARM 1 +#define AAC_HWIF_FALCON 2 +#define AAC_HWIF_UNKNOWN -1 + + struct aac_common *aac_common; + bus_dmamap_t aac_common_map; + u_int32_t aac_common_busaddr; + struct aac_interface aac_if; - struct aac_common *sc_common; - u_int32_t sc_common_busaddr; - struct aac_interface sc_if; + /* command/fib resources */ + TAILQ_HEAD(,aac_fibmap) aac_fibmap_tqh; + u_int total_fibs; + struct aac_command *aac_commands; - /* XXX This should really be dynamic. It is very wasteful now. */ - struct aac_ccb sc_ccbs[AAC_ADAP_NORM_CMD_ENTRIES]; - TAILQ_HEAD(, aac_ccb) sc_free_ccb, sc_ccbq; - /* commands on hold for controller resources */ - TAILQ_HEAD(, aac_ccb) sc_ready; - /* commands which have been returned by the controller */ - TAILQ_HEAD(, aac_ccb) sc_completed; - LIST_HEAD(, scsi_xfer) sc_queue; - struct scsi_xfer *sc_queuelast; + /* command management */ + TAILQ_HEAD(,aac_command) aac_free; /* command structures + * available for reuse */ + TAILQ_HEAD(,aac_command) aac_ready; /* commands on hold for + * controller resources */ + TAILQ_HEAD(,aac_command) aac_busy; + TAILQ_HEAD(,aac_command) aac_bio; /* command management */ - struct aac_queue_table *sc_queues; - struct aac_queue_entry *sc_qentries[AAC_QUEUE_COUNT]; + struct aac_queue_table *aac_queues; + struct aac_queue_entry *aac_qentries[AAC_QUEUE_COUNT]; + + struct aac_qstat aac_qstat[AACQ_COUNT]; /* queue statistics */ + + /* connected containters */ + TAILQ_HEAD(,aac_container) aac_container_tqh; + aac_lock_t aac_container_lock; + + /* Protect the sync fib */ +#define AAC_SYNC_LOCK_FORCE (1 << 0) + aac_lock_t aac_sync_lock; + + aac_lock_t aac_io_lock; struct { u_int8_t hd_present; @@ -260,19 +387,58 @@ struct aac_softc { u_int8_t hd_ldr_no; u_int8_t hd_rw_attribs; u_int32_t hd_start_sec; - } sc_hdr[AAC_MAX_CONTAINERS]; + } aac_hdr[AAC_MAX_CONTAINERS]; + int aac_container_count; + + /* management interface */ + aac_lock_t aac_aifq_lock; + struct aac_aif_command aac_aifq[AAC_AIFQ_LENGTH]; + int aac_aifq_head; + int aac_aifq_tail; + struct selinfo aac_select; + struct proc *aifthread; + int aifflags; +#define AAC_AIFFLAGS_RUNNING (1 << 0) +#define AAC_AIFFLAGS_AIF (1 << 1) +#define AAC_AIFFLAGS_EXIT (1 << 2) +#define AAC_AIFFLAGS_EXITED (1 << 3) +#define AAC_AIFFLAGS_COMPLETE (1 << 4) +#define AAC_AIFFLAGS_PRINTF (1 << 5) +#define AAC_AIFFLAGS_PENDING (AAC_AIFFLAGS_AIF | AAC_AIFFLAGS_COMPLETE | \ + AAC_AIFFLAGS_PRINTF) + + u_int32_t flags; +#define AAC_FLAGS_PERC2QC (1 << 0) +#define AAC_FLAGS_ENABLE_CAM (1 << 1) /* No SCSI passthrough */ +#define AAC_FLAGS_CAM_NORESET (1 << 2) /* Fake SCSI resets */ +#define AAC_FLAGS_CAM_PASSONLY (1 << 3) /* Only create pass devices */ +#define AAC_FLAGS_SG_64BIT (1 << 4) /* Use 64-bit S/G addresses */ +#define AAC_FLAGS_4GB_WINDOW (1 << 5) /* Device can access host mem + * 2GB-4GB range */ +#define AAC_FLAGS_NO4GB (1 << 6) /* Can't access host mem >2GB*/ +#define AAC_FLAGS_256FIBS (1 << 7) /* Can only do 256 commands */ +#define AAC_FLAGS_BROKEN_MEMMAP (1 << 8) /* Broken HostPhysMemPages */ + + u_int32_t supported_options; + int aac_max_fibs; + void *aac_sdh; }; -/* XXX These have to become spinlocks in case of SMP */ -#define AAC_LOCK(sc) splbio() -#define AAC_UNLOCK(sc, lock) splx(lock) -typedef int aac_lock_t; +/* + * Public functions + */ +extern int aac_wait_command(struct aac_command *, int); +extern int aac_alloc_command(struct aac_softc *, struct aac_command **); +extern void aac_release_command(struct aac_command *); +extern int aac_alloc_sync_fib(struct aac_softc *, struct aac_fib **, int); +extern void aac_release_sync_fib(struct aac_softc *); +extern int aac_sync_fib(struct aac_softc *, u_int32_t, u_int32_t, + struct aac_fib *, u_int16_t); void aacminphys(struct buf *); int aac_attach(struct aac_softc *); int aac_intr(void *); -#ifdef __GNUC__ /* These all require correctly aligned buffers */ static __inline__ void aac_enc16(u_int8_t *, u_int16_t); static __inline__ void aac_enc32(u_int8_t *, u_int32_t); @@ -309,33 +475,121 @@ aac_dec32(addr) return letoh32(*(u_int32_t *)addr); } -/* - * Queue primitives - * - * These are broken out individually to make statistics gathering easier. - */ +/* Declarations copied from aac.c */ +#ifdef AAC_DEBUG +void aac_print_fib(struct aac_softc *, struct aac_fib *, const char *); +void aac_print_aif(struct aac_softc *, struct aac_aif_command *); +#endif +void aac_handle_aif(struct aac_softc *, struct aac_fib *); + + -static __inline__ void -aac_enqueue_completed(struct aac_ccb *ccb) -{ - struct aac_softc *sc = ccb->ac_xs->sc_link->adapter_softc; - aac_lock_t lock; - lock = AAC_LOCK(sc); - TAILQ_INSERT_TAIL(&sc->sc_completed, ccb, ac_chain); - AAC_UNLOCK(sc, lock); -} -static __inline__ struct aac_ccb * -aac_dequeue_completed(struct aac_softc *sc) +/* + * Queue primitives for driver queues. + */ +#define AACQ_ADD(sc, qname) \ + do { \ + struct aac_qstat *qs; \ + \ + qs = &(sc)->aac_qstat[qname]; \ + \ + qs->q_length++; \ + if (qs->q_length > qs->q_max) \ + qs->q_max = qs->q_length; \ + } while (0) + +#define AACQ_REMOVE(sc, qname) (sc)->aac_qstat[qname].q_length-- +#define AACQ_INIT(sc, qname) \ + do { \ + sc->aac_qstat[qname].q_length = 0; \ + sc->aac_qstat[qname].q_max = 0; \ + } while (0) + + +#define AACQ_COMMAND_QUEUE(name, index) \ +static __inline void \ +aac_initq_ ## name (struct aac_softc *sc) \ +{ \ + TAILQ_INIT(&sc->aac_ ## name); \ + AACQ_INIT(sc, index); \ +} \ +static __inline void \ +aac_enqueue_ ## name (struct aac_command *cm) \ +{ \ + AAC_DPRINTF(AAC_D_CMD, (": enqueue " #name)); \ + if ((cm->cm_flags & AAC_ON_AACQ_MASK) != 0) { \ + printf("command %p is on another queue, flags = %#x\n", \ + cm, cm->cm_flags); \ + panic("command is on another queue"); \ + } \ + TAILQ_INSERT_TAIL(&cm->cm_sc->aac_ ## name, cm, cm_link); \ + cm->cm_flags |= AAC_ON_ ## index; \ + AACQ_ADD(cm->cm_sc, index); \ +} \ +static __inline void \ +aac_requeue_ ## name (struct aac_command *cm) \ +{ \ + AAC_DPRINTF(AAC_D_CMD, (": requeue " #name)); \ + if ((cm->cm_flags & AAC_ON_AACQ_MASK) != 0) { \ + printf("command %p is on another queue, flags = %#x\n", \ + cm, cm->cm_flags); \ + panic("command is on another queue"); \ + } \ + TAILQ_INSERT_HEAD(&cm->cm_sc->aac_ ## name, cm, cm_link); \ + cm->cm_flags |= AAC_ON_ ## index; \ + AACQ_ADD(cm->cm_sc, index); \ +} \ +static __inline struct aac_command * \ +aac_dequeue_ ## name (struct aac_softc *sc) \ +{ \ + struct aac_command *cm; \ + \ + if ((cm = TAILQ_FIRST(&sc->aac_ ## name)) != NULL) { \ + AAC_DPRINTF(AAC_D_CMD, (": dequeue " #name)); \ + if ((cm->cm_flags & AAC_ON_ ## index) == 0) { \ + printf("dequeue - command %p not in queue, flags = %#x, " \ + "bit = %#x\n", cm, cm->cm_flags, \ + AAC_ON_ ## index); \ + panic("command not in queue"); \ + } \ + TAILQ_REMOVE(&sc->aac_ ## name, cm, cm_link); \ + cm->cm_flags &= ~AAC_ON_ ## index; \ + AACQ_REMOVE(sc, index); \ + } \ + return(cm); \ +} \ +static __inline void \ +aac_remove_ ## name (struct aac_command *cm) \ +{ \ + AAC_DPRINTF(AAC_D_CMD, (": remove " #name)); \ + if ((cm->cm_flags & AAC_ON_ ## index) == 0) { \ + printf("remove - command %p not in queue, flags = %#x, " \ + "bit = %#x\n", cm, cm->cm_flags, \ + AAC_ON_ ## index); \ + panic("command not in queue"); \ + } \ + TAILQ_REMOVE(&cm->cm_sc->aac_ ## name, cm, cm_link); \ + cm->cm_flags &= ~AAC_ON_ ## index; \ + AACQ_REMOVE(cm->cm_sc, index); \ +} \ +struct hack + +AACQ_COMMAND_QUEUE(free, AACQ_FREE); +AACQ_COMMAND_QUEUE(ready, AACQ_READY); +AACQ_COMMAND_QUEUE(busy, AACQ_BUSY); +AACQ_COMMAND_QUEUE(bio, AACQ_BIO); + +static __inline void +aac_print_printf(struct aac_softc *sc) { - struct aac_ccb *ccb; - aac_lock_t lock; - - lock = AAC_LOCK(sc); - if ((ccb = TAILQ_FIRST(&sc->sc_completed)) != NULL) - TAILQ_REMOVE(&sc->sc_completed, ccb, ac_chain); - AAC_UNLOCK(sc, lock); - return (ccb); + /* + * XXX We have the ability to read the length of the printf string + * from out of the mailboxes. + */ + printf("** %s: %.*s", sc->aac_dev.dv_xname, AAC_PRINTF_BUFSIZE, + sc->aac_common->ac_printf); + sc->aac_common->ac_printf[0] = 0; + AAC_QNOTIFY(sc, AAC_DB_PRINTF); } -#endif diff --git a/sys/dev/pci/aac_pci.c b/sys/dev/pci/aac_pci.c index 161b78354d2..f720476d938 100644 --- a/sys/dev/pci/aac_pci.c +++ b/sys/dev/pci/aac_pci.c @@ -1,4 +1,4 @@ -/* $OpenBSD: aac_pci.c,v 1.15 2005/08/09 04:10:10 mickey Exp $ */ +/* $OpenBSD: aac_pci.c,v 1.16 2005/11/18 05:39:10 nate Exp $ */ /*- * Copyright (c) 2000 Michael Smith @@ -44,6 +44,8 @@ #include <sys/kernel.h> #include <sys/malloc.h> #include <sys/queue.h> +#include <sys/select.h> +#include <sys/rwlock.h> #include <machine/bus.h> #include <machine/endian.h> @@ -162,12 +164,10 @@ struct aac_ident { /* HP NetRAID-4M */ { PCI_VENDOR_DEC, PCI_PRODUCT_DEC_CPQ42XX, PCI_VENDOR_HP, PCI_PRODUCT_HP_NETRAID_4M, AAC_HWIF_STRONGARM }, - /* Adaptec ASR-2120S */ { PCI_VENDOR_ADP2, PCI_PRODUCT_ADP2_ASR2200S, PCI_VENDOR_ADP2, - PCI_PRODUCT_ADP2_AACASR2120S, AAC_HWIF_I960RX }, - /* Adaptec ASR-2200S */ + PCI_PRODUCT_ADP2_ASR2120S, AAC_HWIF_I960RX }, { PCI_VENDOR_ADP2, PCI_PRODUCT_ADP2_ASR2200S, PCI_VENDOR_ADP2, - PCI_PRODUCT_ADP2_AACASR2200S, AAC_HWIF_I960RX }, + PCI_PRODUCT_ADP2_ASR2200S, AAC_HWIF_I960RX }, { 0, 0, 0, 0 } }; @@ -250,8 +250,8 @@ aac_pci_attach(parent, self, aux) * Map control/status registers. */ if (pci_mapreg_map(pa, PCI_MAPREG_START, - PCI_MAPREG_TYPE_MEM | PCI_MAPREG_MEM_TYPE_32BIT, 0, &sc->sc_memt, - &sc->sc_memh, &membase, &memsize, AAC_REGSIZE)) { + PCI_MAPREG_TYPE_MEM | PCI_MAPREG_MEM_TYPE_32BIT, 0, &sc->aac_memt, + &sc->aac_memh, &membase, &memsize, AAC_REGSIZE)) { printf("can't find mem space\n"); goto bail_out; } @@ -262,9 +262,9 @@ aac_pci_attach(parent, self, aux) goto bail_out; } intrstr = pci_intr_string(pc, ih); - sc->sc_ih = pci_intr_establish(pc, ih, IPL_BIO, aac_intr, sc, - sc->sc_dev.dv_xname); - if (sc->sc_ih == NULL) { + sc->aac_ih = pci_intr_establish(pc, ih, IPL_BIO, aac_intr, sc, + sc->aac_dev.dv_xname); + if (sc->aac_ih == NULL) { printf("couldn't establish interrupt"); if (intrstr != NULL) printf(" at %s", intrstr); @@ -275,25 +275,30 @@ aac_pci_attach(parent, self, aux) if (intrstr != NULL) printf("%s\n", intrstr); - sc->sc_dmat = pa->pa_dmat; + sc->aac_dmat = pa->pa_dmat; for (m = aac_identifiers; m->vendor != 0; m++) if (m->vendor == PCI_VENDOR(pa->pa_id) && m->device == PCI_PRODUCT(pa->pa_id)) { if (m->subvendor == PCI_VENDOR(subsysid) && m->subdevice == PCI_PRODUCT(subsysid)) { - sc->sc_hwif = m->hwif; - switch(sc->sc_hwif) { + sc->aac_hwif = m->hwif; + switch(sc->aac_hwif) { case AAC_HWIF_I960RX: AAC_DPRINTF(AAC_D_MISC, ("set hardware up for i960Rx")); - sc->sc_if = aac_rx_interface; + sc->aac_if = aac_rx_interface; break; case AAC_HWIF_STRONGARM: AAC_DPRINTF(AAC_D_MISC, ("set hardware up for StrongARM")); - sc->sc_if = aac_sa_interface; + sc->aac_if = aac_sa_interface; + break; + case AAC_HWIF_FALCON: + AAC_DPRINTF(AAC_D_MISC, + ("set hardware up for Falcon/PPC")); + sc->aac_if = aac_fa_interface; break; } break; @@ -307,8 +312,8 @@ aac_pci_attach(parent, self, aux) bail_out: if (state > 1) - pci_intr_disestablish(pc, sc->sc_ih); + pci_intr_disestablish(pc, sc->aac_ih); if (state > 0) - bus_space_unmap(sc->sc_memt, sc->sc_memh, memsize); + bus_space_unmap(sc->aac_memt, sc->aac_memh, memsize); return; } |