xref: /freebsd/sys/dev/aac/aac.c (revision 11a9117871e6037ae7b8011b243939322efce569)
135863739SMike Smith /*-
24d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3718cf2ccSPedro F. Giffuni  *
435863739SMike Smith  * Copyright (c) 2000 Michael Smith
5c6eafcf2SScott Long  * Copyright (c) 2001 Scott Long
635863739SMike Smith  * Copyright (c) 2000 BSDi
7c6eafcf2SScott Long  * Copyright (c) 2001 Adaptec, Inc.
835863739SMike Smith  * All rights reserved.
935863739SMike Smith  *
1035863739SMike Smith  * Redistribution and use in source and binary forms, with or without
1135863739SMike Smith  * modification, are permitted provided that the following conditions
1235863739SMike Smith  * are met:
1335863739SMike Smith  * 1. Redistributions of source code must retain the above copyright
1435863739SMike Smith  *    notice, this list of conditions and the following disclaimer.
1535863739SMike Smith  * 2. Redistributions in binary form must reproduce the above copyright
1635863739SMike Smith  *    notice, this list of conditions and the following disclaimer in the
1735863739SMike Smith  *    documentation and/or other materials provided with the distribution.
1835863739SMike Smith  *
1935863739SMike Smith  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
2035863739SMike Smith  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2135863739SMike Smith  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2235863739SMike Smith  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2335863739SMike Smith  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2435863739SMike Smith  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2535863739SMike Smith  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2635863739SMike Smith  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2735863739SMike Smith  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2835863739SMike Smith  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2935863739SMike Smith  * SUCH DAMAGE.
3035863739SMike Smith  */
3135863739SMike Smith 
32aad970f1SDavid E. O'Brien #include <sys/cdefs.h>
3335863739SMike Smith /*
3435863739SMike Smith  * Driver for the Adaptec 'FSA' family of PCI/SCSI RAID adapters.
3535863739SMike Smith  */
367cb209f5SScott Long #define AAC_DRIVERNAME			"aac"
3735863739SMike Smith 
38f6c4dd3fSScott Long #include "opt_aac.h"
39f6c4dd3fSScott Long 
4036e0bf6eSScott Long /* #include <stddef.h> */
4135863739SMike Smith #include <sys/param.h>
4235863739SMike Smith #include <sys/systm.h>
4335863739SMike Smith #include <sys/malloc.h>
4435863739SMike Smith #include <sys/kernel.h>
4536e0bf6eSScott Long #include <sys/kthread.h>
46f287c3e4SBrooks Davis #include <sys/proc.h>
473d04a9d7SScott Long #include <sys/sysctl.h>
48f287c3e4SBrooks Davis #include <sys/sysent.h>
49b3457b51SScott Long #include <sys/poll.h>
50891619a6SPoul-Henning Kamp #include <sys/ioccom.h>
5135863739SMike Smith 
5235863739SMike Smith #include <sys/bus.h>
5335863739SMike Smith #include <sys/conf.h>
5435863739SMike Smith #include <sys/signalvar.h>
550b94a66eSMike Smith #include <sys/time.h>
5636e0bf6eSScott Long #include <sys/eventhandler.h>
577cb209f5SScott Long #include <sys/rman.h>
5835863739SMike Smith 
5935863739SMike Smith #include <machine/bus.h>
6035863739SMike Smith #include <machine/resource.h>
6135863739SMike Smith 
627cb209f5SScott Long #include <dev/pci/pcireg.h>
637cb209f5SScott Long #include <dev/pci/pcivar.h>
647cb209f5SScott Long 
6535863739SMike Smith #include <dev/aac/aacreg.h>
660b0594cdSScott Long #include <sys/aac_ioctl.h>
6735863739SMike Smith #include <dev/aac/aacvar.h>
6835863739SMike Smith #include <dev/aac/aac_tables.h>
6935863739SMike Smith 
7035863739SMike Smith static void	aac_startup(void *arg);
71914da7d0SScott Long static void	aac_add_container(struct aac_softc *sc,
72cbfd045bSScott Long 				  struct aac_mntinforesp *mir, int f);
73fe3cb0e1SScott Long static void	aac_get_bus_info(struct aac_softc *sc);
74ff0991c4SAttilio Rao static void	aac_daemon(void *arg);
7535863739SMike Smith 
7635863739SMike Smith /* Command Processing */
770b94a66eSMike Smith static void	aac_timeout(struct aac_softc *sc);
7835863739SMike Smith static void	aac_complete(void *context, int pending);
7935863739SMike Smith static int	aac_bio_command(struct aac_softc *sc, struct aac_command **cmp);
8035863739SMike Smith static void	aac_bio_complete(struct aac_command *cm);
81d8a0a473SScott Long static int	aac_wait_command(struct aac_command *cm);
8270545d1aSScott Long static void	aac_command_thread(struct aac_softc *sc);
8335863739SMike Smith 
8435863739SMike Smith /* Command Buffer Management */
85cd481291SScott Long static void	aac_map_command_sg(void *arg, bus_dma_segment_t *segs,
86cd481291SScott Long 				   int nseg, int error);
87c6eafcf2SScott Long static void	aac_map_command_helper(void *arg, bus_dma_segment_t *segs,
88c6eafcf2SScott Long 				       int nseg, int error);
890b94a66eSMike Smith static int	aac_alloc_commands(struct aac_softc *sc);
908480cc63SScott Long static void	aac_free_commands(struct aac_softc *sc);
9135863739SMike Smith static void	aac_unmap_command(struct aac_command *cm);
9235863739SMike Smith 
9335863739SMike Smith /* Hardware Interface */
9404f4d586SEd Maste static int	aac_alloc(struct aac_softc *sc);
95c6eafcf2SScott Long static void	aac_common_map(void *arg, bus_dma_segment_t *segs, int nseg,
96c6eafcf2SScott Long 			       int error);
97fe94b852SScott Long static int	aac_check_firmware(struct aac_softc *sc);
9835863739SMike Smith static int	aac_init(struct aac_softc *sc);
9935863739SMike Smith static int	aac_sync_command(struct aac_softc *sc, u_int32_t command,
100c6eafcf2SScott Long 				 u_int32_t arg0, u_int32_t arg1, u_int32_t arg2,
101c6eafcf2SScott Long 				 u_int32_t arg3, u_int32_t *sp);
10204f4d586SEd Maste static int	aac_setup_intr(struct aac_softc *sc);
103c6eafcf2SScott Long static int	aac_enqueue_fib(struct aac_softc *sc, int queue,
104f6c4dd3fSScott Long 				struct aac_command *cm);
105c6eafcf2SScott Long static int	aac_dequeue_fib(struct aac_softc *sc, int queue,
106914da7d0SScott Long 				u_int32_t *fib_size, struct aac_fib **fib_addr);
10736e0bf6eSScott Long static int	aac_enqueue_response(struct aac_softc *sc, int queue,
10836e0bf6eSScott Long 				     struct aac_fib *fib);
10935863739SMike Smith 
11035863739SMike Smith /* StrongARM interface */
11135863739SMike Smith static int	aac_sa_get_fwstatus(struct aac_softc *sc);
11235863739SMike Smith static void	aac_sa_qnotify(struct aac_softc *sc, int qbit);
11335863739SMike Smith static int	aac_sa_get_istatus(struct aac_softc *sc);
11435863739SMike Smith static void	aac_sa_clear_istatus(struct aac_softc *sc, int mask);
11535863739SMike Smith static void	aac_sa_set_mailbox(struct aac_softc *sc, u_int32_t command,
116c6eafcf2SScott Long 				   u_int32_t arg0, u_int32_t arg1,
117c6eafcf2SScott Long 				   u_int32_t arg2, u_int32_t arg3);
118a6d35632SScott Long static int	aac_sa_get_mailbox(struct aac_softc *sc, int mb);
11935863739SMike Smith static void	aac_sa_set_interrupts(struct aac_softc *sc, int enable);
12035863739SMike Smith 
121da4882c2SMarius Strobl const struct aac_interface aac_sa_interface = {
12235863739SMike Smith 	aac_sa_get_fwstatus,
12335863739SMike Smith 	aac_sa_qnotify,
12435863739SMike Smith 	aac_sa_get_istatus,
12535863739SMike Smith 	aac_sa_clear_istatus,
12635863739SMike Smith 	aac_sa_set_mailbox,
127a6d35632SScott Long 	aac_sa_get_mailbox,
1287cb209f5SScott Long 	aac_sa_set_interrupts,
1297cb209f5SScott Long 	NULL, NULL, NULL
13035863739SMike Smith };
13135863739SMike Smith 
13235863739SMike Smith /* i960Rx interface */
13335863739SMike Smith static int	aac_rx_get_fwstatus(struct aac_softc *sc);
13435863739SMike Smith static void	aac_rx_qnotify(struct aac_softc *sc, int qbit);
13535863739SMike Smith static int	aac_rx_get_istatus(struct aac_softc *sc);
13635863739SMike Smith static void	aac_rx_clear_istatus(struct aac_softc *sc, int mask);
13735863739SMike Smith static void	aac_rx_set_mailbox(struct aac_softc *sc, u_int32_t command,
138c6eafcf2SScott Long 				   u_int32_t arg0, u_int32_t arg1,
139c6eafcf2SScott Long 				   u_int32_t arg2, u_int32_t arg3);
140a6d35632SScott Long static int	aac_rx_get_mailbox(struct aac_softc *sc, int mb);
14135863739SMike Smith static void	aac_rx_set_interrupts(struct aac_softc *sc, int enable);
1427cb209f5SScott Long static int aac_rx_send_command(struct aac_softc *sc, struct aac_command *cm);
1437cb209f5SScott Long static int aac_rx_get_outb_queue(struct aac_softc *sc);
1447cb209f5SScott Long static void aac_rx_set_outb_queue(struct aac_softc *sc, int index);
14535863739SMike Smith 
146da4882c2SMarius Strobl const struct aac_interface aac_rx_interface = {
14735863739SMike Smith 	aac_rx_get_fwstatus,
14835863739SMike Smith 	aac_rx_qnotify,
14935863739SMike Smith 	aac_rx_get_istatus,
15035863739SMike Smith 	aac_rx_clear_istatus,
15135863739SMike Smith 	aac_rx_set_mailbox,
152a6d35632SScott Long 	aac_rx_get_mailbox,
1537cb209f5SScott Long 	aac_rx_set_interrupts,
1547cb209f5SScott Long 	aac_rx_send_command,
1557cb209f5SScott Long 	aac_rx_get_outb_queue,
1567cb209f5SScott Long 	aac_rx_set_outb_queue
15735863739SMike Smith };
15835863739SMike Smith 
1594afedc31SScott Long /* Rocket/MIPS interface */
1604afedc31SScott Long static int	aac_rkt_get_fwstatus(struct aac_softc *sc);
1614afedc31SScott Long static void	aac_rkt_qnotify(struct aac_softc *sc, int qbit);
1624afedc31SScott Long static int	aac_rkt_get_istatus(struct aac_softc *sc);
1634afedc31SScott Long static void	aac_rkt_clear_istatus(struct aac_softc *sc, int mask);
1644afedc31SScott Long static void	aac_rkt_set_mailbox(struct aac_softc *sc, u_int32_t command,
1654afedc31SScott Long 				    u_int32_t arg0, u_int32_t arg1,
1664afedc31SScott Long 				    u_int32_t arg2, u_int32_t arg3);
1674afedc31SScott Long static int	aac_rkt_get_mailbox(struct aac_softc *sc, int mb);
1684afedc31SScott Long static void	aac_rkt_set_interrupts(struct aac_softc *sc, int enable);
1697cb209f5SScott Long static int aac_rkt_send_command(struct aac_softc *sc, struct aac_command *cm);
1707cb209f5SScott Long static int aac_rkt_get_outb_queue(struct aac_softc *sc);
1717cb209f5SScott Long static void aac_rkt_set_outb_queue(struct aac_softc *sc, int index);
1724afedc31SScott Long 
173da4882c2SMarius Strobl const struct aac_interface aac_rkt_interface = {
1744afedc31SScott Long 	aac_rkt_get_fwstatus,
1754afedc31SScott Long 	aac_rkt_qnotify,
1764afedc31SScott Long 	aac_rkt_get_istatus,
1774afedc31SScott Long 	aac_rkt_clear_istatus,
1784afedc31SScott Long 	aac_rkt_set_mailbox,
1794afedc31SScott Long 	aac_rkt_get_mailbox,
1807cb209f5SScott Long 	aac_rkt_set_interrupts,
1817cb209f5SScott Long 	aac_rkt_send_command,
1827cb209f5SScott Long 	aac_rkt_get_outb_queue,
1837cb209f5SScott Long 	aac_rkt_set_outb_queue
1844afedc31SScott Long };
1854afedc31SScott Long 
18635863739SMike Smith /* Debugging and Diagnostics */
18735863739SMike Smith static void		aac_describe_controller(struct aac_softc *sc);
188da4882c2SMarius Strobl static const char	*aac_describe_code(const struct aac_code_lookup *table,
189c6eafcf2SScott Long 				   u_int32_t code);
19035863739SMike Smith 
19135863739SMike Smith /* Management Interface */
19235863739SMike Smith static d_open_t		aac_open;
19335863739SMike Smith static d_ioctl_t	aac_ioctl;
194b3457b51SScott Long static d_poll_t		aac_poll;
195dfe2c294SAttilio Rao static void		aac_cdevpriv_dtor(void *arg);
196c6eafcf2SScott Long static int		aac_ioctl_sendfib(struct aac_softc *sc, caddr_t ufib);
197f355c0e0SEd Maste static int		aac_ioctl_send_raw_srb(struct aac_softc *sc, caddr_t arg);
198c6eafcf2SScott Long static void		aac_handle_aif(struct aac_softc *sc,
19936e0bf6eSScott Long 					   struct aac_fib *fib);
200fb0c27d7SScott Long static int		aac_rev_check(struct aac_softc *sc, caddr_t udata);
201a723a548SEd Maste static int		aac_open_aif(struct aac_softc *sc, caddr_t arg);
202a723a548SEd Maste static int		aac_close_aif(struct aac_softc *sc, caddr_t arg);
203fb0c27d7SScott Long static int		aac_getnext_aif(struct aac_softc *sc, caddr_t arg);
204a723a548SEd Maste static int		aac_return_aif(struct aac_softc *sc,
205a723a548SEd Maste 					struct aac_fib_context *ctx, caddr_t uptr);
20636e0bf6eSScott Long static int		aac_query_disk(struct aac_softc *sc, caddr_t uptr);
2077cb209f5SScott Long static int		aac_get_pci_info(struct aac_softc *sc, caddr_t uptr);
2086d307336SEd Maste static int		aac_supported_features(struct aac_softc *sc, caddr_t uptr);
2097cb209f5SScott Long static void		aac_ioctl_event(struct aac_softc *sc,
2107cb209f5SScott Long 					struct aac_event *event, void *arg);
21104f4d586SEd Maste static struct aac_mntinforesp *
21204f4d586SEd Maste 	aac_get_container_info(struct aac_softc *sc, struct aac_fib *fib, int cid);
21335863739SMike Smith 
21435863739SMike Smith static struct cdevsw aac_cdevsw = {
215dc08ffecSPoul-Henning Kamp 	.d_version =	D_VERSION,
216dfdbb320SWarner Losh 	.d_flags =	0,
2177ac40f5fSPoul-Henning Kamp 	.d_open =	aac_open,
2187ac40f5fSPoul-Henning Kamp 	.d_ioctl =	aac_ioctl,
2197ac40f5fSPoul-Henning Kamp 	.d_poll =	aac_poll,
2207ac40f5fSPoul-Henning Kamp 	.d_name =	"aac",
22135863739SMike Smith };
22235863739SMike Smith 
223d745c852SEd Schouten static MALLOC_DEFINE(M_AACBUF, "aacbuf", "Buffers for the AAC driver");
22436e0bf6eSScott Long 
2253d04a9d7SScott Long /* sysctl node */
2267029da5cSPawel Biernacki SYSCTL_NODE(_hw, OID_AUTO, aac, CTLFLAG_RD | CTLFLAG_MPSAFE, 0,
2277029da5cSPawel Biernacki     "AAC driver parameters");
2283d04a9d7SScott Long 
229914da7d0SScott Long /*
230914da7d0SScott Long  * Device Interface
231914da7d0SScott Long  */
23235863739SMike Smith 
233914da7d0SScott Long /*
2344109ba51SEd Maste  * Initialize the controller and softc
23535863739SMike Smith  */
23635863739SMike Smith int
aac_attach(struct aac_softc * sc)23735863739SMike Smith aac_attach(struct aac_softc *sc)
23835863739SMike Smith {
23935863739SMike Smith 	int error, unit;
24035863739SMike Smith 
24131a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
24235863739SMike Smith 
24335863739SMike Smith 	/*
2444109ba51SEd Maste 	 * Initialize per-controller queues.
24535863739SMike Smith 	 */
2460b94a66eSMike Smith 	aac_initq_free(sc);
2470b94a66eSMike Smith 	aac_initq_ready(sc);
2480b94a66eSMike Smith 	aac_initq_busy(sc);
2490b94a66eSMike Smith 	aac_initq_bio(sc);
25035863739SMike Smith 
25135863739SMike Smith 	/*
2524109ba51SEd Maste 	 * Initialize command-completion task.
25335863739SMike Smith 	 */
25435863739SMike Smith 	TASK_INIT(&sc->aac_task_complete, 0, aac_complete, sc);
25535863739SMike Smith 
25635863739SMike Smith 	/* mark controller as suspended until we get ourselves organised */
25735863739SMike Smith 	sc->aac_state |= AAC_STATE_SUSPEND;
25835863739SMike Smith 
25935863739SMike Smith 	/*
260fe94b852SScott Long 	 * Check that the firmware on the card is supported.
261fe94b852SScott Long 	 */
262fe94b852SScott Long 	if ((error = aac_check_firmware(sc)) != 0)
263fe94b852SScott Long 		return(error);
264fe94b852SScott Long 
265f6b1c44dSScott Long 	/*
266f6b1c44dSScott Long 	 * Initialize locks
267f6b1c44dSScott Long 	 */
268bb6fe253SScott Long 	mtx_init(&sc->aac_aifq_lock, "AAC AIF lock", NULL, MTX_DEF);
269bb6fe253SScott Long 	mtx_init(&sc->aac_io_lock, "AAC I/O lock", NULL, MTX_DEF);
270bb6fe253SScott Long 	mtx_init(&sc->aac_container_lock, "AAC container lock", NULL, MTX_DEF);
271f6b1c44dSScott Long 	TAILQ_INIT(&sc->aac_container_tqh);
272065dd78cSScott Long 	TAILQ_INIT(&sc->aac_ev_cmfree);
273f6b1c44dSScott Long 
274ff0991c4SAttilio Rao 	/* Initialize the clock daemon callout. */
275ff0991c4SAttilio Rao 	callout_init_mtx(&sc->aac_daemontime, &sc->aac_io_lock, 0);
276ff0991c4SAttilio Rao 
2770b94a66eSMike Smith 	/*
2784109ba51SEd Maste 	 * Initialize the adapter.
27935863739SMike Smith 	 */
28004f4d586SEd Maste 	if ((error = aac_alloc(sc)) != 0)
28104f4d586SEd Maste 		return(error);
2820b94a66eSMike Smith 	if ((error = aac_init(sc)) != 0)
28335863739SMike Smith 		return(error);
28435863739SMike Smith 
28535863739SMike Smith 	/*
2867cb209f5SScott Long 	 * Allocate and connect our interrupt.
2877cb209f5SScott Long 	 */
28804f4d586SEd Maste 	if ((error = aac_setup_intr(sc)) != 0)
28904f4d586SEd Maste 		return(error);
2907cb209f5SScott Long 
2917cb209f5SScott Long 	/*
29235863739SMike Smith 	 * Print a little information about the controller.
29335863739SMike Smith 	 */
29435863739SMike Smith 	aac_describe_controller(sc);
29535863739SMike Smith 
29635863739SMike Smith 	/*
2971423dcd6SEd Maste 	 * Add sysctls.
2981423dcd6SEd Maste 	 */
2991423dcd6SEd Maste 	SYSCTL_ADD_INT(device_get_sysctl_ctx(sc->aac_dev),
3001423dcd6SEd Maste 	    SYSCTL_CHILDREN(device_get_sysctl_tree(sc->aac_dev)),
3011423dcd6SEd Maste 	    OID_AUTO, "firmware_build", CTLFLAG_RD,
3021423dcd6SEd Maste 	    &sc->aac_revision.buildNumber, 0,
3031423dcd6SEd Maste 	    "firmware build number");
3041423dcd6SEd Maste 
3051423dcd6SEd Maste 	/*
306ae543596SScott Long 	 * Register to probe our containers later.
307ae543596SScott Long 	 */
30835863739SMike Smith 	sc->aac_ich.ich_func = aac_startup;
30935863739SMike Smith 	sc->aac_ich.ich_arg = sc;
31035863739SMike Smith 	if (config_intrhook_establish(&sc->aac_ich) != 0) {
311914da7d0SScott Long 		device_printf(sc->aac_dev,
312914da7d0SScott Long 			      "can't establish configuration hook\n");
31335863739SMike Smith 		return(ENXIO);
31435863739SMike Smith 	}
31535863739SMike Smith 
31635863739SMike Smith 	/*
31735863739SMike Smith 	 * Make the control device.
31835863739SMike Smith 	 */
31935863739SMike Smith 	unit = device_get_unit(sc->aac_dev);
3209e9466baSRobert Watson 	sc->aac_dev_t = make_dev(&aac_cdevsw, unit, UID_ROOT, GID_OPERATOR,
3219e9466baSRobert Watson 				 0640, "aac%d", unit);
322157fbb2eSScott Long 	(void)make_dev_alias(sc->aac_dev_t, "afa%d", unit);
3234aa620cdSScott Long 	(void)make_dev_alias(sc->aac_dev_t, "hpn%d", unit);
32435863739SMike Smith 	sc->aac_dev_t->si_drv1 = sc;
32535863739SMike Smith 
32636e0bf6eSScott Long 	/* Create the AIF thread */
3273745c395SJulian Elischer 	if (kproc_create((void(*)(void *))aac_command_thread, sc,
328316ec49aSScott Long 		   &sc->aifthread, 0, 0, "aac%daif", unit))
329a620bad0SEd Maste 		panic("Could not create AIF thread");
33036e0bf6eSScott Long 
33136e0bf6eSScott Long 	/* Register the shutdown method to only be called post-dump */
3325f54d522SScott Long 	if ((sc->eh = EVENTHANDLER_REGISTER(shutdown_final, aac_shutdown,
3335f54d522SScott Long 	    sc->aac_dev, SHUTDOWN_PRI_DEFAULT)) == NULL)
3345f54d522SScott Long 		device_printf(sc->aac_dev,
3355f54d522SScott Long 			      "shutdown event registration failed\n");
33636e0bf6eSScott Long 
337fe3cb0e1SScott Long 	/* Register with CAM for the non-DASD devices */
338a6d35632SScott Long 	if ((sc->flags & AAC_FLAGS_ENABLE_CAM) != 0) {
33970545d1aSScott Long 		TAILQ_INIT(&sc->aac_sim_tqh);
340fe3cb0e1SScott Long 		aac_get_bus_info(sc);
34170545d1aSScott Long 	}
342fe3cb0e1SScott Long 
343ff0991c4SAttilio Rao 	mtx_lock(&sc->aac_io_lock);
344867b1d34SEd Maste 	callout_reset(&sc->aac_daemontime, 60 * hz, aac_daemon, sc);
345ff0991c4SAttilio Rao 	mtx_unlock(&sc->aac_io_lock);
346ff0991c4SAttilio Rao 
34735863739SMike Smith 	return(0);
34835863739SMike Smith }
34935863739SMike Smith 
350ff0991c4SAttilio Rao static void
aac_daemon(void * arg)351ff0991c4SAttilio Rao aac_daemon(void *arg)
352ff0991c4SAttilio Rao {
353ff0991c4SAttilio Rao 	struct timeval tv;
354ff0991c4SAttilio Rao 	struct aac_softc *sc;
355ff0991c4SAttilio Rao 	struct aac_fib *fib;
356ff0991c4SAttilio Rao 
357ff0991c4SAttilio Rao 	sc = arg;
358ff0991c4SAttilio Rao 	mtx_assert(&sc->aac_io_lock, MA_OWNED);
359ff0991c4SAttilio Rao 
360ff0991c4SAttilio Rao 	if (callout_pending(&sc->aac_daemontime) ||
361ff0991c4SAttilio Rao 	    callout_active(&sc->aac_daemontime) == 0)
362ff0991c4SAttilio Rao 		return;
363ff0991c4SAttilio Rao 	getmicrotime(&tv);
364ff0991c4SAttilio Rao 	aac_alloc_sync_fib(sc, &fib);
365ff0991c4SAttilio Rao 	*(uint32_t *)fib->data = tv.tv_sec;
366ff0991c4SAttilio Rao 	aac_sync_fib(sc, SendHostTime, 0, fib, sizeof(uint32_t));
367ff0991c4SAttilio Rao 	aac_release_sync_fib(sc);
368ff0991c4SAttilio Rao 	callout_schedule(&sc->aac_daemontime, 30 * 60 * hz);
369ff0991c4SAttilio Rao }
370ff0991c4SAttilio Rao 
3717cb209f5SScott Long void
aac_add_event(struct aac_softc * sc,struct aac_event * event)3727cb209f5SScott Long aac_add_event(struct aac_softc *sc, struct aac_event *event)
3737cb209f5SScott Long {
3747cb209f5SScott Long 
3757cb209f5SScott Long 	switch (event->ev_type & AAC_EVENT_MASK) {
3767cb209f5SScott Long 	case AAC_EVENT_CMFREE:
3777cb209f5SScott Long 		TAILQ_INSERT_TAIL(&sc->aac_ev_cmfree, event, ev_links);
3787cb209f5SScott Long 		break;
3797cb209f5SScott Long 	default:
3807cb209f5SScott Long 		device_printf(sc->aac_dev, "aac_add event: unknown event %d\n",
3817cb209f5SScott Long 		    event->ev_type);
3827cb209f5SScott Long 		break;
3837cb209f5SScott Long 	}
3847cb209f5SScott Long }
3857cb209f5SScott Long 
386914da7d0SScott Long /*
38704f4d586SEd Maste  * Request information of container #cid
38804f4d586SEd Maste  */
38904f4d586SEd Maste static struct aac_mntinforesp *
aac_get_container_info(struct aac_softc * sc,struct aac_fib * fib,int cid)39004f4d586SEd Maste aac_get_container_info(struct aac_softc *sc, struct aac_fib *fib, int cid)
39104f4d586SEd Maste {
39204f4d586SEd Maste 	struct aac_mntinfo *mi;
39304f4d586SEd Maste 
39404f4d586SEd Maste 	mi = (struct aac_mntinfo *)&fib->data[0];
395523da39bSEd Maste 	/* use 64-bit LBA if enabled */
396523da39bSEd Maste 	mi->Command = (sc->flags & AAC_FLAGS_LBA_64BIT) ?
397523da39bSEd Maste 	    VM_NameServe64 : VM_NameServe;
39804f4d586SEd Maste 	mi->MntType = FT_FILESYS;
39904f4d586SEd Maste 	mi->MntCount = cid;
40004f4d586SEd Maste 
40104f4d586SEd Maste 	if (aac_sync_fib(sc, ContainerCommand, 0, fib,
40204f4d586SEd Maste 			 sizeof(struct aac_mntinfo))) {
403dbb34a64SEd Maste 		device_printf(sc->aac_dev, "Error probing container %d\n", cid);
40404f4d586SEd Maste 		return (NULL);
40504f4d586SEd Maste 	}
40604f4d586SEd Maste 
40704f4d586SEd Maste 	return ((struct aac_mntinforesp *)&fib->data[0]);
40804f4d586SEd Maste }
40904f4d586SEd Maste 
41004f4d586SEd Maste /*
41135863739SMike Smith  * Probe for containers, create disks.
41235863739SMike Smith  */
41335863739SMike Smith static void
aac_startup(void * arg)41435863739SMike Smith aac_startup(void *arg)
41535863739SMike Smith {
416914da7d0SScott Long 	struct aac_softc *sc;
417cbfd045bSScott Long 	struct aac_fib *fib;
41804f4d586SEd Maste 	struct aac_mntinforesp *mir;
419795d7dc0SScott Long 	int count = 0, i = 0;
42035863739SMike Smith 
421914da7d0SScott Long 	sc = (struct aac_softc *)arg;
42231a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
423914da7d0SScott Long 
4247cb209f5SScott Long 	mtx_lock(&sc->aac_io_lock);
42503b5fe51SScott Long 	aac_alloc_sync_fib(sc, &fib);
426cbfd045bSScott Long 
42735863739SMike Smith 	/* loop over possible containers */
42836e0bf6eSScott Long 	do {
42904f4d586SEd Maste 		if ((mir = aac_get_container_info(sc, fib, i)) == NULL)
43035863739SMike Smith 			continue;
43104f4d586SEd Maste 		if (i == 0)
432795d7dc0SScott Long 			count = mir->MntRespCount;
433cbfd045bSScott Long 		aac_add_container(sc, mir, 0);
43436e0bf6eSScott Long 		i++;
435795d7dc0SScott Long 	} while ((i < count) && (i < AAC_MAX_CONTAINERS));
436cbfd045bSScott Long 
437cbfd045bSScott Long 	aac_release_sync_fib(sc);
4387cb209f5SScott Long 	mtx_unlock(&sc->aac_io_lock);
43935863739SMike Smith 
440cc336c78SScott Long 	/* mark the controller up */
441cc336c78SScott Long 	sc->aac_state &= ~AAC_STATE_SUSPEND;
442cc336c78SScott Long 
44335863739SMike Smith 	/* poke the bus to actually attach the child devices */
44418250ec6SJohn Baldwin 	bus_attach_children(sc->aac_dev);
44535863739SMike Smith 
446cc336c78SScott Long 	/* disconnect ourselves from the intrhook chain */
447cc336c78SScott Long 	config_intrhook_disestablish(&sc->aac_ich);
44835863739SMike Smith 
44935863739SMike Smith 	/* enable interrupts now */
45035863739SMike Smith 	AAC_UNMASK_INTERRUPTS(sc);
45135863739SMike Smith }
45235863739SMike Smith 
453914da7d0SScott Long /*
4544109ba51SEd Maste  * Create a device to represent a new container
455914da7d0SScott Long  */
456914da7d0SScott Long static void
aac_add_container(struct aac_softc * sc,struct aac_mntinforesp * mir,int f)457cbfd045bSScott Long aac_add_container(struct aac_softc *sc, struct aac_mntinforesp *mir, int f)
458914da7d0SScott Long {
459914da7d0SScott Long 	struct aac_container *co;
460914da7d0SScott Long 	device_t child;
461914da7d0SScott Long 
462914da7d0SScott Long 	/*
463914da7d0SScott Long 	 * Check container volume type for validity.  Note that many of
464914da7d0SScott Long 	 * the possible types may never show up.
465914da7d0SScott Long 	 */
466914da7d0SScott Long 	if ((mir->Status == ST_OK) && (mir->MntTable[0].VolType != CT_NONE)) {
467a761a1caSScott Long 		co = (struct aac_container *)malloc(sizeof *co, M_AACBUF,
468a761a1caSScott Long 		       M_NOWAIT | M_ZERO);
469914da7d0SScott Long 		if (co == NULL)
470a620bad0SEd Maste 			panic("Out of memory?!");
47131a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_INIT_B, "id %x  name '%.16s'  size %u  type %d",
472914da7d0SScott Long 		      mir->MntTable[0].ObjectId,
473914da7d0SScott Long 		      mir->MntTable[0].FileSystemName,
474914da7d0SScott Long 		      mir->MntTable[0].Capacity, mir->MntTable[0].VolType);
475914da7d0SScott Long 
476fe3cb0e1SScott Long 		if ((child = device_add_child(sc->aac_dev, "aacd", -1)) == NULL)
477914da7d0SScott Long 			device_printf(sc->aac_dev, "device_add_child failed\n");
478914da7d0SScott Long 		else
479914da7d0SScott Long 			device_set_ivars(child, co);
480914da7d0SScott Long 		device_set_desc(child, aac_describe_code(aac_container_types,
481914da7d0SScott Long 				mir->MntTable[0].VolType));
482914da7d0SScott Long 		co->co_disk = child;
483914da7d0SScott Long 		co->co_found = f;
484914da7d0SScott Long 		bcopy(&mir->MntTable[0], &co->co_mntobj,
485914da7d0SScott Long 		      sizeof(struct aac_mntobj));
486bb6fe253SScott Long 		mtx_lock(&sc->aac_container_lock);
487914da7d0SScott Long 		TAILQ_INSERT_TAIL(&sc->aac_container_tqh, co, co_link);
488bb6fe253SScott Long 		mtx_unlock(&sc->aac_container_lock);
489914da7d0SScott Long 	}
490914da7d0SScott Long }
491914da7d0SScott Long 
492914da7d0SScott Long /*
49304f4d586SEd Maste  * Allocate resources associated with (sc)
49404f4d586SEd Maste  */
49504f4d586SEd Maste static int
aac_alloc(struct aac_softc * sc)49604f4d586SEd Maste aac_alloc(struct aac_softc *sc)
49704f4d586SEd Maste {
49831a0399eSEd Maste 
49931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
50031a0399eSEd Maste 
50104f4d586SEd Maste 	/*
50204f4d586SEd Maste 	 * Create DMA tag for mapping buffers into controller-addressable space.
50304f4d586SEd Maste 	 */
50404f4d586SEd Maste 	if (bus_dma_tag_create(sc->aac_parent_dmat, 	/* parent */
50504f4d586SEd Maste 			       1, 0, 			/* algnmnt, boundary */
50604f4d586SEd Maste 			       (sc->flags & AAC_FLAGS_SG_64BIT) ?
50704f4d586SEd Maste 			       BUS_SPACE_MAXADDR :
50804f4d586SEd Maste 			       BUS_SPACE_MAXADDR_32BIT,	/* lowaddr */
50904f4d586SEd Maste 			       BUS_SPACE_MAXADDR, 	/* highaddr */
51004f4d586SEd Maste 			       NULL, NULL, 		/* filter, filterarg */
5116f954fb3SAlexander Motin 			       sc->aac_max_sectors << 9, /* maxsize */
51204f4d586SEd Maste 			       sc->aac_sg_tablesize,	/* nsegments */
5136f954fb3SAlexander Motin 			       BUS_SPACE_MAXSIZE_32BIT,	/* maxsegsize */
51404f4d586SEd Maste 			       BUS_DMA_ALLOCNOW,	/* flags */
51504f4d586SEd Maste 			       busdma_lock_mutex,	/* lockfunc */
51604f4d586SEd Maste 			       &sc->aac_io_lock,	/* lockfuncarg */
51704f4d586SEd Maste 			       &sc->aac_buffer_dmat)) {
51804f4d586SEd Maste 		device_printf(sc->aac_dev, "can't allocate buffer DMA tag\n");
51904f4d586SEd Maste 		return (ENOMEM);
52004f4d586SEd Maste 	}
52104f4d586SEd Maste 
52204f4d586SEd Maste 	/*
52304f4d586SEd Maste 	 * Create DMA tag for mapping FIBs into controller-addressable space..
52404f4d586SEd Maste 	 */
52504f4d586SEd Maste 	if (bus_dma_tag_create(sc->aac_parent_dmat,	/* parent */
52604f4d586SEd Maste 			       1, 0, 			/* algnmnt, boundary */
52704f4d586SEd Maste 			       (sc->flags & AAC_FLAGS_4GB_WINDOW) ?
52804f4d586SEd Maste 			       BUS_SPACE_MAXADDR_32BIT :
52904f4d586SEd Maste 			       0x7fffffff,		/* lowaddr */
53004f4d586SEd Maste 			       BUS_SPACE_MAXADDR, 	/* highaddr */
53104f4d586SEd Maste 			       NULL, NULL, 		/* filter, filterarg */
53204f4d586SEd Maste 			       sc->aac_max_fibs_alloc *
53304f4d586SEd Maste 			       sc->aac_max_fib_size,  /* maxsize */
53404f4d586SEd Maste 			       1,			/* nsegments */
53504f4d586SEd Maste 			       sc->aac_max_fibs_alloc *
53604f4d586SEd Maste 			       sc->aac_max_fib_size,	/* maxsize */
53704f4d586SEd Maste 			       0,			/* flags */
53804f4d586SEd Maste 			       NULL, NULL,		/* No locking needed */
53904f4d586SEd Maste 			       &sc->aac_fib_dmat)) {
540c2ede4b3SMartin Blapp 		device_printf(sc->aac_dev, "can't allocate FIB DMA tag\n");
54104f4d586SEd Maste 		return (ENOMEM);
54204f4d586SEd Maste 	}
54304f4d586SEd Maste 
54404f4d586SEd Maste 	/*
54504f4d586SEd Maste 	 * Create DMA tag for the common structure and allocate it.
54604f4d586SEd Maste 	 */
54704f4d586SEd Maste 	if (bus_dma_tag_create(sc->aac_parent_dmat, 	/* parent */
54804f4d586SEd Maste 			       1, 0,			/* algnmnt, boundary */
54904f4d586SEd Maste 			       (sc->flags & AAC_FLAGS_4GB_WINDOW) ?
55004f4d586SEd Maste 			       BUS_SPACE_MAXADDR_32BIT :
55104f4d586SEd Maste 			       0x7fffffff,		/* lowaddr */
55204f4d586SEd Maste 			       BUS_SPACE_MAXADDR, 	/* highaddr */
55304f4d586SEd Maste 			       NULL, NULL, 		/* filter, filterarg */
55404f4d586SEd Maste 			       8192 + sizeof(struct aac_common), /* maxsize */
55504f4d586SEd Maste 			       1,			/* nsegments */
55604f4d586SEd Maste 			       BUS_SPACE_MAXSIZE_32BIT,	/* maxsegsize */
55704f4d586SEd Maste 			       0,			/* flags */
55804f4d586SEd Maste 			       NULL, NULL,		/* No locking needed */
55904f4d586SEd Maste 			       &sc->aac_common_dmat)) {
56004f4d586SEd Maste 		device_printf(sc->aac_dev,
56104f4d586SEd Maste 			      "can't allocate common structure DMA tag\n");
56204f4d586SEd Maste 		return (ENOMEM);
56304f4d586SEd Maste 	}
56404f4d586SEd Maste 	if (bus_dmamem_alloc(sc->aac_common_dmat, (void **)&sc->aac_common,
56504f4d586SEd Maste 			     BUS_DMA_NOWAIT, &sc->aac_common_dmamap)) {
56604f4d586SEd Maste 		device_printf(sc->aac_dev, "can't allocate common structure\n");
56704f4d586SEd Maste 		return (ENOMEM);
56804f4d586SEd Maste 	}
56904f4d586SEd Maste 
57004f4d586SEd Maste 	/*
57104f4d586SEd Maste 	 * Work around a bug in the 2120 and 2200 that cannot DMA commands
57204f4d586SEd Maste 	 * below address 8192 in physical memory.
57304f4d586SEd Maste 	 * XXX If the padding is not needed, can it be put to use instead
57404f4d586SEd Maste 	 * of ignored?
57504f4d586SEd Maste 	 */
57604f4d586SEd Maste 	(void)bus_dmamap_load(sc->aac_common_dmat, sc->aac_common_dmamap,
57704f4d586SEd Maste 			sc->aac_common, 8192 + sizeof(*sc->aac_common),
57804f4d586SEd Maste 			aac_common_map, sc, 0);
57904f4d586SEd Maste 
58004f4d586SEd Maste 	if (sc->aac_common_busaddr < 8192) {
58104f4d586SEd Maste 		sc->aac_common = (struct aac_common *)
58204f4d586SEd Maste 		    ((uint8_t *)sc->aac_common + 8192);
58304f4d586SEd Maste 		sc->aac_common_busaddr += 8192;
58404f4d586SEd Maste 	}
58504f4d586SEd Maste 	bzero(sc->aac_common, sizeof(*sc->aac_common));
58604f4d586SEd Maste 
58704f4d586SEd Maste 	/* Allocate some FIBs and associated command structs */
58804f4d586SEd Maste 	TAILQ_INIT(&sc->aac_fibmap_tqh);
58904f4d586SEd Maste 	sc->aac_commands = malloc(sc->aac_max_fibs * sizeof(struct aac_command),
59004f4d586SEd Maste 				  M_AACBUF, M_WAITOK|M_ZERO);
59123e876b1SJung-uk Kim 	while (sc->total_fibs < sc->aac_max_fibs) {
59204f4d586SEd Maste 		if (aac_alloc_commands(sc) != 0)
59304f4d586SEd Maste 			break;
59404f4d586SEd Maste 	}
59504f4d586SEd Maste 	if (sc->total_fibs == 0)
59604f4d586SEd Maste 		return (ENOMEM);
59704f4d586SEd Maste 
59804f4d586SEd Maste 	return (0);
59904f4d586SEd Maste }
60004f4d586SEd Maste 
60104f4d586SEd Maste /*
60235863739SMike Smith  * Free all of the resources associated with (sc)
60335863739SMike Smith  *
60435863739SMike Smith  * Should not be called if the controller is active.
60535863739SMike Smith  */
60635863739SMike Smith void
aac_free(struct aac_softc * sc)60735863739SMike Smith aac_free(struct aac_softc *sc)
60835863739SMike Smith {
609ffb37f33SScott Long 
61031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
61135863739SMike Smith 
61235863739SMike Smith 	/* remove the control device */
61335863739SMike Smith 	if (sc->aac_dev_t != NULL)
61435863739SMike Smith 		destroy_dev(sc->aac_dev_t);
61535863739SMike Smith 
6160b94a66eSMike Smith 	/* throw away any FIB buffers, discard the FIB DMA tag */
6178480cc63SScott Long 	aac_free_commands(sc);
6180b94a66eSMike Smith 	if (sc->aac_fib_dmat)
6190b94a66eSMike Smith 		bus_dma_tag_destroy(sc->aac_fib_dmat);
62035863739SMike Smith 
621ffb37f33SScott Long 	free(sc->aac_commands, M_AACBUF);
622ffb37f33SScott Long 
62335863739SMike Smith 	/* destroy the common area */
62435863739SMike Smith 	if (sc->aac_common) {
62535863739SMike Smith 		bus_dmamap_unload(sc->aac_common_dmat, sc->aac_common_dmamap);
626c6eafcf2SScott Long 		bus_dmamem_free(sc->aac_common_dmat, sc->aac_common,
627c6eafcf2SScott Long 				sc->aac_common_dmamap);
62835863739SMike Smith 	}
6290b94a66eSMike Smith 	if (sc->aac_common_dmat)
6300b94a66eSMike Smith 		bus_dma_tag_destroy(sc->aac_common_dmat);
63135863739SMike Smith 
63235863739SMike Smith 	/* disconnect the interrupt handler */
63335863739SMike Smith 	if (sc->aac_intr)
63435863739SMike Smith 		bus_teardown_intr(sc->aac_dev, sc->aac_irq, sc->aac_intr);
635bbc03f1bSMarius Strobl 	if (sc->aac_irq != NULL) {
636da4882c2SMarius Strobl 		bus_release_resource(sc->aac_dev, SYS_RES_IRQ,
637da4882c2SMarius Strobl 		    rman_get_rid(sc->aac_irq), sc->aac_irq);
638bbc03f1bSMarius Strobl 		pci_release_msi(sc->aac_dev);
639bbc03f1bSMarius Strobl 	}
64035863739SMike Smith 
64135863739SMike Smith 	/* destroy data-transfer DMA tag */
64235863739SMike Smith 	if (sc->aac_buffer_dmat)
64335863739SMike Smith 		bus_dma_tag_destroy(sc->aac_buffer_dmat);
64435863739SMike Smith 
64535863739SMike Smith 	/* destroy the parent DMA tag */
64635863739SMike Smith 	if (sc->aac_parent_dmat)
64735863739SMike Smith 		bus_dma_tag_destroy(sc->aac_parent_dmat);
64835863739SMike Smith 
64935863739SMike Smith 	/* release the register window mapping */
650ff0991c4SAttilio Rao 	if (sc->aac_regs_res0 != NULL)
651914da7d0SScott Long 		bus_release_resource(sc->aac_dev, SYS_RES_MEMORY,
652da4882c2SMarius Strobl 		    rman_get_rid(sc->aac_regs_res0), sc->aac_regs_res0);
653ff0991c4SAttilio Rao 	if (sc->aac_hwif == AAC_HWIF_NARK && sc->aac_regs_res1 != NULL)
654ff0991c4SAttilio Rao 		bus_release_resource(sc->aac_dev, SYS_RES_MEMORY,
655da4882c2SMarius Strobl 		    rman_get_rid(sc->aac_regs_res1), sc->aac_regs_res1);
65635863739SMike Smith }
65735863739SMike Smith 
658914da7d0SScott Long /*
65935863739SMike Smith  * Disconnect from the controller completely, in preparation for unload.
66035863739SMike Smith  */
66135863739SMike Smith int
aac_detach(device_t dev)66235863739SMike Smith aac_detach(device_t dev)
66335863739SMike Smith {
664914da7d0SScott Long 	struct aac_softc *sc;
66570545d1aSScott Long 	struct aac_container *co;
66670545d1aSScott Long 	struct aac_sim	*sim;
66735863739SMike Smith 	int error;
66835863739SMike Smith 
669914da7d0SScott Long 	sc = device_get_softc(dev);
67031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
671914da7d0SScott Long 
672*11a91178SJohn Baldwin 	error = bus_generic_detach(dev);
673*11a91178SJohn Baldwin 	if (error != 0)
674*11a91178SJohn Baldwin 		return (error);
675*11a91178SJohn Baldwin 
676ff0991c4SAttilio Rao 	callout_drain(&sc->aac_daemontime);
677ff0991c4SAttilio Rao 
6781bd320ecSAttilio Rao 	mtx_lock(&sc->aac_io_lock);
6791bd320ecSAttilio Rao 	while (sc->aifflags & AAC_AIFFLAGS_RUNNING) {
6801bd320ecSAttilio Rao 		sc->aifflags |= AAC_AIFFLAGS_EXIT;
6811bd320ecSAttilio Rao 		wakeup(sc->aifthread);
6821bd320ecSAttilio Rao 		msleep(sc->aac_dev, &sc->aac_io_lock, PUSER, "aacdch", 0);
6831bd320ecSAttilio Rao 	}
6841bd320ecSAttilio Rao 	mtx_unlock(&sc->aac_io_lock);
6851bd320ecSAttilio Rao 	KASSERT((sc->aifflags & AAC_AIFFLAGS_RUNNING) == 0,
6861bd320ecSAttilio Rao 	    ("%s: invalid detach state", __func__));
6871bd320ecSAttilio Rao 
68870545d1aSScott Long 	/* Remove the child containers */
689a761a1caSScott Long 	while ((co = TAILQ_FIRST(&sc->aac_container_tqh)) != NULL) {
69065ac4ed6SScott Long 		TAILQ_REMOVE(&sc->aac_container_tqh, co, co_link);
691a761a1caSScott Long 		free(co, M_AACBUF);
69270545d1aSScott Long 	}
69370545d1aSScott Long 
69470545d1aSScott Long 	/* Remove the CAM SIMs */
695a761a1caSScott Long 	while ((sim = TAILQ_FIRST(&sc->aac_sim_tqh)) != NULL) {
696a761a1caSScott Long 		TAILQ_REMOVE(&sc->aac_sim_tqh, sim, sim_link);
697a761a1caSScott Long 		free(sim, M_AACBUF);
69870545d1aSScott Long 	}
69970545d1aSScott Long 
70035863739SMike Smith 	if ((error = aac_shutdown(dev)))
70135863739SMike Smith 		return(error);
70235863739SMike Smith 
7035f54d522SScott Long 	EVENTHANDLER_DEREGISTER(shutdown_final, sc->eh);
7045f54d522SScott Long 
70535863739SMike Smith 	aac_free(sc);
70635863739SMike Smith 
707dc9efde5SScott Long 	mtx_destroy(&sc->aac_aifq_lock);
708dc9efde5SScott Long 	mtx_destroy(&sc->aac_io_lock);
709dc9efde5SScott Long 	mtx_destroy(&sc->aac_container_lock);
710dc9efde5SScott Long 
71135863739SMike Smith 	return(0);
71235863739SMike Smith }
71335863739SMike Smith 
714914da7d0SScott Long /*
71535863739SMike Smith  * Bring the controller down to a dormant state and detach all child devices.
71635863739SMike Smith  *
71735863739SMike Smith  * This function is called before detach or system shutdown.
71835863739SMike Smith  *
7190b94a66eSMike Smith  * Note that we can assume that the bioq on the controller is empty, as we won't
72035863739SMike Smith  * allow shutdown if any device is open.
72135863739SMike Smith  */
72235863739SMike Smith int
aac_shutdown(device_t dev)72335863739SMike Smith aac_shutdown(device_t dev)
72435863739SMike Smith {
725914da7d0SScott Long 	struct aac_softc *sc;
726cbfd045bSScott Long 	struct aac_fib *fib;
727cbfd045bSScott Long 	struct aac_close_command *cc;
72835863739SMike Smith 
729914da7d0SScott Long 	sc = device_get_softc(dev);
73031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
731914da7d0SScott Long 
73235863739SMike Smith 	sc->aac_state |= AAC_STATE_SUSPEND;
73335863739SMike Smith 
73435863739SMike Smith 	/*
73535863739SMike Smith 	 * Send a Container shutdown followed by a HostShutdown FIB to the
73635863739SMike Smith 	 * controller to convince it that we don't want to talk to it anymore.
73735863739SMike Smith 	 * We've been closed and all I/O completed already
73835863739SMike Smith 	 */
73935863739SMike Smith 	device_printf(sc->aac_dev, "shutting down controller...");
74035863739SMike Smith 
7417cb209f5SScott Long 	mtx_lock(&sc->aac_io_lock);
74203b5fe51SScott Long 	aac_alloc_sync_fib(sc, &fib);
743cbfd045bSScott Long 	cc = (struct aac_close_command *)&fib->data[0];
744cbfd045bSScott Long 
74539ee03c3SScott Long 	bzero(cc, sizeof(struct aac_close_command));
746cbfd045bSScott Long 	cc->Command = VM_CloseAll;
747cbfd045bSScott Long 	cc->ContainerId = 0xffffffff;
748cbfd045bSScott Long 	if (aac_sync_fib(sc, ContainerCommand, 0, fib,
749cbfd045bSScott Long 	    sizeof(struct aac_close_command)))
75035863739SMike Smith 		printf("FAILED.\n");
75170545d1aSScott Long 	else
75270545d1aSScott Long 		printf("done\n");
75370545d1aSScott Long #if 0
754914da7d0SScott Long 	else {
755cbfd045bSScott Long 		fib->data[0] = 0;
75636e0bf6eSScott Long 		/*
757914da7d0SScott Long 		 * XXX Issuing this command to the controller makes it shut down
75836e0bf6eSScott Long 		 * but also keeps it from coming back up without a reset of the
75936e0bf6eSScott Long 		 * PCI bus.  This is not desirable if you are just unloading the
76036e0bf6eSScott Long 		 * driver module with the intent to reload it later.
76136e0bf6eSScott Long 		 */
762cbfd045bSScott Long 		if (aac_sync_fib(sc, FsaHostShutdown, AAC_FIBSTATE_SHUTDOWN,
763cbfd045bSScott Long 		    fib, 1)) {
76435863739SMike Smith 			printf("FAILED.\n");
76535863739SMike Smith 		} else {
76635863739SMike Smith 			printf("done.\n");
76735863739SMike Smith 		}
76835863739SMike Smith 	}
76970545d1aSScott Long #endif
77035863739SMike Smith 
77135863739SMike Smith 	AAC_MASK_INTERRUPTS(sc);
7723576af8fSScott Long 	aac_release_sync_fib(sc);
7737cb209f5SScott Long 	mtx_unlock(&sc->aac_io_lock);
77435863739SMike Smith 
77535863739SMike Smith 	return(0);
77635863739SMike Smith }
77735863739SMike Smith 
778914da7d0SScott Long /*
77935863739SMike Smith  * Bring the controller to a quiescent state, ready for system suspend.
78035863739SMike Smith  */
78135863739SMike Smith int
aac_suspend(device_t dev)78235863739SMike Smith aac_suspend(device_t dev)
78335863739SMike Smith {
784914da7d0SScott Long 	struct aac_softc *sc;
78535863739SMike Smith 
786914da7d0SScott Long 	sc = device_get_softc(dev);
787914da7d0SScott Long 
78831a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
78935863739SMike Smith 	sc->aac_state |= AAC_STATE_SUSPEND;
79035863739SMike Smith 
79135863739SMike Smith 	AAC_MASK_INTERRUPTS(sc);
79235863739SMike Smith 	return(0);
79335863739SMike Smith }
79435863739SMike Smith 
795914da7d0SScott Long /*
79635863739SMike Smith  * Bring the controller back to a state ready for operation.
79735863739SMike Smith  */
79835863739SMike Smith int
aac_resume(device_t dev)79935863739SMike Smith aac_resume(device_t dev)
80035863739SMike Smith {
801914da7d0SScott Long 	struct aac_softc *sc;
80235863739SMike Smith 
803914da7d0SScott Long 	sc = device_get_softc(dev);
804914da7d0SScott Long 
80531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
80635863739SMike Smith 	sc->aac_state &= ~AAC_STATE_SUSPEND;
80735863739SMike Smith 	AAC_UNMASK_INTERRUPTS(sc);
80835863739SMike Smith 	return(0);
80935863739SMike Smith }
81035863739SMike Smith 
811914da7d0SScott Long /*
8127cb209f5SScott Long  * Interrupt handler for NEW_COMM interface.
81335863739SMike Smith  */
81435863739SMike Smith void
aac_new_intr(void * arg)8157cb209f5SScott Long aac_new_intr(void *arg)
8167cb209f5SScott Long {
8177cb209f5SScott Long 	struct aac_softc *sc;
8187cb209f5SScott Long 	u_int32_t index, fast;
8197cb209f5SScott Long 	struct aac_command *cm;
8207cb209f5SScott Long 	struct aac_fib *fib;
8217cb209f5SScott Long 	int i;
8227cb209f5SScott Long 
8237cb209f5SScott Long 	sc = (struct aac_softc *)arg;
8247cb209f5SScott Long 
82531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
8267cb209f5SScott Long 	mtx_lock(&sc->aac_io_lock);
8277cb209f5SScott Long 	while (1) {
8287cb209f5SScott Long 		index = AAC_GET_OUTB_QUEUE(sc);
8297cb209f5SScott Long 		if (index == 0xffffffff)
8307cb209f5SScott Long 			index = AAC_GET_OUTB_QUEUE(sc);
8317cb209f5SScott Long 		if (index == 0xffffffff)
8327cb209f5SScott Long 			break;
8337cb209f5SScott Long 		if (index & 2) {
8347cb209f5SScott Long 			if (index == 0xfffffffe) {
8357cb209f5SScott Long 				/* XXX This means that the controller wants
8367cb209f5SScott Long 				 * more work.  Ignore it for now.
8377cb209f5SScott Long 				 */
8387cb209f5SScott Long 				continue;
8397cb209f5SScott Long 			}
8407cb209f5SScott Long 			/* AIF */
8417cb209f5SScott Long 			fib = (struct aac_fib *)malloc(sizeof *fib, M_AACBUF,
8427cb209f5SScott Long 				   M_NOWAIT | M_ZERO);
8437cb209f5SScott Long 			if (fib == NULL) {
8447cb209f5SScott Long 				/* If we're really this short on memory,
8457cb209f5SScott Long 				 * hopefully breaking out of the handler will
8467cb209f5SScott Long 				 * allow something to get freed.  This
8477cb209f5SScott Long 				 * actually sucks a whole lot.
8487cb209f5SScott Long 				 */
8497cb209f5SScott Long 				break;
8507cb209f5SScott Long 			}
8517cb209f5SScott Long 			index &= ~2;
8527cb209f5SScott Long 			for (i = 0; i < sizeof(struct aac_fib)/4; ++i)
853ff0991c4SAttilio Rao 				((u_int32_t *)fib)[i] = AAC_MEM1_GETREG4(sc, index + i*4);
8547cb209f5SScott Long 			aac_handle_aif(sc, fib);
8557cb209f5SScott Long 			free(fib, M_AACBUF);
8567cb209f5SScott Long 
8577cb209f5SScott Long 			/*
8587cb209f5SScott Long 			 * AIF memory is owned by the adapter, so let it
8597cb209f5SScott Long 			 * know that we are done with it.
8607cb209f5SScott Long 			 */
8617cb209f5SScott Long 			AAC_SET_OUTB_QUEUE(sc, index);
8627cb209f5SScott Long 			AAC_CLEAR_ISTATUS(sc, AAC_DB_RESPONSE_READY);
8637cb209f5SScott Long 		} else {
8647cb209f5SScott Long 			fast = index & 1;
8657cb209f5SScott Long 			cm = sc->aac_commands + (index >> 2);
8667cb209f5SScott Long 			fib = cm->cm_fib;
8677cb209f5SScott Long 			if (fast) {
8687cb209f5SScott Long 				fib->Header.XferState |= AAC_FIBSTATE_DONEADAP;
8697cb209f5SScott Long 				*((u_int32_t *)(fib->data)) = AAC_ERROR_NORMAL;
8707cb209f5SScott Long 			}
8717cb209f5SScott Long 			aac_remove_busy(cm);
8727cb209f5SScott Long  			aac_unmap_command(cm);
8737cb209f5SScott Long 			cm->cm_flags |= AAC_CMD_COMPLETED;
8747cb209f5SScott Long 
8757cb209f5SScott Long 			/* is there a completion handler? */
8767cb209f5SScott Long 			if (cm->cm_complete != NULL) {
8777cb209f5SScott Long 				cm->cm_complete(cm);
8787cb209f5SScott Long 			} else {
8797cb209f5SScott Long 				/* assume that someone is sleeping on this
8807cb209f5SScott Long 				 * command
8817cb209f5SScott Long 				 */
8827cb209f5SScott Long 				wakeup(cm);
8837cb209f5SScott Long 			}
8847cb209f5SScott Long 			sc->flags &= ~AAC_QUEUE_FRZN;
8857cb209f5SScott Long 		}
8867cb209f5SScott Long 	}
8877cb209f5SScott Long 	/* see if we can start some more I/O */
8887cb209f5SScott Long 	if ((sc->flags & AAC_QUEUE_FRZN) == 0)
8897cb209f5SScott Long 		aac_startio(sc);
8907cb209f5SScott Long 
8917cb209f5SScott Long 	mtx_unlock(&sc->aac_io_lock);
8927cb209f5SScott Long }
8937cb209f5SScott Long 
894e46b9eeaSEd Maste /*
895e46b9eeaSEd Maste  * Interrupt filter for !NEW_COMM interface.
896e46b9eeaSEd Maste  */
897ef544f63SPaolo Pisati int
aac_filter(void * arg)898e46b9eeaSEd Maste aac_filter(void *arg)
89935863739SMike Smith {
900914da7d0SScott Long 	struct aac_softc *sc;
90170545d1aSScott Long 	u_int16_t reason;
90235863739SMike Smith 
903914da7d0SScott Long 	sc = (struct aac_softc *)arg;
904914da7d0SScott Long 
90531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
906f30ac74cSScott Long 	/*
9079148fa21SScott Long 	 * Read the status register directly.  This is faster than taking the
9089148fa21SScott Long 	 * driver lock and reading the queues directly.  It also saves having
9099148fa21SScott Long 	 * to turn parts of the driver lock into a spin mutex, which would be
9109148fa21SScott Long 	 * ugly.
911f30ac74cSScott Long 	 */
91235863739SMike Smith 	reason = AAC_GET_ISTATUS(sc);
913f30ac74cSScott Long 	AAC_CLEAR_ISTATUS(sc, reason);
914f30ac74cSScott Long 
9159c3a7fceSScott Long 	/* handle completion processing */
9169148fa21SScott Long 	if (reason & AAC_DB_RESPONSE_READY)
917cbc4d2dbSJohn Baldwin 		taskqueue_enqueue(taskqueue_fast, &sc->aac_task_complete);
91835863739SMike Smith 
9199148fa21SScott Long 	/* controller wants to talk to us */
9209148fa21SScott Long 	if (reason & (AAC_DB_PRINTF | AAC_DB_COMMAND_READY)) {
92170545d1aSScott Long 		/*
9229148fa21SScott Long 		 * XXX Make sure that we don't get fooled by strange messages
9239148fa21SScott Long 		 * that start with a NULL.
92470545d1aSScott Long 		 */
9259148fa21SScott Long 		if ((reason & AAC_DB_PRINTF) &&
9269148fa21SScott Long 			(sc->aac_common->ac_printf[0] == 0))
9279148fa21SScott Long 			sc->aac_common->ac_printf[0] = 32;
92870545d1aSScott Long 
9299148fa21SScott Long 		/*
9309148fa21SScott Long 		 * This might miss doing the actual wakeup.  However, the
931a32a982dSScott Long 		 * msleep that this is waking up has a timeout, so it will
9329148fa21SScott Long 		 * wake up eventually.  AIFs and printfs are low enough
9339148fa21SScott Long 		 * priority that they can handle hanging out for a few seconds
9349148fa21SScott Long 		 * if needed.
9359148fa21SScott Long 		 */
93636e0bf6eSScott Long 		wakeup(sc->aifthread);
93736e0bf6eSScott Long 	}
938ef544f63SPaolo Pisati 	return (FILTER_HANDLED);
9399148fa21SScott Long }
94035863739SMike Smith 
941c6eafcf2SScott Long /*
942914da7d0SScott Long  * Command Processing
943914da7d0SScott Long  */
94435863739SMike Smith 
945914da7d0SScott Long /*
94635863739SMike Smith  * Start as much queued I/O as possible on the controller
94735863739SMike Smith  */
948fe3cb0e1SScott Long void
aac_startio(struct aac_softc * sc)94935863739SMike Smith aac_startio(struct aac_softc *sc)
95035863739SMike Smith {
95135863739SMike Smith 	struct aac_command *cm;
952397fa34fSScott Long 	int error;
95335863739SMike Smith 
95431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
95535863739SMike Smith 
95635863739SMike Smith 	for (;;) {
957914da7d0SScott Long 		/*
958397fa34fSScott Long 		 * This flag might be set if the card is out of resources.
959397fa34fSScott Long 		 * Checking it here prevents an infinite loop of deferrals.
960397fa34fSScott Long 		 */
961397fa34fSScott Long 		if (sc->flags & AAC_QUEUE_FRZN)
962397fa34fSScott Long 			break;
963397fa34fSScott Long 
964397fa34fSScott Long 		/*
965914da7d0SScott Long 		 * Try to get a command that's been put off for lack of
966914da7d0SScott Long 		 * resources
967914da7d0SScott Long 		 */
96835863739SMike Smith 		cm = aac_dequeue_ready(sc);
96935863739SMike Smith 
970914da7d0SScott Long 		/*
971914da7d0SScott Long 		 * Try to build a command off the bio queue (ignore error
972914da7d0SScott Long 		 * return)
973914da7d0SScott Long 		 */
9740b94a66eSMike Smith 		if (cm == NULL)
97535863739SMike Smith 			aac_bio_command(sc, &cm);
97635863739SMike Smith 
97735863739SMike Smith 		/* nothing to do? */
97835863739SMike Smith 		if (cm == NULL)
97935863739SMike Smith 			break;
98035863739SMike Smith 
981cd481291SScott Long 		/* don't map more than once */
982cd481291SScott Long 		if (cm->cm_flags & AAC_CMD_MAPPED)
9834102d44bSScott Long 			panic("aac: command %p already mapped", cm);
98435863739SMike Smith 
985397fa34fSScott Long 		/*
986397fa34fSScott Long 		 * Set up the command to go to the controller.  If there are no
987397fa34fSScott Long 		 * data buffers associated with the command then it can bypass
988397fa34fSScott Long 		 * busdma.
989397fa34fSScott Long 		 */
990cd481291SScott Long 		if (cm->cm_datalen != 0) {
9918fce673cSMarius Strobl 			if (cm->cm_flags & AAC_REQ_BIO)
9928fce673cSMarius Strobl 				error = bus_dmamap_load_bio(
9938fce673cSMarius Strobl 				    sc->aac_buffer_dmat, cm->cm_datamap,
9948fce673cSMarius Strobl 				    (struct bio *)cm->cm_private,
9958fce673cSMarius Strobl 				    aac_map_command_sg, cm, 0);
9968fce673cSMarius Strobl 			else
997397fa34fSScott Long 				error = bus_dmamap_load(sc->aac_buffer_dmat,
998397fa34fSScott Long 				    cm->cm_datamap, cm->cm_data,
9998fce673cSMarius Strobl 				    cm->cm_datalen, aac_map_command_sg, cm, 0);
1000cd481291SScott Long 			if (error == EINPROGRESS) {
100131a0399eSEd Maste 				fwprintf(sc, HBA_FLAGS_DBG_COMM_B, "freezing queue\n");
1002cd481291SScott Long 				sc->flags |= AAC_QUEUE_FRZN;
1003614c22b2SScott Long 			} else if (error != 0)
1004397fa34fSScott Long 				panic("aac_startio: unexpected error %d from "
1005a620bad0SEd Maste 				      "busdma", error);
1006397fa34fSScott Long 		} else
10078778f63dSScott Long 			aac_map_command_sg(cm, NULL, 0, 0);
1008cd481291SScott Long 	}
100935863739SMike Smith }
101035863739SMike Smith 
1011914da7d0SScott Long /*
101235863739SMike Smith  * Handle notification of one or more FIBs coming from the controller.
101335863739SMike Smith  */
101435863739SMike Smith static void
aac_command_thread(struct aac_softc * sc)101570545d1aSScott Long aac_command_thread(struct aac_softc *sc)
101635863739SMike Smith {
101735863739SMike Smith 	struct aac_fib *fib;
101835863739SMike Smith 	u_int32_t fib_size;
10199148fa21SScott Long 	int size, retval;
102035863739SMike Smith 
102131a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
102235863739SMike Smith 
1023bb6fe253SScott Long 	mtx_lock(&sc->aac_io_lock);
1024a32a982dSScott Long 	sc->aifflags = AAC_AIFFLAGS_RUNNING;
102536e0bf6eSScott Long 
1026a32a982dSScott Long 	while ((sc->aifflags & AAC_AIFFLAGS_EXIT) == 0) {
1027a32a982dSScott Long 		retval = 0;
1028a32a982dSScott Long 		if ((sc->aifflags & AAC_AIFFLAGS_PENDING) == 0)
1029a32a982dSScott Long 			retval = msleep(sc->aifthread, &sc->aac_io_lock, PRIBIO,
1030a32a982dSScott Long 					"aifthd", AAC_PERIODIC_INTERVAL * hz);
103136e0bf6eSScott Long 
10329148fa21SScott Long 		/*
10339148fa21SScott Long 		 * First see if any FIBs need to be allocated.  This needs
10349148fa21SScott Long 		 * to be called without the driver lock because contigmalloc
10351bd320ecSAttilio Rao 		 * can sleep.
10369148fa21SScott Long 		 */
10379148fa21SScott Long 		if ((sc->aifflags & AAC_AIFFLAGS_ALLOCFIBS) != 0) {
1038bb6fe253SScott Long 			mtx_unlock(&sc->aac_io_lock);
1039a32a982dSScott Long 			aac_alloc_commands(sc);
1040bb6fe253SScott Long 			mtx_lock(&sc->aac_io_lock);
10414102d44bSScott Long 			sc->aifflags &= ~AAC_AIFFLAGS_ALLOCFIBS;
1042a32a982dSScott Long 			aac_startio(sc);
1043a32a982dSScott Long 		}
10449148fa21SScott Long 
10459148fa21SScott Long 		/*
10469148fa21SScott Long 		 * While we're here, check to see if any commands are stuck.
10479148fa21SScott Long 		 * This is pretty low-priority, so it's ok if it doesn't
10489148fa21SScott Long 		 * always fire.
10499148fa21SScott Long 		 */
10509148fa21SScott Long 		if (retval == EWOULDBLOCK)
105170545d1aSScott Long 			aac_timeout(sc);
105270545d1aSScott Long 
105370545d1aSScott Long 		/* Check the hardware printf message buffer */
10549148fa21SScott Long 		if (sc->aac_common->ac_printf[0] != 0)
105570545d1aSScott Long 			aac_print_printf(sc);
105670545d1aSScott Long 
10579148fa21SScott Long 		/* Also check to see if the adapter has a command for us. */
10587cb209f5SScott Long 		if (sc->flags & AAC_FLAGS_NEW_COMM)
10597cb209f5SScott Long 			continue;
10607cb209f5SScott Long 		for (;;) {
10617cb209f5SScott Long 			if (aac_dequeue_fib(sc, AAC_HOST_NORM_CMD_QUEUE,
10627cb209f5SScott Long 					   &fib_size, &fib))
10637cb209f5SScott Long 				break;
106435863739SMike Smith 
106536e0bf6eSScott Long 			AAC_PRINT_FIB(sc, fib);
106636e0bf6eSScott Long 
106735863739SMike Smith 			switch (fib->Header.Command) {
106835863739SMike Smith 			case AifRequest:
106936e0bf6eSScott Long 				aac_handle_aif(sc, fib);
107035863739SMike Smith 				break;
107135863739SMike Smith 			default:
1072914da7d0SScott Long 				device_printf(sc->aac_dev, "unknown command "
1073914da7d0SScott Long 					      "from controller\n");
107435863739SMike Smith 				break;
107535863739SMike Smith 			}
107635863739SMike Smith 
107736e0bf6eSScott Long 			if ((fib->Header.XferState == 0) ||
10787cb209f5SScott Long 			    (fib->Header.StructType != AAC_FIBTYPE_TFIB)) {
107936e0bf6eSScott Long 				break;
10807cb209f5SScott Long 			}
108136e0bf6eSScott Long 
108270545d1aSScott Long 			/* Return the AIF to the controller. */
108336e0bf6eSScott Long 			if (fib->Header.XferState & AAC_FIBSTATE_FROMADAP) {
108436e0bf6eSScott Long 				fib->Header.XferState |= AAC_FIBSTATE_DONEHOST;
108536e0bf6eSScott Long 				*(AAC_FSAStatus*)fib->data = ST_OK;
108636e0bf6eSScott Long 
108736e0bf6eSScott Long 				/* XXX Compute the Size field? */
108836e0bf6eSScott Long 				size = fib->Header.Size;
108936e0bf6eSScott Long 				if (size > sizeof(struct aac_fib)) {
109036e0bf6eSScott Long 					size = sizeof(struct aac_fib);
109136e0bf6eSScott Long 					fib->Header.Size = size;
109236e0bf6eSScott Long 				}
109336e0bf6eSScott Long 				/*
1094914da7d0SScott Long 				 * Since we did not generate this command, it
1095914da7d0SScott Long 				 * cannot go through the normal
1096914da7d0SScott Long 				 * enqueue->startio chain.
109736e0bf6eSScott Long 				 */
1098914da7d0SScott Long 				aac_enqueue_response(sc,
1099914da7d0SScott Long 						 AAC_ADAP_NORM_RESP_QUEUE,
1100914da7d0SScott Long 						 fib);
110136e0bf6eSScott Long 			}
110236e0bf6eSScott Long 		}
110336e0bf6eSScott Long 	}
110436e0bf6eSScott Long 	sc->aifflags &= ~AAC_AIFFLAGS_RUNNING;
1105bb6fe253SScott Long 	mtx_unlock(&sc->aac_io_lock);
110636e0bf6eSScott Long 	wakeup(sc->aac_dev);
110736e0bf6eSScott Long 
11083745c395SJulian Elischer 	kproc_exit(0);
110935863739SMike Smith }
111035863739SMike Smith 
1111914da7d0SScott Long /*
11129c3a7fceSScott Long  * Process completed commands.
111335863739SMike Smith  */
111435863739SMike Smith static void
aac_complete(void * context,int pending)11159c3a7fceSScott Long aac_complete(void *context, int pending)
111635863739SMike Smith {
11179c3a7fceSScott Long 	struct aac_softc *sc;
111835863739SMike Smith 	struct aac_command *cm;
111935863739SMike Smith 	struct aac_fib *fib;
112035863739SMike Smith 	u_int32_t fib_size;
112135863739SMike Smith 
11229c3a7fceSScott Long 	sc = (struct aac_softc *)context;
112331a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
11249c3a7fceSScott Long 
1125bb6fe253SScott Long 	mtx_lock(&sc->aac_io_lock);
1126ae543596SScott Long 
11279c3a7fceSScott Long 	/* pull completed commands off the queue */
112835863739SMike Smith 	for (;;) {
112935863739SMike Smith 		/* look for completed FIBs on our queue */
1130914da7d0SScott Long 		if (aac_dequeue_fib(sc, AAC_HOST_NORM_RESP_QUEUE, &fib_size,
1131914da7d0SScott Long 							&fib))
113235863739SMike Smith 			break;	/* nothing to do */
113335863739SMike Smith 
1134ecd1c51fSScott Long 		/* get the command, unmap and hand off for processing */
1135cb0d64b9SScott Long 		cm = sc->aac_commands + fib->Header.SenderData;
113635863739SMike Smith 		if (cm == NULL) {
113735863739SMike Smith 			AAC_PRINT_FIB(sc, fib);
11389c3a7fceSScott Long 			break;
11399c3a7fceSScott Long 		}
11403e507710SEd Maste 		if ((cm->cm_flags & AAC_CMD_TIMEDOUT) != 0)
11413e507710SEd Maste 			device_printf(sc->aac_dev,
11423e507710SEd Maste 			    "COMMAND %p COMPLETED AFTER %d SECONDS\n",
11433e507710SEd Maste 			    cm, (int)(time_uptime-cm->cm_timestamp));
11443e507710SEd Maste 
11450b94a66eSMike Smith 		aac_remove_busy(cm);
11467cb209f5SScott Long 
1147ecd1c51fSScott Long  		aac_unmap_command(cm);
114835863739SMike Smith 		cm->cm_flags |= AAC_CMD_COMPLETED;
114935863739SMike Smith 
115035863739SMike Smith 		/* is there a completion handler? */
115135863739SMike Smith 		if (cm->cm_complete != NULL) {
115235863739SMike Smith 			cm->cm_complete(cm);
115335863739SMike Smith 		} else {
115435863739SMike Smith 			/* assume that someone is sleeping on this command */
115535863739SMike Smith 			wakeup(cm);
115635863739SMike Smith 		}
115735863739SMike Smith 	}
11580b94a66eSMike Smith 
11590b94a66eSMike Smith 	/* see if we can start some more I/O */
1160cd481291SScott Long 	sc->flags &= ~AAC_QUEUE_FRZN;
11610b94a66eSMike Smith 	aac_startio(sc);
1162ae543596SScott Long 
1163bb6fe253SScott Long 	mtx_unlock(&sc->aac_io_lock);
116435863739SMike Smith }
116535863739SMike Smith 
1166914da7d0SScott Long /*
116735863739SMike Smith  * Handle a bio submitted from a disk device.
116835863739SMike Smith  */
116935863739SMike Smith void
aac_submit_bio(struct bio * bp)117035863739SMike Smith aac_submit_bio(struct bio *bp)
117135863739SMike Smith {
1172914da7d0SScott Long 	struct aac_disk *ad;
1173914da7d0SScott Long 	struct aac_softc *sc;
117435863739SMike Smith 
11757540e65eSScott Long 	ad = (struct aac_disk *)bp->bio_disk->d_drv1;
1176914da7d0SScott Long 	sc = ad->ad_controller;
117731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
1178914da7d0SScott Long 
117935863739SMike Smith 	/* queue the BIO and try to get some work done */
11800b94a66eSMike Smith 	aac_enqueue_bio(sc, bp);
118135863739SMike Smith 	aac_startio(sc);
118235863739SMike Smith }
118335863739SMike Smith 
1184914da7d0SScott Long /*
118535863739SMike Smith  * Get a bio and build a command to go with it.
118635863739SMike Smith  */
118735863739SMike Smith static int
aac_bio_command(struct aac_softc * sc,struct aac_command ** cmp)118835863739SMike Smith aac_bio_command(struct aac_softc *sc, struct aac_command **cmp)
118935863739SMike Smith {
119035863739SMike Smith 	struct aac_command *cm;
119135863739SMike Smith 	struct aac_fib *fib;
119235863739SMike Smith 	struct aac_disk *ad;
119335863739SMike Smith 	struct bio *bp;
119435863739SMike Smith 
119531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
119635863739SMike Smith 
119735863739SMike Smith 	/* get the resources we will need */
119835863739SMike Smith 	cm = NULL;
1199a32a982dSScott Long 	bp = NULL;
120035863739SMike Smith 	if (aac_alloc_command(sc, &cm))	/* get a command */
120135863739SMike Smith 		goto fail;
1202a32a982dSScott Long 	if ((bp = aac_dequeue_bio(sc)) == NULL)
1203a32a982dSScott Long 		goto fail;
120435863739SMike Smith 
120535863739SMike Smith 	/* fill out the command */
12060b94a66eSMike Smith 	cm->cm_datalen = bp->bio_bcount;
12070b94a66eSMike Smith 	cm->cm_complete = aac_bio_complete;
12088fce673cSMarius Strobl 	cm->cm_flags = AAC_REQ_BIO;
120935863739SMike Smith 	cm->cm_private = bp;
12102b3b0f17SScott Long 	cm->cm_timestamp = time_uptime;
121135863739SMike Smith 
121235863739SMike Smith 	/* build the FIB */
121335863739SMike Smith 	fib = cm->cm_fib;
1214b85f5808SScott Long 	fib->Header.Size = sizeof(struct aac_fib_header);
121535863739SMike Smith 	fib->Header.XferState =
121635863739SMike Smith 		AAC_FIBSTATE_HOSTOWNED   |
121735863739SMike Smith 		AAC_FIBSTATE_INITIALISED |
1218f30ac74cSScott Long 		AAC_FIBSTATE_EMPTY	 |
121935863739SMike Smith 		AAC_FIBSTATE_FROMHOST	 |
122035863739SMike Smith 		AAC_FIBSTATE_REXPECTED   |
1221f30ac74cSScott Long 		AAC_FIBSTATE_NORM	 |
1222f30ac74cSScott Long 		AAC_FIBSTATE_ASYNC	 |
1223f30ac74cSScott Long 		AAC_FIBSTATE_FAST_RESPONSE;
122435863739SMike Smith 
122535863739SMike Smith 	/* build the read/write request */
12267540e65eSScott Long 	ad = (struct aac_disk *)bp->bio_disk->d_drv1;
1227b85f5808SScott Long 
12287cb209f5SScott Long 	if (sc->flags & AAC_FLAGS_RAW_IO) {
12297cb209f5SScott Long 		struct aac_raw_io *raw;
12307cb209f5SScott Long 		raw = (struct aac_raw_io *)&fib->data[0];
12317cb209f5SScott Long 		fib->Header.Command = RawIo;
12327cb209f5SScott Long 		raw->BlockNumber = (u_int64_t)bp->bio_pblkno;
12337cb209f5SScott Long 		raw->ByteCount = bp->bio_bcount;
12347cb209f5SScott Long 		raw->ContainerId = ad->ad_container->co_mntobj.ObjectId;
12357cb209f5SScott Long 		raw->BpTotal = 0;
12367cb209f5SScott Long 		raw->BpComplete = 0;
12377cb209f5SScott Long 		fib->Header.Size += sizeof(struct aac_raw_io);
12387cb209f5SScott Long 		cm->cm_sgtable = (struct aac_sg_table *)&raw->SgMapRaw;
12397cb209f5SScott Long 		if (bp->bio_cmd == BIO_READ) {
12407cb209f5SScott Long 			raw->Flags = 1;
12417cb209f5SScott Long 			cm->cm_flags |= AAC_CMD_DATAIN;
12427cb209f5SScott Long 		} else {
12437cb209f5SScott Long 			raw->Flags = 0;
12447cb209f5SScott Long 			cm->cm_flags |= AAC_CMD_DATAOUT;
12457cb209f5SScott Long 		}
12467cb209f5SScott Long 	} else if ((sc->flags & AAC_FLAGS_SG_64BIT) == 0) {
1247b85f5808SScott Long 		fib->Header.Command = ContainerCommand;
12489e2e96d8SScott Long 		if (bp->bio_cmd == BIO_READ) {
1249b85f5808SScott Long 			struct aac_blockread *br;
125035863739SMike Smith 			br = (struct aac_blockread *)&fib->data[0];
125135863739SMike Smith 			br->Command = VM_CtBlockRead;
125235863739SMike Smith 			br->ContainerId = ad->ad_container->co_mntobj.ObjectId;
125335863739SMike Smith 			br->BlockNumber = bp->bio_pblkno;
125435863739SMike Smith 			br->ByteCount = bp->bio_bcount;
125535863739SMike Smith 			fib->Header.Size += sizeof(struct aac_blockread);
125635863739SMike Smith 			cm->cm_sgtable = &br->SgMap;
125735863739SMike Smith 			cm->cm_flags |= AAC_CMD_DATAIN;
125835863739SMike Smith 		} else {
1259b85f5808SScott Long 			struct aac_blockwrite *bw;
126035863739SMike Smith 			bw = (struct aac_blockwrite *)&fib->data[0];
126135863739SMike Smith 			bw->Command = VM_CtBlockWrite;
126235863739SMike Smith 			bw->ContainerId = ad->ad_container->co_mntobj.ObjectId;
126335863739SMike Smith 			bw->BlockNumber = bp->bio_pblkno;
126435863739SMike Smith 			bw->ByteCount = bp->bio_bcount;
1265b85f5808SScott Long 			bw->Stable = CUNSTABLE;
126635863739SMike Smith 			fib->Header.Size += sizeof(struct aac_blockwrite);
126735863739SMike Smith 			cm->cm_flags |= AAC_CMD_DATAOUT;
126835863739SMike Smith 			cm->cm_sgtable = &bw->SgMap;
126935863739SMike Smith 		}
1270b85f5808SScott Long 	} else {
1271b85f5808SScott Long 		fib->Header.Command = ContainerCommand64;
1272b85f5808SScott Long 		if (bp->bio_cmd == BIO_READ) {
1273b85f5808SScott Long 			struct aac_blockread64 *br;
1274b85f5808SScott Long 			br = (struct aac_blockread64 *)&fib->data[0];
1275b85f5808SScott Long 			br->Command = VM_CtHostRead64;
1276b85f5808SScott Long 			br->ContainerId = ad->ad_container->co_mntobj.ObjectId;
1277b85f5808SScott Long 			br->SectorCount = bp->bio_bcount / AAC_BLOCK_SIZE;
1278b85f5808SScott Long 			br->BlockNumber = bp->bio_pblkno;
1279b85f5808SScott Long 			br->Pad = 0;
1280b85f5808SScott Long 			br->Flags = 0;
1281b85f5808SScott Long 			fib->Header.Size += sizeof(struct aac_blockread64);
128254e2ebdfSEd Maste 			cm->cm_flags |= AAC_CMD_DATAIN;
1283eec256deSAlexander Kabaev 			cm->cm_sgtable = (struct aac_sg_table *)&br->SgMap64;
1284b85f5808SScott Long 		} else {
1285b85f5808SScott Long 			struct aac_blockwrite64 *bw;
1286b85f5808SScott Long 			bw = (struct aac_blockwrite64 *)&fib->data[0];
1287b85f5808SScott Long 			bw->Command = VM_CtHostWrite64;
1288b85f5808SScott Long 			bw->ContainerId = ad->ad_container->co_mntobj.ObjectId;
1289b85f5808SScott Long 			bw->SectorCount = bp->bio_bcount / AAC_BLOCK_SIZE;
1290b85f5808SScott Long 			bw->BlockNumber = bp->bio_pblkno;
1291b85f5808SScott Long 			bw->Pad = 0;
1292b85f5808SScott Long 			bw->Flags = 0;
1293b85f5808SScott Long 			fib->Header.Size += sizeof(struct aac_blockwrite64);
129454e2ebdfSEd Maste 			cm->cm_flags |= AAC_CMD_DATAOUT;
1295eec256deSAlexander Kabaev 			cm->cm_sgtable = (struct aac_sg_table *)&bw->SgMap64;
1296b85f5808SScott Long 		}
1297b85f5808SScott Long 	}
129835863739SMike Smith 
129935863739SMike Smith 	*cmp = cm;
130035863739SMike Smith 	return(0);
130135863739SMike Smith 
130235863739SMike Smith fail:
13037cb209f5SScott Long 	if (bp != NULL)
13047cb209f5SScott Long 		aac_enqueue_bio(sc, bp);
130535863739SMike Smith 	if (cm != NULL)
130635863739SMike Smith 		aac_release_command(cm);
130735863739SMike Smith 	return(ENOMEM);
130835863739SMike Smith }
130935863739SMike Smith 
1310914da7d0SScott Long /*
131135863739SMike Smith  * Handle a bio-instigated command that has been completed.
131235863739SMike Smith  */
131335863739SMike Smith static void
aac_bio_complete(struct aac_command * cm)131435863739SMike Smith aac_bio_complete(struct aac_command *cm)
131535863739SMike Smith {
131635863739SMike Smith 	struct aac_blockread_response *brr;
131735863739SMike Smith 	struct aac_blockwrite_response *bwr;
131835863739SMike Smith 	struct bio *bp;
131935863739SMike Smith 	AAC_FSAStatus status;
132035863739SMike Smith 
132135863739SMike Smith 	/* fetch relevant status and then release the command */
132235863739SMike Smith 	bp = (struct bio *)cm->cm_private;
13239e2e96d8SScott Long 	if (bp->bio_cmd == BIO_READ) {
132435863739SMike Smith 		brr = (struct aac_blockread_response *)&cm->cm_fib->data[0];
132535863739SMike Smith 		status = brr->Status;
132635863739SMike Smith 	} else {
132735863739SMike Smith 		bwr = (struct aac_blockwrite_response *)&cm->cm_fib->data[0];
132835863739SMike Smith 		status = bwr->Status;
132935863739SMike Smith 	}
133035863739SMike Smith 	aac_release_command(cm);
133135863739SMike Smith 
133235863739SMike Smith 	/* fix up the bio based on status */
133335863739SMike Smith 	if (status == ST_OK) {
133435863739SMike Smith 		bp->bio_resid = 0;
133535863739SMike Smith 	} else {
133635863739SMike Smith 		bp->bio_error = EIO;
133735863739SMike Smith 		bp->bio_flags |= BIO_ERROR;
133835863739SMike Smith 	}
13390b94a66eSMike Smith 	aac_biodone(bp);
134035863739SMike Smith }
134135863739SMike Smith 
1342914da7d0SScott Long /*
134335863739SMike Smith  * Submit a command to the controller, return when it completes.
1344b3457b51SScott Long  * XXX This is very dangerous!  If the card has gone out to lunch, we could
1345b3457b51SScott Long  *     be stuck here forever.  At the same time, signals are not caught
1346d8a0a473SScott Long  *     because there is a risk that a signal could wakeup the sleep before
1347d8a0a473SScott Long  *     the card has a chance to complete the command.  Since there is no way
1348d8a0a473SScott Long  *     to cancel a command that is in progress, we can't protect against the
1349d8a0a473SScott Long  *     card completing a command late and spamming the command and data
1350d8a0a473SScott Long  *     memory.  So, we are held hostage until the command completes.
135135863739SMike Smith  */
135235863739SMike Smith static int
aac_wait_command(struct aac_command * cm)1353d8a0a473SScott Long aac_wait_command(struct aac_command *cm)
135435863739SMike Smith {
1355ae543596SScott Long 	struct aac_softc *sc;
1356d8a0a473SScott Long 	int error;
135735863739SMike Smith 
1358ae543596SScott Long 	sc = cm->cm_sc;
135931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
1360ae543596SScott Long 
136135863739SMike Smith 	/* Put the command on the ready queue and get things going */
136235863739SMike Smith 	aac_enqueue_ready(cm);
1363ae543596SScott Long 	aac_startio(sc);
1364ae543596SScott Long 	error = msleep(cm, &sc->aac_io_lock, PRIBIO, "aacwait", 0);
136535863739SMike Smith 	return(error);
136635863739SMike Smith }
136735863739SMike Smith 
1368914da7d0SScott Long /*
1369914da7d0SScott Long  *Command Buffer Management
1370914da7d0SScott Long  */
137135863739SMike Smith 
1372914da7d0SScott Long /*
137335863739SMike Smith  * Allocate a command.
137435863739SMike Smith  */
1375fe3cb0e1SScott Long int
aac_alloc_command(struct aac_softc * sc,struct aac_command ** cmp)137635863739SMike Smith aac_alloc_command(struct aac_softc *sc, struct aac_command **cmp)
137735863739SMike Smith {
137835863739SMike Smith 	struct aac_command *cm;
137935863739SMike Smith 
138031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
138135863739SMike Smith 
1382ffb37f33SScott Long 	if ((cm = aac_dequeue_free(sc)) == NULL) {
1383b85f5808SScott Long 		if (sc->total_fibs < sc->aac_max_fibs) {
13841bd320ecSAttilio Rao 			mtx_lock(&sc->aac_io_lock);
1385ae543596SScott Long 			sc->aifflags |= AAC_AIFFLAGS_ALLOCFIBS;
13861bd320ecSAttilio Rao 			mtx_unlock(&sc->aac_io_lock);
1387ae543596SScott Long 			wakeup(sc->aifthread);
1388b85f5808SScott Long 		}
1389ae543596SScott Long 		return (EBUSY);
1390ffb37f33SScott Long 	}
139135863739SMike Smith 
13920b94a66eSMike Smith 	*cmp = cm;
13930b94a66eSMike Smith 	return(0);
13940b94a66eSMike Smith }
13950b94a66eSMike Smith 
1396914da7d0SScott Long /*
13970b94a66eSMike Smith  * Release a command back to the freelist.
13980b94a66eSMike Smith  */
1399fe3cb0e1SScott Long void
aac_release_command(struct aac_command * cm)14000b94a66eSMike Smith aac_release_command(struct aac_command *cm)
14010b94a66eSMike Smith {
14027cb209f5SScott Long 	struct aac_event *event;
14037cb209f5SScott Long 	struct aac_softc *sc;
14047cb209f5SScott Long 
140531a0399eSEd Maste 	sc = cm->cm_sc;
140631a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
14070b94a66eSMike Smith 
14084109ba51SEd Maste 	/* (re)initialize the command/FIB */
1409b2b39b04SJohn Baldwin 	cm->cm_datalen = 0;
141035863739SMike Smith 	cm->cm_sgtable = NULL;
141135863739SMike Smith 	cm->cm_flags = 0;
141235863739SMike Smith 	cm->cm_complete = NULL;
141335863739SMike Smith 	cm->cm_private = NULL;
1414dbfc5960SEd Maste 	cm->cm_queue = AAC_ADAP_NORM_CMD_QUEUE;
141535863739SMike Smith 	cm->cm_fib->Header.XferState = AAC_FIBSTATE_EMPTY;
141635863739SMike Smith 	cm->cm_fib->Header.StructType = AAC_FIBTYPE_TFIB;
141735863739SMike Smith 	cm->cm_fib->Header.Flags = 0;
14187cb209f5SScott Long 	cm->cm_fib->Header.SenderSize = cm->cm_sc->aac_max_fib_size;
141935863739SMike Smith 
142035863739SMike Smith 	/*
142135863739SMike Smith 	 * These are duplicated in aac_start to cover the case where an
142235863739SMike Smith 	 * intermediate stage may have destroyed them.  They're left
14234109ba51SEd Maste 	 * initialized here for debugging purposes only.
142435863739SMike Smith 	 */
1425f30ac74cSScott Long 	cm->cm_fib->Header.ReceiverFibAddress = (u_int32_t)cm->cm_fibphys;
1426f30ac74cSScott Long 	cm->cm_fib->Header.SenderData = 0;
142735863739SMike Smith 
142835863739SMike Smith 	aac_enqueue_free(cm);
14297cb209f5SScott Long 
14302ad1c92dSEd Maste 	if ((event = TAILQ_FIRST(&sc->aac_ev_cmfree)) != NULL) {
14317cb209f5SScott Long 		TAILQ_REMOVE(&sc->aac_ev_cmfree, event, ev_links);
14327cb209f5SScott Long 		event->ev_callback(sc, event, event->ev_arg);
14337cb209f5SScott Long 	}
143435863739SMike Smith }
143535863739SMike Smith 
1436914da7d0SScott Long /*
14370b94a66eSMike Smith  * Map helper for command/FIB allocation.
143835863739SMike Smith  */
143935863739SMike Smith static void
aac_map_command_helper(void * arg,bus_dma_segment_t * segs,int nseg,int error)14400b94a66eSMike Smith aac_map_command_helper(void *arg, bus_dma_segment_t *segs, int nseg, int error)
144135863739SMike Smith {
14427cb209f5SScott Long 	uint64_t	*fibphys;
1443914da7d0SScott Long 
14447cb209f5SScott Long 	fibphys = (uint64_t *)arg;
144535863739SMike Smith 
1446ffb37f33SScott Long 	*fibphys = segs[0].ds_addr;
144735863739SMike Smith }
144835863739SMike Smith 
1449914da7d0SScott Long /*
14504109ba51SEd Maste  * Allocate and initialize commands/FIBs for this adapter.
145135863739SMike Smith  */
14520b94a66eSMike Smith static int
aac_alloc_commands(struct aac_softc * sc)14530b94a66eSMike Smith aac_alloc_commands(struct aac_softc *sc)
145435863739SMike Smith {
145535863739SMike Smith 	struct aac_command *cm;
1456ffb37f33SScott Long 	struct aac_fibmap *fm;
14577cb209f5SScott Long 	uint64_t fibphys;
1458ffb37f33SScott Long 	int i, error;
145935863739SMike Smith 
146031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
146135863739SMike Smith 
14627cb209f5SScott Long 	if (sc->total_fibs + sc->aac_max_fibs_alloc > sc->aac_max_fibs)
1463ffb37f33SScott Long 		return (ENOMEM);
1464ffb37f33SScott Long 
14658480cc63SScott Long 	fm = malloc(sizeof(struct aac_fibmap), M_AACBUF, M_NOWAIT|M_ZERO);
1466a6d35632SScott Long 	if (fm == NULL)
1467a6d35632SScott Long 		return (ENOMEM);
1468ffb37f33SScott Long 
14690b94a66eSMike Smith 	/* allocate the FIBs in DMAable memory and load them */
1470ffb37f33SScott Long 	if (bus_dmamem_alloc(sc->aac_fib_dmat, (void **)&fm->aac_fibs,
1471ffb37f33SScott Long 			     BUS_DMA_NOWAIT, &fm->aac_fibmap)) {
147270545d1aSScott Long 		device_printf(sc->aac_dev,
147370545d1aSScott Long 			      "Not enough contiguous memory available.\n");
14748480cc63SScott Long 		free(fm, M_AACBUF);
14750b94a66eSMike Smith 		return (ENOMEM);
147635863739SMike Smith 	}
1477128aa5a0SScott Long 
1478cd481291SScott Long 	/* Ignore errors since this doesn't bounce */
1479cd481291SScott Long 	(void)bus_dmamap_load(sc->aac_fib_dmat, fm->aac_fibmap, fm->aac_fibs,
14807cb209f5SScott Long 			      sc->aac_max_fibs_alloc * sc->aac_max_fib_size,
1481ffb37f33SScott Long 			      aac_map_command_helper, &fibphys, 0);
1482128aa5a0SScott Long 
14834109ba51SEd Maste 	/* initialize constant fields in the command structure */
14847cb209f5SScott Long 	bzero(fm->aac_fibs, sc->aac_max_fibs_alloc * sc->aac_max_fib_size);
14857cb209f5SScott Long 	for (i = 0; i < sc->aac_max_fibs_alloc; i++) {
14868480cc63SScott Long 		cm = sc->aac_commands + sc->total_fibs;
1487ffb37f33SScott Long 		fm->aac_commands = cm;
148835863739SMike Smith 		cm->cm_sc = sc;
14897cb209f5SScott Long 		cm->cm_fib = (struct aac_fib *)
14907cb209f5SScott Long 			((u_int8_t *)fm->aac_fibs + i*sc->aac_max_fib_size);
14917cb209f5SScott Long 		cm->cm_fibphys = fibphys + i*sc->aac_max_fib_size;
1492cb0d64b9SScott Long 		cm->cm_index = sc->total_fibs;
149335863739SMike Smith 
1494ffb37f33SScott Long 		if ((error = bus_dmamap_create(sc->aac_buffer_dmat, 0,
149593cfca22SScott Long 					       &cm->cm_datamap)) != 0)
14968480cc63SScott Long 			break;
149793cfca22SScott Long 		mtx_lock(&sc->aac_io_lock);
149893cfca22SScott Long 		aac_release_command(cm);
14998480cc63SScott Long 		sc->total_fibs++;
150093cfca22SScott Long 		mtx_unlock(&sc->aac_io_lock);
150135863739SMike Smith 	}
1502ffb37f33SScott Long 
15038480cc63SScott Long 	if (i > 0) {
150493cfca22SScott Long 		mtx_lock(&sc->aac_io_lock);
1505ffb37f33SScott Long 		TAILQ_INSERT_TAIL(&sc->aac_fibmap_tqh, fm, fm_link);
150631a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_COMM_B, "total_fibs= %d\n", sc->total_fibs);
1507bb6fe253SScott Long 		mtx_unlock(&sc->aac_io_lock);
15080b94a66eSMike Smith 		return (0);
150935863739SMike Smith 	}
151035863739SMike Smith 
15118480cc63SScott Long 	bus_dmamap_unload(sc->aac_fib_dmat, fm->aac_fibmap);
15128480cc63SScott Long 	bus_dmamem_free(sc->aac_fib_dmat, fm->aac_fibs, fm->aac_fibmap);
15138480cc63SScott Long 	free(fm, M_AACBUF);
15148480cc63SScott Long 	return (ENOMEM);
15158480cc63SScott Long }
15168480cc63SScott Long 
1517914da7d0SScott Long /*
15180b94a66eSMike Smith  * Free FIBs owned by this adapter.
151935863739SMike Smith  */
152035863739SMike Smith static void
aac_free_commands(struct aac_softc * sc)15218480cc63SScott Long aac_free_commands(struct aac_softc *sc)
152235863739SMike Smith {
15238480cc63SScott Long 	struct aac_fibmap *fm;
1524ffb37f33SScott Long 	struct aac_command *cm;
152535863739SMike Smith 	int i;
152635863739SMike Smith 
152731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
152835863739SMike Smith 
15298480cc63SScott Long 	while ((fm = TAILQ_FIRST(&sc->aac_fibmap_tqh)) != NULL) {
15308480cc63SScott Long 		TAILQ_REMOVE(&sc->aac_fibmap_tqh, fm, fm_link);
15318480cc63SScott Long 		/*
15328480cc63SScott Long 		 * We check against total_fibs to handle partially
15338480cc63SScott Long 		 * allocated blocks.
15348480cc63SScott Long 		 */
15357cb209f5SScott Long 		for (i = 0; i < sc->aac_max_fibs_alloc && sc->total_fibs--; i++) {
1536ffb37f33SScott Long 			cm = fm->aac_commands + i;
1537ffb37f33SScott Long 			bus_dmamap_destroy(sc->aac_buffer_dmat, cm->cm_datamap);
1538ffb37f33SScott Long 		}
1539ffb37f33SScott Long 		bus_dmamap_unload(sc->aac_fib_dmat, fm->aac_fibmap);
1540ffb37f33SScott Long 		bus_dmamem_free(sc->aac_fib_dmat, fm->aac_fibs, fm->aac_fibmap);
15418480cc63SScott Long 		free(fm, M_AACBUF);
15428480cc63SScott Long 	}
154335863739SMike Smith }
154435863739SMike Smith 
1545914da7d0SScott Long /*
154635863739SMike Smith  * Command-mapping helper function - populate this command's s/g table.
154735863739SMike Smith  */
154835863739SMike Smith static void
aac_map_command_sg(void * arg,bus_dma_segment_t * segs,int nseg,int error)154935863739SMike Smith aac_map_command_sg(void *arg, bus_dma_segment_t *segs, int nseg, int error)
155035863739SMike Smith {
1551cd481291SScott Long 	struct aac_softc *sc;
1552914da7d0SScott Long 	struct aac_command *cm;
1553914da7d0SScott Long 	struct aac_fib *fib;
155435863739SMike Smith 	int i;
155535863739SMike Smith 
1556914da7d0SScott Long 	cm = (struct aac_command *)arg;
1557cd481291SScott Long 	sc = cm->cm_sc;
1558914da7d0SScott Long 	fib = cm->cm_fib;
155931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
1560914da7d0SScott Long 
156135863739SMike Smith 	/* copy into the FIB */
1562b85f5808SScott Long 	if (cm->cm_sgtable != NULL) {
15637cb209f5SScott Long 		if (fib->Header.Command == RawIo) {
15647cb209f5SScott Long 			struct aac_sg_tableraw *sg;
15657cb209f5SScott Long 			sg = (struct aac_sg_tableraw *)cm->cm_sgtable;
15667cb209f5SScott Long 			sg->SgCount = nseg;
15677cb209f5SScott Long 			for (i = 0; i < nseg; i++) {
15687cb209f5SScott Long 				sg->SgEntryRaw[i].SgAddress = segs[i].ds_addr;
15697cb209f5SScott Long 				sg->SgEntryRaw[i].SgByteCount = segs[i].ds_len;
15707cb209f5SScott Long 				sg->SgEntryRaw[i].Next = 0;
15717cb209f5SScott Long 				sg->SgEntryRaw[i].Prev = 0;
15727cb209f5SScott Long 				sg->SgEntryRaw[i].Flags = 0;
15737cb209f5SScott Long 			}
15747cb209f5SScott Long 			/* update the FIB size for the s/g count */
15757cb209f5SScott Long 			fib->Header.Size += nseg*sizeof(struct aac_sg_entryraw);
15767cb209f5SScott Long 		} else if ((cm->cm_sc->flags & AAC_FLAGS_SG_64BIT) == 0) {
1577b85f5808SScott Long 			struct aac_sg_table *sg;
1578b85f5808SScott Long 			sg = cm->cm_sgtable;
157935863739SMike Smith 			sg->SgCount = nseg;
158035863739SMike Smith 			for (i = 0; i < nseg; i++) {
158135863739SMike Smith 				sg->SgEntry[i].SgAddress = segs[i].ds_addr;
158235863739SMike Smith 				sg->SgEntry[i].SgByteCount = segs[i].ds_len;
158335863739SMike Smith 			}
158435863739SMike Smith 			/* update the FIB size for the s/g count */
158535863739SMike Smith 			fib->Header.Size += nseg*sizeof(struct aac_sg_entry);
1586b85f5808SScott Long 		} else {
1587b85f5808SScott Long 			struct aac_sg_table64 *sg;
1588b85f5808SScott Long 			sg = (struct aac_sg_table64 *)cm->cm_sgtable;
1589b85f5808SScott Long 			sg->SgCount = nseg;
1590b85f5808SScott Long 			for (i = 0; i < nseg; i++) {
1591b85f5808SScott Long 				sg->SgEntry64[i].SgAddress = segs[i].ds_addr;
1592b85f5808SScott Long 				sg->SgEntry64[i].SgByteCount = segs[i].ds_len;
159335863739SMike Smith 			}
1594b85f5808SScott Long 			/* update the FIB size for the s/g count */
1595b85f5808SScott Long 			fib->Header.Size += nseg*sizeof(struct aac_sg_entry64);
1596b85f5808SScott Long 		}
1597b85f5808SScott Long 	}
159835863739SMike Smith 
1599cd481291SScott Long 	/* Fix up the address values in the FIB.  Use the command array index
1600cd481291SScott Long 	 * instead of a pointer since these fields are only 32 bits.  Shift
16017cb209f5SScott Long 	 * the SenderFibAddress over to make room for the fast response bit
16027cb209f5SScott Long 	 * and for the AIF bit
160335863739SMike Smith 	 */
16047cb209f5SScott Long 	cm->cm_fib->Header.SenderFibAddress = (cm->cm_index << 2);
16057cb209f5SScott Long 	cm->cm_fib->Header.ReceiverFibAddress = (u_int32_t)cm->cm_fibphys;
160635863739SMike Smith 
1607cd481291SScott Long 	/* save a pointer to the command for speedy reverse-lookup */
1608cd481291SScott Long 	cm->cm_fib->Header.SenderData = cm->cm_index;
160935863739SMike Smith 
161035863739SMike Smith 	if (cm->cm_flags & AAC_CMD_DATAIN)
1611c6eafcf2SScott Long 		bus_dmamap_sync(sc->aac_buffer_dmat, cm->cm_datamap,
1612c6eafcf2SScott Long 				BUS_DMASYNC_PREREAD);
161335863739SMike Smith 	if (cm->cm_flags & AAC_CMD_DATAOUT)
1614c6eafcf2SScott Long 		bus_dmamap_sync(sc->aac_buffer_dmat, cm->cm_datamap,
1615c6eafcf2SScott Long 				BUS_DMASYNC_PREWRITE);
161635863739SMike Smith 	cm->cm_flags |= AAC_CMD_MAPPED;
1617cd481291SScott Long 
16187cb209f5SScott Long 	if (sc->flags & AAC_FLAGS_NEW_COMM) {
16197cb209f5SScott Long 		int count = 10000000L;
16207cb209f5SScott Long 		while (AAC_SEND_COMMAND(sc, cm) != 0) {
16217cb209f5SScott Long 			if (--count == 0) {
16227cb209f5SScott Long 				aac_unmap_command(cm);
16237cb209f5SScott Long 				sc->flags |= AAC_QUEUE_FRZN;
16247cb209f5SScott Long 				aac_requeue_ready(cm);
16257cb209f5SScott Long 			}
16267cb209f5SScott Long 			DELAY(5);			/* wait 5 usec. */
16277cb209f5SScott Long 		}
16287cb209f5SScott Long 	} else {
1629397fa34fSScott Long 		/* Put the FIB on the outbound queue */
16304102d44bSScott Long 		if (aac_enqueue_fib(sc, cm->cm_queue, cm) == EBUSY) {
16314102d44bSScott Long 			aac_unmap_command(cm);
1632397fa34fSScott Long 			sc->flags |= AAC_QUEUE_FRZN;
1633cd481291SScott Long 			aac_requeue_ready(cm);
16344102d44bSScott Long 		}
16357cb209f5SScott Long 	}
163635863739SMike Smith }
163735863739SMike Smith 
1638914da7d0SScott Long /*
163935863739SMike Smith  * Unmap a command from controller-visible space.
164035863739SMike Smith  */
164135863739SMike Smith static void
aac_unmap_command(struct aac_command * cm)164235863739SMike Smith aac_unmap_command(struct aac_command *cm)
164335863739SMike Smith {
1644914da7d0SScott Long 	struct aac_softc *sc;
164535863739SMike Smith 
1646914da7d0SScott Long 	sc = cm->cm_sc;
164731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
1648914da7d0SScott Long 
164935863739SMike Smith 	if (!(cm->cm_flags & AAC_CMD_MAPPED))
165035863739SMike Smith 		return;
165135863739SMike Smith 
165235863739SMike Smith 	if (cm->cm_datalen != 0) {
165335863739SMike Smith 		if (cm->cm_flags & AAC_CMD_DATAIN)
1654c6eafcf2SScott Long 			bus_dmamap_sync(sc->aac_buffer_dmat, cm->cm_datamap,
1655c6eafcf2SScott Long 					BUS_DMASYNC_POSTREAD);
165635863739SMike Smith 		if (cm->cm_flags & AAC_CMD_DATAOUT)
1657c6eafcf2SScott Long 			bus_dmamap_sync(sc->aac_buffer_dmat, cm->cm_datamap,
1658c6eafcf2SScott Long 					BUS_DMASYNC_POSTWRITE);
165935863739SMike Smith 
166035863739SMike Smith 		bus_dmamap_unload(sc->aac_buffer_dmat, cm->cm_datamap);
166135863739SMike Smith 	}
166235863739SMike Smith 	cm->cm_flags &= ~AAC_CMD_MAPPED;
166335863739SMike Smith }
166435863739SMike Smith 
1665914da7d0SScott Long /*
1666914da7d0SScott Long  * Hardware Interface
1667914da7d0SScott Long  */
166835863739SMike Smith 
1669914da7d0SScott Long /*
16704109ba51SEd Maste  * Initialize the adapter.
167135863739SMike Smith  */
167235863739SMike Smith static void
aac_common_map(void * arg,bus_dma_segment_t * segs,int nseg,int error)167335863739SMike Smith aac_common_map(void *arg, bus_dma_segment_t *segs, int nseg, int error)
167435863739SMike Smith {
1675914da7d0SScott Long 	struct aac_softc *sc;
167635863739SMike Smith 
1677914da7d0SScott Long 	sc = (struct aac_softc *)arg;
167831a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
1679914da7d0SScott Long 
168035863739SMike Smith 	sc->aac_common_busaddr = segs[0].ds_addr;
168135863739SMike Smith }
168235863739SMike Smith 
1683a6d35632SScott Long static int
aac_check_firmware(struct aac_softc * sc)1684a6d35632SScott Long aac_check_firmware(struct aac_softc *sc)
1685a6d35632SScott Long {
168604f4d586SEd Maste 	u_int32_t code, major, minor, options = 0, atu_size = 0;
1687da4882c2SMarius Strobl 	int rid, status;
168804f4d586SEd Maste 	time_t then;
1689a6d35632SScott Long 
169031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
169104f4d586SEd Maste 	/*
169204f4d586SEd Maste 	 * Wait for the adapter to come ready.
169304f4d586SEd Maste 	 */
169404f4d586SEd Maste 	then = time_uptime;
169504f4d586SEd Maste 	do {
169604f4d586SEd Maste 		code = AAC_GET_FWSTATUS(sc);
169704f4d586SEd Maste 		if (code & AAC_SELF_TEST_FAILED) {
169804f4d586SEd Maste 			device_printf(sc->aac_dev, "FATAL: selftest failed\n");
169904f4d586SEd Maste 			return(ENXIO);
170004f4d586SEd Maste 		}
170104f4d586SEd Maste 		if (code & AAC_KERNEL_PANIC) {
170204f4d586SEd Maste 			device_printf(sc->aac_dev,
1703a620bad0SEd Maste 				      "FATAL: controller kernel panic");
170404f4d586SEd Maste 			return(ENXIO);
170504f4d586SEd Maste 		}
170604f4d586SEd Maste 		if (time_uptime > (then + AAC_BOOT_TIMEOUT)) {
170704f4d586SEd Maste 			device_printf(sc->aac_dev,
170804f4d586SEd Maste 				      "FATAL: controller not coming ready, "
170904f4d586SEd Maste 					   "status %x\n", code);
171004f4d586SEd Maste 			return(ENXIO);
171104f4d586SEd Maste 		}
171204f4d586SEd Maste 	} while (!(code & AAC_UP_AND_RUNNING));
1713a6d35632SScott Long 
1714fe94b852SScott Long 	/*
1715fe94b852SScott Long 	 * Retrieve the firmware version numbers.  Dell PERC2/QC cards with
1716fe94b852SScott Long 	 * firmware version 1.x are not compatible with this driver.
1717fe94b852SScott Long 	 */
1718a6d35632SScott Long 	if (sc->flags & AAC_FLAGS_PERC2QC) {
1719fe94b852SScott Long 		if (aac_sync_command(sc, AAC_MONKER_GETKERNVER, 0, 0, 0, 0,
1720fe94b852SScott Long 				     NULL)) {
1721fe94b852SScott Long 			device_printf(sc->aac_dev,
1722fe94b852SScott Long 				      "Error reading firmware version\n");
1723fe94b852SScott Long 			return (EIO);
1724fe94b852SScott Long 		}
1725fe94b852SScott Long 
1726fe94b852SScott Long 		/* These numbers are stored as ASCII! */
1727a6d35632SScott Long 		major = (AAC_GET_MAILBOX(sc, 1) & 0xff) - 0x30;
1728a6d35632SScott Long 		minor = (AAC_GET_MAILBOX(sc, 2) & 0xff) - 0x30;
1729fe94b852SScott Long 		if (major == 1) {
1730fe94b852SScott Long 			device_printf(sc->aac_dev,
1731fe94b852SScott Long 			    "Firmware version %d.%d is not supported.\n",
1732fe94b852SScott Long 			    major, minor);
1733fe94b852SScott Long 			return (EINVAL);
1734fe94b852SScott Long 		}
1735fe94b852SScott Long 	}
1736fe94b852SScott Long 
1737a6d35632SScott Long 	/*
1738a6d35632SScott Long 	 * Retrieve the capabilities/supported options word so we know what
1739a441b3fcSScott Long 	 * work-arounds to enable.  Some firmware revs don't support this
1740a441b3fcSScott Long 	 * command.
1741a6d35632SScott Long 	 */
1742a441b3fcSScott Long 	if (aac_sync_command(sc, AAC_MONKER_GETINFO, 0, 0, 0, 0, &status)) {
1743a441b3fcSScott Long 		if (status != AAC_SRB_STS_INVALID_REQUEST) {
1744a441b3fcSScott Long 			device_printf(sc->aac_dev,
1745a441b3fcSScott Long 			     "RequestAdapterInfo failed\n");
1746a6d35632SScott Long 			return (EIO);
1747a6d35632SScott Long 		}
1748a441b3fcSScott Long 	} else {
1749a6d35632SScott Long 		options = AAC_GET_MAILBOX(sc, 1);
17507cb209f5SScott Long 		atu_size = AAC_GET_MAILBOX(sc, 2);
1751a6d35632SScott Long 		sc->supported_options = options;
1752a6d35632SScott Long 
1753a6d35632SScott Long 		if ((options & AAC_SUPPORTED_4GB_WINDOW) != 0 &&
1754a6d35632SScott Long 		    (sc->flags & AAC_FLAGS_NO4GB) == 0)
1755a6d35632SScott Long 			sc->flags |= AAC_FLAGS_4GB_WINDOW;
1756a6d35632SScott Long 		if (options & AAC_SUPPORTED_NONDASD)
1757a6d35632SScott Long 			sc->flags |= AAC_FLAGS_ENABLE_CAM;
1758cd481291SScott Long 		if ((options & AAC_SUPPORTED_SGMAP_HOST64) != 0
1759cd481291SScott Long 		     && (sizeof(bus_addr_t) > 4)) {
1760a441b3fcSScott Long 			device_printf(sc->aac_dev,
1761a441b3fcSScott Long 			    "Enabling 64-bit address support\n");
1762a6d35632SScott Long 			sc->flags |= AAC_FLAGS_SG_64BIT;
1763a6d35632SScott Long 		}
1764a441b3fcSScott Long 		if ((options & AAC_SUPPORTED_NEW_COMM)
1765da4882c2SMarius Strobl 		 && sc->aac_if->aif_send_command)
17667cb209f5SScott Long 			sc->flags |= AAC_FLAGS_NEW_COMM;
17677cb209f5SScott Long 		if (options & AAC_SUPPORTED_64BIT_ARRAYSIZE)
17687cb209f5SScott Long 			sc->flags |= AAC_FLAGS_ARRAY_64BIT;
1769a441b3fcSScott Long 	}
1770a6d35632SScott Long 
1771a6d35632SScott Long 	/* Check for broken hardware that does a lower number of commands */
17727cb209f5SScott Long 	sc->aac_max_fibs = (sc->flags & AAC_FLAGS_256FIBS ? 256:512);
17737cb209f5SScott Long 
17747cb209f5SScott Long 	/* Remap mem. resource, if required */
17757cb209f5SScott Long 	if ((sc->flags & AAC_FLAGS_NEW_COMM) &&
1776ff0991c4SAttilio Rao 	    atu_size > rman_get_size(sc->aac_regs_res1)) {
1777da4882c2SMarius Strobl 		rid = rman_get_rid(sc->aac_regs_res1);
1778da4882c2SMarius Strobl 		bus_release_resource(sc->aac_dev, SYS_RES_MEMORY, rid,
1779da4882c2SMarius Strobl 		    sc->aac_regs_res1);
1780c47476d7SJustin Hibbits 		sc->aac_regs_res1 = bus_alloc_resource_anywhere(sc->aac_dev,
1781c47476d7SJustin Hibbits 		    SYS_RES_MEMORY, &rid, atu_size, RF_ACTIVE);
1782ff0991c4SAttilio Rao 		if (sc->aac_regs_res1 == NULL) {
1783ff0991c4SAttilio Rao 			sc->aac_regs_res1 = bus_alloc_resource_any(
1784da4882c2SMarius Strobl 			    sc->aac_dev, SYS_RES_MEMORY, &rid, RF_ACTIVE);
1785ff0991c4SAttilio Rao 			if (sc->aac_regs_res1 == NULL) {
17867cb209f5SScott Long 				device_printf(sc->aac_dev,
17877cb209f5SScott Long 				    "couldn't allocate register window\n");
17887cb209f5SScott Long 				return (ENXIO);
17897cb209f5SScott Long 			}
17907cb209f5SScott Long 			sc->flags &= ~AAC_FLAGS_NEW_COMM;
17917cb209f5SScott Long 		}
1792ff0991c4SAttilio Rao 		sc->aac_btag1 = rman_get_bustag(sc->aac_regs_res1);
1793ff0991c4SAttilio Rao 		sc->aac_bhandle1 = rman_get_bushandle(sc->aac_regs_res1);
1794ff0991c4SAttilio Rao 
1795ff0991c4SAttilio Rao 		if (sc->aac_hwif == AAC_HWIF_NARK) {
1796ff0991c4SAttilio Rao 			sc->aac_regs_res0 = sc->aac_regs_res1;
1797ff0991c4SAttilio Rao 			sc->aac_btag0 = sc->aac_btag1;
1798ff0991c4SAttilio Rao 			sc->aac_bhandle0 = sc->aac_bhandle1;
1799ff0991c4SAttilio Rao 		}
18007cb209f5SScott Long 	}
18017cb209f5SScott Long 
18027cb209f5SScott Long 	/* Read preferred settings */
18037cb209f5SScott Long 	sc->aac_max_fib_size = sizeof(struct aac_fib);
18047cb209f5SScott Long 	sc->aac_max_sectors = 128;				/* 64KB */
18057cb209f5SScott Long 	if (sc->flags & AAC_FLAGS_SG_64BIT)
1806a441b3fcSScott Long 		sc->aac_sg_tablesize = (AAC_FIB_DATASIZE
18077e7a458eSEd Maste 		 - sizeof(struct aac_blockwrite64))
18087e7a458eSEd Maste 		 / sizeof(struct aac_sg_entry64);
1809a6d35632SScott Long 	else
1810a441b3fcSScott Long 		sc->aac_sg_tablesize = (AAC_FIB_DATASIZE
18117e7a458eSEd Maste 		 - sizeof(struct aac_blockwrite))
18127e7a458eSEd Maste 		 / sizeof(struct aac_sg_entry);
1813a441b3fcSScott Long 
18147cb209f5SScott Long 	if (!aac_sync_command(sc, AAC_MONKER_GETCOMMPREF, 0, 0, 0, 0, NULL)) {
18157cb209f5SScott Long 		options = AAC_GET_MAILBOX(sc, 1);
18167cb209f5SScott Long 		sc->aac_max_fib_size = (options & 0xFFFF);
18177cb209f5SScott Long 		sc->aac_max_sectors = (options >> 16) << 1;
18187cb209f5SScott Long 		options = AAC_GET_MAILBOX(sc, 2);
18197cb209f5SScott Long 		sc->aac_sg_tablesize = (options >> 16);
18207cb209f5SScott Long 		options = AAC_GET_MAILBOX(sc, 3);
18217cb209f5SScott Long 		sc->aac_max_fibs = (options & 0xFFFF);
18227cb209f5SScott Long 	}
18237cb209f5SScott Long 	if (sc->aac_max_fib_size > PAGE_SIZE)
18247cb209f5SScott Long 		sc->aac_max_fib_size = PAGE_SIZE;
18257cb209f5SScott Long 	sc->aac_max_fibs_alloc = PAGE_SIZE / sc->aac_max_fib_size;
1826a6d35632SScott Long 
1827f355c0e0SEd Maste 	if (sc->aac_max_fib_size > sizeof(struct aac_fib)) {
1828f355c0e0SEd Maste 		sc->flags |= AAC_FLAGS_RAW_IO;
1829f355c0e0SEd Maste 		device_printf(sc->aac_dev, "Enable Raw I/O\n");
1830f355c0e0SEd Maste 	}
1831523da39bSEd Maste 	if ((sc->flags & AAC_FLAGS_RAW_IO) &&
1832523da39bSEd Maste 	    (sc->flags & AAC_FLAGS_ARRAY_64BIT)) {
1833523da39bSEd Maste 		sc->flags |= AAC_FLAGS_LBA_64BIT;
1834523da39bSEd Maste 		device_printf(sc->aac_dev, "Enable 64-bit array\n");
1835523da39bSEd Maste 	}
1836f355c0e0SEd Maste 
1837fe94b852SScott Long 	return (0);
1838fe94b852SScott Long }
1839fe94b852SScott Long 
184035863739SMike Smith static int
aac_init(struct aac_softc * sc)184135863739SMike Smith aac_init(struct aac_softc *sc)
184235863739SMike Smith {
184335863739SMike Smith 	struct aac_adapter_init	*ip;
184404f4d586SEd Maste 	u_int32_t qoffset;
1845a6d35632SScott Long 	int error;
184635863739SMike Smith 
184731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
1848ffb37f33SScott Long 
184935863739SMike Smith 	/*
1850914da7d0SScott Long 	 * Fill in the init structure.  This tells the adapter about the
1851914da7d0SScott Long 	 * physical location of various important shared data structures.
185235863739SMike Smith 	 */
185335863739SMike Smith 	ip = &sc->aac_common->ac_init;
185435863739SMike Smith 	ip->InitStructRevision = AAC_INIT_STRUCT_REVISION;
18557cb209f5SScott Long 	if (sc->aac_max_fib_size > sizeof(struct aac_fib)) {
18567cb209f5SScott Long 		ip->InitStructRevision = AAC_INIT_STRUCT_REVISION_4;
18577cb209f5SScott Long 		sc->flags |= AAC_FLAGS_RAW_IO;
18587cb209f5SScott Long 	}
1859f30ac74cSScott Long 	ip->MiniPortRevision = AAC_INIT_STRUCT_MINIPORT_REVISION;
186035863739SMike Smith 
1861c6eafcf2SScott Long 	ip->AdapterFibsPhysicalAddress = sc->aac_common_busaddr +
1862c6eafcf2SScott Long 					 offsetof(struct aac_common, ac_fibs);
1863149af931SScott Long 	ip->AdapterFibsVirtualAddress = 0;
186435863739SMike Smith 	ip->AdapterFibsSize = AAC_ADAPTER_FIBS * sizeof(struct aac_fib);
186535863739SMike Smith 	ip->AdapterFibAlign = sizeof(struct aac_fib);
186635863739SMike Smith 
1867c6eafcf2SScott Long 	ip->PrintfBufferAddress = sc->aac_common_busaddr +
1868c6eafcf2SScott Long 				  offsetof(struct aac_common, ac_printf);
186935863739SMike Smith 	ip->PrintfBufferSize = AAC_PRINTF_BUFSIZE;
187035863739SMike Smith 
18714b00f859SScott Long 	/*
18724b00f859SScott Long 	 * The adapter assumes that pages are 4K in size, except on some
18734b00f859SScott Long  	 * broken firmware versions that do the page->byte conversion twice,
18744b00f859SScott Long 	 * therefore 'assuming' that this value is in 16MB units (2^24).
18754b00f859SScott Long 	 * Round up since the granularity is so high.
18764b00f859SScott Long 	 */
1877f30ac74cSScott Long 	ip->HostPhysMemPages = ctob(physmem) / AAC_PAGE_SIZE;
18784b00f859SScott Long 	if (sc->flags & AAC_FLAGS_BROKEN_MEMMAP) {
18794b00f859SScott Long 		ip->HostPhysMemPages =
18804b00f859SScott Long 		    (ip->HostPhysMemPages + AAC_PAGE_SIZE) / AAC_PAGE_SIZE;
1881204c0befSScott Long 	}
18822b3b0f17SScott Long 	ip->HostElapsedSeconds = time_uptime;	/* reset later if invalid */
188335863739SMike Smith 
18847cb209f5SScott Long 	ip->InitFlags = 0;
18857cb209f5SScott Long 	if (sc->flags & AAC_FLAGS_NEW_COMM) {
1886e71d3b9cSEd Maste 		ip->InitFlags |= AAC_INITFLAGS_NEW_COMM_SUPPORTED;
18877cb209f5SScott Long 		device_printf(sc->aac_dev, "New comm. interface enabled\n");
18887cb209f5SScott Long 	}
18897cb209f5SScott Long 
18907cb209f5SScott Long 	ip->MaxIoCommands = sc->aac_max_fibs;
18917cb209f5SScott Long 	ip->MaxIoSize = sc->aac_max_sectors << 9;
18927cb209f5SScott Long 	ip->MaxFibSize = sc->aac_max_fib_size;
18937cb209f5SScott Long 
189435863739SMike Smith 	/*
18954109ba51SEd Maste 	 * Initialize FIB queues.  Note that it appears that the layout of the
1896c6eafcf2SScott Long 	 * indexes and the segmentation of the entries may be mandated by the
1897c6eafcf2SScott Long 	 * adapter, which is only told about the base of the queue index fields.
189835863739SMike Smith 	 *
189935863739SMike Smith 	 * The initial values of the indices are assumed to inform the adapter
1900914da7d0SScott Long 	 * of the sizes of the respective queues, and theoretically it could
1901914da7d0SScott Long 	 * work out the entire layout of the queue structures from this.  We
1902914da7d0SScott Long 	 * take the easy route and just lay this area out like everyone else
1903914da7d0SScott Long 	 * does.
190435863739SMike Smith 	 *
1905914da7d0SScott Long 	 * The Linux driver uses a much more complex scheme whereby several
1906914da7d0SScott Long 	 * header records are kept for each queue.  We use a couple of generic
1907914da7d0SScott Long 	 * list manipulation functions which 'know' the size of each list by
1908914da7d0SScott Long 	 * virtue of a table.
190935863739SMike Smith 	 */
1910b88ffdc8SScott Long 	qoffset = offsetof(struct aac_common, ac_qbuf) + AAC_QUEUE_ALIGN;
19110bcbebd6SScott Long 	qoffset &= ~(AAC_QUEUE_ALIGN - 1);
19120bcbebd6SScott Long 	sc->aac_queues =
19130bcbebd6SScott Long 	    (struct aac_queue_table *)((uintptr_t)sc->aac_common + qoffset);
1914b88ffdc8SScott Long 	ip->CommHeaderAddress = sc->aac_common_busaddr + qoffset;
191535863739SMike Smith 
1916c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] =
1917c6eafcf2SScott Long 		AAC_HOST_NORM_CMD_ENTRIES;
1918c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] =
1919c6eafcf2SScott Long 		AAC_HOST_NORM_CMD_ENTRIES;
1920c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] =
1921c6eafcf2SScott Long 		AAC_HOST_HIGH_CMD_ENTRIES;
1922c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] =
1923c6eafcf2SScott Long 		AAC_HOST_HIGH_CMD_ENTRIES;
1924c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] =
1925c6eafcf2SScott Long 		AAC_ADAP_NORM_CMD_ENTRIES;
1926c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] =
1927c6eafcf2SScott Long 		AAC_ADAP_NORM_CMD_ENTRIES;
1928c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] =
1929c6eafcf2SScott Long 		AAC_ADAP_HIGH_CMD_ENTRIES;
1930c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] =
1931c6eafcf2SScott Long 		AAC_ADAP_HIGH_CMD_ENTRIES;
1932c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX]=
1933c6eafcf2SScott Long 		AAC_HOST_NORM_RESP_ENTRIES;
1934c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX]=
1935c6eafcf2SScott Long 		AAC_HOST_NORM_RESP_ENTRIES;
1936c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX]=
1937c6eafcf2SScott Long 		AAC_HOST_HIGH_RESP_ENTRIES;
1938c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX]=
1939c6eafcf2SScott Long 		AAC_HOST_HIGH_RESP_ENTRIES;
1940c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX]=
1941c6eafcf2SScott Long 		AAC_ADAP_NORM_RESP_ENTRIES;
1942c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX]=
1943c6eafcf2SScott Long 		AAC_ADAP_NORM_RESP_ENTRIES;
1944c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX]=
1945c6eafcf2SScott Long 		AAC_ADAP_HIGH_RESP_ENTRIES;
1946c6eafcf2SScott Long 	sc->aac_queues->qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX]=
1947c6eafcf2SScott Long 		AAC_ADAP_HIGH_RESP_ENTRIES;
1948c6eafcf2SScott Long 	sc->aac_qentries[AAC_HOST_NORM_CMD_QUEUE] =
1949c6eafcf2SScott Long 		&sc->aac_queues->qt_HostNormCmdQueue[0];
1950c6eafcf2SScott Long 	sc->aac_qentries[AAC_HOST_HIGH_CMD_QUEUE] =
1951c6eafcf2SScott Long 		&sc->aac_queues->qt_HostHighCmdQueue[0];
1952c6eafcf2SScott Long 	sc->aac_qentries[AAC_ADAP_NORM_CMD_QUEUE] =
1953c6eafcf2SScott Long 		&sc->aac_queues->qt_AdapNormCmdQueue[0];
1954c6eafcf2SScott Long 	sc->aac_qentries[AAC_ADAP_HIGH_CMD_QUEUE] =
1955c6eafcf2SScott Long 		&sc->aac_queues->qt_AdapHighCmdQueue[0];
1956c6eafcf2SScott Long 	sc->aac_qentries[AAC_HOST_NORM_RESP_QUEUE] =
1957c6eafcf2SScott Long 		&sc->aac_queues->qt_HostNormRespQueue[0];
1958c6eafcf2SScott Long 	sc->aac_qentries[AAC_HOST_HIGH_RESP_QUEUE] =
1959c6eafcf2SScott Long 		&sc->aac_queues->qt_HostHighRespQueue[0];
1960c6eafcf2SScott Long 	sc->aac_qentries[AAC_ADAP_NORM_RESP_QUEUE] =
1961c6eafcf2SScott Long 		&sc->aac_queues->qt_AdapNormRespQueue[0];
1962c6eafcf2SScott Long 	sc->aac_qentries[AAC_ADAP_HIGH_RESP_QUEUE] =
1963c6eafcf2SScott Long 		&sc->aac_queues->qt_AdapHighRespQueue[0];
196435863739SMike Smith 
196535863739SMike Smith 	/*
196635863739SMike Smith 	 * Do controller-type-specific initialisation
196735863739SMike Smith 	 */
196835863739SMike Smith 	switch (sc->aac_hwif) {
196935863739SMike Smith 	case AAC_HWIF_I960RX:
1970ff0991c4SAttilio Rao 		AAC_MEM0_SETREG4(sc, AAC_RX_ODBR, ~0);
197135863739SMike Smith 		break;
19724afedc31SScott Long 	case AAC_HWIF_RKT:
1973ff0991c4SAttilio Rao 		AAC_MEM0_SETREG4(sc, AAC_RKT_ODBR, ~0);
19744afedc31SScott Long 		break;
19754afedc31SScott Long 	default:
19764afedc31SScott Long 		break;
197735863739SMike Smith 	}
197835863739SMike Smith 
197935863739SMike Smith 	/*
198035863739SMike Smith 	 * Give the init structure to the controller.
198135863739SMike Smith 	 */
198235863739SMike Smith 	if (aac_sync_command(sc, AAC_MONKER_INITSTRUCT,
1983914da7d0SScott Long 			     sc->aac_common_busaddr +
1984914da7d0SScott Long 			     offsetof(struct aac_common, ac_init), 0, 0, 0,
1985914da7d0SScott Long 			     NULL)) {
1986914da7d0SScott Long 		device_printf(sc->aac_dev,
1987914da7d0SScott Long 			      "error establishing init structure\n");
1988a6d35632SScott Long 		error = EIO;
1989a6d35632SScott Long 		goto out;
199035863739SMike Smith 	}
199135863739SMike Smith 
1992a6d35632SScott Long 	error = 0;
1993a6d35632SScott Long out:
1994a6d35632SScott Long 	return(error);
199535863739SMike Smith }
199635863739SMike Smith 
199704f4d586SEd Maste static int
aac_setup_intr(struct aac_softc * sc)199804f4d586SEd Maste aac_setup_intr(struct aac_softc *sc)
199904f4d586SEd Maste {
2000da4882c2SMarius Strobl 
200104f4d586SEd Maste 	if (sc->flags & AAC_FLAGS_NEW_COMM) {
200204f4d586SEd Maste 		if (bus_setup_intr(sc->aac_dev, sc->aac_irq,
200304f4d586SEd Maste 				   INTR_MPSAFE|INTR_TYPE_BIO, NULL,
200404f4d586SEd Maste 				   aac_new_intr, sc, &sc->aac_intr)) {
200504f4d586SEd Maste 			device_printf(sc->aac_dev, "can't set up interrupt\n");
200604f4d586SEd Maste 			return (EINVAL);
200704f4d586SEd Maste 		}
200804f4d586SEd Maste 	} else {
200904f4d586SEd Maste 		if (bus_setup_intr(sc->aac_dev, sc->aac_irq,
2010e46b9eeaSEd Maste 				   INTR_TYPE_BIO, aac_filter, NULL,
201104f4d586SEd Maste 				   sc, &sc->aac_intr)) {
201204f4d586SEd Maste 			device_printf(sc->aac_dev,
2013e46b9eeaSEd Maste 				      "can't set up interrupt filter\n");
201404f4d586SEd Maste 			return (EINVAL);
201504f4d586SEd Maste 		}
201604f4d586SEd Maste 	}
201704f4d586SEd Maste 	return (0);
201804f4d586SEd Maste }
201904f4d586SEd Maste 
2020914da7d0SScott Long /*
202135863739SMike Smith  * Send a synchronous command to the controller and wait for a result.
20227cb209f5SScott Long  * Indicate if the controller completed the command with an error status.
202335863739SMike Smith  */
202435863739SMike Smith static int
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)202535863739SMike Smith aac_sync_command(struct aac_softc *sc, u_int32_t command,
202635863739SMike Smith 		 u_int32_t arg0, u_int32_t arg1, u_int32_t arg2, u_int32_t arg3,
202735863739SMike Smith 		 u_int32_t *sp)
202835863739SMike Smith {
202935863739SMike Smith 	time_t then;
203035863739SMike Smith 	u_int32_t status;
203135863739SMike Smith 
203231a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
203335863739SMike Smith 
203435863739SMike Smith 	/* populate the mailbox */
203535863739SMike Smith 	AAC_SET_MAILBOX(sc, command, arg0, arg1, arg2, arg3);
203635863739SMike Smith 
203735863739SMike Smith 	/* ensure the sync command doorbell flag is cleared */
203835863739SMike Smith 	AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND);
203935863739SMike Smith 
204035863739SMike Smith 	/* then set it to signal the adapter */
204135863739SMike Smith 	AAC_QNOTIFY(sc, AAC_DB_SYNC_COMMAND);
204235863739SMike Smith 
204335863739SMike Smith 	/* spin waiting for the command to complete */
20442b3b0f17SScott Long 	then = time_uptime;
204535863739SMike Smith 	do {
20462b3b0f17SScott Long 		if (time_uptime > (then + AAC_IMMEDIATE_TIMEOUT)) {
204731a0399eSEd Maste 			fwprintf(sc, HBA_FLAGS_DBG_ERROR_B, "timed out");
204835863739SMike Smith 			return(EIO);
204935863739SMike Smith 		}
205035863739SMike Smith 	} while (!(AAC_GET_ISTATUS(sc) & AAC_DB_SYNC_COMMAND));
205135863739SMike Smith 
205235863739SMike Smith 	/* clear the completion flag */
205335863739SMike Smith 	AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND);
205435863739SMike Smith 
205535863739SMike Smith 	/* get the command status */
2056a6d35632SScott Long 	status = AAC_GET_MAILBOX(sc, 0);
205735863739SMike Smith 	if (sp != NULL)
205835863739SMike Smith 		*sp = status;
20597cb209f5SScott Long 
2060a441b3fcSScott Long 	if (status != AAC_SRB_STS_SUCCESS)
20617cb209f5SScott Long 		return (-1);
20620b94a66eSMike Smith 	return(0);
206335863739SMike Smith }
206435863739SMike Smith 
2065cbfd045bSScott Long int
aac_sync_fib(struct aac_softc * sc,u_int32_t command,u_int32_t xferstate,struct aac_fib * fib,u_int16_t datasize)206635863739SMike Smith aac_sync_fib(struct aac_softc *sc, u_int32_t command, u_int32_t xferstate,
2067cbfd045bSScott Long 		 struct aac_fib *fib, u_int16_t datasize)
206835863739SMike Smith {
206931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
20707cb209f5SScott Long 	mtx_assert(&sc->aac_io_lock, MA_OWNED);
207135863739SMike Smith 
207235863739SMike Smith 	if (datasize > AAC_FIB_DATASIZE)
207335863739SMike Smith 		return(EINVAL);
207435863739SMike Smith 
207535863739SMike Smith 	/*
207635863739SMike Smith 	 * Set up the sync FIB
207735863739SMike Smith 	 */
2078914da7d0SScott Long 	fib->Header.XferState = AAC_FIBSTATE_HOSTOWNED |
2079914da7d0SScott Long 				AAC_FIBSTATE_INITIALISED |
2080c6eafcf2SScott Long 				AAC_FIBSTATE_EMPTY;
208135863739SMike Smith 	fib->Header.XferState |= xferstate;
208235863739SMike Smith 	fib->Header.Command = command;
208335863739SMike Smith 	fib->Header.StructType = AAC_FIBTYPE_TFIB;
208442ef13a2SEd Maste 	fib->Header.Size = sizeof(struct aac_fib_header) + datasize;
208535863739SMike Smith 	fib->Header.SenderSize = sizeof(struct aac_fib);
2086b88ffdc8SScott Long 	fib->Header.SenderFibAddress = 0;	/* Not needed */
2087c6eafcf2SScott Long 	fib->Header.ReceiverFibAddress = sc->aac_common_busaddr +
2088914da7d0SScott Long 					 offsetof(struct aac_common,
2089914da7d0SScott Long 						  ac_sync_fib);
209035863739SMike Smith 
209135863739SMike Smith 	/*
209235863739SMike Smith 	 * Give the FIB to the controller, wait for a response.
209335863739SMike Smith 	 */
2094914da7d0SScott Long 	if (aac_sync_command(sc, AAC_MONKER_SYNCFIB,
2095914da7d0SScott Long 			     fib->Header.ReceiverFibAddress, 0, 0, 0, NULL)) {
209631a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_ERROR_B, "IO error");
209735863739SMike Smith 		return(EIO);
209835863739SMike Smith 	}
209935863739SMike Smith 
210035863739SMike Smith 	return (0);
210135863739SMike Smith }
210235863739SMike Smith 
2103914da7d0SScott Long /*
210435863739SMike Smith  * Adapter-space FIB queue manipulation
210535863739SMike Smith  *
210635863739SMike Smith  * Note that the queue implementation here is a little funky; neither the PI or
210735863739SMike Smith  * CI will ever be zero.  This behaviour is a controller feature.
210835863739SMike Smith  */
2109da4882c2SMarius Strobl static const struct {
211035863739SMike Smith 	int		size;
211135863739SMike Smith 	int		notify;
211235863739SMike Smith } aac_qinfo[] = {
211335863739SMike Smith 	{AAC_HOST_NORM_CMD_ENTRIES, AAC_DB_COMMAND_NOT_FULL},
211435863739SMike Smith 	{AAC_HOST_HIGH_CMD_ENTRIES, 0},
211535863739SMike Smith 	{AAC_ADAP_NORM_CMD_ENTRIES, AAC_DB_COMMAND_READY},
211635863739SMike Smith 	{AAC_ADAP_HIGH_CMD_ENTRIES, 0},
211735863739SMike Smith 	{AAC_HOST_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_NOT_FULL},
211835863739SMike Smith 	{AAC_HOST_HIGH_RESP_ENTRIES, 0},
211935863739SMike Smith 	{AAC_ADAP_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_READY},
212035863739SMike Smith 	{AAC_ADAP_HIGH_RESP_ENTRIES, 0}
212135863739SMike Smith };
212235863739SMike Smith 
212335863739SMike Smith /*
2124c6eafcf2SScott Long  * Atomically insert an entry into the nominated queue, returns 0 on success or
2125c6eafcf2SScott Long  * EBUSY if the queue is full.
212635863739SMike Smith  *
21270b94a66eSMike Smith  * Note: it would be more efficient to defer notifying the controller in
2128914da7d0SScott Long  *	 the case where we may be inserting several entries in rapid succession,
2129914da7d0SScott Long  *	 but implementing this usefully may be difficult (it would involve a
2130c6eafcf2SScott Long  *	 separate queue/notify interface).
213135863739SMike Smith  */
213235863739SMike Smith static int
aac_enqueue_fib(struct aac_softc * sc,int queue,struct aac_command * cm)2133f6c4dd3fSScott Long aac_enqueue_fib(struct aac_softc *sc, int queue, struct aac_command *cm)
213435863739SMike Smith {
213535863739SMike Smith 	u_int32_t pi, ci;
21369e2e96d8SScott Long 	int error;
2137f6c4dd3fSScott Long 	u_int32_t fib_size;
2138f6c4dd3fSScott Long 	u_int32_t fib_addr;
2139f6c4dd3fSScott Long 
214031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
214136e0bf6eSScott Long 
2142f6c4dd3fSScott Long 	fib_size = cm->cm_fib->Header.Size;
2143f6c4dd3fSScott Long 	fib_addr = cm->cm_fib->Header.ReceiverFibAddress;
214435863739SMike Smith 
214535863739SMike Smith 	/* get the producer/consumer indices */
214635863739SMike Smith 	pi = sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX];
214735863739SMike Smith 	ci = sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX];
214835863739SMike Smith 
214935863739SMike Smith 	/* wrap the queue? */
215035863739SMike Smith 	if (pi >= aac_qinfo[queue].size)
215135863739SMike Smith 		pi = 0;
215235863739SMike Smith 
215335863739SMike Smith 	/* check for queue full */
215435863739SMike Smith 	if ((pi + 1) == ci) {
215535863739SMike Smith 		error = EBUSY;
215635863739SMike Smith 		goto out;
215735863739SMike Smith 	}
215835863739SMike Smith 
2159614c22b2SScott Long 	/*
2160614c22b2SScott Long 	 * To avoid a race with its completion interrupt, place this command on
2161614c22b2SScott Long 	 * the busy queue prior to advertising it to the controller.
2162614c22b2SScott Long 	 */
2163614c22b2SScott Long 	aac_enqueue_busy(cm);
2164614c22b2SScott Long 
216535863739SMike Smith 	/* populate queue entry */
216635863739SMike Smith 	(sc->aac_qentries[queue] + pi)->aq_fib_size = fib_size;
216735863739SMike Smith 	(sc->aac_qentries[queue] + pi)->aq_fib_addr = fib_addr;
216835863739SMike Smith 
216935863739SMike Smith 	/* update producer index */
217035863739SMike Smith 	sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX] = pi + 1;
217135863739SMike Smith 
217235863739SMike Smith 	/* notify the adapter if we know how */
217335863739SMike Smith 	if (aac_qinfo[queue].notify != 0)
217435863739SMike Smith 		AAC_QNOTIFY(sc, aac_qinfo[queue].notify);
217535863739SMike Smith 
217635863739SMike Smith 	error = 0;
217735863739SMike Smith 
217835863739SMike Smith out:
217935863739SMike Smith 	return(error);
218035863739SMike Smith }
218135863739SMike Smith 
218235863739SMike Smith /*
218336e0bf6eSScott Long  * Atomically remove one entry from the nominated queue, returns 0 on
218436e0bf6eSScott Long  * success or ENOENT if the queue is empty.
218535863739SMike Smith  */
218635863739SMike Smith static int
aac_dequeue_fib(struct aac_softc * sc,int queue,u_int32_t * fib_size,struct aac_fib ** fib_addr)2187c6eafcf2SScott Long aac_dequeue_fib(struct aac_softc *sc, int queue, u_int32_t *fib_size,
2188c6eafcf2SScott Long 		struct aac_fib **fib_addr)
218935863739SMike Smith {
219035863739SMike Smith 	u_int32_t pi, ci;
2191149af931SScott Long 	u_int32_t fib_index;
21929e2e96d8SScott Long 	int error;
2193f6c4dd3fSScott Long 	int notify;
219435863739SMike Smith 
219531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
219635863739SMike Smith 
219735863739SMike Smith 	/* get the producer/consumer indices */
219835863739SMike Smith 	pi = sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX];
219935863739SMike Smith 	ci = sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX];
220035863739SMike Smith 
220135863739SMike Smith 	/* check for queue empty */
220235863739SMike Smith 	if (ci == pi) {
220335863739SMike Smith 		error = ENOENT;
220435863739SMike Smith 		goto out;
220535863739SMike Smith 	}
220635863739SMike Smith 
22077753acd2SScott Long 	/* wrap the pi so the following test works */
22087753acd2SScott Long 	if (pi >= aac_qinfo[queue].size)
22097753acd2SScott Long 		pi = 0;
22107753acd2SScott Long 
2211f6c4dd3fSScott Long 	notify = 0;
2212f6c4dd3fSScott Long 	if (ci == pi + 1)
2213f6c4dd3fSScott Long 		notify++;
2214f6c4dd3fSScott Long 
221535863739SMike Smith 	/* wrap the queue? */
221635863739SMike Smith 	if (ci >= aac_qinfo[queue].size)
221735863739SMike Smith 		ci = 0;
221835863739SMike Smith 
221935863739SMike Smith 	/* fetch the entry */
222035863739SMike Smith 	*fib_size = (sc->aac_qentries[queue] + ci)->aq_fib_size;
2221149af931SScott Long 
2222149af931SScott Long 	switch (queue) {
2223149af931SScott Long 	case AAC_HOST_NORM_CMD_QUEUE:
2224149af931SScott Long 	case AAC_HOST_HIGH_CMD_QUEUE:
2225149af931SScott Long 		/*
2226149af931SScott Long 		 * The aq_fib_addr is only 32 bits wide so it can't be counted
2227149af931SScott Long 		 * on to hold an address.  For AIF's, the adapter assumes
2228149af931SScott Long 		 * that it's giving us an address into the array of AIF fibs.
2229149af931SScott Long 		 * Therefore, we have to convert it to an index.
2230149af931SScott Long 		 */
2231149af931SScott Long 		fib_index = (sc->aac_qentries[queue] + ci)->aq_fib_addr /
2232149af931SScott Long 			sizeof(struct aac_fib);
2233149af931SScott Long 		*fib_addr = &sc->aac_common->ac_fibs[fib_index];
2234149af931SScott Long 		break;
2235149af931SScott Long 
2236149af931SScott Long 	case AAC_HOST_NORM_RESP_QUEUE:
2237149af931SScott Long 	case AAC_HOST_HIGH_RESP_QUEUE:
2238149af931SScott Long 	{
2239149af931SScott Long 		struct aac_command *cm;
2240149af931SScott Long 
2241149af931SScott Long 		/*
2242149af931SScott Long 		 * As above, an index is used instead of an actual address.
2243149af931SScott Long 		 * Gotta shift the index to account for the fast response
2244149af931SScott Long 		 * bit.  No other correction is needed since this value was
2245149af931SScott Long 		 * originally provided by the driver via the SenderFibAddress
2246149af931SScott Long 		 * field.
2247149af931SScott Long 		 */
2248149af931SScott Long 		fib_index = (sc->aac_qentries[queue] + ci)->aq_fib_addr;
22497cb209f5SScott Long 		cm = sc->aac_commands + (fib_index >> 2);
2250149af931SScott Long 		*fib_addr = cm->cm_fib;
225135863739SMike Smith 
2252f30ac74cSScott Long 		/*
2253f30ac74cSScott Long 		 * Is this a fast response? If it is, update the fib fields in
2254149af931SScott Long 		 * local memory since the whole fib isn't DMA'd back up.
2255f30ac74cSScott Long 		 */
2256149af931SScott Long 		if (fib_index & 0x01) {
2257f30ac74cSScott Long 			(*fib_addr)->Header.XferState |= AAC_FIBSTATE_DONEADAP;
2258f30ac74cSScott Long 			*((u_int32_t*)((*fib_addr)->data)) = AAC_ERROR_NORMAL;
2259f30ac74cSScott Long 		}
2260149af931SScott Long 		break;
2261149af931SScott Long 	}
2262149af931SScott Long 	default:
2263149af931SScott Long 		panic("Invalid queue in aac_dequeue_fib()");
2264149af931SScott Long 		break;
2265149af931SScott Long 	}
2266149af931SScott Long 
226735863739SMike Smith 	/* update consumer index */
226835863739SMike Smith 	sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX] = ci + 1;
226935863739SMike Smith 
227035863739SMike Smith 	/* if we have made the queue un-full, notify the adapter */
2271f6c4dd3fSScott Long 	if (notify && (aac_qinfo[queue].notify != 0))
227235863739SMike Smith 		AAC_QNOTIFY(sc, aac_qinfo[queue].notify);
227335863739SMike Smith 	error = 0;
227435863739SMike Smith 
227535863739SMike Smith out:
227635863739SMike Smith 	return(error);
227735863739SMike Smith }
227835863739SMike Smith 
2279914da7d0SScott Long /*
228036e0bf6eSScott Long  * Put our response to an Adapter Initialed Fib on the response queue
228136e0bf6eSScott Long  */
228236e0bf6eSScott Long static int
aac_enqueue_response(struct aac_softc * sc,int queue,struct aac_fib * fib)228336e0bf6eSScott Long aac_enqueue_response(struct aac_softc *sc, int queue, struct aac_fib *fib)
228436e0bf6eSScott Long {
228536e0bf6eSScott Long 	u_int32_t pi, ci;
22869e2e96d8SScott Long 	int error;
228736e0bf6eSScott Long 	u_int32_t fib_size;
228836e0bf6eSScott Long 	u_int32_t fib_addr;
228936e0bf6eSScott Long 
229031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
229136e0bf6eSScott Long 
229236e0bf6eSScott Long 	/* Tell the adapter where the FIB is */
229336e0bf6eSScott Long 	fib_size = fib->Header.Size;
229436e0bf6eSScott Long 	fib_addr = fib->Header.SenderFibAddress;
229536e0bf6eSScott Long 	fib->Header.ReceiverFibAddress = fib_addr;
229636e0bf6eSScott Long 
229736e0bf6eSScott Long 	/* get the producer/consumer indices */
229836e0bf6eSScott Long 	pi = sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX];
229936e0bf6eSScott Long 	ci = sc->aac_queues->qt_qindex[queue][AAC_CONSUMER_INDEX];
230036e0bf6eSScott Long 
230136e0bf6eSScott Long 	/* wrap the queue? */
230236e0bf6eSScott Long 	if (pi >= aac_qinfo[queue].size)
230336e0bf6eSScott Long 		pi = 0;
230436e0bf6eSScott Long 
230536e0bf6eSScott Long 	/* check for queue full */
230636e0bf6eSScott Long 	if ((pi + 1) == ci) {
230736e0bf6eSScott Long 		error = EBUSY;
230836e0bf6eSScott Long 		goto out;
230936e0bf6eSScott Long 	}
231036e0bf6eSScott Long 
231136e0bf6eSScott Long 	/* populate queue entry */
231236e0bf6eSScott Long 	(sc->aac_qentries[queue] + pi)->aq_fib_size = fib_size;
231336e0bf6eSScott Long 	(sc->aac_qentries[queue] + pi)->aq_fib_addr = fib_addr;
231436e0bf6eSScott Long 
231536e0bf6eSScott Long 	/* update producer index */
231636e0bf6eSScott Long 	sc->aac_queues->qt_qindex[queue][AAC_PRODUCER_INDEX] = pi + 1;
231736e0bf6eSScott Long 
231836e0bf6eSScott Long 	/* notify the adapter if we know how */
231936e0bf6eSScott Long 	if (aac_qinfo[queue].notify != 0)
232036e0bf6eSScott Long 		AAC_QNOTIFY(sc, aac_qinfo[queue].notify);
232136e0bf6eSScott Long 
232236e0bf6eSScott Long 	error = 0;
232336e0bf6eSScott Long 
232436e0bf6eSScott Long out:
232536e0bf6eSScott Long 	return(error);
232636e0bf6eSScott Long }
232736e0bf6eSScott Long 
2328914da7d0SScott Long /*
23290b94a66eSMike Smith  * Check for commands that have been outstanding for a suspiciously long time,
23300b94a66eSMike Smith  * and complain about them.
23310b94a66eSMike Smith  */
23320b94a66eSMike Smith static void
aac_timeout(struct aac_softc * sc)23330b94a66eSMike Smith aac_timeout(struct aac_softc *sc)
23340b94a66eSMike Smith {
23350b94a66eSMike Smith 	struct aac_command *cm;
23360b94a66eSMike Smith 	time_t deadline;
233715c37be0SScott Long 	int timedout, code;
23380b94a66eSMike Smith 
2339f6c4dd3fSScott Long 	/*
234070545d1aSScott Long 	 * Traverse the busy command list, bitch about late commands once
2341914da7d0SScott Long 	 * only.
2342914da7d0SScott Long 	 */
234315c37be0SScott Long 	timedout = 0;
23442b3b0f17SScott Long 	deadline = time_uptime - AAC_CMD_TIMEOUT;
23450b94a66eSMike Smith 	TAILQ_FOREACH(cm, &sc->aac_busy, cm_link) {
2346f6c4dd3fSScott Long 		if ((cm->cm_timestamp  < deadline)
23473e507710SEd Maste 		    && !(cm->cm_flags & AAC_CMD_TIMEDOUT)) {
23480b94a66eSMike Smith 			cm->cm_flags |= AAC_CMD_TIMEDOUT;
2349914da7d0SScott Long 			device_printf(sc->aac_dev,
23505aa4bb5bSEd Maste 			    "COMMAND %p (TYPE %d) TIMEOUT AFTER %d SECONDS\n",
23515aa4bb5bSEd Maste 			    cm, cm->cm_fib->Header.Command,
23525aa4bb5bSEd Maste 			    (int)(time_uptime-cm->cm_timestamp));
23530b94a66eSMike Smith 			AAC_PRINT_FIB(sc, cm->cm_fib);
235415c37be0SScott Long 			timedout++;
23550b94a66eSMike Smith 		}
23560b94a66eSMike Smith 	}
23570b94a66eSMike Smith 
235815c37be0SScott Long 	if (timedout) {
235915c37be0SScott Long 		code = AAC_GET_FWSTATUS(sc);
236015c37be0SScott Long 		if (code != AAC_UP_AND_RUNNING) {
236115c37be0SScott Long 			device_printf(sc->aac_dev, "WARNING! Controller is no "
236215c37be0SScott Long 				      "longer running! code= 0x%x\n", code);
236315c37be0SScott Long 		}
236415c37be0SScott Long 	}
23650b94a66eSMike Smith }
23660b94a66eSMike Smith 
2367914da7d0SScott Long /*
2368914da7d0SScott Long  * Interface Function Vectors
2369914da7d0SScott Long  */
237035863739SMike Smith 
2371914da7d0SScott Long /*
237235863739SMike Smith  * Read the current firmware status word.
237335863739SMike Smith  */
237435863739SMike Smith static int
aac_sa_get_fwstatus(struct aac_softc * sc)237535863739SMike Smith aac_sa_get_fwstatus(struct aac_softc *sc)
237635863739SMike Smith {
237731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
237835863739SMike Smith 
2379ff0991c4SAttilio Rao 	return(AAC_MEM0_GETREG4(sc, AAC_SA_FWSTATUS));
238035863739SMike Smith }
238135863739SMike Smith 
238235863739SMike Smith static int
aac_rx_get_fwstatus(struct aac_softc * sc)238335863739SMike Smith aac_rx_get_fwstatus(struct aac_softc *sc)
238435863739SMike Smith {
238531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
238635863739SMike Smith 
23874824be88SEd Maste 	return(AAC_MEM0_GETREG4(sc, sc->flags & AAC_FLAGS_NEW_COMM ?
23884824be88SEd Maste 	    AAC_RX_OMR0 : AAC_RX_FWSTATUS));
238935863739SMike Smith }
239035863739SMike Smith 
2391b3457b51SScott Long static int
aac_rkt_get_fwstatus(struct aac_softc * sc)23924afedc31SScott Long aac_rkt_get_fwstatus(struct aac_softc *sc)
23934afedc31SScott Long {
239431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
23954afedc31SScott Long 
23964824be88SEd Maste 	return(AAC_MEM0_GETREG4(sc, sc->flags & AAC_FLAGS_NEW_COMM ?
23974824be88SEd Maste 	    AAC_RKT_OMR0 : AAC_RKT_FWSTATUS));
23984afedc31SScott Long }
23994afedc31SScott Long 
2400914da7d0SScott Long /*
240135863739SMike Smith  * Notify the controller of a change in a given queue
240235863739SMike Smith  */
240335863739SMike Smith 
240435863739SMike Smith static void
aac_sa_qnotify(struct aac_softc * sc,int qbit)240535863739SMike Smith aac_sa_qnotify(struct aac_softc *sc, int qbit)
240635863739SMike Smith {
240731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
240835863739SMike Smith 
2409ff0991c4SAttilio Rao 	AAC_MEM0_SETREG2(sc, AAC_SA_DOORBELL1_SET, qbit);
241035863739SMike Smith }
241135863739SMike Smith 
241235863739SMike Smith static void
aac_rx_qnotify(struct aac_softc * sc,int qbit)241335863739SMike Smith aac_rx_qnotify(struct aac_softc *sc, int qbit)
241435863739SMike Smith {
241531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
241635863739SMike Smith 
2417ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RX_IDBR, qbit);
241835863739SMike Smith }
241935863739SMike Smith 
2420b3457b51SScott Long static void
aac_rkt_qnotify(struct aac_softc * sc,int qbit)24214afedc31SScott Long aac_rkt_qnotify(struct aac_softc *sc, int qbit)
24224afedc31SScott Long {
242331a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
24244afedc31SScott Long 
2425ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RKT_IDBR, qbit);
24264afedc31SScott Long }
24274afedc31SScott Long 
2428914da7d0SScott Long /*
242935863739SMike Smith  * Get the interrupt reason bits
243035863739SMike Smith  */
243135863739SMike Smith static int
aac_sa_get_istatus(struct aac_softc * sc)243235863739SMike Smith aac_sa_get_istatus(struct aac_softc *sc)
243335863739SMike Smith {
243431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
243535863739SMike Smith 
2436ff0991c4SAttilio Rao 	return(AAC_MEM0_GETREG2(sc, AAC_SA_DOORBELL0));
243735863739SMike Smith }
243835863739SMike Smith 
243935863739SMike Smith static int
aac_rx_get_istatus(struct aac_softc * sc)244035863739SMike Smith aac_rx_get_istatus(struct aac_softc *sc)
244135863739SMike Smith {
244231a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
244335863739SMike Smith 
2444ff0991c4SAttilio Rao 	return(AAC_MEM0_GETREG4(sc, AAC_RX_ODBR));
244535863739SMike Smith }
244635863739SMike Smith 
2447b3457b51SScott Long static int
aac_rkt_get_istatus(struct aac_softc * sc)24484afedc31SScott Long aac_rkt_get_istatus(struct aac_softc *sc)
24494afedc31SScott Long {
245031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
24514afedc31SScott Long 
2452ff0991c4SAttilio Rao 	return(AAC_MEM0_GETREG4(sc, AAC_RKT_ODBR));
24534afedc31SScott Long }
24544afedc31SScott Long 
2455914da7d0SScott Long /*
245635863739SMike Smith  * Clear some interrupt reason bits
245735863739SMike Smith  */
245835863739SMike Smith static void
aac_sa_clear_istatus(struct aac_softc * sc,int mask)245935863739SMike Smith aac_sa_clear_istatus(struct aac_softc *sc, int mask)
246035863739SMike Smith {
246131a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
246235863739SMike Smith 
2463ff0991c4SAttilio Rao 	AAC_MEM0_SETREG2(sc, AAC_SA_DOORBELL0_CLEAR, mask);
246435863739SMike Smith }
246535863739SMike Smith 
246635863739SMike Smith static void
aac_rx_clear_istatus(struct aac_softc * sc,int mask)246735863739SMike Smith aac_rx_clear_istatus(struct aac_softc *sc, int mask)
246835863739SMike Smith {
246931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
247035863739SMike Smith 
2471ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RX_ODBR, mask);
247235863739SMike Smith }
247335863739SMike Smith 
2474b3457b51SScott Long static void
aac_rkt_clear_istatus(struct aac_softc * sc,int mask)24754afedc31SScott Long aac_rkt_clear_istatus(struct aac_softc *sc, int mask)
24764afedc31SScott Long {
247731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
24784afedc31SScott Long 
2479ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RKT_ODBR, mask);
24804afedc31SScott Long }
24814afedc31SScott Long 
2482914da7d0SScott Long /*
248335863739SMike Smith  * Populate the mailbox and set the command word
248435863739SMike Smith  */
248535863739SMike Smith static void
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)248635863739SMike Smith aac_sa_set_mailbox(struct aac_softc *sc, u_int32_t command,
248735863739SMike Smith 		u_int32_t arg0, u_int32_t arg1, u_int32_t arg2, u_int32_t arg3)
248835863739SMike Smith {
248931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
249035863739SMike Smith 
2491ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_SA_MAILBOX, command);
2492ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_SA_MAILBOX + 4, arg0);
2493ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_SA_MAILBOX + 8, arg1);
2494ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_SA_MAILBOX + 12, arg2);
2495ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_SA_MAILBOX + 16, arg3);
249635863739SMike Smith }
249735863739SMike Smith 
249835863739SMike Smith static void
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)249935863739SMike Smith aac_rx_set_mailbox(struct aac_softc *sc, u_int32_t command,
250035863739SMike Smith 		u_int32_t arg0, u_int32_t arg1, u_int32_t arg2, u_int32_t arg3)
250135863739SMike Smith {
250231a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
250335863739SMike Smith 
2504ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RX_MAILBOX, command);
2505ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RX_MAILBOX + 4, arg0);
2506ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RX_MAILBOX + 8, arg1);
2507ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RX_MAILBOX + 12, arg2);
2508ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RX_MAILBOX + 16, arg3);
250935863739SMike Smith }
251035863739SMike Smith 
2511b3457b51SScott Long static void
aac_rkt_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)25124afedc31SScott Long aac_rkt_set_mailbox(struct aac_softc *sc, u_int32_t command, u_int32_t arg0,
25134afedc31SScott Long 		    u_int32_t arg1, u_int32_t arg2, u_int32_t arg3)
25144afedc31SScott Long {
251531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
25164afedc31SScott Long 
2517ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RKT_MAILBOX, command);
2518ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RKT_MAILBOX + 4, arg0);
2519ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RKT_MAILBOX + 8, arg1);
2520ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RKT_MAILBOX + 12, arg2);
2521ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, AAC_RKT_MAILBOX + 16, arg3);
25224afedc31SScott Long }
25234afedc31SScott Long 
2524914da7d0SScott Long /*
252535863739SMike Smith  * Fetch the immediate command status word
252635863739SMike Smith  */
252735863739SMike Smith static int
aac_sa_get_mailbox(struct aac_softc * sc,int mb)2528a6d35632SScott Long aac_sa_get_mailbox(struct aac_softc *sc, int mb)
252935863739SMike Smith {
253031a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
253135863739SMike Smith 
2532ff0991c4SAttilio Rao 	return(AAC_MEM1_GETREG4(sc, AAC_SA_MAILBOX + (mb * 4)));
253335863739SMike Smith }
253435863739SMike Smith 
253535863739SMike Smith static int
aac_rx_get_mailbox(struct aac_softc * sc,int mb)2536a6d35632SScott Long aac_rx_get_mailbox(struct aac_softc *sc, int mb)
253735863739SMike Smith {
253831a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
253935863739SMike Smith 
2540ff0991c4SAttilio Rao 	return(AAC_MEM1_GETREG4(sc, AAC_RX_MAILBOX + (mb * 4)));
254135863739SMike Smith }
254235863739SMike Smith 
2543b3457b51SScott Long static int
aac_rkt_get_mailbox(struct aac_softc * sc,int mb)25444afedc31SScott Long aac_rkt_get_mailbox(struct aac_softc *sc, int mb)
25454afedc31SScott Long {
254631a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
25474afedc31SScott Long 
2548ff0991c4SAttilio Rao 	return(AAC_MEM1_GETREG4(sc, AAC_RKT_MAILBOX + (mb * 4)));
25494afedc31SScott Long }
25504afedc31SScott Long 
2551914da7d0SScott Long /*
255235863739SMike Smith  * Set/clear interrupt masks
255335863739SMike Smith  */
255435863739SMike Smith static void
aac_sa_set_interrupts(struct aac_softc * sc,int enable)255535863739SMike Smith aac_sa_set_interrupts(struct aac_softc *sc, int enable)
255635863739SMike Smith {
255731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "%sable interrupts", enable ? "en" : "dis");
255835863739SMike Smith 
255935863739SMike Smith 	if (enable) {
2560ff0991c4SAttilio Rao 		AAC_MEM0_SETREG2((sc), AAC_SA_MASK0_CLEAR, AAC_DB_INTERRUPTS);
256135863739SMike Smith 	} else {
2562ff0991c4SAttilio Rao 		AAC_MEM0_SETREG2((sc), AAC_SA_MASK0_SET, ~0);
256335863739SMike Smith 	}
256435863739SMike Smith }
256535863739SMike Smith 
256635863739SMike Smith static void
aac_rx_set_interrupts(struct aac_softc * sc,int enable)256735863739SMike Smith aac_rx_set_interrupts(struct aac_softc *sc, int enable)
256835863739SMike Smith {
256931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "%sable interrupts", enable ? "en" : "dis");
257035863739SMike Smith 
257135863739SMike Smith 	if (enable) {
25727cb209f5SScott Long 		if (sc->flags & AAC_FLAGS_NEW_COMM)
2573ff0991c4SAttilio Rao 			AAC_MEM0_SETREG4(sc, AAC_RX_OIMR, ~AAC_DB_INT_NEW_COMM);
25747cb209f5SScott Long 		else
2575ff0991c4SAttilio Rao 			AAC_MEM0_SETREG4(sc, AAC_RX_OIMR, ~AAC_DB_INTERRUPTS);
257635863739SMike Smith 	} else {
2577ff0991c4SAttilio Rao 		AAC_MEM0_SETREG4(sc, AAC_RX_OIMR, ~0);
257835863739SMike Smith 	}
257935863739SMike Smith }
258035863739SMike Smith 
2581b3457b51SScott Long static void
aac_rkt_set_interrupts(struct aac_softc * sc,int enable)25824afedc31SScott Long aac_rkt_set_interrupts(struct aac_softc *sc, int enable)
25834afedc31SScott Long {
258431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "%sable interrupts", enable ? "en" : "dis");
25854afedc31SScott Long 
25864afedc31SScott Long 	if (enable) {
25877cb209f5SScott Long 		if (sc->flags & AAC_FLAGS_NEW_COMM)
2588ff0991c4SAttilio Rao 			AAC_MEM0_SETREG4(sc, AAC_RKT_OIMR, ~AAC_DB_INT_NEW_COMM);
25897cb209f5SScott Long 		else
2590ff0991c4SAttilio Rao 			AAC_MEM0_SETREG4(sc, AAC_RKT_OIMR, ~AAC_DB_INTERRUPTS);
25914afedc31SScott Long 	} else {
2592ff0991c4SAttilio Rao 		AAC_MEM0_SETREG4(sc, AAC_RKT_OIMR, ~0);
25934afedc31SScott Long 	}
25944afedc31SScott Long }
25954afedc31SScott Long 
2596914da7d0SScott Long /*
25977cb209f5SScott Long  * New comm. interface: Send command functions
25987cb209f5SScott Long  */
25997cb209f5SScott Long static int
aac_rx_send_command(struct aac_softc * sc,struct aac_command * cm)26007cb209f5SScott Long aac_rx_send_command(struct aac_softc *sc, struct aac_command *cm)
26017cb209f5SScott Long {
26027cb209f5SScott Long 	u_int32_t index, device;
26037cb209f5SScott Long 
260431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "send command (new comm.)");
26057cb209f5SScott Long 
2606ff0991c4SAttilio Rao 	index = AAC_MEM0_GETREG4(sc, AAC_RX_IQUE);
26077cb209f5SScott Long 	if (index == 0xffffffffL)
2608ff0991c4SAttilio Rao 		index = AAC_MEM0_GETREG4(sc, AAC_RX_IQUE);
26097cb209f5SScott Long 	if (index == 0xffffffffL)
26107cb209f5SScott Long 		return index;
26117cb209f5SScott Long 	aac_enqueue_busy(cm);
26127cb209f5SScott Long 	device = index;
2613ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, device, (u_int32_t)(cm->cm_fibphys & 0xffffffffUL));
26147cb209f5SScott Long 	device += 4;
2615ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, device, (u_int32_t)(cm->cm_fibphys >> 32));
26167cb209f5SScott Long 	device += 4;
2617ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, device, cm->cm_fib->Header.Size);
2618ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RX_IQUE, index);
26197cb209f5SScott Long 	return 0;
26207cb209f5SScott Long }
26217cb209f5SScott Long 
26227cb209f5SScott Long static int
aac_rkt_send_command(struct aac_softc * sc,struct aac_command * cm)26237cb209f5SScott Long aac_rkt_send_command(struct aac_softc *sc, struct aac_command *cm)
26247cb209f5SScott Long {
26257cb209f5SScott Long 	u_int32_t index, device;
26267cb209f5SScott Long 
262731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "send command (new comm.)");
26287cb209f5SScott Long 
2629ff0991c4SAttilio Rao 	index = AAC_MEM0_GETREG4(sc, AAC_RKT_IQUE);
26307cb209f5SScott Long 	if (index == 0xffffffffL)
2631ff0991c4SAttilio Rao 		index = AAC_MEM0_GETREG4(sc, AAC_RKT_IQUE);
26327cb209f5SScott Long 	if (index == 0xffffffffL)
26337cb209f5SScott Long 		return index;
26347cb209f5SScott Long 	aac_enqueue_busy(cm);
26357cb209f5SScott Long 	device = index;
2636ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, device, (u_int32_t)(cm->cm_fibphys & 0xffffffffUL));
26377cb209f5SScott Long 	device += 4;
2638ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, device, (u_int32_t)(cm->cm_fibphys >> 32));
26397cb209f5SScott Long 	device += 4;
2640ff0991c4SAttilio Rao 	AAC_MEM1_SETREG4(sc, device, cm->cm_fib->Header.Size);
2641ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RKT_IQUE, index);
26427cb209f5SScott Long 	return 0;
26437cb209f5SScott Long }
26447cb209f5SScott Long 
26457cb209f5SScott Long /*
26467cb209f5SScott Long  * New comm. interface: get, set outbound queue index
26477cb209f5SScott Long  */
26487cb209f5SScott Long static int
aac_rx_get_outb_queue(struct aac_softc * sc)26497cb209f5SScott Long aac_rx_get_outb_queue(struct aac_softc *sc)
26507cb209f5SScott Long {
265131a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
26527cb209f5SScott Long 
2653ff0991c4SAttilio Rao 	return(AAC_MEM0_GETREG4(sc, AAC_RX_OQUE));
26547cb209f5SScott Long }
26557cb209f5SScott Long 
26567cb209f5SScott Long static int
aac_rkt_get_outb_queue(struct aac_softc * sc)26577cb209f5SScott Long aac_rkt_get_outb_queue(struct aac_softc *sc)
26587cb209f5SScott Long {
265931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
26607cb209f5SScott Long 
2661ff0991c4SAttilio Rao 	return(AAC_MEM0_GETREG4(sc, AAC_RKT_OQUE));
26627cb209f5SScott Long }
26637cb209f5SScott Long 
26647cb209f5SScott Long static void
aac_rx_set_outb_queue(struct aac_softc * sc,int index)26657cb209f5SScott Long aac_rx_set_outb_queue(struct aac_softc *sc, int index)
26667cb209f5SScott Long {
266731a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
26687cb209f5SScott Long 
2669ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RX_OQUE, index);
26707cb209f5SScott Long }
26717cb209f5SScott Long 
26727cb209f5SScott Long static void
aac_rkt_set_outb_queue(struct aac_softc * sc,int index)26737cb209f5SScott Long aac_rkt_set_outb_queue(struct aac_softc *sc, int index)
26747cb209f5SScott Long {
267531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
26767cb209f5SScott Long 
2677ff0991c4SAttilio Rao 	AAC_MEM0_SETREG4(sc, AAC_RKT_OQUE, index);
26787cb209f5SScott Long }
26797cb209f5SScott Long 
26807cb209f5SScott Long /*
2681914da7d0SScott Long  * Debugging and Diagnostics
2682914da7d0SScott Long  */
268335863739SMike Smith 
2684914da7d0SScott Long /*
268535863739SMike Smith  * Print some information about the controller.
268635863739SMike Smith  */
268735863739SMike Smith static void
aac_describe_controller(struct aac_softc * sc)268835863739SMike Smith aac_describe_controller(struct aac_softc *sc)
268935863739SMike Smith {
2690cbfd045bSScott Long 	struct aac_fib *fib;
269135863739SMike Smith 	struct aac_adapter_info	*info;
26927ea2d558SEd Maste 	char *adapter_type = "Adaptec RAID controller";
269335863739SMike Smith 
269431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
269535863739SMike Smith 
269681b3da08SScott Long 	mtx_lock(&sc->aac_io_lock);
269703b5fe51SScott Long 	aac_alloc_sync_fib(sc, &fib);
2698cbfd045bSScott Long 
2699cbfd045bSScott Long 	fib->data[0] = 0;
2700cbfd045bSScott Long 	if (aac_sync_fib(sc, RequestAdapterInfo, 0, fib, 1)) {
270135863739SMike Smith 		device_printf(sc->aac_dev, "RequestAdapterInfo failed\n");
2702fe3cb0e1SScott Long 		aac_release_sync_fib(sc);
270381b3da08SScott Long 		mtx_unlock(&sc->aac_io_lock);
270435863739SMike Smith 		return;
270535863739SMike Smith 	}
270635863739SMike Smith 
2707bd971c49SScott Long 	/* save the kernel revision structure for later use */
2708bd971c49SScott Long 	info = (struct aac_adapter_info *)&fib->data[0];
2709bd971c49SScott Long 	sc->aac_revision = info->KernelRevision;
2710bd971c49SScott Long 
2711bd971c49SScott Long 	if (bootverbose) {
2712b1c56c68SScott Long 		device_printf(sc->aac_dev, "%s %dMHz, %dMB memory "
2713b1c56c68SScott Long 		    "(%dMB cache, %dMB execution), %s\n",
2714c6eafcf2SScott Long 		    aac_describe_code(aac_cpu_variant, info->CpuVariant),
2715b1c56c68SScott Long 		    info->ClockSpeed, info->TotalMem / (1024 * 1024),
2716b1c56c68SScott Long 		    info->BufferMem / (1024 * 1024),
2717b1c56c68SScott Long 		    info->ExecutionMem / (1024 * 1024),
2718914da7d0SScott Long 		    aac_describe_code(aac_battery_platform,
2719914da7d0SScott Long 		    info->batteryPlatform));
272035863739SMike Smith 
2721bd971c49SScott Long 		device_printf(sc->aac_dev,
2722bd971c49SScott Long 		    "Kernel %d.%d-%d, Build %d, S/N %6X\n",
272335863739SMike Smith 		    info->KernelRevision.external.comp.major,
272435863739SMike Smith 		    info->KernelRevision.external.comp.minor,
272535863739SMike Smith 		    info->KernelRevision.external.comp.dash,
272636e0bf6eSScott Long 		    info->KernelRevision.buildNumber,
272736e0bf6eSScott Long 		    (u_int32_t)(info->SerialNumber & 0xffffff));
2728fe3cb0e1SScott Long 
2729a6d35632SScott Long 		device_printf(sc->aac_dev, "Supported Options=%b\n",
2730a6d35632SScott Long 			      sc->supported_options,
2731a6d35632SScott Long 			      "\20"
2732a6d35632SScott Long 			      "\1SNAPSHOT"
2733a6d35632SScott Long 			      "\2CLUSTERS"
2734a6d35632SScott Long 			      "\3WCACHE"
2735a6d35632SScott Long 			      "\4DATA64"
2736a6d35632SScott Long 			      "\5HOSTTIME"
2737a6d35632SScott Long 			      "\6RAID50"
2738a6d35632SScott Long 			      "\7WINDOW4GB"
2739a6d35632SScott Long 			      "\10SCSIUPGD"
2740a6d35632SScott Long 			      "\11SOFTERR"
2741a6d35632SScott Long 			      "\12NORECOND"
2742a6d35632SScott Long 			      "\13SGMAP64"
2743a6d35632SScott Long 			      "\14ALARM"
27447cb209f5SScott Long 			      "\15NONDASD"
27457cb209f5SScott Long 			      "\16SCSIMGT"
27467cb209f5SScott Long 			      "\17RAIDSCSI"
27477cb209f5SScott Long 			      "\21ADPTINFO"
27487cb209f5SScott Long 			      "\22NEWCOMM"
27497cb209f5SScott Long 			      "\23ARRAY64BIT"
27507cb209f5SScott Long 			      "\24HEATSENSOR");
2751a6d35632SScott Long 	}
275255aa1136SEd Maste 
275355aa1136SEd Maste 	if (sc->supported_options & AAC_SUPPORTED_SUPPLEMENT_ADAPTER_INFO) {
275455aa1136SEd Maste 		fib->data[0] = 0;
275555aa1136SEd Maste 		if (aac_sync_fib(sc, RequestSupplementAdapterInfo, 0, fib, 1))
275655aa1136SEd Maste 			device_printf(sc->aac_dev,
275755aa1136SEd Maste 			    "RequestSupplementAdapterInfo failed\n");
275855aa1136SEd Maste 		else
275955aa1136SEd Maste 			adapter_type = ((struct aac_supplement_adapter_info *)
276055aa1136SEd Maste 			    &fib->data[0])->AdapterTypeText;
276155aa1136SEd Maste 	}
276255aa1136SEd Maste 	device_printf(sc->aac_dev, "%s, aac driver %d.%d.%d-%d\n",
276355aa1136SEd Maste 		adapter_type,
27648e7e6335SEd Maste 		AAC_DRIVER_MAJOR_VERSION, AAC_DRIVER_MINOR_VERSION,
27658e7e6335SEd Maste 		AAC_DRIVER_BUGFIX_LEVEL, AAC_DRIVER_BUILD);
276655aa1136SEd Maste 
2767bd971c49SScott Long 	aac_release_sync_fib(sc);
276881b3da08SScott Long 	mtx_unlock(&sc->aac_io_lock);
276935863739SMike Smith }
277035863739SMike Smith 
2771914da7d0SScott Long /*
277235863739SMike Smith  * Look up a text description of a numeric error code and return a pointer to
277335863739SMike Smith  * same.
277435863739SMike Smith  */
2775da4882c2SMarius Strobl static const char *
aac_describe_code(const struct aac_code_lookup * table,u_int32_t code)2776da4882c2SMarius Strobl aac_describe_code(const struct aac_code_lookup *table, u_int32_t code)
277735863739SMike Smith {
277835863739SMike Smith 	int i;
277935863739SMike Smith 
278035863739SMike Smith 	for (i = 0; table[i].string != NULL; i++)
278135863739SMike Smith 		if (table[i].code == code)
278235863739SMike Smith 			return(table[i].string);
278335863739SMike Smith 	return(table[i + 1].string);
278435863739SMike Smith }
278535863739SMike Smith 
2786914da7d0SScott Long /*
2787914da7d0SScott Long  * Management Interface
2788914da7d0SScott Long  */
278935863739SMike Smith 
279035863739SMike Smith static int
aac_open(struct cdev * dev,int flags,int fmt,struct thread * td)279100b4e54aSWarner Losh aac_open(struct cdev *dev, int flags, int fmt, struct thread *td)
279235863739SMike Smith {
2793914da7d0SScott Long 	struct aac_softc *sc;
279435863739SMike Smith 
2795914da7d0SScott Long 	sc = dev->si_drv1;
279631a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
279704f798ecSAttilio Rao 	device_busy(sc->aac_dev);
2798dfe2c294SAttilio Rao 	devfs_set_cdevpriv(sc, aac_cdevpriv_dtor);
279935863739SMike Smith 
280035863739SMike Smith 	return 0;
280135863739SMike Smith }
280235863739SMike Smith 
280335863739SMike Smith static int
aac_ioctl(struct cdev * dev,u_long cmd,caddr_t arg,int flag,struct thread * td)280400b4e54aSWarner Losh aac_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, struct thread *td)
280535863739SMike Smith {
2806914da7d0SScott Long 	union aac_statrequest *as;
2807914da7d0SScott Long 	struct aac_softc *sc;
28080b94a66eSMike Smith 	int error = 0;
280935863739SMike Smith 
2810914da7d0SScott Long 	as = (union aac_statrequest *)arg;
2811914da7d0SScott Long 	sc = dev->si_drv1;
281231a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
2813914da7d0SScott Long 
281435863739SMike Smith 	switch (cmd) {
28150b94a66eSMike Smith 	case AACIO_STATS:
28160b94a66eSMike Smith 		switch (as->as_item) {
28170b94a66eSMike Smith 		case AACQ_FREE:
28180b94a66eSMike Smith 		case AACQ_BIO:
28190b94a66eSMike Smith 		case AACQ_READY:
28200b94a66eSMike Smith 		case AACQ_BUSY:
2821c6eafcf2SScott Long 			bcopy(&sc->aac_qstat[as->as_item], &as->as_qstat,
2822c6eafcf2SScott Long 			      sizeof(struct aac_qstat));
28230b94a66eSMike Smith 			break;
28240b94a66eSMike Smith 		default:
28250b94a66eSMike Smith 			error = ENOENT;
28260b94a66eSMike Smith 			break;
28270b94a66eSMike Smith 		}
28280b94a66eSMike Smith 	break;
28290b94a66eSMike Smith 
283035863739SMike Smith 	case FSACTL_SENDFIB:
2831f355c0e0SEd Maste 	case FSACTL_SEND_LARGE_FIB:
2832fb0c27d7SScott Long 		arg = *(caddr_t*)arg;
2833fb0c27d7SScott Long 	case FSACTL_LNX_SENDFIB:
2834f355c0e0SEd Maste 	case FSACTL_LNX_SEND_LARGE_FIB:
283531a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_SENDFIB");
283635863739SMike Smith 		error = aac_ioctl_sendfib(sc, arg);
283735863739SMike Smith 		break;
2838f355c0e0SEd Maste 	case FSACTL_SEND_RAW_SRB:
2839f355c0e0SEd Maste 		arg = *(caddr_t*)arg;
2840f355c0e0SEd Maste 	case FSACTL_LNX_SEND_RAW_SRB:
284131a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_SEND_RAW_SRB");
2842f355c0e0SEd Maste 		error = aac_ioctl_send_raw_srb(sc, arg);
2843f355c0e0SEd Maste 		break;
284435863739SMike Smith 	case FSACTL_AIF_THREAD:
2845fb0c27d7SScott Long 	case FSACTL_LNX_AIF_THREAD:
284631a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_AIF_THREAD");
284735863739SMike Smith 		error = EINVAL;
284835863739SMike Smith 		break;
284935863739SMike Smith 	case FSACTL_OPEN_GET_ADAPTER_FIB:
2850fb0c27d7SScott Long 		arg = *(caddr_t*)arg;
2851fb0c27d7SScott Long 	case FSACTL_LNX_OPEN_GET_ADAPTER_FIB:
285231a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_OPEN_GET_ADAPTER_FIB");
2853a723a548SEd Maste 		error = aac_open_aif(sc, arg);
285435863739SMike Smith 		break;
285535863739SMike Smith 	case FSACTL_GET_NEXT_ADAPTER_FIB:
2856fb0c27d7SScott Long 		arg = *(caddr_t*)arg;
2857fb0c27d7SScott Long 	case FSACTL_LNX_GET_NEXT_ADAPTER_FIB:
285831a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_GET_NEXT_ADAPTER_FIB");
2859fb0c27d7SScott Long 		error = aac_getnext_aif(sc, arg);
286035863739SMike Smith 		break;
286135863739SMike Smith 	case FSACTL_CLOSE_GET_ADAPTER_FIB:
2862a723a548SEd Maste 		arg = *(caddr_t*)arg;
2863fb0c27d7SScott Long 	case FSACTL_LNX_CLOSE_GET_ADAPTER_FIB:
286431a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_CLOSE_GET_ADAPTER_FIB");
2865a723a548SEd Maste 		error = aac_close_aif(sc, arg);
286635863739SMike Smith 		break;
286735863739SMike Smith 	case FSACTL_MINIPORT_REV_CHECK:
2868fb0c27d7SScott Long 		arg = *(caddr_t*)arg;
2869fb0c27d7SScott Long 	case FSACTL_LNX_MINIPORT_REV_CHECK:
287031a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_MINIPORT_REV_CHECK");
2871fb0c27d7SScott Long 		error = aac_rev_check(sc, arg);
287235863739SMike Smith 		break;
287336e0bf6eSScott Long 	case FSACTL_QUERY_DISK:
287436e0bf6eSScott Long 		arg = *(caddr_t*)arg;
287536e0bf6eSScott Long 	case FSACTL_LNX_QUERY_DISK:
287631a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_QUERY_DISK");
287736e0bf6eSScott Long 		error = aac_query_disk(sc, arg);
287836e0bf6eSScott Long 		break;
287936e0bf6eSScott Long 	case FSACTL_DELETE_DISK:
288036e0bf6eSScott Long 	case FSACTL_LNX_DELETE_DISK:
2881914da7d0SScott Long 		/*
2882914da7d0SScott Long 		 * We don't trust the underland to tell us when to delete a
2883914da7d0SScott Long 		 * container, rather we rely on an AIF coming from the
2884914da7d0SScott Long 		 * controller
2885914da7d0SScott Long 		 */
288636e0bf6eSScott Long 		error = 0;
288736e0bf6eSScott Long 		break;
28887cb209f5SScott Long 	case FSACTL_GET_PCI_INFO:
28897cb209f5SScott Long 		arg = *(caddr_t*)arg;
28907cb209f5SScott Long 	case FSACTL_LNX_GET_PCI_INFO:
289131a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_GET_PCI_INFO");
28927cb209f5SScott Long 		error = aac_get_pci_info(sc, arg);
28937cb209f5SScott Long 		break;
28946d307336SEd Maste 	case FSACTL_GET_FEATURES:
28956d307336SEd Maste 		arg = *(caddr_t*)arg;
28966d307336SEd Maste 	case FSACTL_LNX_GET_FEATURES:
28976d307336SEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "FSACTL_GET_FEATURES");
28986d307336SEd Maste 		error = aac_supported_features(sc, arg);
28996d307336SEd Maste 		break;
290035863739SMike Smith 	default:
290131a0399eSEd Maste 		fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "unsupported cmd 0x%lx\n", cmd);
290235863739SMike Smith 		error = EINVAL;
290335863739SMike Smith 		break;
290435863739SMike Smith 	}
290535863739SMike Smith 	return(error);
290635863739SMike Smith }
290735863739SMike Smith 
2908b3457b51SScott Long static int
aac_poll(struct cdev * dev,int poll_events,struct thread * td)290900b4e54aSWarner Losh aac_poll(struct cdev *dev, int poll_events, struct thread *td)
2910b3457b51SScott Long {
2911b3457b51SScott Long 	struct aac_softc *sc;
2912ef0b687cSEd Maste 	struct aac_fib_context *ctx;
2913b3457b51SScott Long 	int revents;
2914b3457b51SScott Long 
2915b3457b51SScott Long 	sc = dev->si_drv1;
2916b3457b51SScott Long 	revents = 0;
2917b3457b51SScott Long 
2918bb6fe253SScott Long 	mtx_lock(&sc->aac_aifq_lock);
2919b3457b51SScott Long 	if ((poll_events & (POLLRDNORM | POLLIN)) != 0) {
2920ef0b687cSEd Maste 		for (ctx = sc->fibctx; ctx; ctx = ctx->next) {
2921ef0b687cSEd Maste 			if (ctx->ctx_idx != sc->aifq_idx || ctx->ctx_wrap) {
2922b3457b51SScott Long 				revents |= poll_events & (POLLIN | POLLRDNORM);
2923ef0b687cSEd Maste 				break;
2924ef0b687cSEd Maste 			}
2925ef0b687cSEd Maste 		}
2926b3457b51SScott Long 	}
2927bb6fe253SScott Long 	mtx_unlock(&sc->aac_aifq_lock);
2928b3457b51SScott Long 
2929b3457b51SScott Long 	if (revents == 0) {
2930b3457b51SScott Long 		if (poll_events & (POLLIN | POLLRDNORM))
2931b3457b51SScott Long 			selrecord(td, &sc->rcv_select);
2932b3457b51SScott Long 	}
2933b3457b51SScott Long 
2934b3457b51SScott Long 	return (revents);
2935b3457b51SScott Long }
2936b3457b51SScott Long 
29377cb209f5SScott Long static void
aac_ioctl_event(struct aac_softc * sc,struct aac_event * event,void * arg)29387cb209f5SScott Long aac_ioctl_event(struct aac_softc *sc, struct aac_event *event, void *arg)
29397cb209f5SScott Long {
29407cb209f5SScott Long 
29417cb209f5SScott Long 	switch (event->ev_type) {
29427cb209f5SScott Long 	case AAC_EVENT_CMFREE:
29430c40d5beSEd Maste 		mtx_assert(&sc->aac_io_lock, MA_OWNED);
29441a681311SLuoqi Chen 		if (aac_alloc_command(sc, (struct aac_command **)arg)) {
29457cb209f5SScott Long 			aac_add_event(sc, event);
29467cb209f5SScott Long 			return;
29477cb209f5SScott Long 		}
29487cb209f5SScott Long 		free(event, M_AACBUF);
29498eeb2ca6SScott Long 		wakeup(arg);
29507cb209f5SScott Long 		break;
29517cb209f5SScott Long 	default:
29527cb209f5SScott Long 		break;
29537cb209f5SScott Long 	}
29547cb209f5SScott Long }
29557cb209f5SScott Long 
2956914da7d0SScott Long /*
295735863739SMike Smith  * Send a FIB supplied from userspace
295835863739SMike Smith  */
295935863739SMike Smith static int
aac_ioctl_sendfib(struct aac_softc * sc,caddr_t ufib)296035863739SMike Smith aac_ioctl_sendfib(struct aac_softc *sc, caddr_t ufib)
296135863739SMike Smith {
296235863739SMike Smith 	struct aac_command *cm;
296335863739SMike Smith 	int size, error;
296435863739SMike Smith 
296531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
296635863739SMike Smith 
296735863739SMike Smith 	cm = NULL;
296835863739SMike Smith 
296935863739SMike Smith 	/*
297035863739SMike Smith 	 * Get a command
297135863739SMike Smith 	 */
2972bb6fe253SScott Long 	mtx_lock(&sc->aac_io_lock);
297335863739SMike Smith 	if (aac_alloc_command(sc, &cm)) {
29747cb209f5SScott Long 		struct aac_event *event;
29757cb209f5SScott Long 
29767cb209f5SScott Long 		event = malloc(sizeof(struct aac_event), M_AACBUF,
29777cb209f5SScott Long 		    M_NOWAIT | M_ZERO);
29787cb209f5SScott Long 		if (event == NULL) {
297935863739SMike Smith 			error = EBUSY;
2980f16627aaSEd Maste 			mtx_unlock(&sc->aac_io_lock);
298135863739SMike Smith 			goto out;
298235863739SMike Smith 		}
29837cb209f5SScott Long 		event->ev_type = AAC_EVENT_CMFREE;
29847cb209f5SScott Long 		event->ev_callback = aac_ioctl_event;
29857cb209f5SScott Long 		event->ev_arg = &cm;
29867cb209f5SScott Long 		aac_add_event(sc, event);
29878eeb2ca6SScott Long 		msleep(&cm, &sc->aac_io_lock, 0, "sendfib", 0);
29887cb209f5SScott Long 	}
298993cfca22SScott Long 	mtx_unlock(&sc->aac_io_lock);
299035863739SMike Smith 
299135863739SMike Smith 	/*
299235863739SMike Smith 	 * Fetch the FIB header, then re-copy to get data as well.
299335863739SMike Smith 	 */
2994914da7d0SScott Long 	if ((error = copyin(ufib, cm->cm_fib,
2995914da7d0SScott Long 			    sizeof(struct aac_fib_header))) != 0)
299635863739SMike Smith 		goto out;
299735863739SMike Smith 	size = cm->cm_fib->Header.Size + sizeof(struct aac_fib_header);
2998f355c0e0SEd Maste 	if (size > sc->aac_max_fib_size) {
2999f355c0e0SEd Maste 		device_printf(sc->aac_dev, "incoming FIB oversized (%d > %d)\n",
3000f355c0e0SEd Maste 			      size, sc->aac_max_fib_size);
3001f355c0e0SEd Maste 		size = sc->aac_max_fib_size;
300235863739SMike Smith 	}
300335863739SMike Smith 	if ((error = copyin(ufib, cm->cm_fib, size)) != 0)
300435863739SMike Smith 		goto out;
300535863739SMike Smith 	cm->cm_fib->Header.Size = size;
30062b3b0f17SScott Long 	cm->cm_timestamp = time_uptime;
300735863739SMike Smith 
300835863739SMike Smith 	/*
300935863739SMike Smith 	 * Pass the FIB to the controller, wait for it to complete.
301035863739SMike Smith 	 */
301193cfca22SScott Long 	mtx_lock(&sc->aac_io_lock);
3012f16627aaSEd Maste 	error = aac_wait_command(cm);
3013f16627aaSEd Maste 	mtx_unlock(&sc->aac_io_lock);
3014f16627aaSEd Maste 	if (error != 0) {
301570545d1aSScott Long 		device_printf(sc->aac_dev,
301670545d1aSScott Long 			      "aac_wait_command return %d\n", error);
301735863739SMike Smith 		goto out;
3018b3457b51SScott Long 	}
301935863739SMike Smith 
302035863739SMike Smith 	/*
302135863739SMike Smith 	 * Copy the FIB and data back out to the caller.
302235863739SMike Smith 	 */
302335863739SMike Smith 	size = cm->cm_fib->Header.Size;
3024f355c0e0SEd Maste 	if (size > sc->aac_max_fib_size) {
3025f355c0e0SEd Maste 		device_printf(sc->aac_dev, "outbound FIB oversized (%d > %d)\n",
3026f355c0e0SEd Maste 			      size, sc->aac_max_fib_size);
3027f355c0e0SEd Maste 		size = sc->aac_max_fib_size;
302835863739SMike Smith 	}
302935863739SMike Smith 	error = copyout(cm->cm_fib, ufib, size);
303035863739SMike Smith 
303135863739SMike Smith out:
3032f6c4dd3fSScott Long 	if (cm != NULL) {
3033f16627aaSEd Maste 		mtx_lock(&sc->aac_io_lock);
303435863739SMike Smith 		aac_release_command(cm);
3035bb6fe253SScott Long 		mtx_unlock(&sc->aac_io_lock);
3036f16627aaSEd Maste 	}
303735863739SMike Smith 	return(error);
303835863739SMike Smith }
303935863739SMike Smith 
3040914da7d0SScott Long /*
3041f355c0e0SEd Maste  * Send a passthrough FIB supplied from userspace
3042f355c0e0SEd Maste  */
3043f355c0e0SEd Maste static int
aac_ioctl_send_raw_srb(struct aac_softc * sc,caddr_t arg)3044f355c0e0SEd Maste aac_ioctl_send_raw_srb(struct aac_softc *sc, caddr_t arg)
3045f355c0e0SEd Maste {
30467b90e5ecSAttilio Rao 	struct aac_command *cm;
30477b90e5ecSAttilio Rao 	struct aac_event *event;
30487b90e5ecSAttilio Rao 	struct aac_fib *fib;
30497b90e5ecSAttilio Rao 	struct aac_srb *srbcmd, *user_srb;
30507b90e5ecSAttilio Rao 	struct aac_sg_entry *sge;
30517b90e5ecSAttilio Rao 	void *srb_sg_address, *ureply;
30527b90e5ecSAttilio Rao 	uint32_t fibsize, srb_sg_bytecount;
30537b90e5ecSAttilio Rao 	int error, transfer_data;
30547b90e5ecSAttilio Rao 
30557b90e5ecSAttilio Rao 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
30567b90e5ecSAttilio Rao 
30577b90e5ecSAttilio Rao 	cm = NULL;
30587b90e5ecSAttilio Rao 	transfer_data = 0;
30597b90e5ecSAttilio Rao 	fibsize = 0;
30607b90e5ecSAttilio Rao 	user_srb = (struct aac_srb *)arg;
30617b90e5ecSAttilio Rao 
30627b90e5ecSAttilio Rao 	mtx_lock(&sc->aac_io_lock);
30637b90e5ecSAttilio Rao 	if (aac_alloc_command(sc, &cm)) {
30647b90e5ecSAttilio Rao 		 event = malloc(sizeof(struct aac_event), M_AACBUF,
30657b90e5ecSAttilio Rao 		    M_NOWAIT | M_ZERO);
30667b90e5ecSAttilio Rao 		if (event == NULL) {
30677b90e5ecSAttilio Rao 			error = EBUSY;
30687b90e5ecSAttilio Rao 			mtx_unlock(&sc->aac_io_lock);
30697b90e5ecSAttilio Rao 			goto out;
30707b90e5ecSAttilio Rao 		}
30717b90e5ecSAttilio Rao 		event->ev_type = AAC_EVENT_CMFREE;
30727b90e5ecSAttilio Rao 		event->ev_callback = aac_ioctl_event;
30737b90e5ecSAttilio Rao 		event->ev_arg = &cm;
30747b90e5ecSAttilio Rao 		aac_add_event(sc, event);
30757b90e5ecSAttilio Rao 		msleep(cm, &sc->aac_io_lock, 0, "aacraw", 0);
30767b90e5ecSAttilio Rao 	}
30777b90e5ecSAttilio Rao 	mtx_unlock(&sc->aac_io_lock);
30787b90e5ecSAttilio Rao 
30797b90e5ecSAttilio Rao 	cm->cm_data = NULL;
30807b90e5ecSAttilio Rao 	fib = cm->cm_fib;
30817b90e5ecSAttilio Rao 	srbcmd = (struct aac_srb *)fib->data;
30827b90e5ecSAttilio Rao 	error = copyin(&user_srb->data_len, &fibsize, sizeof(uint32_t));
30837b90e5ecSAttilio Rao 	if (error != 0)
30847b90e5ecSAttilio Rao 		goto out;
30857b90e5ecSAttilio Rao 	if (fibsize > (sc->aac_max_fib_size - sizeof(struct aac_fib_header))) {
30867b90e5ecSAttilio Rao 		error = EINVAL;
30877b90e5ecSAttilio Rao 		goto out;
30887b90e5ecSAttilio Rao 	}
30897b90e5ecSAttilio Rao 	error = copyin(user_srb, srbcmd, fibsize);
30907b90e5ecSAttilio Rao 	if (error != 0)
30917b90e5ecSAttilio Rao 		goto out;
30927b90e5ecSAttilio Rao 	srbcmd->function = 0;
30937b90e5ecSAttilio Rao 	srbcmd->retry_limit = 0;
30947b90e5ecSAttilio Rao 	if (srbcmd->sg_map.SgCount > 1) {
30957b90e5ecSAttilio Rao 		error = EINVAL;
30967b90e5ecSAttilio Rao 		goto out;
30977b90e5ecSAttilio Rao 	}
30987b90e5ecSAttilio Rao 
30997b90e5ecSAttilio Rao 	/* Retrieve correct SG entries. */
31007b90e5ecSAttilio Rao 	if (fibsize == (sizeof(struct aac_srb) +
31017b90e5ecSAttilio Rao 	    srbcmd->sg_map.SgCount * sizeof(struct aac_sg_entry))) {
3102f4a18258SSean Bruno 		struct aac_sg_entry sg;
3103f4a18258SSean Bruno 
31047b90e5ecSAttilio Rao 		sge = srbcmd->sg_map.SgEntry;
3105f4a18258SSean Bruno 
3106f4a18258SSean Bruno 		if ((error = copyin(sge, &sg, sizeof(sg))) != 0)
3107f4a18258SSean Bruno 			goto out;
3108f4a18258SSean Bruno 
3109f4a18258SSean Bruno 		srb_sg_bytecount = sg.SgByteCount;
3110f4a18258SSean Bruno 		srb_sg_address = (void *)(uintptr_t)sg.SgAddress;
31117b90e5ecSAttilio Rao 	}
31127b90e5ecSAttilio Rao #ifdef __amd64__
31137b90e5ecSAttilio Rao 	else if (fibsize == (sizeof(struct aac_srb) +
31147b90e5ecSAttilio Rao 	    srbcmd->sg_map.SgCount * sizeof(struct aac_sg_entry64))) {
31155cd9d254SJohn Baldwin 		struct aac_sg_entry64 *sge64;
3116f4a18258SSean Bruno 		struct aac_sg_entry64 sg;
3117f4a18258SSean Bruno 
31187b90e5ecSAttilio Rao 		sge = NULL;
31197b90e5ecSAttilio Rao 		sge64 = (struct aac_sg_entry64 *)srbcmd->sg_map.SgEntry;
3120f4a18258SSean Bruno 
3121f4a18258SSean Bruno 		if ((error = copyin(sge64, &sg, sizeof(sg))) != 0)
3122f4a18258SSean Bruno 			goto out;
3123f4a18258SSean Bruno 
3124f4a18258SSean Bruno 		srb_sg_bytecount = sg.SgByteCount;
3125f4a18258SSean Bruno 		srb_sg_address = (void *)sg.SgAddress;
31267b90e5ecSAttilio Rao 		if (sge64->SgAddress > 0xffffffffull &&
31277b90e5ecSAttilio Rao 		    (sc->flags & AAC_FLAGS_SG_64BIT) == 0) {
31287b90e5ecSAttilio Rao 			error = EINVAL;
31297b90e5ecSAttilio Rao 			goto out;
31307b90e5ecSAttilio Rao 		}
31317b90e5ecSAttilio Rao 	}
31327b90e5ecSAttilio Rao #endif
31337b90e5ecSAttilio Rao 	else {
31347b90e5ecSAttilio Rao 		error = EINVAL;
31357b90e5ecSAttilio Rao 		goto out;
31367b90e5ecSAttilio Rao 	}
31377b90e5ecSAttilio Rao 	ureply = (char *)arg + fibsize;
31387b90e5ecSAttilio Rao 	srbcmd->data_len = srb_sg_bytecount;
31397b90e5ecSAttilio Rao 	if (srbcmd->sg_map.SgCount == 1)
31407b90e5ecSAttilio Rao 		transfer_data = 1;
31417b90e5ecSAttilio Rao 
31427b90e5ecSAttilio Rao 	cm->cm_sgtable = (struct aac_sg_table *)&srbcmd->sg_map;
31437b90e5ecSAttilio Rao 	if (transfer_data) {
31447b90e5ecSAttilio Rao 		cm->cm_datalen = srb_sg_bytecount;
31457b90e5ecSAttilio Rao 		cm->cm_data = malloc(cm->cm_datalen, M_AACBUF, M_NOWAIT);
31467b90e5ecSAttilio Rao 		if (cm->cm_data == NULL) {
31477b90e5ecSAttilio Rao 			error = ENOMEM;
31487b90e5ecSAttilio Rao 			goto out;
31497b90e5ecSAttilio Rao 		}
31507b90e5ecSAttilio Rao 		if (srbcmd->flags & AAC_SRB_FLAGS_DATA_IN)
31517b90e5ecSAttilio Rao 			cm->cm_flags |= AAC_CMD_DATAIN;
31527b90e5ecSAttilio Rao 		if (srbcmd->flags & AAC_SRB_FLAGS_DATA_OUT) {
31537b90e5ecSAttilio Rao 			cm->cm_flags |= AAC_CMD_DATAOUT;
31547b90e5ecSAttilio Rao 			error = copyin(srb_sg_address, cm->cm_data,
31557b90e5ecSAttilio Rao 			    cm->cm_datalen);
31567b90e5ecSAttilio Rao 			if (error != 0)
31577b90e5ecSAttilio Rao 				goto out;
31587b90e5ecSAttilio Rao 		}
31597b90e5ecSAttilio Rao 	}
31607b90e5ecSAttilio Rao 
31617b90e5ecSAttilio Rao 	fib->Header.Size = sizeof(struct aac_fib_header) +
31627b90e5ecSAttilio Rao 	    sizeof(struct aac_srb);
31637b90e5ecSAttilio Rao 	fib->Header.XferState =
31647b90e5ecSAttilio Rao 	    AAC_FIBSTATE_HOSTOWNED   |
31657b90e5ecSAttilio Rao 	    AAC_FIBSTATE_INITIALISED |
31667b90e5ecSAttilio Rao 	    AAC_FIBSTATE_EMPTY       |
31677b90e5ecSAttilio Rao 	    AAC_FIBSTATE_FROMHOST    |
31687b90e5ecSAttilio Rao 	    AAC_FIBSTATE_REXPECTED   |
31697b90e5ecSAttilio Rao 	    AAC_FIBSTATE_NORM        |
31707b90e5ecSAttilio Rao 	    AAC_FIBSTATE_ASYNC       |
31717b90e5ecSAttilio Rao 	    AAC_FIBSTATE_FAST_RESPONSE;
31727b90e5ecSAttilio Rao 	fib->Header.Command = (sc->flags & AAC_FLAGS_SG_64BIT) != 0 ?
31737b90e5ecSAttilio Rao 	    ScsiPortCommandU64 : ScsiPortCommand;
31747b90e5ecSAttilio Rao 
31757b90e5ecSAttilio Rao 	mtx_lock(&sc->aac_io_lock);
31767b90e5ecSAttilio Rao 	aac_wait_command(cm);
31777b90e5ecSAttilio Rao 	mtx_unlock(&sc->aac_io_lock);
31787b90e5ecSAttilio Rao 
31797b90e5ecSAttilio Rao 	if (transfer_data && (srbcmd->flags & AAC_SRB_FLAGS_DATA_IN) != 0) {
31807b90e5ecSAttilio Rao 		error = copyout(cm->cm_data, srb_sg_address, cm->cm_datalen);
31817b90e5ecSAttilio Rao 		if (error != 0)
31827b90e5ecSAttilio Rao 			goto out;
31837b90e5ecSAttilio Rao 	}
31847b90e5ecSAttilio Rao 	error = copyout(fib->data, ureply, sizeof(struct aac_srb_response));
31857b90e5ecSAttilio Rao out:
31867b90e5ecSAttilio Rao 	if (cm != NULL) {
31877b90e5ecSAttilio Rao 		if (cm->cm_data != NULL)
31887b90e5ecSAttilio Rao 			free(cm->cm_data, M_AACBUF);
31897b90e5ecSAttilio Rao 		mtx_lock(&sc->aac_io_lock);
31907b90e5ecSAttilio Rao 		aac_release_command(cm);
31917b90e5ecSAttilio Rao 		mtx_unlock(&sc->aac_io_lock);
31927b90e5ecSAttilio Rao 	}
31937b90e5ecSAttilio Rao 	return(error);
3194f355c0e0SEd Maste }
3195f355c0e0SEd Maste 
3196f355c0e0SEd Maste /*
3197dfe2c294SAttilio Rao  * cdevpriv interface private destructor.
3198dfe2c294SAttilio Rao  */
3199dfe2c294SAttilio Rao static void
aac_cdevpriv_dtor(void * arg)3200dfe2c294SAttilio Rao aac_cdevpriv_dtor(void *arg)
3201dfe2c294SAttilio Rao {
3202dfe2c294SAttilio Rao 	struct aac_softc *sc;
3203dfe2c294SAttilio Rao 
3204dfe2c294SAttilio Rao 	sc = arg;
3205dfe2c294SAttilio Rao 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
3206dfe2c294SAttilio Rao 	device_unbusy(sc->aac_dev);
3207dfe2c294SAttilio Rao }
3208dfe2c294SAttilio Rao 
3209dfe2c294SAttilio Rao /*
321035863739SMike Smith  * Handle an AIF sent to us by the controller; queue it for later reference.
321136e0bf6eSScott Long  * If the queue fills up, then drop the older entries.
321235863739SMike Smith  */
321335863739SMike Smith static void
aac_handle_aif(struct aac_softc * sc,struct aac_fib * fib)321436e0bf6eSScott Long aac_handle_aif(struct aac_softc *sc, struct aac_fib *fib)
321535863739SMike Smith {
321636e0bf6eSScott Long 	struct aac_aif_command *aif;
321736e0bf6eSScott Long 	struct aac_container *co, *co_next;
3218a723a548SEd Maste 	struct aac_fib_context *ctx;
321904f4d586SEd Maste 	struct aac_mntinforesp *mir;
3220a723a548SEd Maste 	int next, current, found;
3221795d7dc0SScott Long 	int count = 0, added = 0, i = 0;
3222851f59d7SEd Maste 	uint32_t channel;
322335863739SMike Smith 
322431a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
322535863739SMike Smith 
322636e0bf6eSScott Long 	aif = (struct aac_aif_command*)&fib->data[0];
322736e0bf6eSScott Long 	aac_print_aif(sc, aif);
322836e0bf6eSScott Long 
322936e0bf6eSScott Long 	/* Is it an event that we should care about? */
323036e0bf6eSScott Long 	switch (aif->command) {
323136e0bf6eSScott Long 	case AifCmdEventNotify:
323236e0bf6eSScott Long 		switch (aif->data.EN.type) {
323336e0bf6eSScott Long 		case AifEnAddContainer:
323436e0bf6eSScott Long 		case AifEnDeleteContainer:
323536e0bf6eSScott Long 			/*
3236914da7d0SScott Long 			 * A container was added or deleted, but the message
3237914da7d0SScott Long 			 * doesn't tell us anything else!  Re-enumerate the
3238914da7d0SScott Long 			 * containers and sort things out.
323936e0bf6eSScott Long 			 */
324003b5fe51SScott Long 			aac_alloc_sync_fib(sc, &fib);
324136e0bf6eSScott Long 			do {
324236e0bf6eSScott Long 				/*
3243914da7d0SScott Long 				 * Ask the controller for its containers one at
3244914da7d0SScott Long 				 * a time.
3245914da7d0SScott Long 				 * XXX What if the controller's list changes
3246914da7d0SScott Long 				 * midway through this enumaration?
324736e0bf6eSScott Long 				 * XXX This should be done async.
324836e0bf6eSScott Long 				 */
324904f4d586SEd Maste 				if ((mir = aac_get_container_info(sc, fib, i)) == NULL)
325036e0bf6eSScott Long 					continue;
325104f4d586SEd Maste 				if (i == 0)
3252795d7dc0SScott Long 					count = mir->MntRespCount;
325336e0bf6eSScott Long 				/*
3254914da7d0SScott Long 				 * Check the container against our list.
3255914da7d0SScott Long 				 * co->co_found was already set to 0 in a
3256914da7d0SScott Long 				 * previous run.
325736e0bf6eSScott Long 				 */
3258cbfd045bSScott Long 				if ((mir->Status == ST_OK) &&
3259cbfd045bSScott Long 				    (mir->MntTable[0].VolType != CT_NONE)) {
326036e0bf6eSScott Long 					found = 0;
3261914da7d0SScott Long 					TAILQ_FOREACH(co,
3262914da7d0SScott Long 						      &sc->aac_container_tqh,
3263914da7d0SScott Long 						      co_link) {
326436e0bf6eSScott Long 						if (co->co_mntobj.ObjectId ==
3265cbfd045bSScott Long 						    mir->MntTable[0].ObjectId) {
326636e0bf6eSScott Long 							co->co_found = 1;
326736e0bf6eSScott Long 							found = 1;
326836e0bf6eSScott Long 							break;
326936e0bf6eSScott Long 						}
327036e0bf6eSScott Long 					}
3271914da7d0SScott Long 					/*
3272914da7d0SScott Long 					 * If the container matched, continue
3273914da7d0SScott Long 					 * in the list.
3274914da7d0SScott Long 					 */
327536e0bf6eSScott Long 					if (found) {
327636e0bf6eSScott Long 						i++;
327736e0bf6eSScott Long 						continue;
327836e0bf6eSScott Long 					}
327936e0bf6eSScott Long 
328036e0bf6eSScott Long 					/*
3281914da7d0SScott Long 					 * This is a new container.  Do all the
328270545d1aSScott Long 					 * appropriate things to set it up.
328370545d1aSScott Long 					 */
3284cbfd045bSScott Long 					aac_add_container(sc, mir, 1);
328536e0bf6eSScott Long 					added = 1;
328636e0bf6eSScott Long 				}
328736e0bf6eSScott Long 				i++;
3288795d7dc0SScott Long 			} while ((i < count) && (i < AAC_MAX_CONTAINERS));
3289cbfd045bSScott Long 			aac_release_sync_fib(sc);
329036e0bf6eSScott Long 
329136e0bf6eSScott Long 			/*
3292914da7d0SScott Long 			 * Go through our list of containers and see which ones
3293914da7d0SScott Long 			 * were not marked 'found'.  Since the controller didn't
3294914da7d0SScott Long 			 * list them they must have been deleted.  Do the
3295914da7d0SScott Long 			 * appropriate steps to destroy the device.  Also reset
3296914da7d0SScott Long 			 * the co->co_found field.
329736e0bf6eSScott Long 			 */
329836e0bf6eSScott Long 			co = TAILQ_FIRST(&sc->aac_container_tqh);
329936e0bf6eSScott Long 			while (co != NULL) {
330036e0bf6eSScott Long 				if (co->co_found == 0) {
33017cb209f5SScott Long 					mtx_unlock(&sc->aac_io_lock);
3302c6df6f53SWarner Losh 					bus_topo_lock();
3303914da7d0SScott Long 					device_delete_child(sc->aac_dev,
3304914da7d0SScott Long 							    co->co_disk);
3305c6df6f53SWarner Losh 					bus_topo_unlock();
33067cb209f5SScott Long 					mtx_lock(&sc->aac_io_lock);
330736e0bf6eSScott Long 					co_next = TAILQ_NEXT(co, co_link);
3308bb6fe253SScott Long 					mtx_lock(&sc->aac_container_lock);
3309914da7d0SScott Long 					TAILQ_REMOVE(&sc->aac_container_tqh, co,
3310914da7d0SScott Long 						     co_link);
3311bb6fe253SScott Long 					mtx_unlock(&sc->aac_container_lock);
3312ba1d57e7SScott Long 					free(co, M_AACBUF);
331336e0bf6eSScott Long 					co = co_next;
331436e0bf6eSScott Long 				} else {
331536e0bf6eSScott Long 					co->co_found = 0;
331636e0bf6eSScott Long 					co = TAILQ_NEXT(co, co_link);
331736e0bf6eSScott Long 				}
331836e0bf6eSScott Long 			}
331936e0bf6eSScott Long 
332036e0bf6eSScott Long 			/* Attach the newly created containers */
33217cb209f5SScott Long 			if (added) {
33227cb209f5SScott Long 				mtx_unlock(&sc->aac_io_lock);
3323c6df6f53SWarner Losh 				bus_topo_lock();
332418250ec6SJohn Baldwin 				bus_attach_children(sc->aac_dev);
3325c6df6f53SWarner Losh 				bus_topo_unlock();
33267cb209f5SScott Long 				mtx_lock(&sc->aac_io_lock);
33277cb209f5SScott Long 			}
332836e0bf6eSScott Long 
332936e0bf6eSScott Long 			break;
333036e0bf6eSScott Long 
3331851f59d7SEd Maste 		case AifEnEnclosureManagement:
3332851f59d7SEd Maste 			switch (aif->data.EN.data.EEE.eventType) {
3333851f59d7SEd Maste 			case AIF_EM_DRIVE_INSERTION:
3334851f59d7SEd Maste 			case AIF_EM_DRIVE_REMOVAL:
3335851f59d7SEd Maste 				channel = aif->data.EN.data.EEE.unitID;
3336851f59d7SEd Maste 				if (sc->cam_rescan_cb != NULL)
3337851f59d7SEd Maste 					sc->cam_rescan_cb(sc,
3338851f59d7SEd Maste 					    (channel >> 24) & 0xF,
3339851f59d7SEd Maste 					    (channel & 0xFFFF));
3340851f59d7SEd Maste 				break;
3341851f59d7SEd Maste 			}
3342851f59d7SEd Maste 			break;
3343851f59d7SEd Maste 
3344851f59d7SEd Maste 		case AifEnAddJBOD:
3345851f59d7SEd Maste 		case AifEnDeleteJBOD:
3346851f59d7SEd Maste 			channel = aif->data.EN.data.ECE.container;
3347851f59d7SEd Maste 			if (sc->cam_rescan_cb != NULL)
3348851f59d7SEd Maste 				sc->cam_rescan_cb(sc, (channel >> 24) & 0xF,
3349851f59d7SEd Maste 				    AAC_CAM_TARGET_WILDCARD);
3350851f59d7SEd Maste 			break;
3351851f59d7SEd Maste 
335236e0bf6eSScott Long 		default:
335336e0bf6eSScott Long 			break;
335436e0bf6eSScott Long 		}
335536e0bf6eSScott Long 
335636e0bf6eSScott Long 	default:
335736e0bf6eSScott Long 		break;
335836e0bf6eSScott Long 	}
335936e0bf6eSScott Long 
336036e0bf6eSScott Long 	/* Copy the AIF data to the AIF queue for ioctl retrieval */
3361bb6fe253SScott Long 	mtx_lock(&sc->aac_aifq_lock);
3362a723a548SEd Maste 	current = sc->aifq_idx;
3363a723a548SEd Maste 	next = (current + 1) % AAC_AIFQ_LENGTH;
3364a723a548SEd Maste 	if (next == 0)
3365a723a548SEd Maste 		sc->aifq_filled = 1;
3366a723a548SEd Maste 	bcopy(fib, &sc->aac_aifq[current], sizeof(struct aac_fib));
3367a723a548SEd Maste 	/* modify AIF contexts */
3368a723a548SEd Maste 	if (sc->aifq_filled) {
3369a723a548SEd Maste 		for (ctx = sc->fibctx; ctx; ctx = ctx->next) {
3370a723a548SEd Maste 			if (next == ctx->ctx_idx)
3371a723a548SEd Maste 				ctx->ctx_wrap = 1;
3372a723a548SEd Maste 			else if (current == ctx->ctx_idx && ctx->ctx_wrap)
3373a723a548SEd Maste 				ctx->ctx_idx = next;
3374a723a548SEd Maste 		}
3375a723a548SEd Maste 	}
3376a723a548SEd Maste 	sc->aifq_idx = next;
3377b3457b51SScott Long 	/* On the off chance that someone is sleeping for an aif... */
337835863739SMike Smith 	if (sc->aac_state & AAC_STATE_AIF_SLEEPER)
337935863739SMike Smith 		wakeup(sc->aac_aifq);
3380b3457b51SScott Long 	/* Wakeup any poll()ers */
3381512824f8SSeigo Tanimura 	selwakeuppri(&sc->rcv_select, PRIBIO);
3382bb6fe253SScott Long 	mtx_unlock(&sc->aac_aifq_lock);
338335863739SMike Smith }
338435863739SMike Smith 
3385914da7d0SScott Long /*
33860b94a66eSMike Smith  * Return the Revision of the driver to userspace and check to see if the
338736e0bf6eSScott Long  * userspace app is possibly compatible.  This is extremely bogus since
338836e0bf6eSScott Long  * our driver doesn't follow Adaptec's versioning system.  Cheat by just
338936e0bf6eSScott Long  * returning what the card reported.
339035863739SMike Smith  */
339135863739SMike Smith static int
aac_rev_check(struct aac_softc * sc,caddr_t udata)3392fb0c27d7SScott Long aac_rev_check(struct aac_softc *sc, caddr_t udata)
339335863739SMike Smith {
339435863739SMike Smith 	struct aac_rev_check rev_check;
339535863739SMike Smith 	struct aac_rev_check_resp rev_check_resp;
339635863739SMike Smith 	int error = 0;
339735863739SMike Smith 
339831a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
339935863739SMike Smith 
340035863739SMike Smith 	/*
340135863739SMike Smith 	 * Copyin the revision struct from userspace
340235863739SMike Smith 	 */
3403c6eafcf2SScott Long 	if ((error = copyin(udata, (caddr_t)&rev_check,
3404c6eafcf2SScott Long 			sizeof(struct aac_rev_check))) != 0) {
340535863739SMike Smith 		return error;
340635863739SMike Smith 	}
340735863739SMike Smith 
340831a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_IOCTL_COMMANDS_B, "Userland revision= %d\n",
3409914da7d0SScott Long 	      rev_check.callingRevision.buildNumber);
341035863739SMike Smith 
341135863739SMike Smith 	/*
341235863739SMike Smith 	 * Doctor up the response struct.
341335863739SMike Smith 	 */
341435863739SMike Smith 	rev_check_resp.possiblyCompatible = 1;
34158e7e6335SEd Maste 	rev_check_resp.adapterSWRevision.external.comp.major =
34168e7e6335SEd Maste 	    AAC_DRIVER_MAJOR_VERSION;
34178e7e6335SEd Maste 	rev_check_resp.adapterSWRevision.external.comp.minor =
34188e7e6335SEd Maste 	    AAC_DRIVER_MINOR_VERSION;
34198e7e6335SEd Maste 	rev_check_resp.adapterSWRevision.external.comp.type =
34208e7e6335SEd Maste 	    AAC_DRIVER_TYPE;
34218e7e6335SEd Maste 	rev_check_resp.adapterSWRevision.external.comp.dash =
34228e7e6335SEd Maste 	    AAC_DRIVER_BUGFIX_LEVEL;
3423914da7d0SScott Long 	rev_check_resp.adapterSWRevision.buildNumber =
34248e7e6335SEd Maste 	    AAC_DRIVER_BUILD;
342535863739SMike Smith 
3426c6eafcf2SScott Long 	return(copyout((caddr_t)&rev_check_resp, udata,
3427c6eafcf2SScott Long 			sizeof(struct aac_rev_check_resp)));
342835863739SMike Smith }
342935863739SMike Smith 
3430914da7d0SScott Long /*
3431a723a548SEd Maste  * Pass the fib context to the caller
3432a723a548SEd Maste  */
3433a723a548SEd Maste static int
aac_open_aif(struct aac_softc * sc,caddr_t arg)3434a723a548SEd Maste aac_open_aif(struct aac_softc *sc, caddr_t arg)
3435a723a548SEd Maste {
3436a723a548SEd Maste 	struct aac_fib_context *fibctx, *ctx;
3437a723a548SEd Maste 	int error = 0;
3438a723a548SEd Maste 
343931a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
3440a723a548SEd Maste 
3441a723a548SEd Maste 	fibctx = malloc(sizeof(struct aac_fib_context), M_AACBUF, M_NOWAIT|M_ZERO);
3442a723a548SEd Maste 	if (fibctx == NULL)
3443a723a548SEd Maste 		return (ENOMEM);
3444a723a548SEd Maste 
3445a723a548SEd Maste 	mtx_lock(&sc->aac_aifq_lock);
3446a723a548SEd Maste 	/* all elements are already 0, add to queue */
3447a723a548SEd Maste 	if (sc->fibctx == NULL)
3448a723a548SEd Maste 		sc->fibctx = fibctx;
3449a723a548SEd Maste 	else {
3450a723a548SEd Maste 		for (ctx = sc->fibctx; ctx->next; ctx = ctx->next)
3451a723a548SEd Maste 			;
3452a723a548SEd Maste 		ctx->next = fibctx;
3453a723a548SEd Maste 		fibctx->prev = ctx;
3454a723a548SEd Maste 	}
3455a723a548SEd Maste 
3456a723a548SEd Maste 	/* evaluate unique value */
3457a723a548SEd Maste 	fibctx->unique = (*(u_int32_t *)&fibctx & 0xffffffff);
3458a723a548SEd Maste 	ctx = sc->fibctx;
3459a723a548SEd Maste 	while (ctx != fibctx) {
3460a723a548SEd Maste 		if (ctx->unique == fibctx->unique) {
3461a723a548SEd Maste 			fibctx->unique++;
3462a723a548SEd Maste 			ctx = sc->fibctx;
3463a723a548SEd Maste 		} else {
3464a723a548SEd Maste 			ctx = ctx->next;
3465a723a548SEd Maste 		}
3466a723a548SEd Maste 	}
3467a723a548SEd Maste 	mtx_unlock(&sc->aac_aifq_lock);
3468a723a548SEd Maste 
3469a723a548SEd Maste 	error = copyout(&fibctx->unique, (void *)arg, sizeof(u_int32_t));
3470a723a548SEd Maste 	if (error)
3471a723a548SEd Maste 		aac_close_aif(sc, (caddr_t)ctx);
3472a723a548SEd Maste 	return error;
3473a723a548SEd Maste }
3474a723a548SEd Maste 
3475a723a548SEd Maste /*
3476a723a548SEd Maste  * Close the caller's fib context
3477a723a548SEd Maste  */
3478a723a548SEd Maste static int
aac_close_aif(struct aac_softc * sc,caddr_t arg)3479a723a548SEd Maste aac_close_aif(struct aac_softc *sc, caddr_t arg)
3480a723a548SEd Maste {
3481a723a548SEd Maste 	struct aac_fib_context *ctx;
3482a723a548SEd Maste 
348331a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
3484a723a548SEd Maste 
3485a723a548SEd Maste 	mtx_lock(&sc->aac_aifq_lock);
3486a723a548SEd Maste 	for (ctx = sc->fibctx; ctx; ctx = ctx->next) {
3487a723a548SEd Maste 		if (ctx->unique == *(uint32_t *)&arg) {
3488a723a548SEd Maste 			if (ctx == sc->fibctx)
3489a723a548SEd Maste 				sc->fibctx = NULL;
3490a723a548SEd Maste 			else {
3491a723a548SEd Maste 				ctx->prev->next = ctx->next;
3492a723a548SEd Maste 				if (ctx->next)
3493a723a548SEd Maste 					ctx->next->prev = ctx->prev;
3494a723a548SEd Maste 			}
3495a723a548SEd Maste 			break;
3496a723a548SEd Maste 		}
3497a723a548SEd Maste 	}
3498a723a548SEd Maste 	mtx_unlock(&sc->aac_aifq_lock);
3499a723a548SEd Maste 	if (ctx)
3500a723a548SEd Maste 		free(ctx, M_AACBUF);
3501a723a548SEd Maste 
3502a723a548SEd Maste 	return 0;
3503a723a548SEd Maste }
3504a723a548SEd Maste 
3505a723a548SEd Maste /*
350635863739SMike Smith  * Pass the caller the next AIF in their queue
350735863739SMike Smith  */
350835863739SMike Smith static int
aac_getnext_aif(struct aac_softc * sc,caddr_t arg)3509fb0c27d7SScott Long aac_getnext_aif(struct aac_softc *sc, caddr_t arg)
351035863739SMike Smith {
351135863739SMike Smith 	struct get_adapter_fib_ioctl agf;
3512a723a548SEd Maste 	struct aac_fib_context *ctx;
35139e2e96d8SScott Long 	int error;
351435863739SMike Smith 
351531a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
351635863739SMike Smith 
3517f287c3e4SBrooks Davis #ifdef COMPAT_FREEBSD32
3518f287c3e4SBrooks Davis 	if (SV_CURPROC_FLAG(SV_ILP32)) {
3519f287c3e4SBrooks Davis 		struct get_adapter_fib_ioctl32 agf32;
3520f287c3e4SBrooks Davis 		error = copyin(arg, &agf32, sizeof(agf32));
3521f287c3e4SBrooks Davis 		if (error == 0) {
3522f287c3e4SBrooks Davis 			agf.AdapterFibContext = agf32.AdapterFibContext;
3523f287c3e4SBrooks Davis 			agf.Wait = agf32.Wait;
3524f287c3e4SBrooks Davis 			agf.AifFib = (caddr_t)(uintptr_t)agf32.AifFib;
3525f287c3e4SBrooks Davis 		}
3526f287c3e4SBrooks Davis 	} else
3527f287c3e4SBrooks Davis #endif
3528f287c3e4SBrooks Davis 		error = copyin(arg, &agf, sizeof(agf));
3529f287c3e4SBrooks Davis 	if (error == 0) {
3530a723a548SEd Maste 		for (ctx = sc->fibctx; ctx; ctx = ctx->next) {
3531a723a548SEd Maste 			if (agf.AdapterFibContext == ctx->unique)
3532a723a548SEd Maste 				break;
3533a723a548SEd Maste 		}
3534a723a548SEd Maste 		if (!ctx)
3535a723a548SEd Maste 			return (EFAULT);
353635863739SMike Smith 
3537a723a548SEd Maste 		error = aac_return_aif(sc, ctx, agf.AifFib);
3538a723a548SEd Maste 		if (error == EAGAIN && agf.Wait) {
353931a0399eSEd Maste 			fwprintf(sc, HBA_FLAGS_DBG_AIF_B, "aac_getnext_aif(): waiting for AIF");
354035863739SMike Smith 			sc->aac_state |= AAC_STATE_AIF_SLEEPER;
354135863739SMike Smith 			while (error == EAGAIN) {
3542914da7d0SScott Long 				error = tsleep(sc->aac_aifq, PRIBIO |
3543914da7d0SScott Long 					       PCATCH, "aacaif", 0);
354435863739SMike Smith 				if (error == 0)
3545a723a548SEd Maste 					error = aac_return_aif(sc, ctx, agf.AifFib);
354635863739SMike Smith 			}
354735863739SMike Smith 			sc->aac_state &= ~AAC_STATE_AIF_SLEEPER;
354835863739SMike Smith 		}
354935863739SMike Smith 	}
355035863739SMike Smith 	return(error);
355135863739SMike Smith }
355235863739SMike Smith 
3553914da7d0SScott Long /*
35540b94a66eSMike Smith  * Hand the next AIF off the top of the queue out to userspace.
35550b94a66eSMike Smith  */
35560b94a66eSMike Smith static int
aac_return_aif(struct aac_softc * sc,struct aac_fib_context * ctx,caddr_t uptr)3557a723a548SEd Maste aac_return_aif(struct aac_softc *sc, struct aac_fib_context *ctx, caddr_t uptr)
35580b94a66eSMike Smith {
3559a723a548SEd Maste 	int current, error;
35600b94a66eSMike Smith 
356131a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
35620b94a66eSMike Smith 
3563bb6fe253SScott Long 	mtx_lock(&sc->aac_aifq_lock);
3564a723a548SEd Maste 	current = ctx->ctx_idx;
3565a723a548SEd Maste 	if (current == sc->aifq_idx && !ctx->ctx_wrap) {
3566a723a548SEd Maste 		/* empty */
3567bb6fe253SScott Long 		mtx_unlock(&sc->aac_aifq_lock);
35683df780cfSScott Long 		return (EAGAIN);
35693df780cfSScott Long 	}
3570a723a548SEd Maste 	error =
3571a723a548SEd Maste 		copyout(&sc->aac_aifq[current], (void *)uptr, sizeof(struct aac_fib));
357236e0bf6eSScott Long 	if (error)
357370545d1aSScott Long 		device_printf(sc->aac_dev,
357470545d1aSScott Long 		    "aac_return_aif: copyout returned %d\n", error);
3575a723a548SEd Maste 	else {
3576a723a548SEd Maste 		ctx->ctx_wrap = 0;
3577a723a548SEd Maste 		ctx->ctx_idx = (current + 1) % AAC_AIFQ_LENGTH;
3578a723a548SEd Maste 	}
3579bb6fe253SScott Long 	mtx_unlock(&sc->aac_aifq_lock);
35800b94a66eSMike Smith 	return(error);
35810b94a66eSMike Smith }
358236e0bf6eSScott Long 
35837cb209f5SScott Long static int
aac_get_pci_info(struct aac_softc * sc,caddr_t uptr)35847cb209f5SScott Long aac_get_pci_info(struct aac_softc *sc, caddr_t uptr)
35857cb209f5SScott Long {
35867cb209f5SScott Long 	struct aac_pci_info {
35877cb209f5SScott Long 		u_int32_t bus;
35887cb209f5SScott Long 		u_int32_t slot;
35897cb209f5SScott Long 	} pciinf;
35907cb209f5SScott Long 	int error;
35917cb209f5SScott Long 
359231a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
35937cb209f5SScott Long 
35947cb209f5SScott Long 	pciinf.bus = pci_get_bus(sc->aac_dev);
35957cb209f5SScott Long 	pciinf.slot = pci_get_slot(sc->aac_dev);
35967cb209f5SScott Long 
35977cb209f5SScott Long 	error = copyout((caddr_t)&pciinf, uptr,
35987cb209f5SScott Long 			sizeof(struct aac_pci_info));
35997cb209f5SScott Long 
36007cb209f5SScott Long 	return (error);
36017cb209f5SScott Long }
36027cb209f5SScott Long 
36036d307336SEd Maste static int
aac_supported_features(struct aac_softc * sc,caddr_t uptr)36046d307336SEd Maste aac_supported_features(struct aac_softc *sc, caddr_t uptr)
36056d307336SEd Maste {
36066d307336SEd Maste 	struct aac_features f;
36076d307336SEd Maste 	int error;
36086d307336SEd Maste 
36096d307336SEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
36106d307336SEd Maste 
36116d307336SEd Maste 	if ((error = copyin(uptr, &f, sizeof (f))) != 0)
36126d307336SEd Maste 		return (error);
36136d307336SEd Maste 
36146d307336SEd Maste 	/*
36156d307336SEd Maste 	 * When the management driver receives FSACTL_GET_FEATURES ioctl with
36166d307336SEd Maste 	 * ALL zero in the featuresState, the driver will return the current
36176d307336SEd Maste 	 * state of all the supported features, the data field will not be
36186d307336SEd Maste 	 * valid.
36196d307336SEd Maste 	 * When the management driver receives FSACTL_GET_FEATURES ioctl with
36206d307336SEd Maste 	 * a specific bit set in the featuresState, the driver will return the
36216d307336SEd Maste 	 * current state of this specific feature and whatever data that are
36226d307336SEd Maste 	 * associated with the feature in the data field or perform whatever
36236d307336SEd Maste 	 * action needed indicates in the data field.
36246d307336SEd Maste 	 */
36256d307336SEd Maste 	if (f.feat.fValue == 0) {
36266d307336SEd Maste 		f.feat.fBits.largeLBA =
36276d307336SEd Maste 		    (sc->flags & AAC_FLAGS_LBA_64BIT) ? 1 : 0;
36286d307336SEd Maste 		/* TODO: In the future, add other features state here as well */
36296d307336SEd Maste 	} else {
36306d307336SEd Maste 		if (f.feat.fBits.largeLBA)
36316d307336SEd Maste 			f.feat.fBits.largeLBA =
36326d307336SEd Maste 			    (sc->flags & AAC_FLAGS_LBA_64BIT) ? 1 : 0;
36336d307336SEd Maste 		/* TODO: Add other features state and data in the future */
36346d307336SEd Maste 	}
36356d307336SEd Maste 
36366d307336SEd Maste 	error = copyout(&f, uptr, sizeof (f));
36376d307336SEd Maste 	return (error);
36386d307336SEd Maste }
36396d307336SEd Maste 
3640914da7d0SScott Long /*
364136e0bf6eSScott Long  * Give the userland some information about the container.  The AAC arch
364236e0bf6eSScott Long  * expects the driver to be a SCSI passthrough type driver, so it expects
364336e0bf6eSScott Long  * the containers to have b:t:l numbers.  Fake it.
364436e0bf6eSScott Long  */
364536e0bf6eSScott Long static int
aac_query_disk(struct aac_softc * sc,caddr_t uptr)364636e0bf6eSScott Long aac_query_disk(struct aac_softc *sc, caddr_t uptr)
364736e0bf6eSScott Long {
364836e0bf6eSScott Long 	struct aac_query_disk query_disk;
364936e0bf6eSScott Long 	struct aac_container *co;
3650914da7d0SScott Long 	struct aac_disk	*disk;
365136e0bf6eSScott Long 	int error, id;
365236e0bf6eSScott Long 
365331a0399eSEd Maste 	fwprintf(sc, HBA_FLAGS_DBG_FUNCTION_ENTRY_B, "");
365436e0bf6eSScott Long 
3655914da7d0SScott Long 	disk = NULL;
3656914da7d0SScott Long 
3657914da7d0SScott Long 	error = copyin(uptr, (caddr_t)&query_disk,
3658914da7d0SScott Long 		       sizeof(struct aac_query_disk));
365936e0bf6eSScott Long 	if (error)
366036e0bf6eSScott Long 		return (error);
366136e0bf6eSScott Long 
366236e0bf6eSScott Long 	id = query_disk.ContainerNumber;
366336e0bf6eSScott Long 	if (id == -1)
366436e0bf6eSScott Long 		return (EINVAL);
366536e0bf6eSScott Long 
3666bb6fe253SScott Long 	mtx_lock(&sc->aac_container_lock);
366736e0bf6eSScott Long 	TAILQ_FOREACH(co, &sc->aac_container_tqh, co_link) {
366836e0bf6eSScott Long 		if (co->co_mntobj.ObjectId == id)
366936e0bf6eSScott Long 			break;
367036e0bf6eSScott Long 		}
367136e0bf6eSScott Long 
367236e0bf6eSScott Long 	if (co == NULL) {
367336e0bf6eSScott Long 			query_disk.Valid = 0;
367436e0bf6eSScott Long 			query_disk.Locked = 0;
367536e0bf6eSScott Long 			query_disk.Deleted = 1;		/* XXX is this right? */
367636e0bf6eSScott Long 	} else {
367736e0bf6eSScott Long 		disk = device_get_softc(co->co_disk);
367836e0bf6eSScott Long 		query_disk.Valid = 1;
3679914da7d0SScott Long 		query_disk.Locked =
3680914da7d0SScott Long 		    (disk->ad_flags & AAC_DISK_OPEN) ? 1 : 0;
368136e0bf6eSScott Long 		query_disk.Deleted = 0;
3682b3457b51SScott Long 		query_disk.Bus = device_get_unit(sc->aac_dev);
368336e0bf6eSScott Long 		query_disk.Target = disk->unit;
368436e0bf6eSScott Long 		query_disk.Lun = 0;
368536e0bf6eSScott Long 		query_disk.UnMapped = 0;
36867540e65eSScott Long 		sprintf(&query_disk.diskDeviceName[0], "%s%d",
36870b7ed341SPoul-Henning Kamp 			disk->ad_disk->d_name, disk->ad_disk->d_unit);
368836e0bf6eSScott Long 	}
3689bb6fe253SScott Long 	mtx_unlock(&sc->aac_container_lock);
369036e0bf6eSScott Long 
3691914da7d0SScott Long 	error = copyout((caddr_t)&query_disk, uptr,
3692914da7d0SScott Long 			sizeof(struct aac_query_disk));
369336e0bf6eSScott Long 
369436e0bf6eSScott Long 	return (error);
369536e0bf6eSScott Long }
369636e0bf6eSScott Long 
3697fe3cb0e1SScott Long static void
aac_get_bus_info(struct aac_softc * sc)3698fe3cb0e1SScott Long aac_get_bus_info(struct aac_softc *sc)
3699fe3cb0e1SScott Long {
3700fe3cb0e1SScott Long 	struct aac_fib *fib;
3701fe3cb0e1SScott Long 	struct aac_ctcfg *c_cmd;
3702fe3cb0e1SScott Long 	struct aac_ctcfg_resp *c_resp;
3703fe3cb0e1SScott Long 	struct aac_vmioctl *vmi;
3704fe3cb0e1SScott Long 	struct aac_vmi_businf_resp *vmi_resp;
3705fe3cb0e1SScott Long 	struct aac_getbusinf businfo;
370670545d1aSScott Long 	struct aac_sim *caminf;
3707fe3cb0e1SScott Long 	device_t child;
3708fe3cb0e1SScott Long 	int i, found, error;
3709fe3cb0e1SScott Long 
37101ffe41c1SChristian S.J. Peron 	mtx_lock(&sc->aac_io_lock);
371103b5fe51SScott Long 	aac_alloc_sync_fib(sc, &fib);
3712fe3cb0e1SScott Long 	c_cmd = (struct aac_ctcfg *)&fib->data[0];
371339ee03c3SScott Long 	bzero(c_cmd, sizeof(struct aac_ctcfg));
3714fe3cb0e1SScott Long 
3715fe3cb0e1SScott Long 	c_cmd->Command = VM_ContainerConfig;
3716fe3cb0e1SScott Long 	c_cmd->cmd = CT_GET_SCSI_METHOD;
3717fe3cb0e1SScott Long 	c_cmd->param = 0;
3718fe3cb0e1SScott Long 
3719fe3cb0e1SScott Long 	error = aac_sync_fib(sc, ContainerCommand, 0, fib,
3720fe3cb0e1SScott Long 	    sizeof(struct aac_ctcfg));
3721fe3cb0e1SScott Long 	if (error) {
3722fe3cb0e1SScott Long 		device_printf(sc->aac_dev, "Error %d sending "
3723fe3cb0e1SScott Long 		    "VM_ContainerConfig command\n", error);
3724fe3cb0e1SScott Long 		aac_release_sync_fib(sc);
37251ffe41c1SChristian S.J. Peron 		mtx_unlock(&sc->aac_io_lock);
3726fe3cb0e1SScott Long 		return;
3727fe3cb0e1SScott Long 	}
3728fe3cb0e1SScott Long 
3729fe3cb0e1SScott Long 	c_resp = (struct aac_ctcfg_resp *)&fib->data[0];
3730fe3cb0e1SScott Long 	if (c_resp->Status != ST_OK) {
3731fe3cb0e1SScott Long 		device_printf(sc->aac_dev, "VM_ContainerConfig returned 0x%x\n",
3732fe3cb0e1SScott Long 		    c_resp->Status);
3733fe3cb0e1SScott Long 		aac_release_sync_fib(sc);
37341ffe41c1SChristian S.J. Peron 		mtx_unlock(&sc->aac_io_lock);
3735fe3cb0e1SScott Long 		return;
3736fe3cb0e1SScott Long 	}
3737fe3cb0e1SScott Long 
3738fe3cb0e1SScott Long 	sc->scsi_method_id = c_resp->param;
3739fe3cb0e1SScott Long 
3740fe3cb0e1SScott Long 	vmi = (struct aac_vmioctl *)&fib->data[0];
374139ee03c3SScott Long 	bzero(vmi, sizeof(struct aac_vmioctl));
374239ee03c3SScott Long 
3743fe3cb0e1SScott Long 	vmi->Command = VM_Ioctl;
3744fe3cb0e1SScott Long 	vmi->ObjType = FT_DRIVE;
3745fe3cb0e1SScott Long 	vmi->MethId = sc->scsi_method_id;
3746fe3cb0e1SScott Long 	vmi->ObjId = 0;
3747fe3cb0e1SScott Long 	vmi->IoctlCmd = GetBusInfo;
3748fe3cb0e1SScott Long 
3749fe3cb0e1SScott Long 	error = aac_sync_fib(sc, ContainerCommand, 0, fib,
375042ef13a2SEd Maste 	    sizeof(struct aac_vmi_businf_resp));
3751fe3cb0e1SScott Long 	if (error) {
3752fe3cb0e1SScott Long 		device_printf(sc->aac_dev, "Error %d sending VMIoctl command\n",
3753fe3cb0e1SScott Long 		    error);
3754fe3cb0e1SScott Long 		aac_release_sync_fib(sc);
37551ffe41c1SChristian S.J. Peron 		mtx_unlock(&sc->aac_io_lock);
3756fe3cb0e1SScott Long 		return;
3757fe3cb0e1SScott Long 	}
3758fe3cb0e1SScott Long 
3759fe3cb0e1SScott Long 	vmi_resp = (struct aac_vmi_businf_resp *)&fib->data[0];
3760fe3cb0e1SScott Long 	if (vmi_resp->Status != ST_OK) {
3761fe3cb0e1SScott Long 		device_printf(sc->aac_dev, "VM_Ioctl returned %d\n",
3762fe3cb0e1SScott Long 		    vmi_resp->Status);
3763fe3cb0e1SScott Long 		aac_release_sync_fib(sc);
37641ffe41c1SChristian S.J. Peron 		mtx_unlock(&sc->aac_io_lock);
3765fe3cb0e1SScott Long 		return;
3766fe3cb0e1SScott Long 	}
3767fe3cb0e1SScott Long 
3768fe3cb0e1SScott Long 	bcopy(&vmi_resp->BusInf, &businfo, sizeof(struct aac_getbusinf));
3769fe3cb0e1SScott Long 	aac_release_sync_fib(sc);
37701ffe41c1SChristian S.J. Peron 	mtx_unlock(&sc->aac_io_lock);
3771fe3cb0e1SScott Long 
3772fe3cb0e1SScott Long 	found = 0;
3773fe3cb0e1SScott Long 	for (i = 0; i < businfo.BusCount; i++) {
3774fe3cb0e1SScott Long 		if (businfo.BusValid[i] != AAC_BUS_VALID)
3775fe3cb0e1SScott Long 			continue;
3776fe3cb0e1SScott Long 
3777a761a1caSScott Long 		caminf = (struct aac_sim *)malloc( sizeof(struct aac_sim),
3778a761a1caSScott Long 		    M_AACBUF, M_NOWAIT | M_ZERO);
3779b5f516cdSScott Long 		if (caminf == NULL) {
3780b5f516cdSScott Long 			device_printf(sc->aac_dev,
3781b5f516cdSScott Long 			    "No memory to add passthrough bus %d\n", i);
3782b5f516cdSScott Long 			break;
378374b8d63dSPedro F. Giffuni 		}
3784fe3cb0e1SScott Long 
37855b56413dSWarner Losh 		child = device_add_child(sc->aac_dev, "aacp", DEVICE_UNIT_ANY);
3786fe3cb0e1SScott Long 		if (child == NULL) {
3787b5f516cdSScott Long 			device_printf(sc->aac_dev,
3788b5f516cdSScott Long 			    "device_add_child failed for passthrough bus %d\n",
3789b5f516cdSScott Long 			    i);
3790b5f516cdSScott Long 			free(caminf, M_AACBUF);
3791b5f516cdSScott Long 			break;
3792fe3cb0e1SScott Long 		}
3793fe3cb0e1SScott Long 
3794fe3cb0e1SScott Long 		caminf->TargetsPerBus = businfo.TargetsPerBus;
3795fe3cb0e1SScott Long 		caminf->BusNumber = i;
3796fe3cb0e1SScott Long 		caminf->InitiatorBusId = businfo.InitiatorBusId[i];
3797fe3cb0e1SScott Long 		caminf->aac_sc = sc;
3798ddb8683eSScott Long 		caminf->sim_dev = child;
3799fe3cb0e1SScott Long 
3800fe3cb0e1SScott Long 		device_set_ivars(child, caminf);
3801fe3cb0e1SScott Long 		device_set_desc(child, "SCSI Passthrough Bus");
380270545d1aSScott Long 		TAILQ_INSERT_TAIL(&sc->aac_sim_tqh, caminf, sim_link);
3803fe3cb0e1SScott Long 
3804fe3cb0e1SScott Long 		found = 1;
3805fe3cb0e1SScott Long 	}
3806fe3cb0e1SScott Long 
3807fe3cb0e1SScott Long 	if (found)
380818250ec6SJohn Baldwin 		bus_attach_children(sc->aac_dev);
3809fe3cb0e1SScott Long }
3810