aac.c revision 1.13 1 1.13 drochner /* $NetBSD: aac.c,v 1.13 2004/09/13 12:55:47 drochner Exp $ */
2 1.1 ad
3 1.1 ad /*-
4 1.1 ad * Copyright (c) 2002 The NetBSD Foundation, Inc.
5 1.1 ad * All rights reserved.
6 1.1 ad *
7 1.1 ad * This code is derived from software contributed to The NetBSD Foundation
8 1.1 ad * by Andrew Doran.
9 1.1 ad *
10 1.1 ad * Redistribution and use in source and binary forms, with or without
11 1.1 ad * modification, are permitted provided that the following conditions
12 1.1 ad * are met:
13 1.1 ad * 1. Redistributions of source code must retain the above copyright
14 1.1 ad * notice, this list of conditions and the following disclaimer.
15 1.1 ad * 2. Redistributions in binary form must reproduce the above copyright
16 1.1 ad * notice, this list of conditions and the following disclaimer in the
17 1.1 ad * documentation and/or other materials provided with the distribution.
18 1.1 ad * 3. All advertising materials mentioning features or use of this software
19 1.1 ad * must display the following acknowledgement:
20 1.1 ad * This product includes software developed by the NetBSD
21 1.1 ad * Foundation, Inc. and its contributors.
22 1.1 ad * 4. Neither the name of The NetBSD Foundation nor the names of its
23 1.1 ad * contributors may be used to endorse or promote products derived
24 1.1 ad * from this software without specific prior written permission.
25 1.1 ad *
26 1.1 ad * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
27 1.1 ad * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
28 1.1 ad * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
29 1.1 ad * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
30 1.1 ad * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
31 1.1 ad * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
32 1.1 ad * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
33 1.1 ad * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
34 1.1 ad * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
35 1.1 ad * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
36 1.1 ad * POSSIBILITY OF SUCH DAMAGE.
37 1.1 ad */
38 1.1 ad
39 1.1 ad /*-
40 1.1 ad * Copyright (c) 2001 Scott Long
41 1.1 ad * Copyright (c) 2001 Adaptec, Inc.
42 1.1 ad * Copyright (c) 2000 Michael Smith
43 1.1 ad * Copyright (c) 2000 BSDi
44 1.1 ad * Copyright (c) 2000 Niklas Hallqvist
45 1.1 ad * All rights reserved.
46 1.1 ad *
47 1.1 ad * Redistribution and use in source and binary forms, with or without
48 1.1 ad * modification, are permitted provided that the following conditions
49 1.1 ad * are met:
50 1.1 ad * 1. Redistributions of source code must retain the above copyright
51 1.1 ad * notice, this list of conditions and the following disclaimer.
52 1.1 ad * 2. Redistributions in binary form must reproduce the above copyright
53 1.1 ad * notice, this list of conditions and the following disclaimer in the
54 1.1 ad * documentation and/or other materials provided with the distribution.
55 1.1 ad *
56 1.1 ad * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
57 1.1 ad * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
58 1.1 ad * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
59 1.1 ad * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
60 1.1 ad * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
61 1.1 ad * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
62 1.1 ad * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
63 1.1 ad * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
64 1.1 ad * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
65 1.1 ad * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
66 1.1 ad * SUCH DAMAGE.
67 1.1 ad */
68 1.1 ad
69 1.1 ad /*
70 1.1 ad * Driver for the Adaptec 'FSA' family of PCI/SCSI RAID adapters.
71 1.1 ad *
72 1.1 ad * TODO:
73 1.1 ad *
74 1.1 ad * o Management interface.
75 1.1 ad * o Look again at some of the portability issues.
76 1.1 ad * o Handle various AIFs (e.g., notification that a container is going away).
77 1.1 ad */
78 1.1 ad
79 1.1 ad #include <sys/cdefs.h>
80 1.13 drochner __KERNEL_RCSID(0, "$NetBSD: aac.c,v 1.13 2004/09/13 12:55:47 drochner Exp $");
81 1.1 ad
82 1.1 ad #include <sys/param.h>
83 1.1 ad #include <sys/systm.h>
84 1.1 ad #include <sys/buf.h>
85 1.1 ad #include <sys/device.h>
86 1.1 ad #include <sys/kernel.h>
87 1.1 ad #include <sys/malloc.h>
88 1.1 ad
89 1.1 ad #include <machine/bus.h>
90 1.1 ad
91 1.1 ad #include <uvm/uvm_extern.h>
92 1.1 ad
93 1.1 ad #include <dev/ic/aacreg.h>
94 1.1 ad #include <dev/ic/aacvar.h>
95 1.1 ad #include <dev/ic/aac_tables.h>
96 1.1 ad
97 1.13 drochner #include "locators.h"
98 1.13 drochner
99 1.12 thorpej static int aac_check_firmware(struct aac_softc *);
100 1.12 thorpej static void aac_describe_controller(struct aac_softc *);
101 1.12 thorpej static int aac_dequeue_fib(struct aac_softc *, int, u_int32_t *,
102 1.12 thorpej struct aac_fib **);
103 1.12 thorpej static int aac_enqueue_fib(struct aac_softc *, int, struct aac_fib *);
104 1.12 thorpej static void aac_host_command(struct aac_softc *);
105 1.12 thorpej static void aac_host_response(struct aac_softc *);
106 1.12 thorpej static int aac_init(struct aac_softc *);
107 1.12 thorpej static int aac_print(void *, const char *);
108 1.12 thorpej static void aac_shutdown(void *);
109 1.12 thorpej static void aac_startup(struct aac_softc *);
110 1.12 thorpej static int aac_sync_command(struct aac_softc *, u_int32_t, u_int32_t,
111 1.12 thorpej u_int32_t, u_int32_t, u_int32_t, u_int32_t *);
112 1.12 thorpej static int aac_sync_fib(struct aac_softc *, u_int32_t, u_int32_t, void *,
113 1.12 thorpej u_int16_t, void *, u_int16_t *);
114 1.13 drochner static int aac_submatch(struct device *, struct cfdata *,
115 1.13 drochner const locdesc_t *, void *);
116 1.1 ad
117 1.1 ad #ifdef AAC_DEBUG
118 1.12 thorpej static void aac_print_fib(struct aac_softc *, struct aac_fib *, char *);
119 1.1 ad #endif
120 1.1 ad
121 1.1 ad /*
122 1.1 ad * Adapter-space FIB queue manipulation.
123 1.1 ad *
124 1.1 ad * Note that the queue implementation here is a little funky; neither the PI or
125 1.1 ad * CI will ever be zero. This behaviour is a controller feature.
126 1.1 ad */
127 1.1 ad static struct {
128 1.1 ad int size;
129 1.1 ad int notify;
130 1.1 ad } const aac_qinfo[] = {
131 1.1 ad { AAC_HOST_NORM_CMD_ENTRIES, AAC_DB_COMMAND_NOT_FULL },
132 1.1 ad { AAC_HOST_HIGH_CMD_ENTRIES, 0 },
133 1.1 ad { AAC_ADAP_NORM_CMD_ENTRIES, AAC_DB_COMMAND_READY },
134 1.1 ad { AAC_ADAP_HIGH_CMD_ENTRIES, 0 },
135 1.1 ad { AAC_HOST_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_NOT_FULL },
136 1.1 ad { AAC_HOST_HIGH_RESP_ENTRIES, 0 },
137 1.1 ad { AAC_ADAP_NORM_RESP_ENTRIES, AAC_DB_RESPONSE_READY },
138 1.1 ad { AAC_ADAP_HIGH_RESP_ENTRIES, 0 }
139 1.1 ad };
140 1.1 ad
141 1.1 ad #ifdef AAC_DEBUG
142 1.1 ad int aac_debug = AAC_DEBUG;
143 1.1 ad #endif
144 1.1 ad
145 1.12 thorpej static void *aac_sdh;
146 1.1 ad
147 1.1 ad extern struct cfdriver aac_cd;
148 1.1 ad
149 1.1 ad int
150 1.1 ad aac_attach(struct aac_softc *sc)
151 1.1 ad {
152 1.1 ad struct aac_attach_args aaca;
153 1.1 ad int nsegs, i, rv, state, size;
154 1.1 ad struct aac_ccb *ac;
155 1.1 ad struct aac_fib *fib;
156 1.1 ad bus_addr_t fibpa;
157 1.13 drochner int help[2];
158 1.13 drochner locdesc_t *ldesc = (void *)help; /* XXX */
159 1.1 ad
160 1.1 ad SIMPLEQ_INIT(&sc->sc_ccb_free);
161 1.1 ad SIMPLEQ_INIT(&sc->sc_ccb_queue);
162 1.1 ad SIMPLEQ_INIT(&sc->sc_ccb_complete);
163 1.1 ad
164 1.1 ad /*
165 1.1 ad * Disable interrupts before we do anything.
166 1.1 ad */
167 1.1 ad AAC_MASK_INTERRUPTS(sc);
168 1.1 ad
169 1.1 ad /*
170 1.1 ad * Initialise the adapter.
171 1.1 ad */
172 1.1 ad if (aac_check_firmware(sc))
173 1.1 ad return (EINVAL);
174 1.1 ad
175 1.1 ad if ((rv = aac_init(sc)) != 0)
176 1.1 ad return (rv);
177 1.1 ad aac_startup(sc);
178 1.1 ad
179 1.1 ad /*
180 1.1 ad * Print a little information about the controller.
181 1.1 ad */
182 1.1 ad aac_describe_controller(sc);
183 1.1 ad
184 1.1 ad /*
185 1.1 ad * Initialize the ccbs.
186 1.1 ad */
187 1.1 ad sc->sc_ccbs = malloc(sizeof(*ac) * AAC_NCCBS, M_DEVBUF,
188 1.1 ad M_NOWAIT | M_ZERO);
189 1.1 ad if (sc->sc_ccbs == NULL) {
190 1.8 thorpej aprint_error("%s: memory allocation failure\n",
191 1.8 thorpej sc->sc_dv.dv_xname);
192 1.1 ad return (ENOMEM);
193 1.1 ad }
194 1.1 ad state = 0;
195 1.1 ad size = sizeof(*fib) * AAC_NCCBS;
196 1.1 ad
197 1.1 ad if ((rv = bus_dmamap_create(sc->sc_dmat, size, 1, size,
198 1.1 ad 0, BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, &sc->sc_fibs_dmamap)) != 0) {
199 1.8 thorpej aprint_error("%s: cannot create fibs dmamap\n",
200 1.1 ad sc->sc_dv.dv_xname);
201 1.1 ad goto bail_out;
202 1.1 ad }
203 1.1 ad state++;
204 1.1 ad if ((rv = bus_dmamem_alloc(sc->sc_dmat, size, PAGE_SIZE, 0,
205 1.1 ad &sc->sc_fibs_seg, 1, &nsegs, BUS_DMA_NOWAIT)) != 0) {
206 1.8 thorpej aprint_error("%s: can't allocate fibs structure\n",
207 1.1 ad sc->sc_dv.dv_xname);
208 1.1 ad goto bail_out;
209 1.1 ad }
210 1.1 ad state++;
211 1.1 ad if ((rv = bus_dmamem_map(sc->sc_dmat, &sc->sc_fibs_seg, nsegs, size,
212 1.1 ad (caddr_t *)&sc->sc_fibs, 0)) != 0) {
213 1.8 thorpej aprint_error("%s: can't map fibs structure\n",
214 1.1 ad sc->sc_dv.dv_xname);
215 1.1 ad goto bail_out;
216 1.1 ad }
217 1.1 ad state++;
218 1.1 ad if ((rv = bus_dmamap_load(sc->sc_dmat, sc->sc_fibs_dmamap, sc->sc_fibs,
219 1.1 ad size, NULL, BUS_DMA_NOWAIT)) != 0) {
220 1.8 thorpej aprint_error("%s: cannot load fibs dmamap\n",
221 1.8 thorpej sc->sc_dv.dv_xname);
222 1.1 ad goto bail_out;
223 1.1 ad }
224 1.1 ad state++;
225 1.1 ad
226 1.1 ad memset(sc->sc_fibs, 0, size);
227 1.1 ad fibpa = sc->sc_fibs_seg.ds_addr;
228 1.1 ad fib = sc->sc_fibs;
229 1.1 ad
230 1.1 ad for (i = 0, ac = sc->sc_ccbs; i < AAC_NCCBS; i++, ac++) {
231 1.1 ad rv = bus_dmamap_create(sc->sc_dmat, AAC_MAX_XFER,
232 1.1 ad AAC_MAX_SGENTRIES, AAC_MAX_XFER, 0,
233 1.1 ad BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, &ac->ac_dmamap_xfer);
234 1.1 ad if (rv) {
235 1.1 ad while (--ac >= sc->sc_ccbs)
236 1.1 ad bus_dmamap_destroy(sc->sc_dmat,
237 1.1 ad ac->ac_dmamap_xfer);
238 1.8 thorpej aprint_error("%s: cannot create ccb dmamap (%d)",
239 1.1 ad sc->sc_dv.dv_xname, rv);
240 1.1 ad goto bail_out;
241 1.1 ad }
242 1.1 ad
243 1.1 ad ac->ac_fib = fib++;
244 1.1 ad ac->ac_fibphys = fibpa;
245 1.1 ad fibpa += sizeof(*fib);
246 1.1 ad aac_ccb_free(sc, ac);
247 1.1 ad }
248 1.1 ad
249 1.1 ad /*
250 1.1 ad * Attach devices.
251 1.1 ad */
252 1.1 ad for (i = 0; i < AAC_MAX_CONTAINERS; i++) {
253 1.1 ad if (!sc->sc_hdr[i].hd_present)
254 1.1 ad continue;
255 1.1 ad aaca.aaca_unit = i;
256 1.13 drochner
257 1.13 drochner ldesc->len = 1;
258 1.13 drochner ldesc->locs[AACCF_UNIT] = i;
259 1.13 drochner
260 1.13 drochner config_found_sm_loc(&sc->sc_dv, "aac", ldesc, &aaca,
261 1.13 drochner aac_print, aac_submatch);
262 1.1 ad }
263 1.1 ad
264 1.1 ad /*
265 1.1 ad * Enable interrupts, and register our shutdown hook.
266 1.1 ad */
267 1.1 ad sc->sc_flags |= AAC_ONLINE;
268 1.1 ad AAC_UNMASK_INTERRUPTS(sc);
269 1.1 ad if (aac_sdh != NULL)
270 1.1 ad shutdownhook_establish(aac_shutdown, NULL);
271 1.1 ad return (0);
272 1.1 ad
273 1.1 ad bail_out:
274 1.1 ad bus_dmamap_unload(sc->sc_dmat, sc->sc_common_dmamap);
275 1.1 ad bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->sc_common,
276 1.1 ad sizeof(*sc->sc_common));
277 1.1 ad bus_dmamem_free(sc->sc_dmat, &sc->sc_common_seg, 1);
278 1.1 ad bus_dmamap_destroy(sc->sc_dmat, sc->sc_common_dmamap);
279 1.1 ad
280 1.1 ad if (state > 3)
281 1.1 ad bus_dmamap_unload(sc->sc_dmat, sc->sc_fibs_dmamap);
282 1.1 ad if (state > 2)
283 1.1 ad bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->sc_fibs, size);
284 1.1 ad if (state > 1)
285 1.1 ad bus_dmamem_free(sc->sc_dmat, &sc->sc_fibs_seg, 1);
286 1.1 ad if (state > 0)
287 1.1 ad bus_dmamap_destroy(sc->sc_dmat, sc->sc_fibs_dmamap);
288 1.1 ad
289 1.1 ad free(sc->sc_ccbs, M_DEVBUF);
290 1.1 ad return (rv);
291 1.1 ad }
292 1.1 ad
293 1.1 ad /*
294 1.1 ad * Print autoconfiguration message for a sub-device.
295 1.1 ad */
296 1.12 thorpej static int
297 1.1 ad aac_print(void *aux, const char *pnp)
298 1.1 ad {
299 1.1 ad struct aac_attach_args *aaca;
300 1.1 ad
301 1.1 ad aaca = aux;
302 1.1 ad
303 1.1 ad if (pnp != NULL)
304 1.7 thorpej aprint_normal("block device at %s", pnp);
305 1.7 thorpej aprint_normal(" unit %d", aaca->aaca_unit);
306 1.1 ad return (UNCONF);
307 1.1 ad }
308 1.1 ad
309 1.1 ad /*
310 1.1 ad * Match a sub-device.
311 1.1 ad */
312 1.12 thorpej static int
313 1.13 drochner aac_submatch(struct device *parent, struct cfdata *cf,
314 1.13 drochner const locdesc_t *ldesc, void *aux)
315 1.1 ad {
316 1.1 ad struct aac_attach_args *aaca;
317 1.1 ad
318 1.1 ad aaca = aux;
319 1.1 ad
320 1.13 drochner if (cf->cf_loc[AACCF_UNIT] != AACCF_UNIT_DEFAULT &&
321 1.13 drochner cf->cf_loc[AACCF_UNIT] != ldesc->locs[AACCF_UNIT])
322 1.1 ad return (0);
323 1.1 ad
324 1.4 thorpej return (config_match(parent, cf, aux));
325 1.1 ad }
326 1.1 ad
327 1.1 ad /*
328 1.1 ad * Look up a text description of a numeric error code and return a pointer to
329 1.1 ad * same.
330 1.1 ad */
331 1.1 ad const char *
332 1.1 ad aac_describe_code(const struct aac_code_lookup *table, u_int32_t code)
333 1.1 ad {
334 1.1 ad int i;
335 1.1 ad
336 1.1 ad for (i = 0; table[i].string != NULL; i++)
337 1.1 ad if (table[i].code == code)
338 1.1 ad return (table[i].string);
339 1.1 ad
340 1.1 ad return (table[i + 1].string);
341 1.1 ad }
342 1.1 ad
343 1.12 thorpej static void
344 1.1 ad aac_describe_controller(struct aac_softc *sc)
345 1.1 ad {
346 1.1 ad u_int8_t buf[AAC_FIB_DATASIZE];
347 1.1 ad u_int16_t bufsize;
348 1.1 ad struct aac_adapter_info *info;
349 1.1 ad u_int8_t arg;
350 1.1 ad
351 1.1 ad arg = 0;
352 1.1 ad if (aac_sync_fib(sc, RequestAdapterInfo, 0, &arg, sizeof(arg), &buf,
353 1.1 ad &bufsize)) {
354 1.8 thorpej aprint_error("%s: RequestAdapterInfo failed\n",
355 1.8 thorpej sc->sc_dv.dv_xname);
356 1.1 ad return;
357 1.1 ad }
358 1.1 ad if (bufsize != sizeof(*info)) {
359 1.8 thorpej aprint_error("%s: "
360 1.6 fvdl "RequestAdapterInfo returned wrong data size (%d != %d)\n",
361 1.6 fvdl sc->sc_dv.dv_xname, bufsize, sizeof(*info));
362 1.1 ad return;
363 1.1 ad }
364 1.1 ad info = (struct aac_adapter_info *)&buf[0];
365 1.1 ad
366 1.8 thorpej aprint_normal("%s: %s at %dMHz, %dMB cache, %s, kernel %d.%d-%d\n",
367 1.1 ad sc->sc_dv.dv_xname,
368 1.1 ad aac_describe_code(aac_cpu_variant, le32toh(info->CpuVariant)),
369 1.1 ad le32toh(info->ClockSpeed),
370 1.1 ad le32toh(info->BufferMem) / (1024 * 1024),
371 1.1 ad aac_describe_code(aac_battery_platform,
372 1.2 thorpej le32toh(info->batteryPlatform)),
373 1.1 ad info->KernelRevision.external.comp.major,
374 1.1 ad info->KernelRevision.external.comp.minor,
375 1.1 ad info->KernelRevision.external.comp.dash);
376 1.1 ad
377 1.1 ad /* Save the kernel revision structure for later use. */
378 1.1 ad sc->sc_revision = info->KernelRevision;
379 1.1 ad }
380 1.1 ad
381 1.1 ad /*
382 1.1 ad * Retrieve the firmware version numbers. Dell PERC2/QC cards with firmware
383 1.1 ad * version 1.x are not compatible with this driver.
384 1.1 ad */
385 1.12 thorpej static int
386 1.1 ad aac_check_firmware(struct aac_softc *sc)
387 1.1 ad {
388 1.1 ad u_int32_t major, minor;
389 1.1 ad
390 1.1 ad if ((sc->sc_quirks & AAC_QUIRK_PERC2QC) != 0) {
391 1.1 ad if (aac_sync_command(sc, AAC_MONKER_GETKERNVER, 0, 0, 0, 0,
392 1.1 ad NULL)) {
393 1.8 thorpej aprint_error("%s: error reading firmware version\n",
394 1.1 ad sc->sc_dv.dv_xname);
395 1.1 ad return (1);
396 1.1 ad }
397 1.1 ad
398 1.1 ad /* These numbers are stored as ASCII! */
399 1.1 ad major = (AAC_GETREG4(sc, AAC_SA_MAILBOX + 4) & 0xff) - 0x30;
400 1.1 ad minor = (AAC_GETREG4(sc, AAC_SA_MAILBOX + 8) & 0xff) - 0x30;
401 1.1 ad if (major == 1) {
402 1.8 thorpej aprint_error(
403 1.8 thorpej "%s: firmware version %d.%d not supported.\n",
404 1.1 ad sc->sc_dv.dv_xname, major, minor);
405 1.1 ad return (1);
406 1.1 ad }
407 1.1 ad }
408 1.1 ad
409 1.1 ad return (0);
410 1.1 ad }
411 1.1 ad
412 1.12 thorpej static int
413 1.1 ad aac_init(struct aac_softc *sc)
414 1.1 ad {
415 1.1 ad int nsegs, i, rv, state, norm, high;
416 1.1 ad struct aac_adapter_init *ip;
417 1.1 ad u_int32_t code;
418 1.1 ad u_int8_t *qaddr;
419 1.1 ad
420 1.1 ad state = 0;
421 1.1 ad
422 1.1 ad /*
423 1.1 ad * First wait for the adapter to come ready.
424 1.1 ad */
425 1.1 ad for (i = 0; i < AAC_BOOT_TIMEOUT * 1000; i++) {
426 1.1 ad code = AAC_GET_FWSTATUS(sc);
427 1.1 ad if ((code & AAC_SELF_TEST_FAILED) != 0) {
428 1.8 thorpej aprint_error("%s: FATAL: selftest failed\n",
429 1.1 ad sc->sc_dv.dv_xname);
430 1.1 ad return (ENXIO);
431 1.1 ad }
432 1.1 ad if ((code & AAC_KERNEL_PANIC) != 0) {
433 1.8 thorpej aprint_error("%s: FATAL: controller kernel panic\n",
434 1.1 ad sc->sc_dv.dv_xname);
435 1.1 ad return (ENXIO);
436 1.1 ad }
437 1.1 ad if ((code & AAC_UP_AND_RUNNING) != 0)
438 1.1 ad break;
439 1.1 ad DELAY(1000);
440 1.1 ad }
441 1.1 ad if (i == AAC_BOOT_TIMEOUT * 1000) {
442 1.8 thorpej aprint_error(
443 1.8 thorpej "%s: FATAL: controller not coming ready, status %x\n",
444 1.1 ad sc->sc_dv.dv_xname, code);
445 1.1 ad return (ENXIO);
446 1.1 ad }
447 1.1 ad
448 1.1 ad if ((rv = bus_dmamap_create(sc->sc_dmat, sizeof(*sc->sc_common), 1,
449 1.1 ad sizeof(*sc->sc_common), 0, BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW,
450 1.1 ad &sc->sc_common_dmamap)) != 0) {
451 1.8 thorpej aprint_error("%s: cannot create common dmamap\n",
452 1.1 ad sc->sc_dv.dv_xname);
453 1.1 ad return (rv);
454 1.1 ad }
455 1.1 ad if ((rv = bus_dmamem_alloc(sc->sc_dmat, sizeof(*sc->sc_common),
456 1.1 ad PAGE_SIZE, 0, &sc->sc_common_seg, 1, &nsegs,
457 1.1 ad BUS_DMA_NOWAIT)) != 0) {
458 1.8 thorpej aprint_error("%s: can't allocate common structure\n",
459 1.1 ad sc->sc_dv.dv_xname);
460 1.1 ad goto bail_out;
461 1.1 ad }
462 1.1 ad state++;
463 1.1 ad if ((rv = bus_dmamem_map(sc->sc_dmat, &sc->sc_common_seg, nsegs,
464 1.1 ad sizeof(*sc->sc_common), (caddr_t *)&sc->sc_common, 0)) != 0) {
465 1.8 thorpej aprint_error("%s: can't map common structure\n",
466 1.1 ad sc->sc_dv.dv_xname);
467 1.1 ad goto bail_out;
468 1.1 ad }
469 1.1 ad state++;
470 1.1 ad if ((rv = bus_dmamap_load(sc->sc_dmat, sc->sc_common_dmamap,
471 1.1 ad sc->sc_common, sizeof(*sc->sc_common), NULL,
472 1.1 ad BUS_DMA_NOWAIT)) != 0) {
473 1.8 thorpej aprint_error("%s: cannot load common dmamap\n",
474 1.8 thorpej sc->sc_dv.dv_xname);
475 1.1 ad goto bail_out;
476 1.1 ad }
477 1.1 ad state++;
478 1.1 ad
479 1.1 ad memset(sc->sc_common, 0, sizeof(*sc->sc_common));
480 1.1 ad
481 1.1 ad /*
482 1.1 ad * Fill in the init structure. This tells the adapter about the
483 1.1 ad * physical location of various important shared data structures.
484 1.1 ad */
485 1.1 ad ip = &sc->sc_common->ac_init;
486 1.1 ad ip->InitStructRevision = htole32(AAC_INIT_STRUCT_REVISION);
487 1.1 ad
488 1.1 ad ip->AdapterFibsPhysicalAddress = htole32(sc->sc_common_seg.ds_addr +
489 1.1 ad offsetof(struct aac_common, ac_fibs));
490 1.1 ad ip->AdapterFibsVirtualAddress = htole32(&sc->sc_common->ac_fibs[0]);
491 1.1 ad ip->AdapterFibsSize =
492 1.1 ad htole32(AAC_ADAPTER_FIBS * sizeof(struct aac_fib));
493 1.1 ad ip->AdapterFibAlign = htole32(sizeof(struct aac_fib));
494 1.1 ad
495 1.1 ad ip->PrintfBufferAddress = htole32(sc->sc_common_seg.ds_addr +
496 1.1 ad offsetof(struct aac_common, ac_printf));
497 1.1 ad ip->PrintfBufferSize = htole32(AAC_PRINTF_BUFSIZE);
498 1.1 ad
499 1.1 ad ip->HostPhysMemPages = 0; /* not used? */
500 1.1 ad ip->HostElapsedSeconds = 0; /* reset later if invalid */
501 1.1 ad
502 1.1 ad /*
503 1.1 ad * Initialise FIB queues. Note that it appears that the layout of
504 1.1 ad * the indexes and the segmentation of the entries is mandated by
505 1.1 ad * the adapter, which is only told about the base of the queue index
506 1.1 ad * fields.
507 1.1 ad *
508 1.1 ad * The initial values of the indices are assumed to inform the
509 1.1 ad * adapter of the sizes of the respective queues.
510 1.1 ad *
511 1.1 ad * The Linux driver uses a much more complex scheme whereby several
512 1.1 ad * header records are kept for each queue. We use a couple of
513 1.1 ad * generic list manipulation functions which 'know' the size of each
514 1.1 ad * list by virtue of a table.
515 1.1 ad */
516 1.1 ad qaddr = &sc->sc_common->ac_qbuf[0] + AAC_QUEUE_ALIGN;
517 1.6 fvdl qaddr -= (u_int32_t)qaddr % AAC_QUEUE_ALIGN; /* XXX not portable */
518 1.1 ad sc->sc_queues = (struct aac_queue_table *)qaddr;
519 1.1 ad ip->CommHeaderAddress = htole32(sc->sc_common_seg.ds_addr +
520 1.1 ad ((caddr_t)sc->sc_queues - (caddr_t)sc->sc_common));
521 1.1 ad memset(sc->sc_queues, 0, sizeof(struct aac_queue_table));
522 1.1 ad
523 1.1 ad norm = htole32(AAC_HOST_NORM_CMD_ENTRIES);
524 1.1 ad high = htole32(AAC_HOST_HIGH_CMD_ENTRIES);
525 1.1 ad
526 1.1 ad sc->sc_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] =
527 1.1 ad norm;
528 1.1 ad sc->sc_queues->qt_qindex[AAC_HOST_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] =
529 1.1 ad norm;
530 1.1 ad sc->sc_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] =
531 1.1 ad high;
532 1.1 ad sc->sc_queues->qt_qindex[AAC_HOST_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] =
533 1.1 ad high;
534 1.1 ad
535 1.1 ad norm = htole32(AAC_ADAP_NORM_CMD_ENTRIES);
536 1.1 ad high = htole32(AAC_ADAP_HIGH_CMD_ENTRIES);
537 1.1 ad
538 1.1 ad sc->sc_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_PRODUCER_INDEX] =
539 1.1 ad norm;
540 1.1 ad sc->sc_queues->qt_qindex[AAC_ADAP_NORM_CMD_QUEUE][AAC_CONSUMER_INDEX] =
541 1.1 ad norm;
542 1.1 ad sc->sc_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_PRODUCER_INDEX] =
543 1.1 ad high;
544 1.1 ad sc->sc_queues->qt_qindex[AAC_ADAP_HIGH_CMD_QUEUE][AAC_CONSUMER_INDEX] =
545 1.1 ad high;
546 1.1 ad
547 1.1 ad norm = htole32(AAC_HOST_NORM_RESP_ENTRIES);
548 1.1 ad high = htole32(AAC_HOST_HIGH_RESP_ENTRIES);
549 1.1 ad
550 1.1 ad sc->sc_queues->
551 1.1 ad qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX] = norm;
552 1.1 ad sc->sc_queues->
553 1.1 ad qt_qindex[AAC_HOST_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX] = norm;
554 1.1 ad sc->sc_queues->
555 1.1 ad qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX] = high;
556 1.1 ad sc->sc_queues->
557 1.1 ad qt_qindex[AAC_HOST_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX] = high;
558 1.1 ad
559 1.1 ad norm = htole32(AAC_ADAP_NORM_RESP_ENTRIES);
560 1.1 ad high = htole32(AAC_ADAP_HIGH_RESP_ENTRIES);
561 1.1 ad
562 1.1 ad sc->sc_queues->
563 1.1 ad qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_PRODUCER_INDEX] = norm;
564 1.1 ad sc->sc_queues->
565 1.1 ad qt_qindex[AAC_ADAP_NORM_RESP_QUEUE][AAC_CONSUMER_INDEX] = norm;
566 1.1 ad sc->sc_queues->
567 1.1 ad qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_PRODUCER_INDEX] = high;
568 1.1 ad sc->sc_queues->
569 1.1 ad qt_qindex[AAC_ADAP_HIGH_RESP_QUEUE][AAC_CONSUMER_INDEX] = high;
570 1.1 ad
571 1.1 ad sc->sc_qentries[AAC_HOST_NORM_CMD_QUEUE] =
572 1.1 ad &sc->sc_queues->qt_HostNormCmdQueue[0];
573 1.1 ad sc->sc_qentries[AAC_HOST_HIGH_CMD_QUEUE] =
574 1.1 ad &sc->sc_queues->qt_HostHighCmdQueue[0];
575 1.1 ad sc->sc_qentries[AAC_ADAP_NORM_CMD_QUEUE] =
576 1.1 ad &sc->sc_queues->qt_AdapNormCmdQueue[0];
577 1.1 ad sc->sc_qentries[AAC_ADAP_HIGH_CMD_QUEUE] =
578 1.1 ad &sc->sc_queues->qt_AdapHighCmdQueue[0];
579 1.1 ad sc->sc_qentries[AAC_HOST_NORM_RESP_QUEUE] =
580 1.1 ad &sc->sc_queues->qt_HostNormRespQueue[0];
581 1.1 ad sc->sc_qentries[AAC_HOST_HIGH_RESP_QUEUE] =
582 1.1 ad &sc->sc_queues->qt_HostHighRespQueue[0];
583 1.1 ad sc->sc_qentries[AAC_ADAP_NORM_RESP_QUEUE] =
584 1.1 ad &sc->sc_queues->qt_AdapNormRespQueue[0];
585 1.1 ad sc->sc_qentries[AAC_ADAP_HIGH_RESP_QUEUE] =
586 1.1 ad &sc->sc_queues->qt_AdapHighRespQueue[0];
587 1.1 ad
588 1.1 ad /*
589 1.1 ad * Do controller-type-specific initialisation
590 1.1 ad */
591 1.1 ad switch (sc->sc_hwif) {
592 1.1 ad case AAC_HWIF_I960RX:
593 1.1 ad AAC_SETREG4(sc, AAC_RX_ODBR, ~0);
594 1.1 ad break;
595 1.1 ad }
596 1.1 ad
597 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap, 0,
598 1.1 ad sizeof(*sc->sc_common),
599 1.1 ad BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
600 1.1 ad
601 1.1 ad /*
602 1.1 ad * Give the init structure to the controller.
603 1.1 ad */
604 1.1 ad if (aac_sync_command(sc, AAC_MONKER_INITSTRUCT,
605 1.1 ad sc->sc_common_seg.ds_addr + offsetof(struct aac_common, ac_init),
606 1.1 ad 0, 0, 0, NULL)) {
607 1.8 thorpej aprint_error("%s: error establishing init structure\n",
608 1.1 ad sc->sc_dv.dv_xname);
609 1.1 ad rv = EIO;
610 1.1 ad goto bail_out;
611 1.1 ad }
612 1.1 ad
613 1.1 ad return (0);
614 1.1 ad
615 1.1 ad bail_out:
616 1.1 ad if (state > 2)
617 1.1 ad bus_dmamap_unload(sc->sc_dmat, sc->sc_common_dmamap);
618 1.1 ad if (state > 1)
619 1.1 ad bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->sc_common,
620 1.1 ad sizeof(*sc->sc_common));
621 1.1 ad if (state > 0)
622 1.1 ad bus_dmamem_free(sc->sc_dmat, &sc->sc_common_seg, 1);
623 1.1 ad bus_dmamap_destroy(sc->sc_dmat, sc->sc_common_dmamap);
624 1.1 ad
625 1.1 ad return (rv);
626 1.1 ad }
627 1.1 ad
628 1.1 ad /*
629 1.1 ad * Probe for containers, create disks.
630 1.1 ad */
631 1.12 thorpej static void
632 1.1 ad aac_startup(struct aac_softc *sc)
633 1.1 ad {
634 1.1 ad struct aac_mntinfo mi;
635 1.1 ad struct aac_mntinforesponse mir;
636 1.1 ad struct aac_drive *hd;
637 1.1 ad u_int16_t rsize;
638 1.1 ad int i;
639 1.1 ad
640 1.1 ad /*
641 1.1 ad * Loop over possible containers.
642 1.1 ad */
643 1.1 ad hd = sc->sc_hdr;
644 1.1 ad
645 1.1 ad for (i = 0; i < AAC_MAX_CONTAINERS; i++, hd++) {
646 1.1 ad /*
647 1.1 ad * Request information on this container.
648 1.1 ad */
649 1.10 ad memset(&mi, 0, sizeof(mi));
650 1.10 ad mi.Command = htole32(VM_NameServe);
651 1.10 ad mi.MntType = htole32(FT_FILESYS);
652 1.1 ad mi.MntCount = htole32(i);
653 1.1 ad if (aac_sync_fib(sc, ContainerCommand, 0, &mi, sizeof(mi), &mir,
654 1.1 ad &rsize)) {
655 1.8 thorpej aprint_error("%s: error probing container %d\n",
656 1.1 ad sc->sc_dv.dv_xname, i);
657 1.1 ad continue;
658 1.1 ad }
659 1.1 ad if (rsize != sizeof(mir)) {
660 1.8 thorpej aprint_error("%s: container info response wrong size "
661 1.6 fvdl "(%d should be %d)\n",
662 1.6 fvdl sc->sc_dv.dv_xname, rsize, sizeof(mir));
663 1.1 ad continue;
664 1.1 ad }
665 1.1 ad
666 1.1 ad /*
667 1.1 ad * Check container volume type for validity. Note that many
668 1.1 ad * of the possible types may never show up.
669 1.1 ad */
670 1.1 ad if (le32toh(mir.Status) != ST_OK ||
671 1.1 ad le32toh(mir.MntTable[0].VolType) == CT_NONE)
672 1.1 ad continue;
673 1.1 ad
674 1.1 ad hd->hd_present = 1;
675 1.1 ad hd->hd_size = le32toh(mir.MntTable[0].Capacity);
676 1.1 ad hd->hd_devtype = le32toh(mir.MntTable[0].VolType);
677 1.1 ad hd->hd_size &= ~0x1f;
678 1.1 ad sc->sc_nunits++;
679 1.1 ad }
680 1.1 ad }
681 1.1 ad
682 1.12 thorpej static void
683 1.1 ad aac_shutdown(void *cookie)
684 1.1 ad {
685 1.1 ad struct aac_softc *sc;
686 1.1 ad struct aac_close_command cc;
687 1.1 ad u_int32_t i;
688 1.1 ad
689 1.1 ad for (i = 0; i < aac_cd.cd_ndevs; i++) {
690 1.1 ad if ((sc = device_lookup(&aac_cd, i)) == NULL)
691 1.1 ad continue;
692 1.1 ad if ((sc->sc_flags & AAC_ONLINE) == 0)
693 1.1 ad continue;
694 1.1 ad
695 1.1 ad AAC_MASK_INTERRUPTS(sc);
696 1.1 ad
697 1.1 ad /*
698 1.1 ad * Send a Container shutdown followed by a HostShutdown FIB
699 1.1 ad * to the controller to convince it that we don't want to
700 1.1 ad * talk to it anymore. We've been closed and all I/O
701 1.1 ad * completed already
702 1.1 ad */
703 1.10 ad memset(&cc, 0, sizeof(cc));
704 1.1 ad cc.Command = htole32(VM_CloseAll);
705 1.1 ad cc.ContainerId = 0xffffffff;
706 1.1 ad if (aac_sync_fib(sc, ContainerCommand, 0, &cc, sizeof(cc),
707 1.1 ad NULL, NULL)) {
708 1.1 ad printf("%s: unable to halt controller\n",
709 1.1 ad sc->sc_dv.dv_xname);
710 1.1 ad continue;
711 1.1 ad }
712 1.1 ad
713 1.1 ad /*
714 1.1 ad * Note that issuing this command to the controller makes it
715 1.1 ad * shut down but also keeps it from coming back up without a
716 1.1 ad * reset of the PCI bus.
717 1.1 ad */
718 1.1 ad if (aac_sync_fib(sc, FsaHostShutdown, AAC_FIBSTATE_SHUTDOWN,
719 1.1 ad &i, sizeof(i), NULL, NULL))
720 1.1 ad printf("%s: unable to halt controller\n",
721 1.1 ad sc->sc_dv.dv_xname);
722 1.1 ad }
723 1.1 ad }
724 1.1 ad
725 1.1 ad /*
726 1.1 ad * Take an interrupt.
727 1.1 ad */
728 1.1 ad int
729 1.1 ad aac_intr(void *cookie)
730 1.1 ad {
731 1.1 ad struct aac_softc *sc;
732 1.1 ad u_int16_t reason;
733 1.1 ad int claimed;
734 1.1 ad
735 1.1 ad sc = cookie;
736 1.1 ad claimed = 0;
737 1.1 ad
738 1.1 ad AAC_DPRINTF(AAC_D_INTR, ("aac_intr(%p) ", sc));
739 1.1 ad
740 1.1 ad reason = AAC_GET_ISTATUS(sc);
741 1.1 ad AAC_DPRINTF(AAC_D_INTR, ("istatus 0x%04x ", reason));
742 1.1 ad
743 1.1 ad /*
744 1.1 ad * Controller wants to talk to the log. XXX Should we defer this?
745 1.1 ad */
746 1.1 ad if ((reason & AAC_DB_PRINTF) != 0) {
747 1.1 ad if (sc->sc_common->ac_printf[0] != '\0') {
748 1.1 ad printf("%s: WARNING: adapter logged message:\n",
749 1.1 ad sc->sc_dv.dv_xname);
750 1.1 ad printf("%s: %.*s", sc->sc_dv.dv_xname,
751 1.1 ad AAC_PRINTF_BUFSIZE, sc->sc_common->ac_printf);
752 1.1 ad sc->sc_common->ac_printf[0] = '\0';
753 1.1 ad }
754 1.1 ad AAC_CLEAR_ISTATUS(sc, AAC_DB_PRINTF);
755 1.1 ad AAC_QNOTIFY(sc, AAC_DB_PRINTF);
756 1.1 ad claimed = 1;
757 1.1 ad }
758 1.1 ad
759 1.1 ad /*
760 1.1 ad * Controller has a message for us?
761 1.1 ad */
762 1.1 ad if ((reason & AAC_DB_COMMAND_READY) != 0) {
763 1.1 ad aac_host_command(sc);
764 1.1 ad AAC_CLEAR_ISTATUS(sc, AAC_DB_COMMAND_READY);
765 1.1 ad claimed = 1;
766 1.1 ad }
767 1.1 ad
768 1.1 ad /*
769 1.1 ad * Controller has a response for us?
770 1.1 ad */
771 1.1 ad if ((reason & AAC_DB_RESPONSE_READY) != 0) {
772 1.1 ad aac_host_response(sc);
773 1.1 ad AAC_CLEAR_ISTATUS(sc, AAC_DB_RESPONSE_READY);
774 1.1 ad claimed = 1;
775 1.1 ad }
776 1.1 ad
777 1.1 ad /*
778 1.1 ad * Spurious interrupts that we don't use - reset the mask and clear
779 1.1 ad * the interrupts.
780 1.1 ad */
781 1.1 ad if ((reason & (AAC_DB_SYNC_COMMAND | AAC_DB_COMMAND_NOT_FULL |
782 1.1 ad AAC_DB_RESPONSE_NOT_FULL)) != 0) {
783 1.1 ad AAC_UNMASK_INTERRUPTS(sc);
784 1.1 ad AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND |
785 1.1 ad AAC_DB_COMMAND_NOT_FULL | AAC_DB_RESPONSE_NOT_FULL);
786 1.1 ad claimed = 1;
787 1.1 ad }
788 1.1 ad
789 1.1 ad return (claimed);
790 1.1 ad }
791 1.1 ad
792 1.1 ad /*
793 1.1 ad * Handle notification of one or more FIBs coming from the controller.
794 1.1 ad */
795 1.12 thorpej static void
796 1.1 ad aac_host_command(struct aac_softc *sc)
797 1.1 ad {
798 1.1 ad struct aac_fib *fib;
799 1.1 ad u_int32_t fib_size;
800 1.1 ad
801 1.1 ad for (;;) {
802 1.1 ad if (aac_dequeue_fib(sc, AAC_HOST_NORM_CMD_QUEUE, &fib_size,
803 1.1 ad &fib))
804 1.1 ad break; /* nothing to do */
805 1.1 ad
806 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
807 1.1 ad (caddr_t)fib - (caddr_t)sc->sc_common, sizeof(*fib),
808 1.1 ad BUS_DMASYNC_POSTREAD);
809 1.1 ad
810 1.1 ad switch (le16toh(fib->Header.Command)) {
811 1.1 ad case AifRequest:
812 1.1 ad #ifdef notyet
813 1.1 ad aac_handle_aif(sc,
814 1.1 ad (struct aac_aif_command *)&fib->data[0]);
815 1.1 ad #endif
816 1.1 ad break;
817 1.1 ad default:
818 1.1 ad printf("%s: unknown command from controller\n",
819 1.1 ad sc->sc_dv.dv_xname);
820 1.1 ad AAC_PRINT_FIB(sc, fib);
821 1.1 ad break;
822 1.1 ad }
823 1.1 ad
824 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
825 1.1 ad (caddr_t)fib - (caddr_t)sc->sc_common, sizeof(*fib),
826 1.1 ad BUS_DMASYNC_PREREAD);
827 1.1 ad
828 1.1 ad /* XXX reply to FIBs requesting responses ?? */
829 1.1 ad /* XXX how do we return these FIBs to the controller? */
830 1.1 ad }
831 1.1 ad }
832 1.1 ad
833 1.1 ad /*
834 1.1 ad * Handle notification of one or more FIBs completed by the controller
835 1.1 ad */
836 1.12 thorpej static void
837 1.1 ad aac_host_response(struct aac_softc *sc)
838 1.1 ad {
839 1.1 ad struct aac_ccb *ac;
840 1.1 ad struct aac_fib *fib;
841 1.1 ad u_int32_t fib_size;
842 1.1 ad
843 1.1 ad /*
844 1.1 ad * Look for completed FIBs on our queue.
845 1.1 ad */
846 1.1 ad for (;;) {
847 1.1 ad if (aac_dequeue_fib(sc, AAC_HOST_NORM_RESP_QUEUE, &fib_size,
848 1.1 ad &fib))
849 1.1 ad break; /* nothing to do */
850 1.1 ad
851 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_fibs_dmamap,
852 1.1 ad (caddr_t)fib - (caddr_t)sc->sc_fibs, sizeof(*fib),
853 1.1 ad BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
854 1.1 ad
855 1.1 ad if ((fib->Header.SenderData & 0x80000000) == 0) {
856 1.1 ad /* Not valid; not sent by us. */
857 1.1 ad AAC_PRINT_FIB(sc, fib);
858 1.1 ad } else {
859 1.1 ad ac = (struct aac_ccb *)((caddr_t)sc->sc_ccbs +
860 1.1 ad (fib->Header.SenderData & 0x7fffffff));
861 1.1 ad fib->Header.SenderData = 0;
862 1.1 ad SIMPLEQ_INSERT_TAIL(&sc->sc_ccb_complete, ac, ac_chain);
863 1.1 ad }
864 1.1 ad }
865 1.1 ad
866 1.1 ad /*
867 1.1 ad * Deal with any completed commands.
868 1.1 ad */
869 1.1 ad while ((ac = SIMPLEQ_FIRST(&sc->sc_ccb_complete)) != NULL) {
870 1.3 lukem SIMPLEQ_REMOVE_HEAD(&sc->sc_ccb_complete, ac_chain);
871 1.1 ad ac->ac_flags |= AAC_CCB_COMPLETED;
872 1.1 ad
873 1.1 ad if (ac->ac_intr != NULL)
874 1.1 ad (*ac->ac_intr)(ac);
875 1.1 ad }
876 1.1 ad
877 1.1 ad /*
878 1.1 ad * Try to submit more commands.
879 1.1 ad */
880 1.3 lukem if (! SIMPLEQ_EMPTY(&sc->sc_ccb_queue))
881 1.1 ad aac_ccb_enqueue(sc, NULL);
882 1.1 ad }
883 1.1 ad
884 1.1 ad /*
885 1.1 ad * Send a synchronous command to the controller and wait for a result.
886 1.1 ad */
887 1.12 thorpej static int
888 1.1 ad aac_sync_command(struct aac_softc *sc, u_int32_t command, u_int32_t arg0,
889 1.1 ad u_int32_t arg1, u_int32_t arg2, u_int32_t arg3, u_int32_t *sp)
890 1.1 ad {
891 1.1 ad int i;
892 1.1 ad u_int32_t status;
893 1.1 ad int s;
894 1.1 ad
895 1.1 ad s = splbio();
896 1.1 ad
897 1.1 ad /* Populate the mailbox. */
898 1.1 ad AAC_SET_MAILBOX(sc, command, arg0, arg1, arg2, arg3);
899 1.1 ad
900 1.1 ad /* Ensure the sync command doorbell flag is cleared. */
901 1.1 ad AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND);
902 1.1 ad
903 1.1 ad /* ... then set it to signal the adapter. */
904 1.1 ad AAC_QNOTIFY(sc, AAC_DB_SYNC_COMMAND);
905 1.1 ad DELAY(AAC_SYNC_DELAY);
906 1.1 ad
907 1.1 ad /* Spin waiting for the command to complete. */
908 1.1 ad for (i = 0; i < AAC_IMMEDIATE_TIMEOUT * 1000; i++) {
909 1.11 christos if (AAC_GET_ISTATUS(sc) & AAC_DB_SYNC_COMMAND)
910 1.1 ad break;
911 1.1 ad DELAY(1000);
912 1.1 ad }
913 1.1 ad if (i == AAC_IMMEDIATE_TIMEOUT * 1000) {
914 1.1 ad splx(s);
915 1.1 ad return (EIO);
916 1.1 ad }
917 1.1 ad
918 1.1 ad /* Clear the completion flag. */
919 1.1 ad AAC_CLEAR_ISTATUS(sc, AAC_DB_SYNC_COMMAND);
920 1.1 ad
921 1.1 ad /* Get the command status. */
922 1.1 ad status = AAC_GET_MAILBOXSTATUS(sc);
923 1.1 ad splx(s);
924 1.1 ad if (sp != NULL)
925 1.1 ad *sp = status;
926 1.1 ad
927 1.1 ad return (0); /* XXX Check command return status? */
928 1.1 ad }
929 1.1 ad
930 1.1 ad /*
931 1.1 ad * Send a synchronous FIB to the controller and wait for a result.
932 1.1 ad */
933 1.12 thorpej static int
934 1.1 ad aac_sync_fib(struct aac_softc *sc, u_int32_t command, u_int32_t xferstate,
935 1.1 ad void *data, u_int16_t datasize, void *result,
936 1.1 ad u_int16_t *resultsize)
937 1.1 ad {
938 1.1 ad struct aac_fib *fib;
939 1.1 ad u_int32_t fibpa, status;
940 1.1 ad
941 1.1 ad fib = &sc->sc_common->ac_sync_fib;
942 1.1 ad fibpa = sc->sc_common_seg.ds_addr +
943 1.1 ad offsetof(struct aac_common, ac_sync_fib);
944 1.1 ad
945 1.1 ad if (datasize > AAC_FIB_DATASIZE)
946 1.1 ad return (EINVAL);
947 1.1 ad
948 1.1 ad /*
949 1.1 ad * Set up the sync FIB.
950 1.1 ad */
951 1.1 ad fib->Header.XferState = htole32(AAC_FIBSTATE_HOSTOWNED |
952 1.1 ad AAC_FIBSTATE_INITIALISED | AAC_FIBSTATE_EMPTY | xferstate);
953 1.1 ad fib->Header.Command = htole16(command);
954 1.1 ad fib->Header.StructType = AAC_FIBTYPE_TFIB;
955 1.1 ad fib->Header.Size = htole16(sizeof(*fib) + datasize);
956 1.1 ad fib->Header.SenderSize = htole16(sizeof(*fib));
957 1.6 fvdl fib->Header.SenderFibAddress = htole32((u_int32_t)fib); /* XXX */
958 1.1 ad fib->Header.ReceiverFibAddress = htole32(fibpa);
959 1.1 ad
960 1.1 ad /*
961 1.1 ad * Copy in data.
962 1.1 ad */
963 1.1 ad if (data != NULL) {
964 1.1 ad memcpy(fib->data, data, datasize);
965 1.1 ad fib->Header.XferState |=
966 1.1 ad htole32(AAC_FIBSTATE_FROMHOST | AAC_FIBSTATE_NORM);
967 1.1 ad }
968 1.1 ad
969 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
970 1.1 ad (caddr_t)fib - (caddr_t)sc->sc_common, sizeof(*fib),
971 1.1 ad BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
972 1.1 ad
973 1.1 ad /*
974 1.1 ad * Give the FIB to the controller, wait for a response.
975 1.1 ad */
976 1.1 ad if (aac_sync_command(sc, AAC_MONKER_SYNCFIB, fibpa, 0, 0, 0, &status))
977 1.1 ad return (EIO);
978 1.1 ad
979 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
980 1.1 ad (caddr_t)fib - (caddr_t)sc->sc_common, sizeof(*fib),
981 1.1 ad BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
982 1.1 ad
983 1.1 ad /*
984 1.1 ad * Copy out the result
985 1.1 ad */
986 1.1 ad if (result != NULL) {
987 1.1 ad *resultsize = le16toh(fib->Header.Size) - sizeof(fib->Header);
988 1.1 ad memcpy(result, fib->data, *resultsize);
989 1.1 ad }
990 1.1 ad
991 1.1 ad return (0);
992 1.1 ad }
993 1.1 ad
994 1.1 ad struct aac_ccb *
995 1.1 ad aac_ccb_alloc(struct aac_softc *sc, int flags)
996 1.1 ad {
997 1.1 ad struct aac_ccb *ac;
998 1.1 ad int s;
999 1.1 ad
1000 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_alloc(%p, 0x%x) ", sc, flags));
1001 1.1 ad
1002 1.1 ad s = splbio();
1003 1.1 ad ac = SIMPLEQ_FIRST(&sc->sc_ccb_free);
1004 1.1 ad #ifdef DIAGNOSTIC
1005 1.1 ad if (ac == NULL)
1006 1.1 ad panic("aac_ccb_get: no free CCBS");
1007 1.1 ad #endif
1008 1.3 lukem SIMPLEQ_REMOVE_HEAD(&sc->sc_ccb_free, ac_chain);
1009 1.1 ad splx(s);
1010 1.1 ad
1011 1.1 ad ac->ac_flags = flags;
1012 1.1 ad return (ac);
1013 1.1 ad }
1014 1.1 ad
1015 1.1 ad void
1016 1.1 ad aac_ccb_free(struct aac_softc *sc, struct aac_ccb *ac)
1017 1.1 ad {
1018 1.1 ad int s;
1019 1.1 ad
1020 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_free(%p, %p) ", sc, ac));
1021 1.1 ad
1022 1.1 ad ac->ac_flags = 0;
1023 1.1 ad ac->ac_intr = NULL;
1024 1.1 ad ac->ac_fib->Header.XferState = htole32(AAC_FIBSTATE_EMPTY);
1025 1.1 ad ac->ac_fib->Header.StructType = AAC_FIBTYPE_TFIB;
1026 1.1 ad ac->ac_fib->Header.Flags = 0;
1027 1.1 ad ac->ac_fib->Header.SenderSize = htole16(sizeof(*ac->ac_fib));
1028 1.1 ad
1029 1.1 ad #ifdef AAC_DEBUG
1030 1.1 ad /*
1031 1.1 ad * These are duplicated in aac_ccb_submit() to cover the case where
1032 1.1 ad * an intermediate stage may have destroyed them. They're left
1033 1.1 ad * initialised here for debugging purposes only.
1034 1.1 ad */
1035 1.1 ad ac->ac_fib->Header.SenderFibAddress = htole32((u_int32_t)ac->ac_fib);
1036 1.1 ad ac->ac_fib->Header.ReceiverFibAddress = htole32(ac->ac_fibphys);
1037 1.1 ad #endif
1038 1.1 ad
1039 1.1 ad s = splbio();
1040 1.1 ad SIMPLEQ_INSERT_HEAD(&sc->sc_ccb_free, ac, ac_chain);
1041 1.1 ad splx(s);
1042 1.1 ad }
1043 1.1 ad
1044 1.1 ad int
1045 1.1 ad aac_ccb_map(struct aac_softc *sc, struct aac_ccb *ac)
1046 1.1 ad {
1047 1.1 ad int error;
1048 1.1 ad
1049 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_map(%p, %p) ", sc, ac));
1050 1.1 ad
1051 1.1 ad #ifdef DIAGNOSTIC
1052 1.1 ad if ((ac->ac_flags & AAC_CCB_MAPPED) != 0)
1053 1.1 ad panic("aac_ccb_map: already mapped");
1054 1.1 ad #endif
1055 1.1 ad
1056 1.1 ad error = bus_dmamap_load(sc->sc_dmat, ac->ac_dmamap_xfer, ac->ac_data,
1057 1.1 ad ac->ac_datalen, NULL, BUS_DMA_NOWAIT | BUS_DMA_STREAMING |
1058 1.1 ad ((ac->ac_flags & AAC_CCB_DATA_IN) ? BUS_DMA_READ : BUS_DMA_WRITE));
1059 1.1 ad if (error) {
1060 1.1 ad printf("%s: aac_ccb_map: ", sc->sc_dv.dv_xname);
1061 1.1 ad if (error == EFBIG)
1062 1.9 wiz printf("more than %d DMA segs\n", AAC_MAX_SGENTRIES);
1063 1.1 ad else
1064 1.9 wiz printf("error %d loading DMA map\n", error);
1065 1.1 ad return (error);
1066 1.1 ad }
1067 1.1 ad
1068 1.1 ad bus_dmamap_sync(sc->sc_dmat, ac->ac_dmamap_xfer, 0, ac->ac_datalen,
1069 1.1 ad (ac->ac_flags & AAC_CCB_DATA_IN) ? BUS_DMASYNC_PREREAD :
1070 1.1 ad BUS_DMASYNC_PREWRITE);
1071 1.1 ad
1072 1.1 ad #ifdef DIAGNOSTIC
1073 1.1 ad ac->ac_flags |= AAC_CCB_MAPPED;
1074 1.1 ad #endif
1075 1.1 ad return (0);
1076 1.1 ad }
1077 1.1 ad
1078 1.1 ad void
1079 1.1 ad aac_ccb_unmap(struct aac_softc *sc, struct aac_ccb *ac)
1080 1.1 ad {
1081 1.1 ad
1082 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_unmap(%p, %p) ", sc, ac));
1083 1.1 ad
1084 1.1 ad #ifdef DIAGNOSTIC
1085 1.1 ad if ((ac->ac_flags & AAC_CCB_MAPPED) == 0)
1086 1.1 ad panic("aac_ccb_unmap: not mapped");
1087 1.1 ad #endif
1088 1.1 ad
1089 1.1 ad bus_dmamap_sync(sc->sc_dmat, ac->ac_dmamap_xfer, 0, ac->ac_datalen,
1090 1.1 ad (ac->ac_flags & AAC_CCB_DATA_IN) ? BUS_DMASYNC_POSTREAD :
1091 1.1 ad BUS_DMASYNC_POSTWRITE);
1092 1.1 ad bus_dmamap_unload(sc->sc_dmat, ac->ac_dmamap_xfer);
1093 1.1 ad
1094 1.1 ad #ifdef DIAGNOSTIC
1095 1.1 ad ac->ac_flags &= ~AAC_CCB_MAPPED;
1096 1.1 ad #endif
1097 1.1 ad }
1098 1.1 ad
1099 1.1 ad void
1100 1.1 ad aac_ccb_enqueue(struct aac_softc *sc, struct aac_ccb *ac)
1101 1.1 ad {
1102 1.1 ad int s;
1103 1.1 ad
1104 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_enqueue(%p, %p) ", sc, ac));
1105 1.1 ad
1106 1.1 ad s = splbio();
1107 1.1 ad
1108 1.1 ad if (ac != NULL)
1109 1.1 ad SIMPLEQ_INSERT_TAIL(&sc->sc_ccb_queue, ac, ac_chain);
1110 1.1 ad
1111 1.1 ad while ((ac = SIMPLEQ_FIRST(&sc->sc_ccb_queue)) != NULL) {
1112 1.1 ad if (aac_ccb_submit(sc, ac))
1113 1.1 ad break;
1114 1.3 lukem SIMPLEQ_REMOVE_HEAD(&sc->sc_ccb_queue, ac_chain);
1115 1.1 ad }
1116 1.1 ad
1117 1.1 ad splx(s);
1118 1.1 ad }
1119 1.1 ad
1120 1.1 ad int
1121 1.1 ad aac_ccb_submit(struct aac_softc *sc, struct aac_ccb *ac)
1122 1.1 ad {
1123 1.1 ad
1124 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_submit(%p, %p) ", sc, ac));
1125 1.1 ad
1126 1.1 ad /* Fix up the address values. */
1127 1.6 fvdl ac->ac_fib->Header.SenderFibAddress = htole32((u_int32_t)ac->ac_fib);
1128 1.1 ad ac->ac_fib->Header.ReceiverFibAddress = htole32(ac->ac_fibphys);
1129 1.1 ad
1130 1.1 ad /* Save a pointer to the command for speedy reverse-lookup. */
1131 1.1 ad ac->ac_fib->Header.SenderData =
1132 1.1 ad (u_int32_t)((caddr_t)ac - (caddr_t)sc->sc_ccbs) | 0x80000000;
1133 1.1 ad
1134 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_fibs_dmamap,
1135 1.1 ad (caddr_t)ac->ac_fib - (caddr_t)sc->sc_fibs, sizeof(*ac->ac_fib),
1136 1.1 ad BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
1137 1.1 ad
1138 1.1 ad /* Put the FIB on the outbound queue. */
1139 1.1 ad return (aac_enqueue_fib(sc, AAC_ADAP_NORM_CMD_QUEUE, ac->ac_fib));
1140 1.1 ad }
1141 1.1 ad
1142 1.1 ad int
1143 1.1 ad aac_ccb_poll(struct aac_softc *sc, struct aac_ccb *ac, int timo)
1144 1.1 ad {
1145 1.1 ad int rv, s;
1146 1.1 ad
1147 1.1 ad AAC_DPRINTF(AAC_D_QUEUE, ("aac_ccb_poll(%p, %p, %d) ", sc, ac, timo));
1148 1.1 ad
1149 1.1 ad s = splbio();
1150 1.1 ad
1151 1.1 ad if ((rv = aac_ccb_submit(sc, ac)) != 0) {
1152 1.1 ad splx(s);
1153 1.1 ad return (rv);
1154 1.1 ad }
1155 1.1 ad
1156 1.1 ad for (timo *= 1000; timo != 0; timo--) {
1157 1.1 ad aac_intr(sc);
1158 1.1 ad if ((ac->ac_flags & AAC_CCB_COMPLETED) != 0)
1159 1.1 ad break;
1160 1.1 ad DELAY(100);
1161 1.1 ad }
1162 1.1 ad
1163 1.1 ad splx(s);
1164 1.1 ad return (timo == 0);
1165 1.1 ad }
1166 1.1 ad
1167 1.1 ad /*
1168 1.1 ad * Atomically insert an entry into the nominated queue, returns 0 on success
1169 1.1 ad * or EBUSY if the queue is full.
1170 1.1 ad *
1171 1.1 ad * XXX Note that it would be more efficient to defer notifying the
1172 1.1 ad * controller in the case where we may be inserting several entries in rapid
1173 1.1 ad * succession, but implementing this usefully is difficult.
1174 1.1 ad */
1175 1.12 thorpej static int
1176 1.1 ad aac_enqueue_fib(struct aac_softc *sc, int queue, struct aac_fib *fib)
1177 1.1 ad {
1178 1.1 ad u_int32_t fib_size, fib_addr, pi, ci;
1179 1.1 ad
1180 1.1 ad fib_size = le16toh(fib->Header.Size);
1181 1.1 ad fib_addr = le32toh(fib->Header.ReceiverFibAddress);
1182 1.1 ad
1183 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
1184 1.1 ad (caddr_t)sc->sc_common->ac_qbuf - (caddr_t)sc->sc_common,
1185 1.1 ad sizeof(sc->sc_common->ac_qbuf),
1186 1.1 ad BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
1187 1.1 ad
1188 1.1 ad /* Get the producer/consumer indices. */
1189 1.1 ad pi = le32toh(sc->sc_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]);
1190 1.1 ad ci = le32toh(sc->sc_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]);
1191 1.1 ad
1192 1.1 ad /* Wrap the queue? */
1193 1.1 ad if (pi >= aac_qinfo[queue].size)
1194 1.1 ad pi = 0;
1195 1.1 ad
1196 1.1 ad /* Check for queue full. */
1197 1.1 ad if ((pi + 1) == ci)
1198 1.1 ad return (EAGAIN);
1199 1.1 ad
1200 1.1 ad /* Populate queue entry. */
1201 1.1 ad (sc->sc_qentries[queue] + pi)->aq_fib_size = htole32(fib_size);
1202 1.1 ad (sc->sc_qentries[queue] + pi)->aq_fib_addr = htole32(fib_addr);
1203 1.1 ad
1204 1.1 ad /* Update producer index. */
1205 1.1 ad sc->sc_queues->qt_qindex[queue][AAC_PRODUCER_INDEX] = htole32(pi + 1);
1206 1.1 ad
1207 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
1208 1.1 ad (caddr_t)sc->sc_common->ac_qbuf - (caddr_t)sc->sc_common,
1209 1.1 ad sizeof(sc->sc_common->ac_qbuf),
1210 1.1 ad BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
1211 1.1 ad
1212 1.1 ad /* Notify the adapter if we know how. */
1213 1.1 ad if (aac_qinfo[queue].notify != 0)
1214 1.1 ad AAC_QNOTIFY(sc, aac_qinfo[queue].notify);
1215 1.1 ad
1216 1.1 ad return (0);
1217 1.1 ad }
1218 1.1 ad
1219 1.1 ad /*
1220 1.1 ad * Atomically remove one entry from the nominated queue, returns 0 on success
1221 1.1 ad * or ENOENT if the queue is empty.
1222 1.1 ad */
1223 1.12 thorpej static int
1224 1.1 ad aac_dequeue_fib(struct aac_softc *sc, int queue, u_int32_t *fib_size,
1225 1.1 ad struct aac_fib **fib_addr)
1226 1.1 ad {
1227 1.1 ad u_int32_t pi, ci;
1228 1.1 ad int notify;
1229 1.1 ad
1230 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
1231 1.1 ad (caddr_t)sc->sc_common->ac_qbuf - (caddr_t)sc->sc_common,
1232 1.1 ad sizeof(sc->sc_common->ac_qbuf),
1233 1.1 ad BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD);
1234 1.1 ad
1235 1.1 ad /* Get the producer/consumer indices. */
1236 1.1 ad pi = le32toh(sc->sc_queues->qt_qindex[queue][AAC_PRODUCER_INDEX]);
1237 1.1 ad ci = le32toh(sc->sc_queues->qt_qindex[queue][AAC_CONSUMER_INDEX]);
1238 1.1 ad
1239 1.1 ad /* Check for queue empty. */
1240 1.1 ad if (ci == pi)
1241 1.1 ad return (ENOENT);
1242 1.1 ad
1243 1.1 ad notify = 0;
1244 1.1 ad if (ci == pi + 1)
1245 1.1 ad notify = 1;
1246 1.1 ad
1247 1.1 ad /* Wrap the queue? */
1248 1.1 ad if (ci >= aac_qinfo[queue].size)
1249 1.1 ad ci = 0;
1250 1.1 ad
1251 1.1 ad /* Fetch the entry. */
1252 1.1 ad *fib_size = le32toh((sc->sc_qentries[queue] + ci)->aq_fib_size);
1253 1.6 fvdl *fib_addr = le32toh((struct aac_fib *)
1254 1.6 fvdl (sc->sc_qentries[queue] + ci)->aq_fib_addr);
1255 1.1 ad
1256 1.1 ad /* Update consumer index. */
1257 1.1 ad sc->sc_queues->qt_qindex[queue][AAC_CONSUMER_INDEX] = ci + 1;
1258 1.1 ad
1259 1.1 ad bus_dmamap_sync(sc->sc_dmat, sc->sc_common_dmamap,
1260 1.1 ad (caddr_t)sc->sc_common->ac_qbuf - (caddr_t)sc->sc_common,
1261 1.1 ad sizeof(sc->sc_common->ac_qbuf),
1262 1.1 ad BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
1263 1.1 ad
1264 1.1 ad /* If we have made the queue un-full, notify the adapter. */
1265 1.1 ad if (notify && (aac_qinfo[queue].notify != 0))
1266 1.1 ad AAC_QNOTIFY(sc, aac_qinfo[queue].notify);
1267 1.1 ad
1268 1.1 ad return (0);
1269 1.1 ad }
1270 1.1 ad
1271 1.1 ad #ifdef AAC_DEBUG
1272 1.1 ad /*
1273 1.1 ad * Print a FIB
1274 1.1 ad */
1275 1.12 thorpej static void
1276 1.1 ad aac_print_fib(struct aac_softc *sc, struct aac_fib *fib, char *caller)
1277 1.1 ad {
1278 1.1 ad struct aac_blockread *br;
1279 1.1 ad struct aac_blockwrite *bw;
1280 1.1 ad struct aac_sg_table *sg;
1281 1.1 ad char buf[512];
1282 1.1 ad int i;
1283 1.1 ad
1284 1.1 ad printf("%s: FIB @ %p\n", caller, fib);
1285 1.1 ad bitmask_snprintf(le32toh(fib->Header.XferState),
1286 1.1 ad "\20"
1287 1.1 ad "\1HOSTOWNED"
1288 1.1 ad "\2ADAPTEROWNED"
1289 1.1 ad "\3INITIALISED"
1290 1.1 ad "\4EMPTY"
1291 1.1 ad "\5FROMPOOL"
1292 1.1 ad "\6FROMHOST"
1293 1.1 ad "\7FROMADAP"
1294 1.1 ad "\10REXPECTED"
1295 1.1 ad "\11RNOTEXPECTED"
1296 1.1 ad "\12DONEADAP"
1297 1.1 ad "\13DONEHOST"
1298 1.1 ad "\14HIGH"
1299 1.1 ad "\15NORM"
1300 1.1 ad "\16ASYNC"
1301 1.1 ad "\17PAGEFILEIO"
1302 1.1 ad "\20SHUTDOWN"
1303 1.1 ad "\21LAZYWRITE"
1304 1.1 ad "\22ADAPMICROFIB"
1305 1.1 ad "\23BIOSFIB"
1306 1.1 ad "\24FAST_RESPONSE"
1307 1.1 ad "\25APIFIB\n",
1308 1.1 ad buf,
1309 1.1 ad sizeof(buf));
1310 1.1 ad
1311 1.1 ad printf(" XferState %s\n", buf);
1312 1.1 ad printf(" Command %d\n", le16toh(fib->Header.Command));
1313 1.1 ad printf(" StructType %d\n", fib->Header.StructType);
1314 1.1 ad printf(" Flags 0x%x\n", fib->Header.Flags);
1315 1.1 ad printf(" Size %d\n", le16toh(fib->Header.Size));
1316 1.1 ad printf(" SenderSize %d\n", le16toh(fib->Header.SenderSize));
1317 1.1 ad printf(" SenderAddress 0x%x\n",
1318 1.1 ad le32toh(fib->Header.SenderFibAddress));
1319 1.1 ad printf(" ReceiverAddress 0x%x\n",
1320 1.1 ad le32toh(fib->Header.ReceiverFibAddress));
1321 1.1 ad printf(" SenderData 0x%x\n", fib->Header.SenderData);
1322 1.1 ad
1323 1.1 ad switch (fib->Header.Command) {
1324 1.1 ad case ContainerCommand: {
1325 1.1 ad br = (struct aac_blockread *)fib->data;
1326 1.1 ad bw = (struct aac_blockwrite *)fib->data;
1327 1.1 ad sg = NULL;
1328 1.1 ad
1329 1.1 ad if (le32toh(br->Command) == VM_CtBlockRead) {
1330 1.1 ad printf(" BlockRead: container %d 0x%x/%d\n",
1331 1.1 ad le32toh(br->ContainerId), le32toh(br->BlockNumber),
1332 1.1 ad le32toh(br->ByteCount));
1333 1.1 ad sg = &br->SgMap;
1334 1.1 ad }
1335 1.1 ad if (le32toh(bw->Command) == VM_CtBlockWrite) {
1336 1.1 ad printf(" BlockWrite: container %d 0x%x/%d (%s)\n",
1337 1.1 ad le32toh(bw->ContainerId), le32toh(bw->BlockNumber),
1338 1.1 ad le32toh(bw->ByteCount),
1339 1.1 ad le32toh(bw->Stable) == CSTABLE ?
1340 1.1 ad "stable" : "unstable");
1341 1.1 ad sg = &bw->SgMap;
1342 1.1 ad }
1343 1.1 ad if (sg != NULL) {
1344 1.1 ad printf(" %d s/g entries\n", le32toh(sg->SgCount));
1345 1.1 ad for (i = 0; i < le32toh(sg->SgCount); i++)
1346 1.1 ad printf(" 0x%08x/%d\n",
1347 1.1 ad le32toh(sg->SgEntry[i].SgAddress),
1348 1.1 ad le32toh(sg->SgEntry[i].SgByteCount));
1349 1.1 ad }
1350 1.1 ad break;
1351 1.1 ad }
1352 1.1 ad default:
1353 1.1 ad printf(" %16D\n", fib->data, " ");
1354 1.1 ad printf(" %16D\n", fib->data + 16, " ");
1355 1.1 ad break;
1356 1.1 ad }
1357 1.1 ad }
1358 1.12 thorpej #endif /* AAC_DEBUG */
1359