1991554f2SKenneth D. Merry /*-
2991554f2SKenneth D. Merry * Copyright (c) 2009 Yahoo! Inc.
3a2c14879SStephen McConnell * Copyright (c) 2011-2015 LSI Corp.
47a2a6a1aSStephen McConnell * Copyright (c) 2013-2016 Avago Technologies
546b23587SKashyap D Desai * Copyright 2000-2020 Broadcom Inc.
6991554f2SKenneth D. Merry * All rights reserved.
7991554f2SKenneth D. Merry *
8991554f2SKenneth D. Merry * Redistribution and use in source and binary forms, with or without
9991554f2SKenneth D. Merry * modification, are permitted provided that the following conditions
10991554f2SKenneth D. Merry * are met:
11991554f2SKenneth D. Merry * 1. Redistributions of source code must retain the above copyright
12991554f2SKenneth D. Merry * notice, this list of conditions and the following disclaimer.
13991554f2SKenneth D. Merry * 2. Redistributions in binary form must reproduce the above copyright
14991554f2SKenneth D. Merry * notice, this list of conditions and the following disclaimer in the
15991554f2SKenneth D. Merry * documentation and/or other materials provided with the distribution.
16991554f2SKenneth D. Merry *
17991554f2SKenneth D. Merry * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18991554f2SKenneth D. Merry * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19991554f2SKenneth D. Merry * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20991554f2SKenneth D. Merry * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21991554f2SKenneth D. Merry * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22991554f2SKenneth D. Merry * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23991554f2SKenneth D. Merry * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24991554f2SKenneth D. Merry * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25991554f2SKenneth D. Merry * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26991554f2SKenneth D. Merry * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27991554f2SKenneth D. Merry * SUCH DAMAGE.
28a2c14879SStephen McConnell *
2946b23587SKashyap D Desai * Broadcom Inc. (LSI) MPT-Fusion Host Adapter FreeBSD
30a2c14879SStephen McConnell *
31991554f2SKenneth D. Merry */
32991554f2SKenneth D. Merry
33991554f2SKenneth D. Merry #include <sys/cdefs.h>
34a2c14879SStephen McConnell /* Communications core for Avago Technologies (LSI) MPT3 */
35991554f2SKenneth D. Merry
36991554f2SKenneth D. Merry /* TODO Move headers to mprvar */
37991554f2SKenneth D. Merry #include <sys/types.h>
38991554f2SKenneth D. Merry #include <sys/param.h>
39991554f2SKenneth D. Merry #include <sys/systm.h>
40991554f2SKenneth D. Merry #include <sys/kernel.h>
41991554f2SKenneth D. Merry #include <sys/selinfo.h>
42991554f2SKenneth D. Merry #include <sys/module.h>
43991554f2SKenneth D. Merry #include <sys/bus.h>
44991554f2SKenneth D. Merry #include <sys/conf.h>
45991554f2SKenneth D. Merry #include <sys/bio.h>
46991554f2SKenneth D. Merry #include <sys/malloc.h>
47991554f2SKenneth D. Merry #include <sys/uio.h>
48991554f2SKenneth D. Merry #include <sys/sysctl.h>
49991554f2SKenneth D. Merry #include <sys/endian.h>
50991554f2SKenneth D. Merry #include <sys/queue.h>
51991554f2SKenneth D. Merry #include <sys/kthread.h>
52991554f2SKenneth D. Merry #include <sys/taskqueue.h>
53991554f2SKenneth D. Merry #include <sys/sbuf.h>
54991554f2SKenneth D. Merry
55991554f2SKenneth D. Merry #include <machine/bus.h>
56991554f2SKenneth D. Merry #include <machine/resource.h>
57991554f2SKenneth D. Merry #include <sys/rman.h>
58991554f2SKenneth D. Merry
59991554f2SKenneth D. Merry #include <machine/stdarg.h>
60991554f2SKenneth D. Merry
61991554f2SKenneth D. Merry #include <cam/cam.h>
62991554f2SKenneth D. Merry #include <cam/cam_ccb.h>
63991554f2SKenneth D. Merry #include <cam/cam_debug.h>
64991554f2SKenneth D. Merry #include <cam/cam_sim.h>
65991554f2SKenneth D. Merry #include <cam/cam_xpt_sim.h>
66991554f2SKenneth D. Merry #include <cam/cam_xpt_periph.h>
67991554f2SKenneth D. Merry #include <cam/cam_periph.h>
68991554f2SKenneth D. Merry #include <cam/scsi/scsi_all.h>
69991554f2SKenneth D. Merry #include <cam/scsi/scsi_message.h>
70991554f2SKenneth D. Merry #include <cam/scsi/smp_all.h>
71991554f2SKenneth D. Merry
7267feec50SStephen McConnell #include <dev/nvme/nvme.h>
7367feec50SStephen McConnell
74991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2_type.h>
75991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2.h>
76991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2_ioc.h>
77991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2_sas.h>
7867feec50SStephen McConnell #include <dev/mpr/mpi/mpi2_pci.h>
79991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2_cnfg.h>
80991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2_init.h>
81991554f2SKenneth D. Merry #include <dev/mpr/mpi/mpi2_tool.h>
82991554f2SKenneth D. Merry #include <dev/mpr/mpr_ioctl.h>
83991554f2SKenneth D. Merry #include <dev/mpr/mprvar.h>
84991554f2SKenneth D. Merry #include <dev/mpr/mpr_table.h>
85991554f2SKenneth D. Merry #include <dev/mpr/mpr_sas.h>
86991554f2SKenneth D. Merry
87991554f2SKenneth D. Merry #define MPRSAS_DISCOVERY_TIMEOUT 20
88991554f2SKenneth D. Merry #define MPRSAS_MAX_DISCOVERY_TIMEOUTS 10 /* 200 seconds */
89991554f2SKenneth D. Merry
90991554f2SKenneth D. Merry /*
91991554f2SKenneth D. Merry * static array to check SCSI OpCode for EEDP protection bits
92991554f2SKenneth D. Merry */
93991554f2SKenneth D. Merry #define PRO_R MPI2_SCSIIO_EEDPFLAGS_CHECK_REMOVE_OP
94991554f2SKenneth D. Merry #define PRO_W MPI2_SCSIIO_EEDPFLAGS_INSERT_OP
95991554f2SKenneth D. Merry #define PRO_V MPI2_SCSIIO_EEDPFLAGS_INSERT_OP
96991554f2SKenneth D. Merry static uint8_t op_code_prot[256] = {
97991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
98991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
99991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, PRO_R, 0, PRO_W, 0, 0, 0, PRO_W, PRO_V,
100991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
101991554f2SKenneth D. Merry 0, PRO_W, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
102991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
103991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
104991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
105991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, PRO_R, 0, PRO_W, 0, 0, 0, PRO_W, PRO_V,
106991554f2SKenneth D. Merry 0, 0, 0, PRO_W, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
107991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, PRO_R, 0, PRO_W, 0, 0, 0, PRO_W, PRO_V,
108991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
109991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
110991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
111991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
112991554f2SKenneth D. Merry 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
113991554f2SKenneth D. Merry };
114991554f2SKenneth D. Merry
115991554f2SKenneth D. Merry MALLOC_DEFINE(M_MPRSAS, "MPRSAS", "MPR SAS memory");
116991554f2SKenneth D. Merry
117991554f2SKenneth D. Merry static void mprsas_remove_device(struct mpr_softc *, struct mpr_command *);
118991554f2SKenneth D. Merry static void mprsas_remove_complete(struct mpr_softc *, struct mpr_command *);
119991554f2SKenneth D. Merry static void mprsas_action(struct cam_sim *sim, union ccb *ccb);
120991554f2SKenneth D. Merry static void mprsas_poll(struct cam_sim *sim);
121991554f2SKenneth D. Merry static void mprsas_scsiio_timeout(void *data);
1227a2a6a1aSStephen McConnell static void mprsas_abort_complete(struct mpr_softc *sc, struct mpr_command *cm);
123991554f2SKenneth D. Merry static void mprsas_action_scsiio(struct mprsas_softc *, union ccb *);
124991554f2SKenneth D. Merry static void mprsas_scsiio_complete(struct mpr_softc *, struct mpr_command *);
125991554f2SKenneth D. Merry static void mprsas_action_resetdev(struct mprsas_softc *, union ccb *);
1267a2a6a1aSStephen McConnell static void mprsas_resetdev_complete(struct mpr_softc *, struct mpr_command *);
127991554f2SKenneth D. Merry static int mprsas_send_abort(struct mpr_softc *sc, struct mpr_command *tm,
128991554f2SKenneth D. Merry struct mpr_command *cm);
129991554f2SKenneth D. Merry static void mprsas_async(void *callback_arg, uint32_t code,
130991554f2SKenneth D. Merry struct cam_path *path, void *arg);
131991554f2SKenneth D. Merry static int mprsas_send_portenable(struct mpr_softc *sc);
132991554f2SKenneth D. Merry static void mprsas_portenable_complete(struct mpr_softc *sc,
133991554f2SKenneth D. Merry struct mpr_command *cm);
134991554f2SKenneth D. Merry
1357a2a6a1aSStephen McConnell static void mprsas_smpio_complete(struct mpr_softc *sc, struct mpr_command *cm);
1367a2a6a1aSStephen McConnell static void mprsas_send_smpcmd(struct mprsas_softc *sassc, union ccb *ccb,
1377a2a6a1aSStephen McConnell uint64_t sasaddr);
138a2c14879SStephen McConnell static void mprsas_action_smpio(struct mprsas_softc *sassc, union ccb *ccb);
139991554f2SKenneth D. Merry
140991554f2SKenneth D. Merry struct mprsas_target *
mprsas_find_target_by_handle(struct mprsas_softc * sassc,int start,uint16_t handle)141991554f2SKenneth D. Merry mprsas_find_target_by_handle(struct mprsas_softc *sassc, int start,
142991554f2SKenneth D. Merry uint16_t handle)
143991554f2SKenneth D. Merry {
144991554f2SKenneth D. Merry struct mprsas_target *target;
145991554f2SKenneth D. Merry int i;
146991554f2SKenneth D. Merry
147991554f2SKenneth D. Merry for (i = start; i < sassc->maxtargets; i++) {
148991554f2SKenneth D. Merry target = &sassc->targets[i];
149991554f2SKenneth D. Merry if (target->handle == handle)
150991554f2SKenneth D. Merry return (target);
151991554f2SKenneth D. Merry }
152991554f2SKenneth D. Merry
153991554f2SKenneth D. Merry return (NULL);
154991554f2SKenneth D. Merry }
155991554f2SKenneth D. Merry
156991554f2SKenneth D. Merry /* we need to freeze the simq during attach and diag reset, to avoid failing
157991554f2SKenneth D. Merry * commands before device handles have been found by discovery. Since
158991554f2SKenneth D. Merry * discovery involves reading config pages and possibly sending commands,
159991554f2SKenneth D. Merry * discovery actions may continue even after we receive the end of discovery
160991554f2SKenneth D. Merry * event, so refcount discovery actions instead of assuming we can unfreeze
161991554f2SKenneth D. Merry * the simq when we get the event.
162991554f2SKenneth D. Merry */
163991554f2SKenneth D. Merry void
mprsas_startup_increment(struct mprsas_softc * sassc)164991554f2SKenneth D. Merry mprsas_startup_increment(struct mprsas_softc *sassc)
165991554f2SKenneth D. Merry {
166991554f2SKenneth D. Merry MPR_FUNCTRACE(sassc->sc);
167991554f2SKenneth D. Merry
168991554f2SKenneth D. Merry if ((sassc->flags & MPRSAS_IN_STARTUP) != 0) {
169991554f2SKenneth D. Merry if (sassc->startup_refcount++ == 0) {
170991554f2SKenneth D. Merry /* just starting, freeze the simq */
171991554f2SKenneth D. Merry mpr_dprint(sassc->sc, MPR_INIT,
172991554f2SKenneth D. Merry "%s freezing simq\n", __func__);
173991554f2SKenneth D. Merry xpt_hold_boot();
174991554f2SKenneth D. Merry xpt_freeze_simq(sassc->sim, 1);
175991554f2SKenneth D. Merry }
176991554f2SKenneth D. Merry mpr_dprint(sassc->sc, MPR_INIT, "%s refcount %u\n", __func__,
177991554f2SKenneth D. Merry sassc->startup_refcount);
178991554f2SKenneth D. Merry }
179991554f2SKenneth D. Merry }
180991554f2SKenneth D. Merry
181991554f2SKenneth D. Merry void
mprsas_release_simq_reinit(struct mprsas_softc * sassc)182991554f2SKenneth D. Merry mprsas_release_simq_reinit(struct mprsas_softc *sassc)
183991554f2SKenneth D. Merry {
184991554f2SKenneth D. Merry if (sassc->flags & MPRSAS_QUEUE_FROZEN) {
185991554f2SKenneth D. Merry sassc->flags &= ~MPRSAS_QUEUE_FROZEN;
186991554f2SKenneth D. Merry xpt_release_simq(sassc->sim, 1);
187991554f2SKenneth D. Merry mpr_dprint(sassc->sc, MPR_INFO, "Unfreezing SIM queue\n");
188991554f2SKenneth D. Merry }
189991554f2SKenneth D. Merry }
190991554f2SKenneth D. Merry
191991554f2SKenneth D. Merry void
mprsas_startup_decrement(struct mprsas_softc * sassc)192991554f2SKenneth D. Merry mprsas_startup_decrement(struct mprsas_softc *sassc)
193991554f2SKenneth D. Merry {
194991554f2SKenneth D. Merry MPR_FUNCTRACE(sassc->sc);
195991554f2SKenneth D. Merry
196991554f2SKenneth D. Merry if ((sassc->flags & MPRSAS_IN_STARTUP) != 0) {
197991554f2SKenneth D. Merry if (--sassc->startup_refcount == 0) {
198991554f2SKenneth D. Merry /* finished all discovery-related actions, release
199991554f2SKenneth D. Merry * the simq and rescan for the latest topology.
200991554f2SKenneth D. Merry */
201991554f2SKenneth D. Merry mpr_dprint(sassc->sc, MPR_INIT,
202991554f2SKenneth D. Merry "%s releasing simq\n", __func__);
203991554f2SKenneth D. Merry sassc->flags &= ~MPRSAS_IN_STARTUP;
204991554f2SKenneth D. Merry xpt_release_simq(sassc->sim, 1);
205991554f2SKenneth D. Merry xpt_release_boot();
206991554f2SKenneth D. Merry }
207991554f2SKenneth D. Merry mpr_dprint(sassc->sc, MPR_INIT, "%s refcount %u\n", __func__,
208991554f2SKenneth D. Merry sassc->startup_refcount);
209991554f2SKenneth D. Merry }
210991554f2SKenneth D. Merry }
211991554f2SKenneth D. Merry
212b7f1ee79SScott Long /*
213b7f1ee79SScott Long * The firmware requires us to stop sending commands when we're doing task
214b7f1ee79SScott Long * management.
215991554f2SKenneth D. Merry * use.
216b7f1ee79SScott Long * XXX The logic for serializing the device has been made lazy and moved to
217b7f1ee79SScott Long * mprsas_prepare_for_tm().
218991554f2SKenneth D. Merry */
219991554f2SKenneth D. Merry struct mpr_command *
mprsas_alloc_tm(struct mpr_softc * sc)220991554f2SKenneth D. Merry mprsas_alloc_tm(struct mpr_softc *sc)
221991554f2SKenneth D. Merry {
22246b9415fSScott Long MPI2_SCSI_TASK_MANAGE_REQUEST *req;
223991554f2SKenneth D. Merry struct mpr_command *tm;
224991554f2SKenneth D. Merry
225991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
226991554f2SKenneth D. Merry tm = mpr_alloc_high_priority_command(sc);
22746b9415fSScott Long if (tm == NULL)
22846b9415fSScott Long return (NULL);
22946b9415fSScott Long
23046b9415fSScott Long req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
23146b9415fSScott Long req->Function = MPI2_FUNCTION_SCSI_TASK_MGMT;
232991554f2SKenneth D. Merry return tm;
233991554f2SKenneth D. Merry }
234991554f2SKenneth D. Merry
235991554f2SKenneth D. Merry void
mprsas_free_tm(struct mpr_softc * sc,struct mpr_command * tm)236991554f2SKenneth D. Merry mprsas_free_tm(struct mpr_softc *sc, struct mpr_command *tm)
237991554f2SKenneth D. Merry {
23888619392SStephen McConnell
239a2c14879SStephen McConnell MPR_FUNCTRACE(sc);
240991554f2SKenneth D. Merry if (tm == NULL)
241991554f2SKenneth D. Merry return;
242991554f2SKenneth D. Merry
243a2c14879SStephen McConnell /*
244a2c14879SStephen McConnell * For TM's the devq is frozen for the device. Unfreeze it here and
245a2c14879SStephen McConnell * free the resources used for freezing the devq. Must clear the
246a2c14879SStephen McConnell * INRESET flag as well or scsi I/O will not work.
247991554f2SKenneth D. Merry */
248a2c14879SStephen McConnell if (tm->cm_ccb) {
249a8837c77SWarner Losh mpr_dprint(sc, MPR_XINFO | MPR_RECOVERY,
250a8837c77SWarner Losh "Unfreezing devq for target ID %d\n",
2519781c28cSAlexander Motin tm->cm_targ->tid);
2529781c28cSAlexander Motin tm->cm_targ->flags &= ~MPRSAS_TARGET_INRESET;
253a2c14879SStephen McConnell xpt_release_devq(tm->cm_ccb->ccb_h.path, 1, TRUE);
254a2c14879SStephen McConnell xpt_free_path(tm->cm_ccb->ccb_h.path);
255a2c14879SStephen McConnell xpt_free_ccb(tm->cm_ccb);
256a2c14879SStephen McConnell }
257991554f2SKenneth D. Merry
258991554f2SKenneth D. Merry mpr_free_high_priority_command(sc, tm);
259991554f2SKenneth D. Merry }
260991554f2SKenneth D. Merry
261991554f2SKenneth D. Merry void
mprsas_rescan_target(struct mpr_softc * sc,struct mprsas_target * targ)262991554f2SKenneth D. Merry mprsas_rescan_target(struct mpr_softc *sc, struct mprsas_target *targ)
263991554f2SKenneth D. Merry {
264991554f2SKenneth D. Merry struct mprsas_softc *sassc = sc->sassc;
265991554f2SKenneth D. Merry path_id_t pathid;
266991554f2SKenneth D. Merry target_id_t targetid;
267991554f2SKenneth D. Merry union ccb *ccb;
268991554f2SKenneth D. Merry
269991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
270991554f2SKenneth D. Merry pathid = cam_sim_path(sassc->sim);
271991554f2SKenneth D. Merry if (targ == NULL)
272991554f2SKenneth D. Merry targetid = CAM_TARGET_WILDCARD;
273991554f2SKenneth D. Merry else
274991554f2SKenneth D. Merry targetid = targ - sassc->targets;
275991554f2SKenneth D. Merry
276991554f2SKenneth D. Merry /*
277991554f2SKenneth D. Merry * Allocate a CCB and schedule a rescan.
278991554f2SKenneth D. Merry */
279991554f2SKenneth D. Merry ccb = xpt_alloc_ccb_nowait();
280991554f2SKenneth D. Merry if (ccb == NULL) {
281991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "unable to alloc CCB for rescan\n");
282991554f2SKenneth D. Merry return;
283991554f2SKenneth D. Merry }
284991554f2SKenneth D. Merry
285a2c14879SStephen McConnell if (xpt_create_path(&ccb->ccb_h.path, NULL, pathid, targetid,
286a2c14879SStephen McConnell CAM_LUN_WILDCARD) != CAM_REQ_CMP) {
287991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "unable to create path for rescan\n");
288991554f2SKenneth D. Merry xpt_free_ccb(ccb);
289991554f2SKenneth D. Merry return;
290991554f2SKenneth D. Merry }
291991554f2SKenneth D. Merry
292991554f2SKenneth D. Merry if (targetid == CAM_TARGET_WILDCARD)
293991554f2SKenneth D. Merry ccb->ccb_h.func_code = XPT_SCAN_BUS;
294991554f2SKenneth D. Merry else
295991554f2SKenneth D. Merry ccb->ccb_h.func_code = XPT_SCAN_TGT;
296991554f2SKenneth D. Merry
297991554f2SKenneth D. Merry mpr_dprint(sc, MPR_TRACE, "%s targetid %u\n", __func__, targetid);
298991554f2SKenneth D. Merry xpt_rescan(ccb);
299991554f2SKenneth D. Merry }
300991554f2SKenneth D. Merry
301991554f2SKenneth D. Merry static void
mprsas_log_command(struct mpr_command * cm,u_int level,const char * fmt,...)302991554f2SKenneth D. Merry mprsas_log_command(struct mpr_command *cm, u_int level, const char *fmt, ...)
303991554f2SKenneth D. Merry {
304991554f2SKenneth D. Merry struct sbuf sb;
305991554f2SKenneth D. Merry va_list ap;
306a2386b6fSAlexander Motin char str[224];
307991554f2SKenneth D. Merry
308991554f2SKenneth D. Merry if (cm == NULL)
309991554f2SKenneth D. Merry return;
310991554f2SKenneth D. Merry
311991554f2SKenneth D. Merry /* No need to be in here if debugging isn't enabled */
312991554f2SKenneth D. Merry if ((cm->cm_sc->mpr_debug & level) == 0)
313991554f2SKenneth D. Merry return;
314991554f2SKenneth D. Merry
315991554f2SKenneth D. Merry sbuf_new(&sb, str, sizeof(str), 0);
316991554f2SKenneth D. Merry
317991554f2SKenneth D. Merry va_start(ap, fmt);
318991554f2SKenneth D. Merry
319991554f2SKenneth D. Merry if (cm->cm_ccb != NULL) {
320*8c4ee0b2SAlexander Motin xpt_path_sbuf(cm->cm_ccb->csio.ccb_h.path, &sb);
321991554f2SKenneth D. Merry if (cm->cm_ccb->ccb_h.func_code == XPT_SCSI_IO) {
322991554f2SKenneth D. Merry scsi_command_string(&cm->cm_ccb->csio, &sb);
323991554f2SKenneth D. Merry sbuf_printf(&sb, "length %d ",
324991554f2SKenneth D. Merry cm->cm_ccb->csio.dxfer_len);
325991554f2SKenneth D. Merry }
326991554f2SKenneth D. Merry } else {
327991554f2SKenneth D. Merry sbuf_printf(&sb, "(noperiph:%s%d:%u:%u:%u): ",
328991554f2SKenneth D. Merry cam_sim_name(cm->cm_sc->sassc->sim),
329991554f2SKenneth D. Merry cam_sim_unit(cm->cm_sc->sassc->sim),
330991554f2SKenneth D. Merry cam_sim_bus(cm->cm_sc->sassc->sim),
331991554f2SKenneth D. Merry cm->cm_targ ? cm->cm_targ->tid : 0xFFFFFFFF,
332991554f2SKenneth D. Merry cm->cm_lun);
333991554f2SKenneth D. Merry }
334991554f2SKenneth D. Merry
335991554f2SKenneth D. Merry sbuf_printf(&sb, "SMID %u ", cm->cm_desc.Default.SMID);
336991554f2SKenneth D. Merry sbuf_vprintf(&sb, fmt, ap);
337991554f2SKenneth D. Merry sbuf_finish(&sb);
338c11c484fSScott Long mpr_print_field(cm->cm_sc, "%s", sbuf_data(&sb));
339991554f2SKenneth D. Merry
340991554f2SKenneth D. Merry va_end(ap);
341991554f2SKenneth D. Merry }
342991554f2SKenneth D. Merry
343991554f2SKenneth D. Merry static void
mprsas_remove_volume(struct mpr_softc * sc,struct mpr_command * tm)344991554f2SKenneth D. Merry mprsas_remove_volume(struct mpr_softc *sc, struct mpr_command *tm)
345991554f2SKenneth D. Merry {
346991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REPLY *reply;
347991554f2SKenneth D. Merry struct mprsas_target *targ;
348991554f2SKenneth D. Merry uint16_t handle;
349991554f2SKenneth D. Merry
350991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
351991554f2SKenneth D. Merry
352991554f2SKenneth D. Merry reply = (MPI2_SCSI_TASK_MANAGE_REPLY *)tm->cm_reply;
353991554f2SKenneth D. Merry handle = (uint16_t)(uintptr_t)tm->cm_complete_data;
354991554f2SKenneth D. Merry targ = tm->cm_targ;
355991554f2SKenneth D. Merry
356991554f2SKenneth D. Merry if (reply == NULL) {
357991554f2SKenneth D. Merry /* XXX retry the remove after the diag reset completes? */
358991554f2SKenneth D. Merry mpr_dprint(sc, MPR_FAULT, "%s NULL reply resetting device "
359991554f2SKenneth D. Merry "0x%04x\n", __func__, handle);
360991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
361991554f2SKenneth D. Merry return;
362991554f2SKenneth D. Merry }
363991554f2SKenneth D. Merry
364d3f6eabfSStephen McConnell if ((le16toh(reply->IOCStatus) & MPI2_IOCSTATUS_MASK) !=
365d3f6eabfSStephen McConnell MPI2_IOCSTATUS_SUCCESS) {
36658581c13SStephen McConnell mpr_dprint(sc, MPR_ERROR, "IOCStatus = 0x%x while resetting "
367d3f6eabfSStephen McConnell "device 0x%x\n", le16toh(reply->IOCStatus), handle);
368991554f2SKenneth D. Merry }
369991554f2SKenneth D. Merry
370991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "Reset aborted %u commands\n",
371d3f6eabfSStephen McConnell le32toh(reply->TerminationCount));
372991554f2SKenneth D. Merry mpr_free_reply(sc, tm->cm_reply_data);
373991554f2SKenneth D. Merry tm->cm_reply = NULL; /* Ensures the reply won't get re-freed */
374991554f2SKenneth D. Merry
375991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "clearing target %u handle 0x%04x\n",
376991554f2SKenneth D. Merry targ->tid, handle);
377991554f2SKenneth D. Merry
378991554f2SKenneth D. Merry /*
379991554f2SKenneth D. Merry * Don't clear target if remove fails because things will get confusing.
380991554f2SKenneth D. Merry * Leave the devname and sasaddr intact so that we know to avoid reusing
381991554f2SKenneth D. Merry * this target id if possible, and so we can assign the same target id
382991554f2SKenneth D. Merry * to this device if it comes back in the future.
383991554f2SKenneth D. Merry */
384d3f6eabfSStephen McConnell if ((le16toh(reply->IOCStatus) & MPI2_IOCSTATUS_MASK) ==
385d3f6eabfSStephen McConnell MPI2_IOCSTATUS_SUCCESS) {
386991554f2SKenneth D. Merry targ = tm->cm_targ;
387991554f2SKenneth D. Merry targ->handle = 0x0;
388991554f2SKenneth D. Merry targ->encl_handle = 0x0;
389991554f2SKenneth D. Merry targ->encl_level_valid = 0x0;
390991554f2SKenneth D. Merry targ->encl_level = 0x0;
391991554f2SKenneth D. Merry targ->connector_name[0] = ' ';
392991554f2SKenneth D. Merry targ->connector_name[1] = ' ';
393991554f2SKenneth D. Merry targ->connector_name[2] = ' ';
394991554f2SKenneth D. Merry targ->connector_name[3] = ' ';
395991554f2SKenneth D. Merry targ->encl_slot = 0x0;
396991554f2SKenneth D. Merry targ->exp_dev_handle = 0x0;
397991554f2SKenneth D. Merry targ->phy_num = 0x0;
398991554f2SKenneth D. Merry targ->linkrate = 0x0;
399991554f2SKenneth D. Merry targ->devinfo = 0x0;
400991554f2SKenneth D. Merry targ->flags = 0x0;
401991554f2SKenneth D. Merry targ->scsi_req_desc_type = 0;
402991554f2SKenneth D. Merry }
403991554f2SKenneth D. Merry
404991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
405991554f2SKenneth D. Merry }
406991554f2SKenneth D. Merry
407991554f2SKenneth D. Merry /*
408e3c5965cSAlexander Motin * Retry mprsas_prepare_remove() if some previous attempt failed to allocate
409e3c5965cSAlexander Motin * high priority command due to limit reached.
410e3c5965cSAlexander Motin */
411e3c5965cSAlexander Motin void
mprsas_prepare_remove_retry(struct mprsas_softc * sassc)412e3c5965cSAlexander Motin mprsas_prepare_remove_retry(struct mprsas_softc *sassc)
413e3c5965cSAlexander Motin {
414e3c5965cSAlexander Motin struct mprsas_target *target;
415e3c5965cSAlexander Motin int i;
416e3c5965cSAlexander Motin
417e3c5965cSAlexander Motin if ((sassc->flags & MPRSAS_TOREMOVE) == 0)
418e3c5965cSAlexander Motin return;
419e3c5965cSAlexander Motin
420e3c5965cSAlexander Motin for (i = 0; i < sassc->maxtargets; i++) {
421e3c5965cSAlexander Motin target = &sassc->targets[i];
422e3c5965cSAlexander Motin if ((target->flags & MPRSAS_TARGET_TOREMOVE) == 0)
423e3c5965cSAlexander Motin continue;
424e3c5965cSAlexander Motin if (TAILQ_EMPTY(&sassc->sc->high_priority_req_list))
425e3c5965cSAlexander Motin return;
426e3c5965cSAlexander Motin target->flags &= ~MPRSAS_TARGET_TOREMOVE;
427e3c5965cSAlexander Motin if (target->flags & MPR_TARGET_FLAGS_VOLUME)
428e3c5965cSAlexander Motin mprsas_prepare_volume_remove(sassc, target->handle);
429e3c5965cSAlexander Motin else
430e3c5965cSAlexander Motin mprsas_prepare_remove(sassc, target->handle);
431e3c5965cSAlexander Motin }
432e3c5965cSAlexander Motin sassc->flags &= ~MPRSAS_TOREMOVE;
433e3c5965cSAlexander Motin }
434e3c5965cSAlexander Motin
435e3c5965cSAlexander Motin /*
436991554f2SKenneth D. Merry * No Need to call "MPI2_SAS_OP_REMOVE_DEVICE" For Volume removal.
437991554f2SKenneth D. Merry * Otherwise Volume Delete is same as Bare Drive Removal.
438991554f2SKenneth D. Merry */
439991554f2SKenneth D. Merry void
mprsas_prepare_volume_remove(struct mprsas_softc * sassc,uint16_t handle)440991554f2SKenneth D. Merry mprsas_prepare_volume_remove(struct mprsas_softc *sassc, uint16_t handle)
441991554f2SKenneth D. Merry {
442991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
443991554f2SKenneth D. Merry struct mpr_softc *sc;
444991554f2SKenneth D. Merry struct mpr_command *cm;
445991554f2SKenneth D. Merry struct mprsas_target *targ = NULL;
446991554f2SKenneth D. Merry
447991554f2SKenneth D. Merry MPR_FUNCTRACE(sassc->sc);
448991554f2SKenneth D. Merry sc = sassc->sc;
449991554f2SKenneth D. Merry
450991554f2SKenneth D. Merry targ = mprsas_find_target_by_handle(sassc, 0, handle);
451991554f2SKenneth D. Merry if (targ == NULL) {
452991554f2SKenneth D. Merry /* FIXME: what is the action? */
453991554f2SKenneth D. Merry /* We don't know about this device? */
454991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR,
455991554f2SKenneth D. Merry "%s %d : invalid handle 0x%x \n", __func__,__LINE__, handle);
456991554f2SKenneth D. Merry return;
457991554f2SKenneth D. Merry }
458991554f2SKenneth D. Merry
459991554f2SKenneth D. Merry targ->flags |= MPRSAS_TARGET_INREMOVAL;
460991554f2SKenneth D. Merry
461991554f2SKenneth D. Merry cm = mprsas_alloc_tm(sc);
462991554f2SKenneth D. Merry if (cm == NULL) {
463e3c5965cSAlexander Motin targ->flags |= MPRSAS_TARGET_TOREMOVE;
464e3c5965cSAlexander Motin sassc->flags |= MPRSAS_TOREMOVE;
465991554f2SKenneth D. Merry return;
466991554f2SKenneth D. Merry }
467991554f2SKenneth D. Merry
468991554f2SKenneth D. Merry mprsas_rescan_target(sc, targ);
469991554f2SKenneth D. Merry
470991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)cm->cm_req;
471991554f2SKenneth D. Merry req->DevHandle = targ->handle;
472991554f2SKenneth D. Merry req->TaskType = MPI2_SCSITASKMGMT_TASKTYPE_TARGET_RESET;
473991554f2SKenneth D. Merry
47489d1c21fSKashyap D Desai if (!targ->is_nvme || sc->custom_nvme_tm_handling) {
475991554f2SKenneth D. Merry /* SAS Hard Link Reset / SATA Link Reset */
476991554f2SKenneth D. Merry req->MsgFlags = MPI2_SCSITASKMGMT_MSGFLAGS_LINK_RESET;
47789d1c21fSKashyap D Desai } else {
47889d1c21fSKashyap D Desai /* PCIe Protocol Level Reset*/
47989d1c21fSKashyap D Desai req->MsgFlags =
48089d1c21fSKashyap D Desai MPI26_SCSITASKMGMT_MSGFLAGS_PROTOCOL_LVL_RST_PCIE;
48189d1c21fSKashyap D Desai }
482991554f2SKenneth D. Merry
483991554f2SKenneth D. Merry cm->cm_targ = targ;
484991554f2SKenneth D. Merry cm->cm_data = NULL;
485991554f2SKenneth D. Merry cm->cm_complete = mprsas_remove_volume;
486991554f2SKenneth D. Merry cm->cm_complete_data = (void *)(uintptr_t)handle;
487a2c14879SStephen McConnell
488a2c14879SStephen McConnell mpr_dprint(sc, MPR_INFO, "%s: Sending reset for target ID %d\n",
489a2c14879SStephen McConnell __func__, targ->tid);
490a2c14879SStephen McConnell mprsas_prepare_for_tm(sc, cm, targ, CAM_LUN_WILDCARD);
491a2c14879SStephen McConnell
492991554f2SKenneth D. Merry mpr_map_command(sc, cm);
493991554f2SKenneth D. Merry }
494991554f2SKenneth D. Merry
495991554f2SKenneth D. Merry /*
49667feec50SStephen McConnell * The firmware performs debounce on the link to avoid transient link errors
49767feec50SStephen McConnell * and false removals. When it does decide that link has been lost and a
49867feec50SStephen McConnell * device needs to go away, it expects that the host will perform a target reset
49967feec50SStephen McConnell * and then an op remove. The reset has the side-effect of aborting any
50067feec50SStephen McConnell * outstanding requests for the device, which is required for the op-remove to
50167feec50SStephen McConnell * succeed. It's not clear if the host should check for the device coming back
50267feec50SStephen McConnell * alive after the reset.
503991554f2SKenneth D. Merry */
504991554f2SKenneth D. Merry void
mprsas_prepare_remove(struct mprsas_softc * sassc,uint16_t handle)505991554f2SKenneth D. Merry mprsas_prepare_remove(struct mprsas_softc *sassc, uint16_t handle)
506991554f2SKenneth D. Merry {
507991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
508991554f2SKenneth D. Merry struct mpr_softc *sc;
50946b9415fSScott Long struct mpr_command *tm;
510991554f2SKenneth D. Merry struct mprsas_target *targ = NULL;
511991554f2SKenneth D. Merry
512991554f2SKenneth D. Merry MPR_FUNCTRACE(sassc->sc);
513991554f2SKenneth D. Merry
514991554f2SKenneth D. Merry sc = sassc->sc;
515991554f2SKenneth D. Merry
516991554f2SKenneth D. Merry targ = mprsas_find_target_by_handle(sassc, 0, handle);
517991554f2SKenneth D. Merry if (targ == NULL) {
518991554f2SKenneth D. Merry /* FIXME: what is the action? */
519991554f2SKenneth D. Merry /* We don't know about this device? */
520991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s : invalid handle 0x%x \n",
521991554f2SKenneth D. Merry __func__, handle);
522991554f2SKenneth D. Merry return;
523991554f2SKenneth D. Merry }
524991554f2SKenneth D. Merry
525991554f2SKenneth D. Merry targ->flags |= MPRSAS_TARGET_INREMOVAL;
526991554f2SKenneth D. Merry
52746b9415fSScott Long tm = mprsas_alloc_tm(sc);
52846b9415fSScott Long if (tm == NULL) {
529e3c5965cSAlexander Motin targ->flags |= MPRSAS_TARGET_TOREMOVE;
530e3c5965cSAlexander Motin sassc->flags |= MPRSAS_TOREMOVE;
531991554f2SKenneth D. Merry return;
532991554f2SKenneth D. Merry }
533991554f2SKenneth D. Merry
534991554f2SKenneth D. Merry mprsas_rescan_target(sc, targ);
535991554f2SKenneth D. Merry
53646b9415fSScott Long req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
537991554f2SKenneth D. Merry req->DevHandle = htole16(targ->handle);
538991554f2SKenneth D. Merry req->TaskType = MPI2_SCSITASKMGMT_TASKTYPE_TARGET_RESET;
539991554f2SKenneth D. Merry
540991554f2SKenneth D. Merry /* SAS Hard Link Reset / SATA Link Reset */
541991554f2SKenneth D. Merry req->MsgFlags = MPI2_SCSITASKMGMT_MSGFLAGS_LINK_RESET;
542991554f2SKenneth D. Merry
54346b9415fSScott Long tm->cm_targ = targ;
54446b9415fSScott Long tm->cm_data = NULL;
54546b9415fSScott Long tm->cm_complete = mprsas_remove_device;
54646b9415fSScott Long tm->cm_complete_data = (void *)(uintptr_t)handle;
547a2c14879SStephen McConnell
548a2c14879SStephen McConnell mpr_dprint(sc, MPR_INFO, "%s: Sending reset for target ID %d\n",
549a2c14879SStephen McConnell __func__, targ->tid);
55046b9415fSScott Long mprsas_prepare_for_tm(sc, tm, targ, CAM_LUN_WILDCARD);
551a2c14879SStephen McConnell
55246b9415fSScott Long mpr_map_command(sc, tm);
553991554f2SKenneth D. Merry }
554991554f2SKenneth D. Merry
555991554f2SKenneth D. Merry static void
mprsas_remove_device(struct mpr_softc * sc,struct mpr_command * tm)556991554f2SKenneth D. Merry mprsas_remove_device(struct mpr_softc *sc, struct mpr_command *tm)
557991554f2SKenneth D. Merry {
558991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REPLY *reply;
559991554f2SKenneth D. Merry MPI2_SAS_IOUNIT_CONTROL_REQUEST *req;
560991554f2SKenneth D. Merry struct mprsas_target *targ;
561991554f2SKenneth D. Merry uint16_t handle;
562991554f2SKenneth D. Merry
563991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
564991554f2SKenneth D. Merry
565991554f2SKenneth D. Merry reply = (MPI2_SCSI_TASK_MANAGE_REPLY *)tm->cm_reply;
566991554f2SKenneth D. Merry handle = (uint16_t)(uintptr_t)tm->cm_complete_data;
567991554f2SKenneth D. Merry targ = tm->cm_targ;
568991554f2SKenneth D. Merry
569991554f2SKenneth D. Merry /*
570991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
571991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
572991554f2SKenneth D. Merry * task management commands don't have S/G lists.
573991554f2SKenneth D. Merry */
574991554f2SKenneth D. Merry if ((tm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
575991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: cm_flags = %#x for remove of "
576991554f2SKenneth D. Merry "handle %#04x! This should not happen!\n", __func__,
577991554f2SKenneth D. Merry tm->cm_flags, handle);
578991554f2SKenneth D. Merry }
579991554f2SKenneth D. Merry
580991554f2SKenneth D. Merry if (reply == NULL) {
581991554f2SKenneth D. Merry /* XXX retry the remove after the diag reset completes? */
582991554f2SKenneth D. Merry mpr_dprint(sc, MPR_FAULT, "%s NULL reply resetting device "
583991554f2SKenneth D. Merry "0x%04x\n", __func__, handle);
584991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
585991554f2SKenneth D. Merry return;
586991554f2SKenneth D. Merry }
587991554f2SKenneth D. Merry
588d3f6eabfSStephen McConnell if ((le16toh(reply->IOCStatus) & MPI2_IOCSTATUS_MASK) !=
589d3f6eabfSStephen McConnell MPI2_IOCSTATUS_SUCCESS) {
59058581c13SStephen McConnell mpr_dprint(sc, MPR_ERROR, "IOCStatus = 0x%x while resetting "
591991554f2SKenneth D. Merry "device 0x%x\n", le16toh(reply->IOCStatus), handle);
592991554f2SKenneth D. Merry }
593991554f2SKenneth D. Merry
594991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "Reset aborted %u commands\n",
595991554f2SKenneth D. Merry le32toh(reply->TerminationCount));
596991554f2SKenneth D. Merry mpr_free_reply(sc, tm->cm_reply_data);
597991554f2SKenneth D. Merry tm->cm_reply = NULL; /* Ensures the reply won't get re-freed */
598991554f2SKenneth D. Merry
599991554f2SKenneth D. Merry /* Reuse the existing command */
600991554f2SKenneth D. Merry req = (MPI2_SAS_IOUNIT_CONTROL_REQUEST *)tm->cm_req;
601991554f2SKenneth D. Merry memset(req, 0, sizeof(*req));
602991554f2SKenneth D. Merry req->Function = MPI2_FUNCTION_SAS_IO_UNIT_CONTROL;
603991554f2SKenneth D. Merry req->Operation = MPI2_SAS_OP_REMOVE_DEVICE;
604991554f2SKenneth D. Merry req->DevHandle = htole16(handle);
605991554f2SKenneth D. Merry tm->cm_data = NULL;
606991554f2SKenneth D. Merry tm->cm_desc.Default.RequestFlags = MPI2_REQ_DESCRIPT_FLAGS_DEFAULT_TYPE;
607991554f2SKenneth D. Merry tm->cm_complete = mprsas_remove_complete;
608991554f2SKenneth D. Merry tm->cm_complete_data = (void *)(uintptr_t)handle;
609991554f2SKenneth D. Merry
6104c1cdd4aSWarner Losh /*
6114c1cdd4aSWarner Losh * Wait to send the REMOVE_DEVICE until all the commands have cleared.
6124c1cdd4aSWarner Losh * They should be aborted or time out and we'll kick thus off there
6134c1cdd4aSWarner Losh * if so.
6144c1cdd4aSWarner Losh */
6154c1cdd4aSWarner Losh if (TAILQ_FIRST(&targ->commands) == NULL) {
616ca420b4eSWarner Losh mpr_dprint(sc, MPR_INFO,
617ca420b4eSWarner Losh "No pending commands: starting remove_device for target %u handle 0x%04x\n",
618ca420b4eSWarner Losh targ->tid, handle);
619991554f2SKenneth D. Merry mpr_map_command(sc, tm);
6204c1cdd4aSWarner Losh targ->pending_remove_tm = NULL;
6214c1cdd4aSWarner Losh } else {
6224c1cdd4aSWarner Losh targ->pending_remove_tm = tm;
6234c1cdd4aSWarner Losh }
624991554f2SKenneth D. Merry
625a2c14879SStephen McConnell mpr_dprint(sc, MPR_INFO, "clearing target %u handle 0x%04x\n",
626991554f2SKenneth D. Merry targ->tid, handle);
627991554f2SKenneth D. Merry if (targ->encl_level_valid) {
628a2c14879SStephen McConnell mpr_dprint(sc, MPR_INFO, "At enclosure level %d, slot %d, "
629991554f2SKenneth D. Merry "connector name (%4s)\n", targ->encl_level, targ->encl_slot,
630991554f2SKenneth D. Merry targ->connector_name);
631991554f2SKenneth D. Merry }
632991554f2SKenneth D. Merry }
633991554f2SKenneth D. Merry
634991554f2SKenneth D. Merry static void
mprsas_remove_complete(struct mpr_softc * sc,struct mpr_command * tm)635991554f2SKenneth D. Merry mprsas_remove_complete(struct mpr_softc *sc, struct mpr_command *tm)
636991554f2SKenneth D. Merry {
637991554f2SKenneth D. Merry MPI2_SAS_IOUNIT_CONTROL_REPLY *reply;
638991554f2SKenneth D. Merry uint16_t handle;
639991554f2SKenneth D. Merry struct mprsas_target *targ;
640991554f2SKenneth D. Merry struct mprsas_lun *lun;
641991554f2SKenneth D. Merry
642991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
643991554f2SKenneth D. Merry
644991554f2SKenneth D. Merry reply = (MPI2_SAS_IOUNIT_CONTROL_REPLY *)tm->cm_reply;
645991554f2SKenneth D. Merry handle = (uint16_t)(uintptr_t)tm->cm_complete_data;
646991554f2SKenneth D. Merry
6474c1cdd4aSWarner Losh targ = tm->cm_targ;
6484c1cdd4aSWarner Losh
6494c1cdd4aSWarner Losh /*
6504c1cdd4aSWarner Losh * At this point, we should have no pending commands for the target.
6514c1cdd4aSWarner Losh * The remove target has just completed.
6524c1cdd4aSWarner Losh */
6534c1cdd4aSWarner Losh KASSERT(TAILQ_FIRST(&targ->commands) == NULL,
6544c1cdd4aSWarner Losh ("%s: no commands should be pending\n", __func__));
6554c1cdd4aSWarner Losh
656991554f2SKenneth D. Merry /*
657991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
658991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
659991554f2SKenneth D. Merry * task management commands don't have S/G lists.
660991554f2SKenneth D. Merry */
661991554f2SKenneth D. Merry if ((tm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
662991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "%s: cm_flags = %#x for remove of "
663991554f2SKenneth D. Merry "handle %#04x! This should not happen!\n", __func__,
664991554f2SKenneth D. Merry tm->cm_flags, handle);
665991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
666991554f2SKenneth D. Merry return;
667991554f2SKenneth D. Merry }
668991554f2SKenneth D. Merry
669991554f2SKenneth D. Merry if (reply == NULL) {
670991554f2SKenneth D. Merry /* most likely a chip reset */
671991554f2SKenneth D. Merry mpr_dprint(sc, MPR_FAULT, "%s NULL reply removing device "
672991554f2SKenneth D. Merry "0x%04x\n", __func__, handle);
673991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
674991554f2SKenneth D. Merry return;
675991554f2SKenneth D. Merry }
676991554f2SKenneth D. Merry
677991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "%s on handle 0x%04x, IOCStatus= 0x%x\n",
678991554f2SKenneth D. Merry __func__, handle, le16toh(reply->IOCStatus));
679991554f2SKenneth D. Merry
680991554f2SKenneth D. Merry /*
681991554f2SKenneth D. Merry * Don't clear target if remove fails because things will get confusing.
682991554f2SKenneth D. Merry * Leave the devname and sasaddr intact so that we know to avoid reusing
683991554f2SKenneth D. Merry * this target id if possible, and so we can assign the same target id
684991554f2SKenneth D. Merry * to this device if it comes back in the future.
685991554f2SKenneth D. Merry */
686d3f6eabfSStephen McConnell if ((le16toh(reply->IOCStatus) & MPI2_IOCSTATUS_MASK) ==
687d3f6eabfSStephen McConnell MPI2_IOCSTATUS_SUCCESS) {
688991554f2SKenneth D. Merry targ->handle = 0x0;
689991554f2SKenneth D. Merry targ->encl_handle = 0x0;
690991554f2SKenneth D. Merry targ->encl_level_valid = 0x0;
691991554f2SKenneth D. Merry targ->encl_level = 0x0;
692991554f2SKenneth D. Merry targ->connector_name[0] = ' ';
693991554f2SKenneth D. Merry targ->connector_name[1] = ' ';
694991554f2SKenneth D. Merry targ->connector_name[2] = ' ';
695991554f2SKenneth D. Merry targ->connector_name[3] = ' ';
696991554f2SKenneth D. Merry targ->encl_slot = 0x0;
697991554f2SKenneth D. Merry targ->exp_dev_handle = 0x0;
698991554f2SKenneth D. Merry targ->phy_num = 0x0;
699991554f2SKenneth D. Merry targ->linkrate = 0x0;
700991554f2SKenneth D. Merry targ->devinfo = 0x0;
701991554f2SKenneth D. Merry targ->flags = 0x0;
702991554f2SKenneth D. Merry targ->scsi_req_desc_type = 0;
703991554f2SKenneth D. Merry
704991554f2SKenneth D. Merry while (!SLIST_EMPTY(&targ->luns)) {
705991554f2SKenneth D. Merry lun = SLIST_FIRST(&targ->luns);
706991554f2SKenneth D. Merry SLIST_REMOVE_HEAD(&targ->luns, lun_link);
707991554f2SKenneth D. Merry free(lun, M_MPR);
708991554f2SKenneth D. Merry }
709991554f2SKenneth D. Merry }
710991554f2SKenneth D. Merry
711991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
712991554f2SKenneth D. Merry }
713991554f2SKenneth D. Merry
714991554f2SKenneth D. Merry static int
mprsas_register_events(struct mpr_softc * sc)715991554f2SKenneth D. Merry mprsas_register_events(struct mpr_softc *sc)
716991554f2SKenneth D. Merry {
717991554f2SKenneth D. Merry uint8_t events[16];
718991554f2SKenneth D. Merry
719991554f2SKenneth D. Merry bzero(events, 16);
720991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_DEVICE_STATUS_CHANGE);
721991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_DISCOVERY);
722991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_BROADCAST_PRIMITIVE);
723991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_INIT_DEVICE_STATUS_CHANGE);
724991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_INIT_TABLE_OVERFLOW);
725991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_TOPOLOGY_CHANGE_LIST);
726991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_SAS_ENCL_DEVICE_STATUS_CHANGE);
727991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_IR_CONFIGURATION_CHANGE_LIST);
728991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_IR_VOLUME);
729991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_IR_PHYSICAL_DISK);
730991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_IR_OPERATION_STATUS);
731991554f2SKenneth D. Merry setbit(events, MPI2_EVENT_TEMP_THRESHOLD);
7325f5baf0eSAlexander Motin setbit(events, MPI2_EVENT_SAS_DEVICE_DISCOVERY_ERROR);
73367feec50SStephen McConnell if (sc->facts->MsgVersion >= MPI2_VERSION_02_06) {
7342bbc5fcbSStephen McConnell setbit(events, MPI2_EVENT_ACTIVE_CABLE_EXCEPTION);
73567feec50SStephen McConnell if (sc->mpr_flags & MPR_FLAGS_GEN35_IOC) {
73667feec50SStephen McConnell setbit(events, MPI2_EVENT_PCIE_DEVICE_STATUS_CHANGE);
73767feec50SStephen McConnell setbit(events, MPI2_EVENT_PCIE_ENUMERATION);
73867feec50SStephen McConnell setbit(events, MPI2_EVENT_PCIE_TOPOLOGY_CHANGE_LIST);
73967feec50SStephen McConnell }
74067feec50SStephen McConnell }
741991554f2SKenneth D. Merry
742991554f2SKenneth D. Merry mpr_register_events(sc, events, mprsas_evt_handler, NULL,
743991554f2SKenneth D. Merry &sc->sassc->mprsas_eh);
744991554f2SKenneth D. Merry
745991554f2SKenneth D. Merry return (0);
746991554f2SKenneth D. Merry }
747991554f2SKenneth D. Merry
748991554f2SKenneth D. Merry int
mpr_attach_sas(struct mpr_softc * sc)749991554f2SKenneth D. Merry mpr_attach_sas(struct mpr_softc *sc)
750991554f2SKenneth D. Merry {
751991554f2SKenneth D. Merry struct mprsas_softc *sassc;
752991554f2SKenneth D. Merry cam_status status;
75362a09ee9SAlexander Motin int unit, error = 0, reqs;
754991554f2SKenneth D. Merry
755991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
756757ff642SScott Long mpr_dprint(sc, MPR_INIT, "%s entered\n", __func__);
757991554f2SKenneth D. Merry
758991554f2SKenneth D. Merry sassc = malloc(sizeof(struct mprsas_softc), M_MPR, M_WAITOK|M_ZERO);
759991554f2SKenneth D. Merry
760991554f2SKenneth D. Merry /*
761a2c14879SStephen McConnell * XXX MaxTargets could change during a reinit. Since we don't
762991554f2SKenneth D. Merry * resize the targets[] array during such an event, cache the value
763991554f2SKenneth D. Merry * of MaxTargets here so that we don't get into trouble later. This
764991554f2SKenneth D. Merry * should move into the reinit logic.
765991554f2SKenneth D. Merry */
766327f2e6cSStephen McConnell sassc->maxtargets = sc->facts->MaxTargets + sc->facts->MaxVolumes;
767991554f2SKenneth D. Merry sassc->targets = malloc(sizeof(struct mprsas_target) *
768991554f2SKenneth D. Merry sassc->maxtargets, M_MPR, M_WAITOK|M_ZERO);
769991554f2SKenneth D. Merry sc->sassc = sassc;
770991554f2SKenneth D. Merry sassc->sc = sc;
771991554f2SKenneth D. Merry
77262a09ee9SAlexander Motin reqs = sc->num_reqs - sc->num_prireqs - 1;
77362a09ee9SAlexander Motin if ((sassc->devq = cam_simq_alloc(reqs)) == NULL) {
774757ff642SScott Long mpr_dprint(sc, MPR_INIT|MPR_ERROR, "Cannot allocate SIMQ\n");
775991554f2SKenneth D. Merry error = ENOMEM;
776991554f2SKenneth D. Merry goto out;
777991554f2SKenneth D. Merry }
778991554f2SKenneth D. Merry
779991554f2SKenneth D. Merry unit = device_get_unit(sc->mpr_dev);
780991554f2SKenneth D. Merry sassc->sim = cam_sim_alloc(mprsas_action, mprsas_poll, "mpr", sassc,
78162a09ee9SAlexander Motin unit, &sc->mpr_mtx, reqs, reqs, sassc->devq);
782991554f2SKenneth D. Merry if (sassc->sim == NULL) {
783757ff642SScott Long mpr_dprint(sc, MPR_INIT|MPR_ERROR, "Cannot allocate SIM\n");
784991554f2SKenneth D. Merry error = EINVAL;
785991554f2SKenneth D. Merry goto out;
786991554f2SKenneth D. Merry }
787991554f2SKenneth D. Merry
788991554f2SKenneth D. Merry TAILQ_INIT(&sassc->ev_queue);
789991554f2SKenneth D. Merry
790991554f2SKenneth D. Merry /* Initialize taskqueue for Event Handling */
791991554f2SKenneth D. Merry TASK_INIT(&sassc->ev_task, 0, mprsas_firmware_event_work, sc);
792991554f2SKenneth D. Merry sassc->ev_tq = taskqueue_create("mpr_taskq", M_NOWAIT | M_ZERO,
793991554f2SKenneth D. Merry taskqueue_thread_enqueue, &sassc->ev_tq);
794c2a0f07aSAlexander Motin taskqueue_start_threads(&sassc->ev_tq, 1, PRIBIO, "%s taskq",
795991554f2SKenneth D. Merry device_get_nameunit(sc->mpr_dev));
796991554f2SKenneth D. Merry
797991554f2SKenneth D. Merry mpr_lock(sc);
798991554f2SKenneth D. Merry
799991554f2SKenneth D. Merry /*
800991554f2SKenneth D. Merry * XXX There should be a bus for every port on the adapter, but since
801991554f2SKenneth D. Merry * we're just going to fake the topology for now, we'll pretend that
802991554f2SKenneth D. Merry * everything is just a target on a single bus.
803991554f2SKenneth D. Merry */
804991554f2SKenneth D. Merry if ((error = xpt_bus_register(sassc->sim, sc->mpr_dev, 0)) != 0) {
805757ff642SScott Long mpr_dprint(sc, MPR_INIT|MPR_ERROR,
806757ff642SScott Long "Error %d registering SCSI bus\n", error);
807991554f2SKenneth D. Merry mpr_unlock(sc);
808991554f2SKenneth D. Merry goto out;
809991554f2SKenneth D. Merry }
810991554f2SKenneth D. Merry
811991554f2SKenneth D. Merry /*
812a2c14879SStephen McConnell * Assume that discovery events will start right away.
813991554f2SKenneth D. Merry *
814991554f2SKenneth D. Merry * Hold off boot until discovery is complete.
815991554f2SKenneth D. Merry */
816991554f2SKenneth D. Merry sassc->flags |= MPRSAS_IN_STARTUP | MPRSAS_IN_DISCOVERY;
817991554f2SKenneth D. Merry sc->sassc->startup_refcount = 0;
818991554f2SKenneth D. Merry mprsas_startup_increment(sassc);
819991554f2SKenneth D. Merry
82002d81940SAlexander Motin mpr_unlock(sc);
82102d81940SAlexander Motin
822991554f2SKenneth D. Merry /*
823991554f2SKenneth D. Merry * Register for async events so we can determine the EEDP
824991554f2SKenneth D. Merry * capabilities of devices.
825991554f2SKenneth D. Merry */
826991554f2SKenneth D. Merry status = xpt_create_path(&sassc->path, /*periph*/NULL,
827991554f2SKenneth D. Merry cam_sim_path(sc->sassc->sim), CAM_TARGET_WILDCARD,
828991554f2SKenneth D. Merry CAM_LUN_WILDCARD);
829991554f2SKenneth D. Merry if (status != CAM_REQ_CMP) {
830757ff642SScott Long mpr_dprint(sc, MPR_INIT|MPR_ERROR,
831757ff642SScott Long "Error %#x creating sim path\n", status);
832991554f2SKenneth D. Merry sassc->path = NULL;
833991554f2SKenneth D. Merry } else {
834991554f2SKenneth D. Merry int event;
835991554f2SKenneth D. Merry
83602d81940SAlexander Motin event = AC_ADVINFO_CHANGED;
837991554f2SKenneth D. Merry status = xpt_register_async(event, mprsas_async, sc,
838991554f2SKenneth D. Merry sassc->path);
83907aa4de1SKenneth D. Merry
840991554f2SKenneth D. Merry if (status != CAM_REQ_CMP) {
841991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR,
842991554f2SKenneth D. Merry "Error %#x registering async handler for "
843991554f2SKenneth D. Merry "AC_ADVINFO_CHANGED events\n", status);
844991554f2SKenneth D. Merry xpt_free_path(sassc->path);
845991554f2SKenneth D. Merry sassc->path = NULL;
846991554f2SKenneth D. Merry }
847991554f2SKenneth D. Merry }
848991554f2SKenneth D. Merry if (status != CAM_REQ_CMP) {
849991554f2SKenneth D. Merry /*
850991554f2SKenneth D. Merry * EEDP use is the exception, not the rule.
851991554f2SKenneth D. Merry * Warn the user, but do not fail to attach.
852991554f2SKenneth D. Merry */
853991554f2SKenneth D. Merry mpr_printf(sc, "EEDP capabilities disabled.\n");
854991554f2SKenneth D. Merry }
855991554f2SKenneth D. Merry
856991554f2SKenneth D. Merry mprsas_register_events(sc);
857991554f2SKenneth D. Merry out:
858991554f2SKenneth D. Merry if (error)
859991554f2SKenneth D. Merry mpr_detach_sas(sc);
860757ff642SScott Long
861757ff642SScott Long mpr_dprint(sc, MPR_INIT, "%s exit, error= %d\n", __func__, error);
862991554f2SKenneth D. Merry return (error);
863991554f2SKenneth D. Merry }
864991554f2SKenneth D. Merry
865991554f2SKenneth D. Merry int
mpr_detach_sas(struct mpr_softc * sc)866991554f2SKenneth D. Merry mpr_detach_sas(struct mpr_softc *sc)
867991554f2SKenneth D. Merry {
868991554f2SKenneth D. Merry struct mprsas_softc *sassc;
869991554f2SKenneth D. Merry struct mprsas_lun *lun, *lun_tmp;
870991554f2SKenneth D. Merry struct mprsas_target *targ;
871991554f2SKenneth D. Merry int i;
872991554f2SKenneth D. Merry
873991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
874991554f2SKenneth D. Merry
875991554f2SKenneth D. Merry if (sc->sassc == NULL)
876991554f2SKenneth D. Merry return (0);
877991554f2SKenneth D. Merry
878991554f2SKenneth D. Merry sassc = sc->sassc;
879991554f2SKenneth D. Merry mpr_deregister_events(sc, sassc->mprsas_eh);
880991554f2SKenneth D. Merry
881991554f2SKenneth D. Merry /*
882991554f2SKenneth D. Merry * Drain and free the event handling taskqueue with the lock
883991554f2SKenneth D. Merry * unheld so that any parallel processing tasks drain properly
884991554f2SKenneth D. Merry * without deadlocking.
885991554f2SKenneth D. Merry */
886991554f2SKenneth D. Merry if (sassc->ev_tq != NULL)
887991554f2SKenneth D. Merry taskqueue_free(sassc->ev_tq);
888991554f2SKenneth D. Merry
889991554f2SKenneth D. Merry /* Deregister our async handler */
890991554f2SKenneth D. Merry if (sassc->path != NULL) {
891991554f2SKenneth D. Merry xpt_register_async(0, mprsas_async, sc, sassc->path);
892991554f2SKenneth D. Merry xpt_free_path(sassc->path);
893991554f2SKenneth D. Merry sassc->path = NULL;
894991554f2SKenneth D. Merry }
895991554f2SKenneth D. Merry
89602d81940SAlexander Motin /* Make sure CAM doesn't wedge if we had to bail out early. */
89702d81940SAlexander Motin mpr_lock(sc);
89802d81940SAlexander Motin
89902d81940SAlexander Motin while (sassc->startup_refcount != 0)
90002d81940SAlexander Motin mprsas_startup_decrement(sassc);
90102d81940SAlexander Motin
902991554f2SKenneth D. Merry if (sassc->flags & MPRSAS_IN_STARTUP)
903991554f2SKenneth D. Merry xpt_release_simq(sassc->sim, 1);
904991554f2SKenneth D. Merry
905991554f2SKenneth D. Merry if (sassc->sim != NULL) {
906991554f2SKenneth D. Merry xpt_bus_deregister(cam_sim_path(sassc->sim));
907991554f2SKenneth D. Merry cam_sim_free(sassc->sim, FALSE);
908991554f2SKenneth D. Merry }
909991554f2SKenneth D. Merry
910991554f2SKenneth D. Merry mpr_unlock(sc);
911991554f2SKenneth D. Merry
912991554f2SKenneth D. Merry if (sassc->devq != NULL)
913991554f2SKenneth D. Merry cam_simq_free(sassc->devq);
914991554f2SKenneth D. Merry
915991554f2SKenneth D. Merry for (i = 0; i < sassc->maxtargets; i++) {
916991554f2SKenneth D. Merry targ = &sassc->targets[i];
917991554f2SKenneth D. Merry SLIST_FOREACH_SAFE(lun, &targ->luns, lun_link, lun_tmp) {
918991554f2SKenneth D. Merry free(lun, M_MPR);
919991554f2SKenneth D. Merry }
920991554f2SKenneth D. Merry }
921991554f2SKenneth D. Merry free(sassc->targets, M_MPR);
922991554f2SKenneth D. Merry free(sassc, M_MPR);
923991554f2SKenneth D. Merry sc->sassc = NULL;
924991554f2SKenneth D. Merry
925991554f2SKenneth D. Merry return (0);
926991554f2SKenneth D. Merry }
927991554f2SKenneth D. Merry
928991554f2SKenneth D. Merry void
mprsas_discovery_end(struct mprsas_softc * sassc)929991554f2SKenneth D. Merry mprsas_discovery_end(struct mprsas_softc *sassc)
930991554f2SKenneth D. Merry {
931991554f2SKenneth D. Merry struct mpr_softc *sc = sassc->sc;
932991554f2SKenneth D. Merry
933991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
934991554f2SKenneth D. Merry
935327f2e6cSStephen McConnell /*
936327f2e6cSStephen McConnell * After discovery has completed, check the mapping table for any
937327f2e6cSStephen McConnell * missing devices and update their missing counts. Only do this once
938327f2e6cSStephen McConnell * whenever the driver is initialized so that missing counts aren't
939327f2e6cSStephen McConnell * updated unnecessarily. Note that just because discovery has
940327f2e6cSStephen McConnell * completed doesn't mean that events have been processed yet. The
941327f2e6cSStephen McConnell * check_devices function is a callout timer that checks if ALL devices
942327f2e6cSStephen McConnell * are missing. If so, it will wait a little longer for events to
943327f2e6cSStephen McConnell * complete and keep resetting itself until some device in the mapping
944327f2e6cSStephen McConnell * table is not missing, meaning that event processing has started.
945327f2e6cSStephen McConnell */
946327f2e6cSStephen McConnell if (sc->track_mapping_events) {
947327f2e6cSStephen McConnell mpr_dprint(sc, MPR_XINFO | MPR_MAPPING, "Discovery has "
948327f2e6cSStephen McConnell "completed. Check for missing devices in the mapping "
949327f2e6cSStephen McConnell "table.\n");
950327f2e6cSStephen McConnell callout_reset(&sc->device_check_callout,
951327f2e6cSStephen McConnell MPR_MISSING_CHECK_DELAY * hz, mpr_mapping_check_devices,
952327f2e6cSStephen McConnell sc);
953327f2e6cSStephen McConnell }
954991554f2SKenneth D. Merry }
955991554f2SKenneth D. Merry
956991554f2SKenneth D. Merry static void
mprsas_action(struct cam_sim * sim,union ccb * ccb)957991554f2SKenneth D. Merry mprsas_action(struct cam_sim *sim, union ccb *ccb)
958991554f2SKenneth D. Merry {
959991554f2SKenneth D. Merry struct mprsas_softc *sassc;
960991554f2SKenneth D. Merry
961991554f2SKenneth D. Merry sassc = cam_sim_softc(sim);
962991554f2SKenneth D. Merry
963991554f2SKenneth D. Merry MPR_FUNCTRACE(sassc->sc);
964a2c14879SStephen McConnell mpr_dprint(sassc->sc, MPR_TRACE, "ccb func_code 0x%x\n",
965991554f2SKenneth D. Merry ccb->ccb_h.func_code);
966991554f2SKenneth D. Merry mtx_assert(&sassc->sc->mpr_mtx, MA_OWNED);
967991554f2SKenneth D. Merry
968991554f2SKenneth D. Merry switch (ccb->ccb_h.func_code) {
969991554f2SKenneth D. Merry case XPT_PATH_INQ:
970991554f2SKenneth D. Merry {
971991554f2SKenneth D. Merry struct ccb_pathinq *cpi = &ccb->cpi;
97232b0a21eSStephen McConnell struct mpr_softc *sc = sassc->sc;
973991554f2SKenneth D. Merry
974991554f2SKenneth D. Merry cpi->version_num = 1;
975991554f2SKenneth D. Merry cpi->hba_inquiry = PI_SDTR_ABLE|PI_TAG_ABLE|PI_WIDE_16;
976991554f2SKenneth D. Merry cpi->target_sprt = 0;
977991554f2SKenneth D. Merry cpi->hba_misc = PIM_NOBUSRESET | PIM_UNMAPPED | PIM_NOSCAN;
978991554f2SKenneth D. Merry cpi->hba_eng_cnt = 0;
979991554f2SKenneth D. Merry cpi->max_target = sassc->maxtargets - 1;
980991554f2SKenneth D. Merry cpi->max_lun = 255;
981327f2e6cSStephen McConnell
982327f2e6cSStephen McConnell /*
983327f2e6cSStephen McConnell * initiator_id is set here to an ID outside the set of valid
984327f2e6cSStephen McConnell * target IDs (including volumes).
985327f2e6cSStephen McConnell */
986327f2e6cSStephen McConnell cpi->initiator_id = sassc->maxtargets;
9874195c7deSAlan Somers strlcpy(cpi->sim_vid, "FreeBSD", SIM_IDLEN);
9884195c7deSAlan Somers strlcpy(cpi->hba_vid, "Avago Tech", HBA_IDLEN);
9894195c7deSAlan Somers strlcpy(cpi->dev_name, cam_sim_name(sim), DEV_IDLEN);
990991554f2SKenneth D. Merry cpi->unit_number = cam_sim_unit(sim);
991991554f2SKenneth D. Merry cpi->bus_id = cam_sim_bus(sim);
992991554f2SKenneth D. Merry /*
993991554f2SKenneth D. Merry * XXXSLM-I think this needs to change based on config page or
994991554f2SKenneth D. Merry * something instead of hardcoded to 150000.
995991554f2SKenneth D. Merry */
996991554f2SKenneth D. Merry cpi->base_transfer_speed = 150000;
997991554f2SKenneth D. Merry cpi->transport = XPORT_SAS;
998991554f2SKenneth D. Merry cpi->transport_version = 0;
999991554f2SKenneth D. Merry cpi->protocol = PROTO_SCSI;
1000991554f2SKenneth D. Merry cpi->protocol_version = SCSI_REV_SPC;
10014f5d6573SAlexander Motin cpi->maxio = sc->maxio;
1002a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
1003991554f2SKenneth D. Merry break;
1004991554f2SKenneth D. Merry }
1005991554f2SKenneth D. Merry case XPT_GET_TRAN_SETTINGS:
1006991554f2SKenneth D. Merry {
1007991554f2SKenneth D. Merry struct ccb_trans_settings *cts;
1008991554f2SKenneth D. Merry struct ccb_trans_settings_sas *sas;
1009991554f2SKenneth D. Merry struct ccb_trans_settings_scsi *scsi;
1010991554f2SKenneth D. Merry struct mprsas_target *targ;
1011991554f2SKenneth D. Merry
1012991554f2SKenneth D. Merry cts = &ccb->cts;
1013991554f2SKenneth D. Merry sas = &cts->xport_specific.sas;
1014991554f2SKenneth D. Merry scsi = &cts->proto_specific.scsi;
1015991554f2SKenneth D. Merry
1016991554f2SKenneth D. Merry KASSERT(cts->ccb_h.target_id < sassc->maxtargets,
1017991554f2SKenneth D. Merry ("Target %d out of bounds in XPT_GET_TRAN_SETTINGS\n",
1018991554f2SKenneth D. Merry cts->ccb_h.target_id));
1019991554f2SKenneth D. Merry targ = &sassc->targets[cts->ccb_h.target_id];
1020991554f2SKenneth D. Merry if (targ->handle == 0x0) {
1021a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
1022991554f2SKenneth D. Merry break;
1023991554f2SKenneth D. Merry }
1024991554f2SKenneth D. Merry
1025991554f2SKenneth D. Merry cts->protocol_version = SCSI_REV_SPC2;
1026991554f2SKenneth D. Merry cts->transport = XPORT_SAS;
1027991554f2SKenneth D. Merry cts->transport_version = 0;
1028991554f2SKenneth D. Merry
1029991554f2SKenneth D. Merry sas->valid = CTS_SAS_VALID_SPEED;
1030991554f2SKenneth D. Merry switch (targ->linkrate) {
1031991554f2SKenneth D. Merry case 0x08:
1032991554f2SKenneth D. Merry sas->bitrate = 150000;
1033991554f2SKenneth D. Merry break;
1034991554f2SKenneth D. Merry case 0x09:
1035991554f2SKenneth D. Merry sas->bitrate = 300000;
1036991554f2SKenneth D. Merry break;
1037991554f2SKenneth D. Merry case 0x0a:
1038991554f2SKenneth D. Merry sas->bitrate = 600000;
1039991554f2SKenneth D. Merry break;
104082315915SAlexander Motin case 0x0b:
104182315915SAlexander Motin sas->bitrate = 1200000;
104282315915SAlexander Motin break;
1043991554f2SKenneth D. Merry default:
1044991554f2SKenneth D. Merry sas->valid = 0;
1045991554f2SKenneth D. Merry }
1046991554f2SKenneth D. Merry
1047991554f2SKenneth D. Merry cts->protocol = PROTO_SCSI;
1048991554f2SKenneth D. Merry scsi->valid = CTS_SCSI_VALID_TQ;
1049991554f2SKenneth D. Merry scsi->flags = CTS_SCSI_FLAGS_TAG_ENB;
1050991554f2SKenneth D. Merry
1051a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
1052991554f2SKenneth D. Merry break;
1053991554f2SKenneth D. Merry }
1054991554f2SKenneth D. Merry case XPT_CALC_GEOMETRY:
1055991554f2SKenneth D. Merry cam_calc_geometry(&ccb->ccg, /*extended*/1);
1056a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
1057991554f2SKenneth D. Merry break;
1058991554f2SKenneth D. Merry case XPT_RESET_DEV:
10597a2a6a1aSStephen McConnell mpr_dprint(sassc->sc, MPR_XINFO, "mprsas_action "
10607a2a6a1aSStephen McConnell "XPT_RESET_DEV\n");
1061991554f2SKenneth D. Merry mprsas_action_resetdev(sassc, ccb);
1062991554f2SKenneth D. Merry return;
1063991554f2SKenneth D. Merry case XPT_RESET_BUS:
1064991554f2SKenneth D. Merry case XPT_ABORT:
1065991554f2SKenneth D. Merry case XPT_TERM_IO:
10667a2a6a1aSStephen McConnell mpr_dprint(sassc->sc, MPR_XINFO, "mprsas_action faking success "
10677a2a6a1aSStephen McConnell "for abort or reset\n");
1068a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
1069991554f2SKenneth D. Merry break;
1070991554f2SKenneth D. Merry case XPT_SCSI_IO:
1071991554f2SKenneth D. Merry mprsas_action_scsiio(sassc, ccb);
1072991554f2SKenneth D. Merry return;
1073991554f2SKenneth D. Merry case XPT_SMP_IO:
1074991554f2SKenneth D. Merry mprsas_action_smpio(sassc, ccb);
1075991554f2SKenneth D. Merry return;
1076991554f2SKenneth D. Merry default:
1077a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_FUNC_NOTAVAIL);
1078991554f2SKenneth D. Merry break;
1079991554f2SKenneth D. Merry }
1080991554f2SKenneth D. Merry xpt_done(ccb);
1081991554f2SKenneth D. Merry
1082991554f2SKenneth D. Merry }
1083991554f2SKenneth D. Merry
1084991554f2SKenneth D. Merry static void
mprsas_announce_reset(struct mpr_softc * sc,uint32_t ac_code,target_id_t target_id,lun_id_t lun_id)1085991554f2SKenneth D. Merry mprsas_announce_reset(struct mpr_softc *sc, uint32_t ac_code,
1086991554f2SKenneth D. Merry target_id_t target_id, lun_id_t lun_id)
1087991554f2SKenneth D. Merry {
1088991554f2SKenneth D. Merry path_id_t path_id = cam_sim_path(sc->sassc->sim);
1089991554f2SKenneth D. Merry struct cam_path *path;
1090991554f2SKenneth D. Merry
1091991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "%s code %x target %d lun %jx\n", __func__,
1092991554f2SKenneth D. Merry ac_code, target_id, (uintmax_t)lun_id);
1093991554f2SKenneth D. Merry
1094991554f2SKenneth D. Merry if (xpt_create_path(&path, NULL,
1095991554f2SKenneth D. Merry path_id, target_id, lun_id) != CAM_REQ_CMP) {
1096991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "unable to create path for reset "
1097991554f2SKenneth D. Merry "notification\n");
1098991554f2SKenneth D. Merry return;
1099991554f2SKenneth D. Merry }
1100991554f2SKenneth D. Merry
1101991554f2SKenneth D. Merry xpt_async(ac_code, path, NULL);
1102991554f2SKenneth D. Merry xpt_free_path(path);
1103991554f2SKenneth D. Merry }
1104991554f2SKenneth D. Merry
1105991554f2SKenneth D. Merry static void
mprsas_complete_all_commands(struct mpr_softc * sc)1106991554f2SKenneth D. Merry mprsas_complete_all_commands(struct mpr_softc *sc)
1107991554f2SKenneth D. Merry {
1108991554f2SKenneth D. Merry struct mpr_command *cm;
1109991554f2SKenneth D. Merry int i;
1110991554f2SKenneth D. Merry int completed;
1111991554f2SKenneth D. Merry
1112991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
1113991554f2SKenneth D. Merry mtx_assert(&sc->mpr_mtx, MA_OWNED);
1114991554f2SKenneth D. Merry
1115991554f2SKenneth D. Merry /* complete all commands with a NULL reply */
1116991554f2SKenneth D. Merry for (i = 1; i < sc->num_reqs; i++) {
1117991554f2SKenneth D. Merry cm = &sc->commands[i];
1118f0779b04SScott Long if (cm->cm_state == MPR_CM_STATE_FREE)
1119f0779b04SScott Long continue;
1120f0779b04SScott Long
1121f0779b04SScott Long cm->cm_state = MPR_CM_STATE_BUSY;
1122991554f2SKenneth D. Merry cm->cm_reply = NULL;
1123991554f2SKenneth D. Merry completed = 0;
1124991554f2SKenneth D. Merry
11258277ce2bSConrad Meyer if (cm->cm_flags & MPR_CM_FLAGS_SATA_ID_TIMEOUT) {
11268277ce2bSConrad Meyer MPASS(cm->cm_data);
11278277ce2bSConrad Meyer free(cm->cm_data, M_MPR);
11288277ce2bSConrad Meyer cm->cm_data = NULL;
11298277ce2bSConrad Meyer }
11308277ce2bSConrad Meyer
1131991554f2SKenneth D. Merry if (cm->cm_flags & MPR_CM_FLAGS_POLLED)
1132991554f2SKenneth D. Merry cm->cm_flags |= MPR_CM_FLAGS_COMPLETE;
1133991554f2SKenneth D. Merry
1134991554f2SKenneth D. Merry if (cm->cm_complete != NULL) {
1135991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
11367a2a6a1aSStephen McConnell "completing cm %p state %x ccb %p for diag reset\n",
11377a2a6a1aSStephen McConnell cm, cm->cm_state, cm->cm_ccb);
1138991554f2SKenneth D. Merry cm->cm_complete(sc, cm);
1139991554f2SKenneth D. Merry completed = 1;
1140f0779b04SScott Long } else if (cm->cm_flags & MPR_CM_FLAGS_WAKEUP) {
1141991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
1142991554f2SKenneth D. Merry "waking up cm %p state %x ccb %p for diag reset\n",
1143991554f2SKenneth D. Merry cm, cm->cm_state, cm->cm_ccb);
1144991554f2SKenneth D. Merry wakeup(cm);
1145991554f2SKenneth D. Merry completed = 1;
1146991554f2SKenneth D. Merry }
1147991554f2SKenneth D. Merry
1148991554f2SKenneth D. Merry if ((completed == 0) && (cm->cm_state != MPR_CM_STATE_FREE)) {
1149991554f2SKenneth D. Merry /* this should never happen, but if it does, log */
1150991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
1151991554f2SKenneth D. Merry "cm %p state %x flags 0x%x ccb %p during diag "
1152991554f2SKenneth D. Merry "reset\n", cm, cm->cm_state, cm->cm_flags,
1153991554f2SKenneth D. Merry cm->cm_ccb);
1154991554f2SKenneth D. Merry }
1155991554f2SKenneth D. Merry }
1156f0779b04SScott Long
1157f0779b04SScott Long sc->io_cmds_active = 0;
1158991554f2SKenneth D. Merry }
1159991554f2SKenneth D. Merry
1160991554f2SKenneth D. Merry void
mprsas_handle_reinit(struct mpr_softc * sc)1161991554f2SKenneth D. Merry mprsas_handle_reinit(struct mpr_softc *sc)
1162991554f2SKenneth D. Merry {
1163991554f2SKenneth D. Merry int i;
1164991554f2SKenneth D. Merry
1165991554f2SKenneth D. Merry /* Go back into startup mode and freeze the simq, so that CAM
1166991554f2SKenneth D. Merry * doesn't send any commands until after we've rediscovered all
1167991554f2SKenneth D. Merry * targets and found the proper device handles for them.
1168991554f2SKenneth D. Merry *
1169991554f2SKenneth D. Merry * After the reset, portenable will trigger discovery, and after all
1170991554f2SKenneth D. Merry * discovery-related activities have finished, the simq will be
1171991554f2SKenneth D. Merry * released.
1172991554f2SKenneth D. Merry */
1173991554f2SKenneth D. Merry mpr_dprint(sc, MPR_INIT, "%s startup\n", __func__);
1174991554f2SKenneth D. Merry sc->sassc->flags |= MPRSAS_IN_STARTUP;
1175991554f2SKenneth D. Merry sc->sassc->flags |= MPRSAS_IN_DISCOVERY;
1176991554f2SKenneth D. Merry mprsas_startup_increment(sc->sassc);
1177991554f2SKenneth D. Merry
1178991554f2SKenneth D. Merry /* notify CAM of a bus reset */
1179991554f2SKenneth D. Merry mprsas_announce_reset(sc, AC_BUS_RESET, CAM_TARGET_WILDCARD,
1180991554f2SKenneth D. Merry CAM_LUN_WILDCARD);
1181991554f2SKenneth D. Merry
1182991554f2SKenneth D. Merry /* complete and cleanup after all outstanding commands */
1183991554f2SKenneth D. Merry mprsas_complete_all_commands(sc);
1184991554f2SKenneth D. Merry
1185a2c14879SStephen McConnell mpr_dprint(sc, MPR_INIT, "%s startup %u after command completion\n",
1186a2c14879SStephen McConnell __func__, sc->sassc->startup_refcount);
1187991554f2SKenneth D. Merry
1188991554f2SKenneth D. Merry /* zero all the target handles, since they may change after the
1189991554f2SKenneth D. Merry * reset, and we have to rediscover all the targets and use the new
1190991554f2SKenneth D. Merry * handles.
1191991554f2SKenneth D. Merry */
1192991554f2SKenneth D. Merry for (i = 0; i < sc->sassc->maxtargets; i++) {
1193991554f2SKenneth D. Merry if (sc->sassc->targets[i].outstanding != 0)
1194991554f2SKenneth D. Merry mpr_dprint(sc, MPR_INIT, "target %u outstanding %u\n",
1195991554f2SKenneth D. Merry i, sc->sassc->targets[i].outstanding);
1196991554f2SKenneth D. Merry sc->sassc->targets[i].handle = 0x0;
1197991554f2SKenneth D. Merry sc->sassc->targets[i].exp_dev_handle = 0x0;
1198991554f2SKenneth D. Merry sc->sassc->targets[i].outstanding = 0;
1199991554f2SKenneth D. Merry sc->sassc->targets[i].flags = MPRSAS_TARGET_INDIAGRESET;
1200991554f2SKenneth D. Merry }
1201991554f2SKenneth D. Merry }
1202991554f2SKenneth D. Merry static void
mprsas_tm_timeout(void * data)1203991554f2SKenneth D. Merry mprsas_tm_timeout(void *data)
1204991554f2SKenneth D. Merry {
1205991554f2SKenneth D. Merry struct mpr_command *tm = data;
1206991554f2SKenneth D. Merry struct mpr_softc *sc = tm->cm_sc;
1207991554f2SKenneth D. Merry
1208991554f2SKenneth D. Merry mtx_assert(&sc->mpr_mtx, MA_OWNED);
1209991554f2SKenneth D. Merry
12107a2a6a1aSStephen McConnell mprsas_log_command(tm, MPR_INFO|MPR_RECOVERY, "task mgmt %p timed "
12117a2a6a1aSStephen McConnell "out\n", tm);
1212f0779b04SScott Long
1213f0779b04SScott Long KASSERT(tm->cm_state == MPR_CM_STATE_INQUEUE,
1214175ad3d0SKenneth D. Merry ("command not inqueue, state = %u\n", tm->cm_state));
1215f0779b04SScott Long
1216f0779b04SScott Long tm->cm_state = MPR_CM_STATE_BUSY;
1217991554f2SKenneth D. Merry mpr_reinit(sc);
1218991554f2SKenneth D. Merry }
1219991554f2SKenneth D. Merry
1220991554f2SKenneth D. Merry static void
mprsas_logical_unit_reset_complete(struct mpr_softc * sc,struct mpr_command * tm)12217a2a6a1aSStephen McConnell mprsas_logical_unit_reset_complete(struct mpr_softc *sc, struct mpr_command *tm)
1222991554f2SKenneth D. Merry {
1223991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REPLY *reply;
1224991554f2SKenneth D. Merry unsigned int cm_count = 0;
1225991554f2SKenneth D. Merry struct mpr_command *cm;
1226991554f2SKenneth D. Merry struct mprsas_target *targ;
1227991554f2SKenneth D. Merry
1228991554f2SKenneth D. Merry callout_stop(&tm->cm_callout);
1229991554f2SKenneth D. Merry
1230991554f2SKenneth D. Merry reply = (MPI2_SCSI_TASK_MANAGE_REPLY *)tm->cm_reply;
1231991554f2SKenneth D. Merry targ = tm->cm_targ;
1232991554f2SKenneth D. Merry
1233991554f2SKenneth D. Merry /*
1234991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
1235991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
1236991554f2SKenneth D. Merry * task management commands don't have S/G lists.
1237991554f2SKenneth D. Merry */
1238991554f2SKenneth D. Merry if ((tm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
12396eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_ERROR,
12406eea4f46SScott Long "%s: cm_flags = %#x for LUN reset! "
1241991554f2SKenneth D. Merry "This should not happen!\n", __func__, tm->cm_flags);
1242991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
1243991554f2SKenneth D. Merry return;
1244991554f2SKenneth D. Merry }
1245991554f2SKenneth D. Merry
1246991554f2SKenneth D. Merry if (reply == NULL) {
12476eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY, "NULL reset reply for tm %p\n",
12486eea4f46SScott Long tm);
1249991554f2SKenneth D. Merry if ((sc->mpr_flags & MPR_FLAGS_DIAGRESET) != 0) {
1250991554f2SKenneth D. Merry /* this completion was due to a reset, just cleanup */
12516eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY, "Hardware undergoing "
12526eea4f46SScott Long "reset, ignoring NULL LUN reset reply\n");
1253991554f2SKenneth D. Merry targ->tm = NULL;
1254991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
1255991554f2SKenneth D. Merry }
1256991554f2SKenneth D. Merry else {
1257991554f2SKenneth D. Merry /* we should have gotten a reply. */
12586eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY, "NULL reply on "
12596eea4f46SScott Long "LUN reset attempt, resetting controller\n");
1260991554f2SKenneth D. Merry mpr_reinit(sc);
1261991554f2SKenneth D. Merry }
1262991554f2SKenneth D. Merry return;
1263991554f2SKenneth D. Merry }
1264991554f2SKenneth D. Merry
12656eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY,
1266991554f2SKenneth D. Merry "logical unit reset status 0x%x code 0x%x count %u\n",
1267991554f2SKenneth D. Merry le16toh(reply->IOCStatus), le32toh(reply->ResponseCode),
1268991554f2SKenneth D. Merry le32toh(reply->TerminationCount));
1269991554f2SKenneth D. Merry
12706eea4f46SScott Long /*
12716eea4f46SScott Long * See if there are any outstanding commands for this LUN.
1272991554f2SKenneth D. Merry * This could be made more efficient by using a per-LU data
1273991554f2SKenneth D. Merry * structure of some sort.
1274991554f2SKenneth D. Merry */
1275991554f2SKenneth D. Merry TAILQ_FOREACH(cm, &targ->commands, cm_link) {
1276991554f2SKenneth D. Merry if (cm->cm_lun == tm->cm_lun)
1277991554f2SKenneth D. Merry cm_count++;
1278991554f2SKenneth D. Merry }
1279991554f2SKenneth D. Merry
1280991554f2SKenneth D. Merry if (cm_count == 0) {
12816eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_INFO,
12826eea4f46SScott Long "Finished recovery after LUN reset for target %u\n",
12836eea4f46SScott Long targ->tid);
1284991554f2SKenneth D. Merry
12856eea4f46SScott Long mprsas_announce_reset(sc, AC_SENT_BDR, targ->tid,
1286991554f2SKenneth D. Merry tm->cm_lun);
1287991554f2SKenneth D. Merry
12886eea4f46SScott Long /*
12896eea4f46SScott Long * We've finished recovery for this logical unit. check and
1290991554f2SKenneth D. Merry * see if some other logical unit has a timedout command
1291991554f2SKenneth D. Merry * that needs to be processed.
1292991554f2SKenneth D. Merry */
1293991554f2SKenneth D. Merry cm = TAILQ_FIRST(&targ->timedout_commands);
1294991554f2SKenneth D. Merry if (cm) {
12956eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
12966eea4f46SScott Long "More commands to abort for target %u\n", targ->tid);
1297991554f2SKenneth D. Merry mprsas_send_abort(sc, tm, cm);
12986eea4f46SScott Long } else {
1299991554f2SKenneth D. Merry targ->tm = NULL;
1300991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
1301991554f2SKenneth D. Merry }
13026eea4f46SScott Long } else {
1303991554f2SKenneth D. Merry /* if we still have commands for this LUN, the reset
1304991554f2SKenneth D. Merry * effectively failed, regardless of the status reported.
1305991554f2SKenneth D. Merry * Escalate to a target reset.
1306991554f2SKenneth D. Merry */
13076eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
13086eea4f46SScott Long "logical unit reset complete for target %u, but still "
13096eea4f46SScott Long "have %u command(s), sending target reset\n", targ->tid,
13106eea4f46SScott Long cm_count);
131189d1c21fSKashyap D Desai if (!targ->is_nvme || sc->custom_nvme_tm_handling)
1312991554f2SKenneth D. Merry mprsas_send_reset(sc, tm,
1313991554f2SKenneth D. Merry MPI2_SCSITASKMGMT_TASKTYPE_TARGET_RESET);
131489d1c21fSKashyap D Desai else
131589d1c21fSKashyap D Desai mpr_reinit(sc);
1316991554f2SKenneth D. Merry }
1317991554f2SKenneth D. Merry }
1318991554f2SKenneth D. Merry
1319991554f2SKenneth D. Merry static void
mprsas_target_reset_complete(struct mpr_softc * sc,struct mpr_command * tm)1320991554f2SKenneth D. Merry mprsas_target_reset_complete(struct mpr_softc *sc, struct mpr_command *tm)
1321991554f2SKenneth D. Merry {
1322991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REPLY *reply;
1323991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
1324991554f2SKenneth D. Merry struct mprsas_target *targ;
1325991554f2SKenneth D. Merry
1326991554f2SKenneth D. Merry callout_stop(&tm->cm_callout);
1327991554f2SKenneth D. Merry
1328991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
1329991554f2SKenneth D. Merry reply = (MPI2_SCSI_TASK_MANAGE_REPLY *)tm->cm_reply;
1330991554f2SKenneth D. Merry targ = tm->cm_targ;
1331991554f2SKenneth D. Merry
1332991554f2SKenneth D. Merry /*
1333991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
1334991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
1335991554f2SKenneth D. Merry * task management commands don't have S/G lists.
1336991554f2SKenneth D. Merry */
1337991554f2SKenneth D. Merry if ((tm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
1338a2c14879SStephen McConnell mpr_dprint(sc, MPR_ERROR, "%s: cm_flags = %#x for target "
1339a2c14879SStephen McConnell "reset! This should not happen!\n", __func__, tm->cm_flags);
1340991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
1341991554f2SKenneth D. Merry return;
1342991554f2SKenneth D. Merry }
1343991554f2SKenneth D. Merry
1344991554f2SKenneth D. Merry if (reply == NULL) {
13456eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY,
13466eea4f46SScott Long "NULL target reset reply for tm %p TaskMID %u\n",
13476eea4f46SScott Long tm, le16toh(req->TaskMID));
1348991554f2SKenneth D. Merry if ((sc->mpr_flags & MPR_FLAGS_DIAGRESET) != 0) {
1349991554f2SKenneth D. Merry /* this completion was due to a reset, just cleanup */
13506eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY, "Hardware undergoing "
13516eea4f46SScott Long "reset, ignoring NULL target reset reply\n");
1352991554f2SKenneth D. Merry targ->tm = NULL;
1353991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
1354991554f2SKenneth D. Merry }
1355991554f2SKenneth D. Merry else {
1356991554f2SKenneth D. Merry /* we should have gotten a reply. */
13576eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY, "NULL reply on "
13586eea4f46SScott Long "target reset attempt, resetting controller\n");
1359991554f2SKenneth D. Merry mpr_reinit(sc);
1360991554f2SKenneth D. Merry }
1361991554f2SKenneth D. Merry return;
1362991554f2SKenneth D. Merry }
1363991554f2SKenneth D. Merry
13646eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY,
1365991554f2SKenneth D. Merry "target reset status 0x%x code 0x%x count %u\n",
1366991554f2SKenneth D. Merry le16toh(reply->IOCStatus), le32toh(reply->ResponseCode),
1367991554f2SKenneth D. Merry le32toh(reply->TerminationCount));
1368991554f2SKenneth D. Merry
1369991554f2SKenneth D. Merry if (targ->outstanding == 0) {
13706eea4f46SScott Long /*
13716eea4f46SScott Long * We've finished recovery for this target and all
1372991554f2SKenneth D. Merry * of its logical units.
1373991554f2SKenneth D. Merry */
13746eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_INFO,
13756eea4f46SScott Long "Finished reset recovery for target %u\n", targ->tid);
1376991554f2SKenneth D. Merry
1377991554f2SKenneth D. Merry mprsas_announce_reset(sc, AC_SENT_BDR, tm->cm_targ->tid,
1378991554f2SKenneth D. Merry CAM_LUN_WILDCARD);
1379991554f2SKenneth D. Merry
1380991554f2SKenneth D. Merry targ->tm = NULL;
1381991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
13826eea4f46SScott Long } else {
13836eea4f46SScott Long /*
13846eea4f46SScott Long * After a target reset, if this target still has
1385991554f2SKenneth D. Merry * outstanding commands, the reset effectively failed,
1386991554f2SKenneth D. Merry * regardless of the status reported. escalate.
1387991554f2SKenneth D. Merry */
13886eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
13896eea4f46SScott Long "Target reset complete for target %u, but still have %u "
13906eea4f46SScott Long "command(s), resetting controller\n", targ->tid,
13916eea4f46SScott Long targ->outstanding);
1392991554f2SKenneth D. Merry mpr_reinit(sc);
1393991554f2SKenneth D. Merry }
1394991554f2SKenneth D. Merry }
1395991554f2SKenneth D. Merry
1396991554f2SKenneth D. Merry #define MPR_RESET_TIMEOUT 30
1397991554f2SKenneth D. Merry
1398a2c14879SStephen McConnell int
mprsas_send_reset(struct mpr_softc * sc,struct mpr_command * tm,uint8_t type)1399991554f2SKenneth D. Merry mprsas_send_reset(struct mpr_softc *sc, struct mpr_command *tm, uint8_t type)
1400991554f2SKenneth D. Merry {
1401991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
1402991554f2SKenneth D. Merry struct mprsas_target *target;
140389d1c21fSKashyap D Desai int err, timeout;
1404991554f2SKenneth D. Merry
1405991554f2SKenneth D. Merry target = tm->cm_targ;
1406991554f2SKenneth D. Merry if (target->handle == 0) {
1407a2c14879SStephen McConnell mpr_dprint(sc, MPR_ERROR, "%s null devhandle for target_id "
1408a2c14879SStephen McConnell "%d\n", __func__, target->tid);
1409991554f2SKenneth D. Merry return -1;
1410991554f2SKenneth D. Merry }
1411991554f2SKenneth D. Merry
1412991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
1413991554f2SKenneth D. Merry req->DevHandle = htole16(target->handle);
1414991554f2SKenneth D. Merry req->TaskType = type;
1415991554f2SKenneth D. Merry
141689d1c21fSKashyap D Desai if (!target->is_nvme || sc->custom_nvme_tm_handling) {
141789d1c21fSKashyap D Desai timeout = MPR_RESET_TIMEOUT;
141889d1c21fSKashyap D Desai /*
141989d1c21fSKashyap D Desai * Target reset method =
142089d1c21fSKashyap D Desai * SAS Hard Link Reset / SATA Link Reset
142189d1c21fSKashyap D Desai */
142289d1c21fSKashyap D Desai req->MsgFlags = MPI2_SCSITASKMGMT_MSGFLAGS_LINK_RESET;
142389d1c21fSKashyap D Desai } else {
142489d1c21fSKashyap D Desai timeout = (target->controller_reset_timeout) ? (
142589d1c21fSKashyap D Desai target->controller_reset_timeout) : (MPR_RESET_TIMEOUT);
142689d1c21fSKashyap D Desai /* PCIe Protocol Level Reset*/
142789d1c21fSKashyap D Desai req->MsgFlags =
142889d1c21fSKashyap D Desai MPI26_SCSITASKMGMT_MSGFLAGS_PROTOCOL_LVL_RST_PCIE;
142989d1c21fSKashyap D Desai }
143089d1c21fSKashyap D Desai
1431991554f2SKenneth D. Merry if (type == MPI2_SCSITASKMGMT_TASKTYPE_LOGICAL_UNIT_RESET) {
1432991554f2SKenneth D. Merry /* XXX Need to handle invalid LUNs */
1433991554f2SKenneth D. Merry MPR_SET_LUN(req->LUN, tm->cm_lun);
1434991554f2SKenneth D. Merry tm->cm_targ->logical_unit_resets++;
14356eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_INFO,
14366eea4f46SScott Long "Sending logical unit reset to target %u lun %d\n",
14376eea4f46SScott Long target->tid, tm->cm_lun);
1438991554f2SKenneth D. Merry tm->cm_complete = mprsas_logical_unit_reset_complete;
1439a2c14879SStephen McConnell mprsas_prepare_for_tm(sc, tm, target, tm->cm_lun);
14406eea4f46SScott Long } else if (type == MPI2_SCSITASKMGMT_TASKTYPE_TARGET_RESET) {
1441991554f2SKenneth D. Merry tm->cm_targ->target_resets++;
14426eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_INFO,
14436eea4f46SScott Long "Sending target reset to target %u\n", target->tid);
1444991554f2SKenneth D. Merry tm->cm_complete = mprsas_target_reset_complete;
1445a2c14879SStephen McConnell mprsas_prepare_for_tm(sc, tm, target, CAM_LUN_WILDCARD);
1446991554f2SKenneth D. Merry }
1447991554f2SKenneth D. Merry else {
1448991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "unexpected reset type 0x%x\n", type);
1449991554f2SKenneth D. Merry return -1;
1450991554f2SKenneth D. Merry }
1451991554f2SKenneth D. Merry
1452991554f2SKenneth D. Merry if (target->encl_level_valid) {
14536eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_INFO,
14546eea4f46SScott Long "At enclosure level %d, slot %d, connector name (%4s)\n",
14556eea4f46SScott Long target->encl_level, target->encl_slot,
14566eea4f46SScott Long target->connector_name);
1457991554f2SKenneth D. Merry }
1458991554f2SKenneth D. Merry
1459991554f2SKenneth D. Merry tm->cm_data = NULL;
1460991554f2SKenneth D. Merry tm->cm_complete_data = (void *)tm;
1461991554f2SKenneth D. Merry
146289d1c21fSKashyap D Desai callout_reset(&tm->cm_callout, timeout * hz,
1463991554f2SKenneth D. Merry mprsas_tm_timeout, tm);
1464991554f2SKenneth D. Merry
1465991554f2SKenneth D. Merry err = mpr_map_command(sc, tm);
1466991554f2SKenneth D. Merry if (err)
14676eea4f46SScott Long mpr_dprint(sc, MPR_ERROR|MPR_RECOVERY,
1468a2c14879SStephen McConnell "error %d sending reset type %u\n", err, type);
1469991554f2SKenneth D. Merry
1470991554f2SKenneth D. Merry return err;
1471991554f2SKenneth D. Merry }
1472991554f2SKenneth D. Merry
1473991554f2SKenneth D. Merry static void
mprsas_abort_complete(struct mpr_softc * sc,struct mpr_command * tm)1474991554f2SKenneth D. Merry mprsas_abort_complete(struct mpr_softc *sc, struct mpr_command *tm)
1475991554f2SKenneth D. Merry {
1476991554f2SKenneth D. Merry struct mpr_command *cm;
1477991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REPLY *reply;
1478991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
1479991554f2SKenneth D. Merry struct mprsas_target *targ;
1480991554f2SKenneth D. Merry
1481991554f2SKenneth D. Merry callout_stop(&tm->cm_callout);
1482991554f2SKenneth D. Merry
1483991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
1484991554f2SKenneth D. Merry reply = (MPI2_SCSI_TASK_MANAGE_REPLY *)tm->cm_reply;
1485991554f2SKenneth D. Merry targ = tm->cm_targ;
1486991554f2SKenneth D. Merry
1487991554f2SKenneth D. Merry /*
1488991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
1489991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
1490991554f2SKenneth D. Merry * task management commands don't have S/G lists.
1491991554f2SKenneth D. Merry */
1492991554f2SKenneth D. Merry if ((tm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
14936eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_ERROR,
1494991554f2SKenneth D. Merry "cm_flags = %#x for abort %p TaskMID %u!\n",
1495991554f2SKenneth D. Merry tm->cm_flags, tm, le16toh(req->TaskMID));
1496991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
1497991554f2SKenneth D. Merry return;
1498991554f2SKenneth D. Merry }
1499991554f2SKenneth D. Merry
1500991554f2SKenneth D. Merry if (reply == NULL) {
15016eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY,
1502991554f2SKenneth D. Merry "NULL abort reply for tm %p TaskMID %u\n",
1503991554f2SKenneth D. Merry tm, le16toh(req->TaskMID));
1504991554f2SKenneth D. Merry if ((sc->mpr_flags & MPR_FLAGS_DIAGRESET) != 0) {
1505991554f2SKenneth D. Merry /* this completion was due to a reset, just cleanup */
15066eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY, "Hardware undergoing "
15076eea4f46SScott Long "reset, ignoring NULL abort reply\n");
1508991554f2SKenneth D. Merry targ->tm = NULL;
1509991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
15106eea4f46SScott Long } else {
1511991554f2SKenneth D. Merry /* we should have gotten a reply. */
15126eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY, "NULL reply on "
15136eea4f46SScott Long "abort attempt, resetting controller\n");
1514991554f2SKenneth D. Merry mpr_reinit(sc);
1515991554f2SKenneth D. Merry }
1516991554f2SKenneth D. Merry return;
1517991554f2SKenneth D. Merry }
1518991554f2SKenneth D. Merry
15196eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY,
1520991554f2SKenneth D. Merry "abort TaskMID %u status 0x%x code 0x%x count %u\n",
1521991554f2SKenneth D. Merry le16toh(req->TaskMID),
1522991554f2SKenneth D. Merry le16toh(reply->IOCStatus), le32toh(reply->ResponseCode),
1523991554f2SKenneth D. Merry le32toh(reply->TerminationCount));
1524991554f2SKenneth D. Merry
1525991554f2SKenneth D. Merry cm = TAILQ_FIRST(&tm->cm_targ->timedout_commands);
1526991554f2SKenneth D. Merry if (cm == NULL) {
15276eea4f46SScott Long /*
15286eea4f46SScott Long * if there are no more timedout commands, we're done with
1529991554f2SKenneth D. Merry * error recovery for this target.
1530991554f2SKenneth D. Merry */
15316eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
15326eea4f46SScott Long "Finished abort recovery for target %u\n", targ->tid);
1533991554f2SKenneth D. Merry targ->tm = NULL;
1534991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
15356eea4f46SScott Long } else if (le16toh(req->TaskMID) != cm->cm_desc.Default.SMID) {
1536991554f2SKenneth D. Merry /* abort success, but we have more timedout commands to abort */
15376eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
15386eea4f46SScott Long "Continuing abort recovery for target %u\n", targ->tid);
1539991554f2SKenneth D. Merry mprsas_send_abort(sc, tm, cm);
15406eea4f46SScott Long } else {
15416eea4f46SScott Long /*
15426eea4f46SScott Long * we didn't get a command completion, so the abort
1543991554f2SKenneth D. Merry * failed as far as we're concerned. escalate.
1544991554f2SKenneth D. Merry */
15456eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
15466eea4f46SScott Long "Abort failed for target %u, sending logical unit reset\n",
15476eea4f46SScott Long targ->tid);
1548991554f2SKenneth D. Merry
1549991554f2SKenneth D. Merry mprsas_send_reset(sc, tm,
1550991554f2SKenneth D. Merry MPI2_SCSITASKMGMT_TASKTYPE_LOGICAL_UNIT_RESET);
1551991554f2SKenneth D. Merry }
1552991554f2SKenneth D. Merry }
1553991554f2SKenneth D. Merry
1554991554f2SKenneth D. Merry #define MPR_ABORT_TIMEOUT 5
1555991554f2SKenneth D. Merry
1556991554f2SKenneth D. Merry static int
mprsas_send_abort(struct mpr_softc * sc,struct mpr_command * tm,struct mpr_command * cm)1557991554f2SKenneth D. Merry mprsas_send_abort(struct mpr_softc *sc, struct mpr_command *tm,
1558991554f2SKenneth D. Merry struct mpr_command *cm)
1559991554f2SKenneth D. Merry {
1560991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
1561991554f2SKenneth D. Merry struct mprsas_target *targ;
156289d1c21fSKashyap D Desai int err, timeout;
1563991554f2SKenneth D. Merry
1564991554f2SKenneth D. Merry targ = cm->cm_targ;
1565991554f2SKenneth D. Merry if (targ->handle == 0) {
15666eea4f46SScott Long mpr_dprint(sc, MPR_ERROR|MPR_RECOVERY,
15676eea4f46SScott Long "%s null devhandle for target_id %d\n",
1568991554f2SKenneth D. Merry __func__, cm->cm_ccb->ccb_h.target_id);
1569991554f2SKenneth D. Merry return -1;
1570991554f2SKenneth D. Merry }
1571991554f2SKenneth D. Merry
1572855fe445SScott Long mprsas_log_command(cm, MPR_RECOVERY|MPR_INFO,
1573991554f2SKenneth D. Merry "Aborting command %p\n", cm);
1574991554f2SKenneth D. Merry
1575991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
1576991554f2SKenneth D. Merry req->DevHandle = htole16(targ->handle);
1577991554f2SKenneth D. Merry req->TaskType = MPI2_SCSITASKMGMT_TASKTYPE_ABORT_TASK;
1578991554f2SKenneth D. Merry
1579991554f2SKenneth D. Merry /* XXX Need to handle invalid LUNs */
1580991554f2SKenneth D. Merry MPR_SET_LUN(req->LUN, cm->cm_ccb->ccb_h.target_lun);
1581991554f2SKenneth D. Merry
1582991554f2SKenneth D. Merry req->TaskMID = htole16(cm->cm_desc.Default.SMID);
1583991554f2SKenneth D. Merry
1584991554f2SKenneth D. Merry tm->cm_data = NULL;
1585991554f2SKenneth D. Merry tm->cm_complete = mprsas_abort_complete;
1586991554f2SKenneth D. Merry tm->cm_complete_data = (void *)tm;
1587991554f2SKenneth D. Merry tm->cm_targ = cm->cm_targ;
1588991554f2SKenneth D. Merry tm->cm_lun = cm->cm_lun;
1589991554f2SKenneth D. Merry
159089d1c21fSKashyap D Desai if (!targ->is_nvme || sc->custom_nvme_tm_handling)
159189d1c21fSKashyap D Desai timeout = MPR_ABORT_TIMEOUT;
159289d1c21fSKashyap D Desai else
159389d1c21fSKashyap D Desai timeout = sc->nvme_abort_timeout;
159489d1c21fSKashyap D Desai
159589d1c21fSKashyap D Desai callout_reset(&tm->cm_callout, timeout * hz,
1596991554f2SKenneth D. Merry mprsas_tm_timeout, tm);
1597991554f2SKenneth D. Merry
1598991554f2SKenneth D. Merry targ->aborts++;
1599991554f2SKenneth D. Merry
1600a2c14879SStephen McConnell mprsas_prepare_for_tm(sc, tm, targ, tm->cm_lun);
1601a2c14879SStephen McConnell
1602991554f2SKenneth D. Merry err = mpr_map_command(sc, tm);
1603991554f2SKenneth D. Merry if (err)
16046eea4f46SScott Long mpr_dprint(sc, MPR_ERROR|MPR_RECOVERY,
1605991554f2SKenneth D. Merry "error %d sending abort for cm %p SMID %u\n",
1606991554f2SKenneth D. Merry err, cm, req->TaskMID);
1607991554f2SKenneth D. Merry return err;
1608991554f2SKenneth D. Merry }
1609991554f2SKenneth D. Merry
1610991554f2SKenneth D. Merry static void
mprsas_scsiio_timeout(void * data)1611991554f2SKenneth D. Merry mprsas_scsiio_timeout(void *data)
1612991554f2SKenneth D. Merry {
16136eea4f46SScott Long sbintime_t elapsed, now;
16146eea4f46SScott Long union ccb *ccb;
1615991554f2SKenneth D. Merry struct mpr_softc *sc;
1616991554f2SKenneth D. Merry struct mpr_command *cm;
1617991554f2SKenneth D. Merry struct mprsas_target *targ;
1618991554f2SKenneth D. Merry
1619991554f2SKenneth D. Merry cm = (struct mpr_command *)data;
1620991554f2SKenneth D. Merry sc = cm->cm_sc;
16216eea4f46SScott Long ccb = cm->cm_ccb;
16226eea4f46SScott Long now = sbinuptime();
1623991554f2SKenneth D. Merry
1624991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
1625991554f2SKenneth D. Merry mtx_assert(&sc->mpr_mtx, MA_OWNED);
1626991554f2SKenneth D. Merry
16276eea4f46SScott Long mpr_dprint(sc, MPR_XINFO|MPR_RECOVERY, "Timeout checking cm %p\n", cm);
1628991554f2SKenneth D. Merry
1629991554f2SKenneth D. Merry /*
1630991554f2SKenneth D. Merry * Run the interrupt handler to make sure it's not pending. This
1631991554f2SKenneth D. Merry * isn't perfect because the command could have already completed
1632991554f2SKenneth D. Merry * and been re-used, though this is unlikely.
1633991554f2SKenneth D. Merry */
1634991554f2SKenneth D. Merry mpr_intr_locked(sc);
16358fe7bf06SWarner Losh if (cm->cm_flags & MPR_CM_FLAGS_ON_RECOVERY) {
1636991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_XINFO,
1637991554f2SKenneth D. Merry "SCSI command %p almost timed out\n", cm);
1638991554f2SKenneth D. Merry return;
1639991554f2SKenneth D. Merry }
1640991554f2SKenneth D. Merry
1641991554f2SKenneth D. Merry if (cm->cm_ccb == NULL) {
1642991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "command timeout with NULL ccb\n");
1643991554f2SKenneth D. Merry return;
1644991554f2SKenneth D. Merry }
1645991554f2SKenneth D. Merry
1646991554f2SKenneth D. Merry targ = cm->cm_targ;
1647991554f2SKenneth D. Merry targ->timeouts++;
1648991554f2SKenneth D. Merry
16496eea4f46SScott Long elapsed = now - ccb->ccb_h.qos.sim_data;
16506eea4f46SScott Long mprsas_log_command(cm, MPR_INFO|MPR_RECOVERY,
16516eea4f46SScott Long "Command timeout on target %u(0x%04x), %d set, %d.%d elapsed\n",
16526eea4f46SScott Long targ->tid, targ->handle, ccb->ccb_h.timeout,
16536eea4f46SScott Long sbintime_getsec(elapsed), elapsed & 0xffffffff);
1654991554f2SKenneth D. Merry if (targ->encl_level_valid) {
16556eea4f46SScott Long mpr_dprint(sc, MPR_INFO|MPR_RECOVERY,
16566eea4f46SScott Long "At enclosure level %d, slot %d, connector name (%4s)\n",
16576eea4f46SScott Long targ->encl_level, targ->encl_slot, targ->connector_name);
1658991554f2SKenneth D. Merry }
1659991554f2SKenneth D. Merry
1660991554f2SKenneth D. Merry /* XXX first, check the firmware state, to see if it's still
1661991554f2SKenneth D. Merry * operational. if not, do a diag reset.
1662991554f2SKenneth D. Merry */
1663a2c14879SStephen McConnell mprsas_set_ccbstatus(cm->cm_ccb, CAM_CMD_TIMEOUT);
16648fe7bf06SWarner Losh cm->cm_flags |= MPR_CM_FLAGS_ON_RECOVERY | MPR_CM_FLAGS_TIMEDOUT;
1665991554f2SKenneth D. Merry TAILQ_INSERT_TAIL(&targ->timedout_commands, cm, cm_recovery);
1666991554f2SKenneth D. Merry
1667991554f2SKenneth D. Merry if (targ->tm != NULL) {
1668991554f2SKenneth D. Merry /* target already in recovery, just queue up another
1669991554f2SKenneth D. Merry * timedout command to be processed later.
1670991554f2SKenneth D. Merry */
16712bbaed4dSWarner Losh mpr_dprint(sc, MPR_RECOVERY,
16722bbaed4dSWarner Losh "queued timedout cm %p for processing by tm %p\n",
16732bbaed4dSWarner Losh cm, targ->tm);
16742bbaed4dSWarner Losh } else if ((targ->tm = mprsas_alloc_tm(sc)) != NULL) {
16756eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY|MPR_INFO,
16766eea4f46SScott Long "Sending abort to target %u for SMID %d\n", targ->tid,
16776eea4f46SScott Long cm->cm_desc.Default.SMID);
16786eea4f46SScott Long mpr_dprint(sc, MPR_RECOVERY, "timedout cm %p allocated tm %p\n",
16796eea4f46SScott Long cm, targ->tm);
16802bbaed4dSWarner Losh
16812bbaed4dSWarner Losh /* start recovery by aborting the first timedout command */
1682991554f2SKenneth D. Merry mprsas_send_abort(sc, targ->tm, cm);
16832bbaed4dSWarner Losh } else {
1684991554f2SKenneth D. Merry /* XXX queue this target up for recovery once a TM becomes
1685991554f2SKenneth D. Merry * available. The firmware only has a limited number of
1686991554f2SKenneth D. Merry * HighPriority credits for the high priority requests used
1687991554f2SKenneth D. Merry * for task management, and we ran out.
1688991554f2SKenneth D. Merry *
1689991554f2SKenneth D. Merry * Isilon: don't worry about this for now, since we have
1690991554f2SKenneth D. Merry * more credits than disks in an enclosure, and limit
1691991554f2SKenneth D. Merry * ourselves to one TM per target for recovery.
1692991554f2SKenneth D. Merry */
16936eea4f46SScott Long mpr_dprint(sc, MPR_ERROR|MPR_RECOVERY,
16946eea4f46SScott Long "timedout cm %p failed to allocate a tm\n", cm);
1695991554f2SKenneth D. Merry }
1696991554f2SKenneth D. Merry }
1697991554f2SKenneth D. Merry
169867feec50SStephen McConnell /**
169967feec50SStephen McConnell * mprsas_build_nvme_unmap - Build Native NVMe DSM command equivalent
170067feec50SStephen McConnell * to SCSI Unmap.
170167feec50SStephen McConnell * Return 0 - for success,
170267feec50SStephen McConnell * 1 - to immediately return back the command with success status to CAM
170367feec50SStephen McConnell * negative value - to fallback to firmware path i.e. issue scsi unmap
170467feec50SStephen McConnell * to FW without any translation.
170567feec50SStephen McConnell */
170667feec50SStephen McConnell static int
mprsas_build_nvme_unmap(struct mpr_softc * sc,struct mpr_command * cm,union ccb * ccb,struct mprsas_target * targ)170767feec50SStephen McConnell mprsas_build_nvme_unmap(struct mpr_softc *sc, struct mpr_command *cm,
170867feec50SStephen McConnell union ccb *ccb, struct mprsas_target *targ)
170967feec50SStephen McConnell {
171067feec50SStephen McConnell Mpi26NVMeEncapsulatedRequest_t *req = NULL;
171167feec50SStephen McConnell struct ccb_scsiio *csio;
171267feec50SStephen McConnell struct unmap_parm_list *plist;
171367feec50SStephen McConnell struct nvme_dsm_range *nvme_dsm_ranges = NULL;
171467feec50SStephen McConnell struct nvme_command *c;
171567feec50SStephen McConnell int i, res;
171667feec50SStephen McConnell uint16_t ndesc, list_len, data_length;
171767feec50SStephen McConnell struct mpr_prp_page *prp_page_info;
171867feec50SStephen McConnell uint64_t nvme_dsm_ranges_dma_handle;
171967feec50SStephen McConnell
172067feec50SStephen McConnell csio = &ccb->csio;
172167feec50SStephen McConnell list_len = (scsiio_cdb_ptr(csio)[7] << 8 | scsiio_cdb_ptr(csio)[8]);
172267feec50SStephen McConnell if (!list_len) {
172367feec50SStephen McConnell mpr_dprint(sc, MPR_ERROR, "Parameter list length is Zero\n");
172467feec50SStephen McConnell return -EINVAL;
172567feec50SStephen McConnell }
172667feec50SStephen McConnell
172767feec50SStephen McConnell plist = malloc(csio->dxfer_len, M_MPR, M_ZERO|M_NOWAIT);
172867feec50SStephen McConnell if (!plist) {
172967feec50SStephen McConnell mpr_dprint(sc, MPR_ERROR, "Unable to allocate memory to "
173067feec50SStephen McConnell "save UNMAP data\n");
173167feec50SStephen McConnell return -ENOMEM;
173267feec50SStephen McConnell }
173367feec50SStephen McConnell
173467feec50SStephen McConnell /* Copy SCSI unmap data to a local buffer */
173567feec50SStephen McConnell bcopy(csio->data_ptr, plist, csio->dxfer_len);
173667feec50SStephen McConnell
173767feec50SStephen McConnell /* return back the unmap command to CAM with success status,
173867feec50SStephen McConnell * if number of descripts is zero.
173967feec50SStephen McConnell */
174067feec50SStephen McConnell ndesc = be16toh(plist->unmap_blk_desc_data_len) >> 4;
174167feec50SStephen McConnell if (!ndesc) {
174267feec50SStephen McConnell mpr_dprint(sc, MPR_XINFO, "Number of descriptors in "
174367feec50SStephen McConnell "UNMAP cmd is Zero\n");
174467feec50SStephen McConnell res = 1;
174567feec50SStephen McConnell goto out;
174667feec50SStephen McConnell }
174767feec50SStephen McConnell
174867feec50SStephen McConnell data_length = ndesc * sizeof(struct nvme_dsm_range);
174967feec50SStephen McConnell if (data_length > targ->MDTS) {
175067feec50SStephen McConnell mpr_dprint(sc, MPR_ERROR, "data length: %d is greater than "
175167feec50SStephen McConnell "Device's MDTS: %d\n", data_length, targ->MDTS);
175267feec50SStephen McConnell res = -EINVAL;
175367feec50SStephen McConnell goto out;
175467feec50SStephen McConnell }
175567feec50SStephen McConnell
175667feec50SStephen McConnell prp_page_info = mpr_alloc_prp_page(sc);
175767feec50SStephen McConnell KASSERT(prp_page_info != NULL, ("%s: There is no PRP Page for "
175867feec50SStephen McConnell "UNMAP command.\n", __func__));
175967feec50SStephen McConnell
176067feec50SStephen McConnell /*
176167feec50SStephen McConnell * Insert the allocated PRP page into the command's PRP page list. This
176267feec50SStephen McConnell * will be freed when the command is freed.
176367feec50SStephen McConnell */
176467feec50SStephen McConnell TAILQ_INSERT_TAIL(&cm->cm_prp_page_list, prp_page_info, prp_page_link);
176567feec50SStephen McConnell
176667feec50SStephen McConnell nvme_dsm_ranges = (struct nvme_dsm_range *)prp_page_info->prp_page;
176767feec50SStephen McConnell nvme_dsm_ranges_dma_handle = prp_page_info->prp_page_busaddr;
176867feec50SStephen McConnell
176967feec50SStephen McConnell bzero(nvme_dsm_ranges, data_length);
177067feec50SStephen McConnell
177167feec50SStephen McConnell /* Convert SCSI unmap's descriptor data to NVMe DSM specific Range data
177267feec50SStephen McConnell * for each descriptors contained in SCSI UNMAP data.
177367feec50SStephen McConnell */
177467feec50SStephen McConnell for (i = 0; i < ndesc; i++) {
177567feec50SStephen McConnell nvme_dsm_ranges[i].length =
177667feec50SStephen McConnell htole32(be32toh(plist->desc[i].nlb));
177767feec50SStephen McConnell nvme_dsm_ranges[i].starting_lba =
177867feec50SStephen McConnell htole64(be64toh(plist->desc[i].slba));
177967feec50SStephen McConnell nvme_dsm_ranges[i].attributes = 0;
178067feec50SStephen McConnell }
178167feec50SStephen McConnell
178267feec50SStephen McConnell /* Build MPI2.6's NVMe Encapsulated Request Message */
178367feec50SStephen McConnell req = (Mpi26NVMeEncapsulatedRequest_t *)cm->cm_req;
178467feec50SStephen McConnell bzero(req, sizeof(*req));
178567feec50SStephen McConnell req->DevHandle = htole16(targ->handle);
178667feec50SStephen McConnell req->Function = MPI2_FUNCTION_NVME_ENCAPSULATED;
178767feec50SStephen McConnell req->Flags = MPI26_NVME_FLAGS_WRITE;
178867feec50SStephen McConnell req->ErrorResponseBaseAddress.High =
178967feec50SStephen McConnell htole32((uint32_t)((uint64_t)cm->cm_sense_busaddr >> 32));
179067feec50SStephen McConnell req->ErrorResponseBaseAddress.Low =
179167feec50SStephen McConnell htole32(cm->cm_sense_busaddr);
179267feec50SStephen McConnell req->ErrorResponseAllocationLength =
179367feec50SStephen McConnell htole16(sizeof(struct nvme_completion));
179467feec50SStephen McConnell req->EncapsulatedCommandLength =
179567feec50SStephen McConnell htole16(sizeof(struct nvme_command));
179667feec50SStephen McConnell req->DataLength = htole32(data_length);
179767feec50SStephen McConnell
179867feec50SStephen McConnell /* Build NVMe DSM command */
179967feec50SStephen McConnell c = (struct nvme_command *) req->NVMe_Command;
18009544e6dcSChuck Tuffli c->opc = NVME_OPC_DATASET_MANAGEMENT;
180167feec50SStephen McConnell c->nsid = htole32(csio->ccb_h.target_lun + 1);
180267feec50SStephen McConnell c->cdw10 = htole32(ndesc - 1);
180367feec50SStephen McConnell c->cdw11 = htole32(NVME_DSM_ATTR_DEALLOCATE);
180467feec50SStephen McConnell
180567feec50SStephen McConnell cm->cm_length = data_length;
180667feec50SStephen McConnell cm->cm_data = NULL;
180767feec50SStephen McConnell
180867feec50SStephen McConnell cm->cm_complete = mprsas_scsiio_complete;
180967feec50SStephen McConnell cm->cm_complete_data = ccb;
181067feec50SStephen McConnell cm->cm_targ = targ;
181167feec50SStephen McConnell cm->cm_lun = csio->ccb_h.target_lun;
181267feec50SStephen McConnell cm->cm_ccb = ccb;
181367feec50SStephen McConnell
181467feec50SStephen McConnell cm->cm_desc.Default.RequestFlags =
181567feec50SStephen McConnell MPI26_REQ_DESCRIPT_FLAGS_PCIE_ENCAPSULATED;
181667feec50SStephen McConnell
18176eea4f46SScott Long csio->ccb_h.qos.sim_data = sbinuptime();
181867feec50SStephen McConnell callout_reset_sbt(&cm->cm_callout, SBT_1MS * ccb->ccb_h.timeout, 0,
181967feec50SStephen McConnell mprsas_scsiio_timeout, cm, 0);
182067feec50SStephen McConnell
182167feec50SStephen McConnell targ->issued++;
182267feec50SStephen McConnell targ->outstanding++;
182367feec50SStephen McConnell TAILQ_INSERT_TAIL(&targ->commands, cm, cm_link);
182467feec50SStephen McConnell ccb->ccb_h.status |= CAM_SIM_QUEUED;
182567feec50SStephen McConnell
182667feec50SStephen McConnell mprsas_log_command(cm, MPR_XINFO, "%s cm %p ccb %p outstanding %u\n",
182767feec50SStephen McConnell __func__, cm, ccb, targ->outstanding);
182867feec50SStephen McConnell
182918982e8fSStephen McConnell mpr_build_nvme_prp(sc, cm, req,
183018982e8fSStephen McConnell (void *)(uintptr_t)nvme_dsm_ranges_dma_handle, 0, data_length);
183167feec50SStephen McConnell mpr_map_command(sc, cm);
183261f17c5fSScott Long res = 0;
183367feec50SStephen McConnell
183467feec50SStephen McConnell out:
183567feec50SStephen McConnell free(plist, M_MPR);
183661f17c5fSScott Long return (res);
183767feec50SStephen McConnell }
183867feec50SStephen McConnell
1839991554f2SKenneth D. Merry static void
mprsas_action_scsiio(struct mprsas_softc * sassc,union ccb * ccb)1840991554f2SKenneth D. Merry mprsas_action_scsiio(struct mprsas_softc *sassc, union ccb *ccb)
1841991554f2SKenneth D. Merry {
1842991554f2SKenneth D. Merry MPI2_SCSI_IO_REQUEST *req;
1843991554f2SKenneth D. Merry struct ccb_scsiio *csio;
1844991554f2SKenneth D. Merry struct mpr_softc *sc;
1845991554f2SKenneth D. Merry struct mprsas_target *targ;
1846991554f2SKenneth D. Merry struct mprsas_lun *lun;
1847991554f2SKenneth D. Merry struct mpr_command *cm;
184867feec50SStephen McConnell uint8_t i, lba_byte, *ref_tag_addr, scsi_opcode;
1849991554f2SKenneth D. Merry uint16_t eedp_flags;
1850991554f2SKenneth D. Merry uint32_t mpi_control;
185167feec50SStephen McConnell int rc;
1852991554f2SKenneth D. Merry
1853991554f2SKenneth D. Merry sc = sassc->sc;
1854991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
1855991554f2SKenneth D. Merry mtx_assert(&sc->mpr_mtx, MA_OWNED);
1856991554f2SKenneth D. Merry
1857991554f2SKenneth D. Merry csio = &ccb->csio;
1858a2c14879SStephen McConnell KASSERT(csio->ccb_h.target_id < sassc->maxtargets,
1859a2c14879SStephen McConnell ("Target %d out of bounds in XPT_SCSI_IO\n",
1860a2c14879SStephen McConnell csio->ccb_h.target_id));
1861991554f2SKenneth D. Merry targ = &sassc->targets[csio->ccb_h.target_id];
1862991554f2SKenneth D. Merry mpr_dprint(sc, MPR_TRACE, "ccb %p target flag %x\n", ccb, targ->flags);
1863991554f2SKenneth D. Merry if (targ->handle == 0x0) {
1864e35816c1SWarner Losh if (targ->flags & MPRSAS_TARGET_INDIAGRESET) {
1865e35816c1SWarner Losh mpr_dprint(sc, MPR_ERROR,
1866e35816c1SWarner Losh "%s NULL handle for target %u in diag reset freezing queue\n",
1867e35816c1SWarner Losh __func__, csio->ccb_h.target_id);
1868e35816c1SWarner Losh ccb->ccb_h.status = CAM_REQUEUE_REQ | CAM_DEV_QFRZN;
1869e35816c1SWarner Losh xpt_freeze_devq(ccb->ccb_h.path, 1);
1870e35816c1SWarner Losh xpt_done(ccb);
1871e35816c1SWarner Losh return;
1872e35816c1SWarner Losh }
1873991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s NULL handle for target %u\n",
1874991554f2SKenneth D. Merry __func__, csio->ccb_h.target_id);
1875a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
1876991554f2SKenneth D. Merry xpt_done(ccb);
1877991554f2SKenneth D. Merry return;
1878991554f2SKenneth D. Merry }
1879991554f2SKenneth D. Merry if (targ->flags & MPR_TARGET_FLAGS_RAID_COMPONENT) {
1880a2c14879SStephen McConnell mpr_dprint(sc, MPR_ERROR, "%s Raid component no SCSI IO "
1881991554f2SKenneth D. Merry "supported %u\n", __func__, csio->ccb_h.target_id);
1882a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
1883991554f2SKenneth D. Merry xpt_done(ccb);
1884991554f2SKenneth D. Merry return;
1885991554f2SKenneth D. Merry }
1886991554f2SKenneth D. Merry /*
1887991554f2SKenneth D. Merry * Sometimes, it is possible to get a command that is not "In
1888991554f2SKenneth D. Merry * Progress" and was actually aborted by the upper layer. Check for
1889991554f2SKenneth D. Merry * this here and complete the command without error.
1890991554f2SKenneth D. Merry */
1891a2c14879SStephen McConnell if (mprsas_get_ccbstatus(ccb) != CAM_REQ_INPROG) {
1892991554f2SKenneth D. Merry mpr_dprint(sc, MPR_TRACE, "%s Command is not in progress for "
1893991554f2SKenneth D. Merry "target %u\n", __func__, csio->ccb_h.target_id);
1894991554f2SKenneth D. Merry xpt_done(ccb);
1895991554f2SKenneth D. Merry return;
1896991554f2SKenneth D. Merry }
1897991554f2SKenneth D. Merry /*
1898991554f2SKenneth D. Merry * If devinfo is 0 this will be a volume. In that case don't tell CAM
1899991554f2SKenneth D. Merry * that the volume has timed out. We want volumes to be enumerated
19004c1cdd4aSWarner Losh * until they are deleted/removed, not just failed. In either event,
19014c1cdd4aSWarner Losh * we're removing the target due to a firmware event telling us
19024c1cdd4aSWarner Losh * the device is now gone (as opposed to some transient event). Since
19034c1cdd4aSWarner Losh * we're opting to remove failed devices from the OS's view, we need
19044c1cdd4aSWarner Losh * to propagate that status up the stack.
1905991554f2SKenneth D. Merry */
1906991554f2SKenneth D. Merry if (targ->flags & MPRSAS_TARGET_INREMOVAL) {
1907991554f2SKenneth D. Merry if (targ->devinfo == 0)
1908a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
1909991554f2SKenneth D. Merry else
19104c1cdd4aSWarner Losh mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
1911991554f2SKenneth D. Merry xpt_done(ccb);
1912991554f2SKenneth D. Merry return;
1913991554f2SKenneth D. Merry }
1914991554f2SKenneth D. Merry
1915991554f2SKenneth D. Merry if ((sc->mpr_flags & MPR_FLAGS_SHUTDOWN) != 0) {
1916a2c14879SStephen McConnell mpr_dprint(sc, MPR_INFO, "%s shutting down\n", __func__);
1917a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
1918a2c14879SStephen McConnell xpt_done(ccb);
1919a2c14879SStephen McConnell return;
1920a2c14879SStephen McConnell }
1921a2c14879SStephen McConnell
1922a2c14879SStephen McConnell /*
19239781c28cSAlexander Motin * If target has a reset in progress, the devq should be frozen.
19249781c28cSAlexander Motin * Geting here we likely hit a race, so just requeue.
1925a2c14879SStephen McConnell */
1926a2c14879SStephen McConnell if (targ->flags & MPRSAS_TARGET_INRESET) {
19279781c28cSAlexander Motin ccb->ccb_h.status = CAM_REQUEUE_REQ | CAM_DEV_QFRZN;
1928a8837c77SWarner Losh mpr_dprint(sc, MPR_XINFO | MPR_RECOVERY,
1929a8837c77SWarner Losh "%s: Freezing devq for target ID %d\n",
1930a8837c77SWarner Losh __func__, targ->tid);
1931a2c14879SStephen McConnell xpt_freeze_devq(ccb->ccb_h.path, 1);
1932991554f2SKenneth D. Merry xpt_done(ccb);
1933991554f2SKenneth D. Merry return;
1934991554f2SKenneth D. Merry }
1935991554f2SKenneth D. Merry
1936991554f2SKenneth D. Merry cm = mpr_alloc_command(sc);
1937991554f2SKenneth D. Merry if (cm == NULL || (sc->mpr_flags & MPR_FLAGS_DIAGRESET)) {
1938991554f2SKenneth D. Merry if (cm != NULL) {
1939991554f2SKenneth D. Merry mpr_free_command(sc, cm);
1940991554f2SKenneth D. Merry }
1941991554f2SKenneth D. Merry if ((sassc->flags & MPRSAS_QUEUE_FROZEN) == 0) {
1942991554f2SKenneth D. Merry xpt_freeze_simq(sassc->sim, 1);
1943991554f2SKenneth D. Merry sassc->flags |= MPRSAS_QUEUE_FROZEN;
1944991554f2SKenneth D. Merry }
1945991554f2SKenneth D. Merry ccb->ccb_h.status &= ~CAM_SIM_QUEUED;
1946991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_REQUEUE_REQ;
1947991554f2SKenneth D. Merry xpt_done(ccb);
1948991554f2SKenneth D. Merry return;
1949991554f2SKenneth D. Merry }
1950991554f2SKenneth D. Merry
195167feec50SStephen McConnell /* For NVME device's issue UNMAP command directly to NVME drives by
195267feec50SStephen McConnell * constructing equivalent native NVMe DataSetManagement command.
195367feec50SStephen McConnell */
195467feec50SStephen McConnell scsi_opcode = scsiio_cdb_ptr(csio)[0];
195567feec50SStephen McConnell if (scsi_opcode == UNMAP &&
195667feec50SStephen McConnell targ->is_nvme &&
195767feec50SStephen McConnell (csio->ccb_h.flags & CAM_DATA_MASK) == CAM_DATA_VADDR) {
195867feec50SStephen McConnell rc = mprsas_build_nvme_unmap(sc, cm, ccb, targ);
195967feec50SStephen McConnell if (rc == 1) { /* return command to CAM with success status */
196067feec50SStephen McConnell mpr_free_command(sc, cm);
196167feec50SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
196267feec50SStephen McConnell xpt_done(ccb);
196367feec50SStephen McConnell return;
196467feec50SStephen McConnell } else if (!rc) /* Issued NVMe Encapsulated Request Message */
196567feec50SStephen McConnell return;
196667feec50SStephen McConnell }
196767feec50SStephen McConnell
1968991554f2SKenneth D. Merry req = (MPI2_SCSI_IO_REQUEST *)cm->cm_req;
1969991554f2SKenneth D. Merry bzero(req, sizeof(*req));
1970991554f2SKenneth D. Merry req->DevHandle = htole16(targ->handle);
1971991554f2SKenneth D. Merry req->Function = MPI2_FUNCTION_SCSI_IO_REQUEST;
1972991554f2SKenneth D. Merry req->MsgFlags = 0;
1973991554f2SKenneth D. Merry req->SenseBufferLowAddress = htole32(cm->cm_sense_busaddr);
1974991554f2SKenneth D. Merry req->SenseBufferLength = MPR_SENSE_LEN;
1975991554f2SKenneth D. Merry req->SGLFlags = 0;
1976991554f2SKenneth D. Merry req->ChainOffset = 0;
1977991554f2SKenneth D. Merry req->SGLOffset0 = 24; /* 32bit word offset to the SGL */
1978991554f2SKenneth D. Merry req->SGLOffset1= 0;
1979991554f2SKenneth D. Merry req->SGLOffset2= 0;
1980991554f2SKenneth D. Merry req->SGLOffset3= 0;
1981991554f2SKenneth D. Merry req->SkipCount = 0;
1982991554f2SKenneth D. Merry req->DataLength = htole32(csio->dxfer_len);
1983991554f2SKenneth D. Merry req->BidirectionalDataLength = 0;
1984991554f2SKenneth D. Merry req->IoFlags = htole16(csio->cdb_len);
1985991554f2SKenneth D. Merry req->EEDPFlags = 0;
1986991554f2SKenneth D. Merry
1987991554f2SKenneth D. Merry /* Note: BiDirectional transfers are not supported */
1988991554f2SKenneth D. Merry switch (csio->ccb_h.flags & CAM_DIR_MASK) {
1989991554f2SKenneth D. Merry case CAM_DIR_IN:
1990991554f2SKenneth D. Merry mpi_control = MPI2_SCSIIO_CONTROL_READ;
1991991554f2SKenneth D. Merry cm->cm_flags |= MPR_CM_FLAGS_DATAIN;
1992991554f2SKenneth D. Merry break;
1993991554f2SKenneth D. Merry case CAM_DIR_OUT:
1994991554f2SKenneth D. Merry mpi_control = MPI2_SCSIIO_CONTROL_WRITE;
1995991554f2SKenneth D. Merry cm->cm_flags |= MPR_CM_FLAGS_DATAOUT;
1996991554f2SKenneth D. Merry break;
1997991554f2SKenneth D. Merry case CAM_DIR_NONE:
1998991554f2SKenneth D. Merry default:
1999991554f2SKenneth D. Merry mpi_control = MPI2_SCSIIO_CONTROL_NODATATRANSFER;
2000991554f2SKenneth D. Merry break;
2001991554f2SKenneth D. Merry }
2002991554f2SKenneth D. Merry
2003991554f2SKenneth D. Merry if (csio->cdb_len == 32)
2004991554f2SKenneth D. Merry mpi_control |= 4 << MPI2_SCSIIO_CONTROL_ADDCDBLEN_SHIFT;
2005991554f2SKenneth D. Merry /*
2006991554f2SKenneth D. Merry * It looks like the hardware doesn't require an explicit tag
2007991554f2SKenneth D. Merry * number for each transaction. SAM Task Management not supported
2008991554f2SKenneth D. Merry * at the moment.
2009991554f2SKenneth D. Merry */
2010991554f2SKenneth D. Merry switch (csio->tag_action) {
2011991554f2SKenneth D. Merry case MSG_HEAD_OF_Q_TAG:
2012991554f2SKenneth D. Merry mpi_control |= MPI2_SCSIIO_CONTROL_HEADOFQ;
2013991554f2SKenneth D. Merry break;
2014991554f2SKenneth D. Merry case MSG_ORDERED_Q_TAG:
2015991554f2SKenneth D. Merry mpi_control |= MPI2_SCSIIO_CONTROL_ORDEREDQ;
2016991554f2SKenneth D. Merry break;
2017991554f2SKenneth D. Merry case MSG_ACA_TASK:
2018991554f2SKenneth D. Merry mpi_control |= MPI2_SCSIIO_CONTROL_ACAQ;
2019991554f2SKenneth D. Merry break;
2020991554f2SKenneth D. Merry case CAM_TAG_ACTION_NONE:
2021991554f2SKenneth D. Merry case MSG_SIMPLE_Q_TAG:
2022991554f2SKenneth D. Merry default:
2023991554f2SKenneth D. Merry mpi_control |= MPI2_SCSIIO_CONTROL_SIMPLEQ;
2024991554f2SKenneth D. Merry break;
2025991554f2SKenneth D. Merry }
202688364968SAlexander Motin mpi_control |= (csio->priority << MPI2_SCSIIO_CONTROL_CMDPRI_SHIFT) &
202788364968SAlexander Motin MPI2_SCSIIO_CONTROL_CMDPRI_MASK;
2028991554f2SKenneth D. Merry mpi_control |= sc->mapping_table[csio->ccb_h.target_id].TLR_bits;
2029991554f2SKenneth D. Merry req->Control = htole32(mpi_control);
2030991554f2SKenneth D. Merry
2031991554f2SKenneth D. Merry if (MPR_SET_LUN(req->LUN, csio->ccb_h.target_lun) != 0) {
2032991554f2SKenneth D. Merry mpr_free_command(sc, cm);
2033a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_LUN_INVALID);
2034991554f2SKenneth D. Merry xpt_done(ccb);
2035991554f2SKenneth D. Merry return;
2036991554f2SKenneth D. Merry }
2037991554f2SKenneth D. Merry
2038991554f2SKenneth D. Merry if (csio->ccb_h.flags & CAM_CDB_POINTER)
2039991554f2SKenneth D. Merry bcopy(csio->cdb_io.cdb_ptr, &req->CDB.CDB32[0], csio->cdb_len);
2040fa699bb2SAlan Somers else {
2041fa699bb2SAlan Somers KASSERT(csio->cdb_len <= IOCDBLEN,
204267feec50SStephen McConnell ("cdb_len %d is greater than IOCDBLEN but CAM_CDB_POINTER "
204367feec50SStephen McConnell "is not set", csio->cdb_len));
2044991554f2SKenneth D. Merry bcopy(csio->cdb_io.cdb_bytes, &req->CDB.CDB32[0],csio->cdb_len);
2045fa699bb2SAlan Somers }
2046991554f2SKenneth D. Merry req->IoFlags = htole16(csio->cdb_len);
2047991554f2SKenneth D. Merry
2048991554f2SKenneth D. Merry /*
2049991554f2SKenneth D. Merry * Check if EEDP is supported and enabled. If it is then check if the
2050991554f2SKenneth D. Merry * SCSI opcode could be using EEDP. If so, make sure the LUN exists and
2051991554f2SKenneth D. Merry * is formatted for EEDP support. If all of this is true, set CDB up
2052991554f2SKenneth D. Merry * for EEDP transfer.
2053991554f2SKenneth D. Merry */
2054991554f2SKenneth D. Merry eedp_flags = op_code_prot[req->CDB.CDB32[0]];
2055991554f2SKenneth D. Merry if (sc->eedp_enabled && eedp_flags) {
2056991554f2SKenneth D. Merry SLIST_FOREACH(lun, &targ->luns, lun_link) {
2057991554f2SKenneth D. Merry if (lun->lun_id == csio->ccb_h.target_lun) {
2058991554f2SKenneth D. Merry break;
2059991554f2SKenneth D. Merry }
2060991554f2SKenneth D. Merry }
2061991554f2SKenneth D. Merry
2062991554f2SKenneth D. Merry if ((lun != NULL) && (lun->eedp_formatted)) {
206371900a79SAlfredo Dal'Ava Junior req->EEDPBlockSize = htole32(lun->eedp_block_size);
2064991554f2SKenneth D. Merry eedp_flags |= (MPI2_SCSIIO_EEDPFLAGS_INC_PRI_REFTAG |
2065991554f2SKenneth D. Merry MPI2_SCSIIO_EEDPFLAGS_CHECK_REFTAG |
2066991554f2SKenneth D. Merry MPI2_SCSIIO_EEDPFLAGS_CHECK_GUARD);
206767feec50SStephen McConnell if (sc->mpr_flags & MPR_FLAGS_GEN35_IOC) {
206867feec50SStephen McConnell eedp_flags |=
206967feec50SStephen McConnell MPI25_SCSIIO_EEDPFLAGS_APPTAG_DISABLE_MODE;
207067feec50SStephen McConnell }
2071991554f2SKenneth D. Merry req->EEDPFlags = htole16(eedp_flags);
2072991554f2SKenneth D. Merry
2073991554f2SKenneth D. Merry /*
2074991554f2SKenneth D. Merry * If CDB less than 32, fill in Primary Ref Tag with
2075991554f2SKenneth D. Merry * low 4 bytes of LBA. If CDB is 32, tag stuff is
2076991554f2SKenneth D. Merry * already there. Also, set protection bit. FreeBSD
2077991554f2SKenneth D. Merry * currently does not support CDBs bigger than 16, but
2078991554f2SKenneth D. Merry * the code doesn't hurt, and will be here for the
2079991554f2SKenneth D. Merry * future.
2080991554f2SKenneth D. Merry */
2081991554f2SKenneth D. Merry if (csio->cdb_len != 32) {
2082991554f2SKenneth D. Merry lba_byte = (csio->cdb_len == 16) ? 6 : 2;
2083991554f2SKenneth D. Merry ref_tag_addr = (uint8_t *)&req->CDB.EEDP32.
2084991554f2SKenneth D. Merry PrimaryReferenceTag;
2085991554f2SKenneth D. Merry for (i = 0; i < 4; i++) {
2086991554f2SKenneth D. Merry *ref_tag_addr =
2087991554f2SKenneth D. Merry req->CDB.CDB32[lba_byte + i];
2088991554f2SKenneth D. Merry ref_tag_addr++;
2089991554f2SKenneth D. Merry }
2090991554f2SKenneth D. Merry req->CDB.EEDP32.PrimaryReferenceTag =
2091991554f2SKenneth D. Merry htole32(req->
2092991554f2SKenneth D. Merry CDB.EEDP32.PrimaryReferenceTag);
2093991554f2SKenneth D. Merry req->CDB.EEDP32.PrimaryApplicationTagMask =
2094991554f2SKenneth D. Merry 0xFFFF;
20958881681bSKenneth D. Merry req->CDB.CDB32[1] =
20968881681bSKenneth D. Merry (req->CDB.CDB32[1] & 0x1F) | 0x20;
2097991554f2SKenneth D. Merry } else {
2098991554f2SKenneth D. Merry eedp_flags |=
2099991554f2SKenneth D. Merry MPI2_SCSIIO_EEDPFLAGS_INC_PRI_APPTAG;
2100991554f2SKenneth D. Merry req->EEDPFlags = htole16(eedp_flags);
2101991554f2SKenneth D. Merry req->CDB.CDB32[10] = (req->CDB.CDB32[10] &
2102991554f2SKenneth D. Merry 0x1F) | 0x20;
2103991554f2SKenneth D. Merry }
2104991554f2SKenneth D. Merry }
2105991554f2SKenneth D. Merry }
2106991554f2SKenneth D. Merry
2107991554f2SKenneth D. Merry cm->cm_length = csio->dxfer_len;
2108991554f2SKenneth D. Merry if (cm->cm_length != 0) {
2109991554f2SKenneth D. Merry cm->cm_data = ccb;
2110991554f2SKenneth D. Merry cm->cm_flags |= MPR_CM_FLAGS_USE_CCB;
2111991554f2SKenneth D. Merry } else {
2112991554f2SKenneth D. Merry cm->cm_data = NULL;
2113991554f2SKenneth D. Merry }
2114991554f2SKenneth D. Merry cm->cm_sge = &req->SGL;
2115991554f2SKenneth D. Merry cm->cm_sglsize = (32 - 24) * 4;
2116991554f2SKenneth D. Merry cm->cm_complete = mprsas_scsiio_complete;
2117991554f2SKenneth D. Merry cm->cm_complete_data = ccb;
2118991554f2SKenneth D. Merry cm->cm_targ = targ;
2119991554f2SKenneth D. Merry cm->cm_lun = csio->ccb_h.target_lun;
2120991554f2SKenneth D. Merry cm->cm_ccb = ccb;
2121991554f2SKenneth D. Merry /*
2122991554f2SKenneth D. Merry * If using FP desc type, need to set a bit in IoFlags (SCSI IO is 0)
2123991554f2SKenneth D. Merry * and set descriptor type.
2124991554f2SKenneth D. Merry */
2125991554f2SKenneth D. Merry if (targ->scsi_req_desc_type ==
2126991554f2SKenneth D. Merry MPI25_REQ_DESCRIPT_FLAGS_FAST_PATH_SCSI_IO) {
2127991554f2SKenneth D. Merry req->IoFlags |= MPI25_SCSIIO_IOFLAGS_FAST_PATH;
2128991554f2SKenneth D. Merry cm->cm_desc.FastPathSCSIIO.RequestFlags =
2129991554f2SKenneth D. Merry MPI25_REQ_DESCRIPT_FLAGS_FAST_PATH_SCSI_IO;
213067feec50SStephen McConnell if (!sc->atomic_desc_capable) {
213167feec50SStephen McConnell cm->cm_desc.FastPathSCSIIO.DevHandle =
213267feec50SStephen McConnell htole16(targ->handle);
213367feec50SStephen McConnell }
2134991554f2SKenneth D. Merry } else {
2135991554f2SKenneth D. Merry cm->cm_desc.SCSIIO.RequestFlags =
2136991554f2SKenneth D. Merry MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO;
213767feec50SStephen McConnell if (!sc->atomic_desc_capable)
2138991554f2SKenneth D. Merry cm->cm_desc.SCSIIO.DevHandle = htole16(targ->handle);
2139991554f2SKenneth D. Merry }
2140991554f2SKenneth D. Merry
21416eea4f46SScott Long csio->ccb_h.qos.sim_data = sbinuptime();
214285c9dd9dSSteven Hartland callout_reset_sbt(&cm->cm_callout, SBT_1MS * ccb->ccb_h.timeout, 0,
214385c9dd9dSSteven Hartland mprsas_scsiio_timeout, cm, 0);
2144991554f2SKenneth D. Merry
2145991554f2SKenneth D. Merry targ->issued++;
2146991554f2SKenneth D. Merry targ->outstanding++;
2147991554f2SKenneth D. Merry TAILQ_INSERT_TAIL(&targ->commands, cm, cm_link);
2148991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_SIM_QUEUED;
2149991554f2SKenneth D. Merry
2150991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_XINFO, "%s cm %p ccb %p outstanding %u\n",
2151991554f2SKenneth D. Merry __func__, cm, ccb, targ->outstanding);
2152991554f2SKenneth D. Merry
2153991554f2SKenneth D. Merry mpr_map_command(sc, cm);
2154991554f2SKenneth D. Merry return;
2155991554f2SKenneth D. Merry }
2156991554f2SKenneth D. Merry
2157991554f2SKenneth D. Merry /**
2158991554f2SKenneth D. Merry * mpr_sc_failed_io_info - translated non-succesfull SCSI_IO request
2159991554f2SKenneth D. Merry */
2160991554f2SKenneth D. Merry static void
mpr_sc_failed_io_info(struct mpr_softc * sc,struct ccb_scsiio * csio,Mpi2SCSIIOReply_t * mpi_reply,struct mprsas_target * targ)2161991554f2SKenneth D. Merry mpr_sc_failed_io_info(struct mpr_softc *sc, struct ccb_scsiio *csio,
2162991554f2SKenneth D. Merry Mpi2SCSIIOReply_t *mpi_reply, struct mprsas_target *targ)
2163991554f2SKenneth D. Merry {
2164991554f2SKenneth D. Merry u32 response_info;
2165991554f2SKenneth D. Merry u8 *response_bytes;
2166991554f2SKenneth D. Merry u16 ioc_status = le16toh(mpi_reply->IOCStatus) &
2167991554f2SKenneth D. Merry MPI2_IOCSTATUS_MASK;
2168991554f2SKenneth D. Merry u8 scsi_state = mpi_reply->SCSIState;
2169991554f2SKenneth D. Merry u8 scsi_status = mpi_reply->SCSIStatus;
2170991554f2SKenneth D. Merry char *desc_ioc_state = NULL;
2171991554f2SKenneth D. Merry char *desc_scsi_status = NULL;
2172991554f2SKenneth D. Merry u32 log_info = le32toh(mpi_reply->IOCLogInfo);
2173991554f2SKenneth D. Merry
2174991554f2SKenneth D. Merry if (log_info == 0x31170000)
2175991554f2SKenneth D. Merry return;
2176991554f2SKenneth D. Merry
21772bf620cbSScott Long desc_ioc_state = mpr_describe_table(mpr_iocstatus_string,
21782bf620cbSScott Long ioc_status);
21792bf620cbSScott Long desc_scsi_status = mpr_describe_table(mpr_scsi_status_string,
21802bf620cbSScott Long scsi_status);
2181991554f2SKenneth D. Merry
2182991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "\thandle(0x%04x), ioc_status(%s)(0x%04x)\n",
2183991554f2SKenneth D. Merry le16toh(mpi_reply->DevHandle), desc_ioc_state, ioc_status);
2184991554f2SKenneth D. Merry if (targ->encl_level_valid) {
2185991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "At enclosure level %d, slot %d, "
2186991554f2SKenneth D. Merry "connector name (%4s)\n", targ->encl_level, targ->encl_slot,
2187991554f2SKenneth D. Merry targ->connector_name);
2188991554f2SKenneth D. Merry }
21892bf620cbSScott Long
21902bf620cbSScott Long /*
21912bf620cbSScott Long * We can add more detail about underflow data here
2192991554f2SKenneth D. Merry * TO-DO
21932bf620cbSScott Long */
2194991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "\tscsi_status(%s)(0x%02x), "
21952bf620cbSScott Long "scsi_state %b\n", desc_scsi_status, scsi_status,
21962bf620cbSScott Long scsi_state, "\20" "\1AutosenseValid" "\2AutosenseFailed"
21972bf620cbSScott Long "\3NoScsiStatus" "\4Terminated" "\5Response InfoValid");
2198991554f2SKenneth D. Merry
2199991554f2SKenneth D. Merry if (sc->mpr_debug & MPR_XINFO &&
2200991554f2SKenneth D. Merry scsi_state & MPI2_SCSI_STATE_AUTOSENSE_VALID) {
2201991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "-> Sense Buffer Data : Start :\n");
2202991554f2SKenneth D. Merry scsi_sense_print(csio);
2203991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "-> Sense Buffer Data : End :\n");
2204991554f2SKenneth D. Merry }
2205991554f2SKenneth D. Merry
2206991554f2SKenneth D. Merry if (scsi_state & MPI2_SCSI_STATE_RESPONSE_INFO_VALID) {
2207991554f2SKenneth D. Merry response_info = le32toh(mpi_reply->ResponseInfo);
2208991554f2SKenneth D. Merry response_bytes = (u8 *)&response_info;
22092bf620cbSScott Long mpr_dprint(sc, MPR_XINFO, "response code(0x%01x): %s\n",
22102bf620cbSScott Long response_bytes[0],
22112bf620cbSScott Long mpr_describe_table(mpr_scsi_taskmgmt_string,
22122bf620cbSScott Long response_bytes[0]));
2213991554f2SKenneth D. Merry }
2214991554f2SKenneth D. Merry }
2215991554f2SKenneth D. Merry
221667feec50SStephen McConnell /** mprsas_nvme_trans_status_code
221767feec50SStephen McConnell *
221867feec50SStephen McConnell * Convert Native NVMe command error status to
221967feec50SStephen McConnell * equivalent SCSI error status.
222067feec50SStephen McConnell *
222167feec50SStephen McConnell * Returns appropriate scsi_status
222267feec50SStephen McConnell */
222367feec50SStephen McConnell static u8
mprsas_nvme_trans_status_code(uint16_t nvme_status,struct mpr_command * cm)22240d787e9bSWojciech Macek mprsas_nvme_trans_status_code(uint16_t nvme_status,
222567feec50SStephen McConnell struct mpr_command *cm)
222667feec50SStephen McConnell {
222767feec50SStephen McConnell u8 status = MPI2_SCSI_STATUS_GOOD;
222867feec50SStephen McConnell int skey, asc, ascq;
222967feec50SStephen McConnell union ccb *ccb = cm->cm_complete_data;
223067feec50SStephen McConnell int returned_sense_len;
22310d787e9bSWojciech Macek uint8_t sct, sc;
22320d787e9bSWojciech Macek
22330d787e9bSWojciech Macek sct = NVME_STATUS_GET_SCT(nvme_status);
22340d787e9bSWojciech Macek sc = NVME_STATUS_GET_SC(nvme_status);
223567feec50SStephen McConnell
223667feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
223767feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
223867feec50SStephen McConnell asc = SCSI_ASC_NO_SENSE;
223967feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
224067feec50SStephen McConnell
22410d787e9bSWojciech Macek switch (sct) {
224267feec50SStephen McConnell case NVME_SCT_GENERIC:
22430d787e9bSWojciech Macek switch (sc) {
224467feec50SStephen McConnell case NVME_SC_SUCCESS:
224567feec50SStephen McConnell status = MPI2_SCSI_STATUS_GOOD;
224667feec50SStephen McConnell skey = SSD_KEY_NO_SENSE;
224767feec50SStephen McConnell asc = SCSI_ASC_NO_SENSE;
224867feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
224967feec50SStephen McConnell break;
225067feec50SStephen McConnell case NVME_SC_INVALID_OPCODE:
225167feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
225267feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
225367feec50SStephen McConnell asc = SCSI_ASC_ILLEGAL_COMMAND;
225467feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
225567feec50SStephen McConnell break;
225667feec50SStephen McConnell case NVME_SC_INVALID_FIELD:
225767feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
225867feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
225967feec50SStephen McConnell asc = SCSI_ASC_INVALID_CDB;
226067feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
226167feec50SStephen McConnell break;
226267feec50SStephen McConnell case NVME_SC_DATA_TRANSFER_ERROR:
226367feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
226467feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
226567feec50SStephen McConnell asc = SCSI_ASC_NO_SENSE;
226667feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
226767feec50SStephen McConnell break;
226867feec50SStephen McConnell case NVME_SC_ABORTED_POWER_LOSS:
226967feec50SStephen McConnell status = MPI2_SCSI_STATUS_TASK_ABORTED;
227067feec50SStephen McConnell skey = SSD_KEY_ABORTED_COMMAND;
227167feec50SStephen McConnell asc = SCSI_ASC_WARNING;
227267feec50SStephen McConnell ascq = SCSI_ASCQ_POWER_LOSS_EXPECTED;
227367feec50SStephen McConnell break;
227467feec50SStephen McConnell case NVME_SC_INTERNAL_DEVICE_ERROR:
227567feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
227667feec50SStephen McConnell skey = SSD_KEY_HARDWARE_ERROR;
227767feec50SStephen McConnell asc = SCSI_ASC_INTERNAL_TARGET_FAILURE;
227867feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
227967feec50SStephen McConnell break;
228067feec50SStephen McConnell case NVME_SC_ABORTED_BY_REQUEST:
228167feec50SStephen McConnell case NVME_SC_ABORTED_SQ_DELETION:
228267feec50SStephen McConnell case NVME_SC_ABORTED_FAILED_FUSED:
228367feec50SStephen McConnell case NVME_SC_ABORTED_MISSING_FUSED:
228467feec50SStephen McConnell status = MPI2_SCSI_STATUS_TASK_ABORTED;
228567feec50SStephen McConnell skey = SSD_KEY_ABORTED_COMMAND;
228667feec50SStephen McConnell asc = SCSI_ASC_NO_SENSE;
228767feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
228867feec50SStephen McConnell break;
228967feec50SStephen McConnell case NVME_SC_INVALID_NAMESPACE_OR_FORMAT:
229067feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
229167feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
229267feec50SStephen McConnell asc = SCSI_ASC_ACCESS_DENIED_INVALID_LUN_ID;
229367feec50SStephen McConnell ascq = SCSI_ASCQ_INVALID_LUN_ID;
229467feec50SStephen McConnell break;
229567feec50SStephen McConnell case NVME_SC_LBA_OUT_OF_RANGE:
229667feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
229767feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
229867feec50SStephen McConnell asc = SCSI_ASC_ILLEGAL_BLOCK;
229967feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
230067feec50SStephen McConnell break;
230167feec50SStephen McConnell case NVME_SC_CAPACITY_EXCEEDED:
230267feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
230367feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
230467feec50SStephen McConnell asc = SCSI_ASC_NO_SENSE;
230567feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
230667feec50SStephen McConnell break;
230767feec50SStephen McConnell case NVME_SC_NAMESPACE_NOT_READY:
230867feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
230967feec50SStephen McConnell skey = SSD_KEY_NOT_READY;
231067feec50SStephen McConnell asc = SCSI_ASC_LUN_NOT_READY;
231167feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
231267feec50SStephen McConnell break;
231367feec50SStephen McConnell }
231467feec50SStephen McConnell break;
231567feec50SStephen McConnell case NVME_SCT_COMMAND_SPECIFIC:
23160d787e9bSWojciech Macek switch (sc) {
231767feec50SStephen McConnell case NVME_SC_INVALID_FORMAT:
231867feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
231967feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
232067feec50SStephen McConnell asc = SCSI_ASC_FORMAT_COMMAND_FAILED;
232167feec50SStephen McConnell ascq = SCSI_ASCQ_FORMAT_COMMAND_FAILED;
232267feec50SStephen McConnell break;
232367feec50SStephen McConnell case NVME_SC_CONFLICTING_ATTRIBUTES:
232467feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
232567feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
232667feec50SStephen McConnell asc = SCSI_ASC_INVALID_CDB;
232767feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
232867feec50SStephen McConnell break;
232967feec50SStephen McConnell }
233067feec50SStephen McConnell break;
233167feec50SStephen McConnell case NVME_SCT_MEDIA_ERROR:
23320d787e9bSWojciech Macek switch (sc) {
233367feec50SStephen McConnell case NVME_SC_WRITE_FAULTS:
233467feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
233567feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
233667feec50SStephen McConnell asc = SCSI_ASC_PERIPHERAL_DEV_WRITE_FAULT;
233767feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
233867feec50SStephen McConnell break;
233967feec50SStephen McConnell case NVME_SC_UNRECOVERED_READ_ERROR:
234067feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
234167feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
234267feec50SStephen McConnell asc = SCSI_ASC_UNRECOVERED_READ_ERROR;
234367feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
234467feec50SStephen McConnell break;
234567feec50SStephen McConnell case NVME_SC_GUARD_CHECK_ERROR:
234667feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
234767feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
234867feec50SStephen McConnell asc = SCSI_ASC_LOG_BLOCK_GUARD_CHECK_FAILED;
234967feec50SStephen McConnell ascq = SCSI_ASCQ_LOG_BLOCK_GUARD_CHECK_FAILED;
235067feec50SStephen McConnell break;
235167feec50SStephen McConnell case NVME_SC_APPLICATION_TAG_CHECK_ERROR:
235267feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
235367feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
235467feec50SStephen McConnell asc = SCSI_ASC_LOG_BLOCK_APPTAG_CHECK_FAILED;
235567feec50SStephen McConnell ascq = SCSI_ASCQ_LOG_BLOCK_APPTAG_CHECK_FAILED;
235667feec50SStephen McConnell break;
235767feec50SStephen McConnell case NVME_SC_REFERENCE_TAG_CHECK_ERROR:
235867feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
235967feec50SStephen McConnell skey = SSD_KEY_MEDIUM_ERROR;
236067feec50SStephen McConnell asc = SCSI_ASC_LOG_BLOCK_REFTAG_CHECK_FAILED;
236167feec50SStephen McConnell ascq = SCSI_ASCQ_LOG_BLOCK_REFTAG_CHECK_FAILED;
236267feec50SStephen McConnell break;
236367feec50SStephen McConnell case NVME_SC_COMPARE_FAILURE:
236467feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
236567feec50SStephen McConnell skey = SSD_KEY_MISCOMPARE;
236667feec50SStephen McConnell asc = SCSI_ASC_MISCOMPARE_DURING_VERIFY;
236767feec50SStephen McConnell ascq = SCSI_ASCQ_CAUSE_NOT_REPORTABLE;
236867feec50SStephen McConnell break;
236967feec50SStephen McConnell case NVME_SC_ACCESS_DENIED:
237067feec50SStephen McConnell status = MPI2_SCSI_STATUS_CHECK_CONDITION;
237167feec50SStephen McConnell skey = SSD_KEY_ILLEGAL_REQUEST;
237267feec50SStephen McConnell asc = SCSI_ASC_ACCESS_DENIED_INVALID_LUN_ID;
237367feec50SStephen McConnell ascq = SCSI_ASCQ_INVALID_LUN_ID;
237467feec50SStephen McConnell break;
237567feec50SStephen McConnell }
237667feec50SStephen McConnell break;
237767feec50SStephen McConnell }
237867feec50SStephen McConnell
237967feec50SStephen McConnell returned_sense_len = sizeof(struct scsi_sense_data);
238067feec50SStephen McConnell if (returned_sense_len < ccb->csio.sense_len)
238167feec50SStephen McConnell ccb->csio.sense_resid = ccb->csio.sense_len -
238267feec50SStephen McConnell returned_sense_len;
238367feec50SStephen McConnell else
238467feec50SStephen McConnell ccb->csio.sense_resid = 0;
238567feec50SStephen McConnell
238667feec50SStephen McConnell scsi_set_sense_data(&ccb->csio.sense_data, SSD_TYPE_FIXED,
238767feec50SStephen McConnell 1, skey, asc, ascq, SSD_ELEM_NONE);
238867feec50SStephen McConnell ccb->ccb_h.status |= CAM_AUTOSNS_VALID;
238967feec50SStephen McConnell
239067feec50SStephen McConnell return status;
239167feec50SStephen McConnell }
239267feec50SStephen McConnell
239367feec50SStephen McConnell /** mprsas_complete_nvme_unmap
239467feec50SStephen McConnell *
239567feec50SStephen McConnell * Complete native NVMe command issued using NVMe Encapsulated
239667feec50SStephen McConnell * Request Message.
239767feec50SStephen McConnell */
239867feec50SStephen McConnell static u8
mprsas_complete_nvme_unmap(struct mpr_softc * sc,struct mpr_command * cm)239967feec50SStephen McConnell mprsas_complete_nvme_unmap(struct mpr_softc *sc, struct mpr_command *cm)
240067feec50SStephen McConnell {
240167feec50SStephen McConnell Mpi26NVMeEncapsulatedErrorReply_t *mpi_reply;
240267feec50SStephen McConnell struct nvme_completion *nvme_completion = NULL;
240367feec50SStephen McConnell u8 scsi_status = MPI2_SCSI_STATUS_GOOD;
240467feec50SStephen McConnell
240567feec50SStephen McConnell mpi_reply =(Mpi26NVMeEncapsulatedErrorReply_t *)cm->cm_reply;
240667feec50SStephen McConnell if (le16toh(mpi_reply->ErrorResponseCount)){
240767feec50SStephen McConnell nvme_completion = (struct nvme_completion *)cm->cm_sense;
240867feec50SStephen McConnell scsi_status = mprsas_nvme_trans_status_code(
240967feec50SStephen McConnell nvme_completion->status, cm);
241067feec50SStephen McConnell }
241167feec50SStephen McConnell return scsi_status;
241267feec50SStephen McConnell }
241367feec50SStephen McConnell
2414991554f2SKenneth D. Merry static void
mprsas_scsiio_complete(struct mpr_softc * sc,struct mpr_command * cm)2415991554f2SKenneth D. Merry mprsas_scsiio_complete(struct mpr_softc *sc, struct mpr_command *cm)
2416991554f2SKenneth D. Merry {
2417991554f2SKenneth D. Merry MPI2_SCSI_IO_REPLY *rep;
2418991554f2SKenneth D. Merry union ccb *ccb;
2419991554f2SKenneth D. Merry struct ccb_scsiio *csio;
2420991554f2SKenneth D. Merry struct mprsas_softc *sassc;
2421991554f2SKenneth D. Merry struct scsi_vpd_supported_page_list *vpd_list = NULL;
242267feec50SStephen McConnell u8 *TLR_bits, TLR_on, *scsi_cdb;
2423991554f2SKenneth D. Merry int dir = 0, i;
2424991554f2SKenneth D. Merry u16 alloc_len;
2425a2c14879SStephen McConnell struct mprsas_target *target;
2426a2c14879SStephen McConnell target_id_t target_id;
2427991554f2SKenneth D. Merry
2428991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
2429991554f2SKenneth D. Merry
2430991554f2SKenneth D. Merry callout_stop(&cm->cm_callout);
2431991554f2SKenneth D. Merry mtx_assert(&sc->mpr_mtx, MA_OWNED);
2432991554f2SKenneth D. Merry
2433991554f2SKenneth D. Merry sassc = sc->sassc;
2434991554f2SKenneth D. Merry ccb = cm->cm_complete_data;
2435991554f2SKenneth D. Merry csio = &ccb->csio;
2436a2c14879SStephen McConnell target_id = csio->ccb_h.target_id;
2437991554f2SKenneth D. Merry rep = (MPI2_SCSI_IO_REPLY *)cm->cm_reply;
243871900a79SAlfredo Dal'Ava Junior mpr_dprint(sc, MPR_TRACE,
243971900a79SAlfredo Dal'Ava Junior "cm %p SMID %u ccb %p reply %p outstanding %u csio->scsi_status 0x%x,"
244071900a79SAlfredo Dal'Ava Junior "csio->dxfer_len 0x%x, csio->msg_le 0x%xn\n", cm,
244171900a79SAlfredo Dal'Ava Junior cm->cm_desc.Default.SMID, cm->cm_ccb, cm->cm_reply,
244271900a79SAlfredo Dal'Ava Junior cm->cm_targ->outstanding, csio->scsi_status,
244371900a79SAlfredo Dal'Ava Junior csio->dxfer_len, csio->msg_len);
2444991554f2SKenneth D. Merry /*
2445991554f2SKenneth D. Merry * XXX KDM if the chain allocation fails, does it matter if we do
2446991554f2SKenneth D. Merry * the sync and unload here? It is simpler to do it in every case,
2447991554f2SKenneth D. Merry * assuming it doesn't cause problems.
2448991554f2SKenneth D. Merry */
2449991554f2SKenneth D. Merry if (cm->cm_data != NULL) {
2450991554f2SKenneth D. Merry if (cm->cm_flags & MPR_CM_FLAGS_DATAIN)
2451991554f2SKenneth D. Merry dir = BUS_DMASYNC_POSTREAD;
2452991554f2SKenneth D. Merry else if (cm->cm_flags & MPR_CM_FLAGS_DATAOUT)
2453991554f2SKenneth D. Merry dir = BUS_DMASYNC_POSTWRITE;
2454991554f2SKenneth D. Merry bus_dmamap_sync(sc->buffer_dmat, cm->cm_dmamap, dir);
2455991554f2SKenneth D. Merry bus_dmamap_unload(sc->buffer_dmat, cm->cm_dmamap);
2456991554f2SKenneth D. Merry }
2457991554f2SKenneth D. Merry
2458991554f2SKenneth D. Merry cm->cm_targ->completed++;
2459991554f2SKenneth D. Merry cm->cm_targ->outstanding--;
2460991554f2SKenneth D. Merry TAILQ_REMOVE(&cm->cm_targ->commands, cm, cm_link);
2461991554f2SKenneth D. Merry ccb->ccb_h.status &= ~(CAM_STATUS_MASK | CAM_SIM_QUEUED);
2462991554f2SKenneth D. Merry
24638fe7bf06SWarner Losh if (cm->cm_flags & MPR_CM_FLAGS_ON_RECOVERY) {
2464991554f2SKenneth D. Merry TAILQ_REMOVE(&cm->cm_targ->timedout_commands, cm, cm_recovery);
24658fe7bf06SWarner Losh KASSERT(cm->cm_state == MPR_CM_STATE_BUSY,
2466175ad3d0SKenneth D. Merry ("Not busy for CM_FLAGS_TIMEDOUT: %u\n", cm->cm_state));
24678fe7bf06SWarner Losh cm->cm_flags &= ~MPR_CM_FLAGS_ON_RECOVERY;
2468991554f2SKenneth D. Merry if (cm->cm_reply != NULL)
2469991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
2470991554f2SKenneth D. Merry "completed timedout cm %p ccb %p during recovery "
2471991554f2SKenneth D. Merry "ioc %x scsi %x state %x xfer %u\n", cm, cm->cm_ccb,
2472991554f2SKenneth D. Merry le16toh(rep->IOCStatus), rep->SCSIStatus,
2473991554f2SKenneth D. Merry rep->SCSIState, le32toh(rep->TransferCount));
2474991554f2SKenneth D. Merry else
2475991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
2476991554f2SKenneth D. Merry "completed timedout cm %p ccb %p during recovery\n",
2477991554f2SKenneth D. Merry cm, cm->cm_ccb);
2478991554f2SKenneth D. Merry } else if (cm->cm_targ->tm != NULL) {
2479991554f2SKenneth D. Merry if (cm->cm_reply != NULL)
2480991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
2481991554f2SKenneth D. Merry "completed cm %p ccb %p during recovery "
2482991554f2SKenneth D. Merry "ioc %x scsi %x state %x xfer %u\n",
2483991554f2SKenneth D. Merry cm, cm->cm_ccb, le16toh(rep->IOCStatus),
2484991554f2SKenneth D. Merry rep->SCSIStatus, rep->SCSIState,
2485991554f2SKenneth D. Merry le32toh(rep->TransferCount));
2486991554f2SKenneth D. Merry else
2487991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
2488991554f2SKenneth D. Merry "completed cm %p ccb %p during recovery\n",
2489991554f2SKenneth D. Merry cm, cm->cm_ccb);
2490991554f2SKenneth D. Merry } else if ((sc->mpr_flags & MPR_FLAGS_DIAGRESET) != 0) {
2491991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_RECOVERY,
2492991554f2SKenneth D. Merry "reset completed cm %p ccb %p\n", cm, cm->cm_ccb);
2493991554f2SKenneth D. Merry }
2494991554f2SKenneth D. Merry
2495991554f2SKenneth D. Merry if ((cm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
2496991554f2SKenneth D. Merry /*
2497991554f2SKenneth D. Merry * We ran into an error after we tried to map the command,
2498991554f2SKenneth D. Merry * so we're getting a callback without queueing the command
2499991554f2SKenneth D. Merry * to the hardware. So we set the status here, and it will
2500991554f2SKenneth D. Merry * be retained below. We'll go through the "fast path",
2501991554f2SKenneth D. Merry * because there can be no reply when we haven't actually
2502991554f2SKenneth D. Merry * gone out to the hardware.
2503991554f2SKenneth D. Merry */
2504a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQUEUE_REQ);
2505991554f2SKenneth D. Merry
2506991554f2SKenneth D. Merry /*
2507991554f2SKenneth D. Merry * Currently the only error included in the mask is
2508991554f2SKenneth D. Merry * MPR_CM_FLAGS_CHAIN_FAILED, which means we're out of
2509991554f2SKenneth D. Merry * chain frames. We need to freeze the queue until we get
2510991554f2SKenneth D. Merry * a command that completed without this error, which will
2511991554f2SKenneth D. Merry * hopefully have some chain frames attached that we can
2512991554f2SKenneth D. Merry * use. If we wanted to get smarter about it, we would
2513991554f2SKenneth D. Merry * only unfreeze the queue in this condition when we're
2514991554f2SKenneth D. Merry * sure that we're getting some chain frames back. That's
2515991554f2SKenneth D. Merry * probably unnecessary.
2516991554f2SKenneth D. Merry */
2517991554f2SKenneth D. Merry if ((sassc->flags & MPRSAS_QUEUE_FROZEN) == 0) {
2518991554f2SKenneth D. Merry xpt_freeze_simq(sassc->sim, 1);
2519991554f2SKenneth D. Merry sassc->flags |= MPRSAS_QUEUE_FROZEN;
2520a8837c77SWarner Losh mpr_dprint(sc, MPR_XINFO | MPR_RECOVERY,
2521a8837c77SWarner Losh "Error sending command, freezing SIM queue\n");
2522991554f2SKenneth D. Merry }
2523991554f2SKenneth D. Merry }
2524991554f2SKenneth D. Merry
2525991554f2SKenneth D. Merry /*
252667feec50SStephen McConnell * Point to the SCSI CDB, which is dependent on the CAM_CDB_POINTER
252767feec50SStephen McConnell * flag, and use it in a few places in the rest of this function for
252867feec50SStephen McConnell * convenience. Use the macro if available.
252967feec50SStephen McConnell */
253067feec50SStephen McConnell scsi_cdb = scsiio_cdb_ptr(csio);
253167feec50SStephen McConnell
253267feec50SStephen McConnell /*
2533991554f2SKenneth D. Merry * If this is a Start Stop Unit command and it was issued by the driver
2534991554f2SKenneth D. Merry * during shutdown, decrement the refcount to account for all of the
2535991554f2SKenneth D. Merry * commands that were sent. All SSU commands should be completed before
2536991554f2SKenneth D. Merry * shutdown completes, meaning SSU_refcount will be 0 after SSU_started
2537991554f2SKenneth D. Merry * is TRUE.
2538991554f2SKenneth D. Merry */
253967feec50SStephen McConnell if (sc->SSU_started && (scsi_cdb[0] == START_STOP_UNIT)) {
2540991554f2SKenneth D. Merry mpr_dprint(sc, MPR_INFO, "Decrementing SSU count.\n");
2541991554f2SKenneth D. Merry sc->SSU_refcount--;
2542991554f2SKenneth D. Merry }
2543991554f2SKenneth D. Merry
2544991554f2SKenneth D. Merry /* Take the fast path to completion */
2545991554f2SKenneth D. Merry if (cm->cm_reply == NULL) {
2546a2c14879SStephen McConnell if (mprsas_get_ccbstatus(ccb) == CAM_REQ_INPROG) {
2547991554f2SKenneth D. Merry if ((sc->mpr_flags & MPR_FLAGS_DIAGRESET) != 0)
2548a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_SCSI_BUS_RESET);
2549991554f2SKenneth D. Merry else {
2550a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
2551a2c14879SStephen McConnell csio->scsi_status = SCSI_STATUS_OK;
2552991554f2SKenneth D. Merry }
2553991554f2SKenneth D. Merry if (sassc->flags & MPRSAS_QUEUE_FROZEN) {
2554991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_RELEASE_SIMQ;
2555991554f2SKenneth D. Merry sassc->flags &= ~MPRSAS_QUEUE_FROZEN;
2556a8837c77SWarner Losh mpr_dprint(sc, MPR_XINFO | MPR_RECOVERY,
2557991554f2SKenneth D. Merry "Unfreezing SIM queue\n");
2558991554f2SKenneth D. Merry }
2559991554f2SKenneth D. Merry }
2560991554f2SKenneth D. Merry
2561991554f2SKenneth D. Merry /*
2562991554f2SKenneth D. Merry * There are two scenarios where the status won't be
2563991554f2SKenneth D. Merry * CAM_REQ_CMP. The first is if MPR_CM_FLAGS_ERROR_MASK is
2564991554f2SKenneth D. Merry * set, the second is in the MPR_FLAGS_DIAGRESET above.
2565991554f2SKenneth D. Merry */
2566a2c14879SStephen McConnell if (mprsas_get_ccbstatus(ccb) != CAM_REQ_CMP) {
2567991554f2SKenneth D. Merry /*
2568991554f2SKenneth D. Merry * Freeze the dev queue so that commands are
2569a2c14879SStephen McConnell * executed in the correct order after error
2570991554f2SKenneth D. Merry * recovery.
2571991554f2SKenneth D. Merry */
2572991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_DEV_QFRZN;
2573991554f2SKenneth D. Merry xpt_freeze_devq(ccb->ccb_h.path, /*count*/ 1);
2574991554f2SKenneth D. Merry }
2575991554f2SKenneth D. Merry mpr_free_command(sc, cm);
2576991554f2SKenneth D. Merry xpt_done(ccb);
2577991554f2SKenneth D. Merry return;
2578991554f2SKenneth D. Merry }
2579991554f2SKenneth D. Merry
258067feec50SStephen McConnell target = &sassc->targets[target_id];
258167feec50SStephen McConnell if (scsi_cdb[0] == UNMAP &&
258267feec50SStephen McConnell target->is_nvme &&
258367feec50SStephen McConnell (csio->ccb_h.flags & CAM_DATA_MASK) == CAM_DATA_VADDR) {
258467feec50SStephen McConnell rep->SCSIStatus = mprsas_complete_nvme_unmap(sc, cm);
258567feec50SStephen McConnell csio->scsi_status = rep->SCSIStatus;
258667feec50SStephen McConnell }
258767feec50SStephen McConnell
2588991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_XINFO,
2589991554f2SKenneth D. Merry "ioc %x scsi %x state %x xfer %u\n",
2590991554f2SKenneth D. Merry le16toh(rep->IOCStatus), rep->SCSIStatus, rep->SCSIState,
2591991554f2SKenneth D. Merry le32toh(rep->TransferCount));
2592991554f2SKenneth D. Merry
2593991554f2SKenneth D. Merry switch (le16toh(rep->IOCStatus) & MPI2_IOCSTATUS_MASK) {
2594991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_DATA_UNDERRUN:
2595991554f2SKenneth D. Merry csio->resid = cm->cm_length - le32toh(rep->TransferCount);
2596991554f2SKenneth D. Merry /* FALLTHROUGH */
2597991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SUCCESS:
2598991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_RECOVERED_ERROR:
2599991554f2SKenneth D. Merry if ((le16toh(rep->IOCStatus) & MPI2_IOCSTATUS_MASK) ==
2600991554f2SKenneth D. Merry MPI2_IOCSTATUS_SCSI_RECOVERED_ERROR)
2601991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_XINFO, "recovered error\n");
2602991554f2SKenneth D. Merry
2603991554f2SKenneth D. Merry /* Completion failed at the transport level. */
2604991554f2SKenneth D. Merry if (rep->SCSIState & (MPI2_SCSI_STATE_NO_SCSI_STATUS |
2605991554f2SKenneth D. Merry MPI2_SCSI_STATE_TERMINATED)) {
2606a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
2607991554f2SKenneth D. Merry break;
2608991554f2SKenneth D. Merry }
2609991554f2SKenneth D. Merry
2610991554f2SKenneth D. Merry /* In a modern packetized environment, an autosense failure
2611991554f2SKenneth D. Merry * implies that there's not much else that can be done to
2612991554f2SKenneth D. Merry * recover the command.
2613991554f2SKenneth D. Merry */
2614991554f2SKenneth D. Merry if (rep->SCSIState & MPI2_SCSI_STATE_AUTOSENSE_FAILED) {
2615a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_AUTOSENSE_FAIL);
2616991554f2SKenneth D. Merry break;
2617991554f2SKenneth D. Merry }
2618991554f2SKenneth D. Merry
2619991554f2SKenneth D. Merry /*
2620991554f2SKenneth D. Merry * CAM doesn't care about SAS Response Info data, but if this is
2621991554f2SKenneth D. Merry * the state check if TLR should be done. If not, clear the
2622991554f2SKenneth D. Merry * TLR_bits for the target.
2623991554f2SKenneth D. Merry */
2624991554f2SKenneth D. Merry if ((rep->SCSIState & MPI2_SCSI_STATE_RESPONSE_INFO_VALID) &&
2625991554f2SKenneth D. Merry ((le32toh(rep->ResponseInfo) & MPI2_SCSI_RI_MASK_REASONCODE)
2626991554f2SKenneth D. Merry == MPR_SCSI_RI_INVALID_FRAME)) {
2627a2c14879SStephen McConnell sc->mapping_table[target_id].TLR_bits =
2628991554f2SKenneth D. Merry (u8)MPI2_SCSIIO_CONTROL_NO_TLR;
2629991554f2SKenneth D. Merry }
2630991554f2SKenneth D. Merry
2631991554f2SKenneth D. Merry /*
2632991554f2SKenneth D. Merry * Intentionally override the normal SCSI status reporting
2633991554f2SKenneth D. Merry * for these two cases. These are likely to happen in a
2634991554f2SKenneth D. Merry * multi-initiator environment, and we want to make sure that
2635991554f2SKenneth D. Merry * CAM retries these commands rather than fail them.
2636991554f2SKenneth D. Merry */
2637991554f2SKenneth D. Merry if ((rep->SCSIStatus == MPI2_SCSI_STATUS_COMMAND_TERMINATED) ||
2638991554f2SKenneth D. Merry (rep->SCSIStatus == MPI2_SCSI_STATUS_TASK_ABORTED)) {
2639a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_ABORTED);
2640991554f2SKenneth D. Merry break;
2641991554f2SKenneth D. Merry }
2642991554f2SKenneth D. Merry
2643991554f2SKenneth D. Merry /* Handle normal status and sense */
2644991554f2SKenneth D. Merry csio->scsi_status = rep->SCSIStatus;
2645991554f2SKenneth D. Merry if (rep->SCSIStatus == MPI2_SCSI_STATUS_GOOD)
2646a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
2647991554f2SKenneth D. Merry else
2648a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_SCSI_STATUS_ERROR);
2649991554f2SKenneth D. Merry
2650991554f2SKenneth D. Merry if (rep->SCSIState & MPI2_SCSI_STATE_AUTOSENSE_VALID) {
2651991554f2SKenneth D. Merry int sense_len, returned_sense_len;
2652991554f2SKenneth D. Merry
2653991554f2SKenneth D. Merry returned_sense_len = min(le32toh(rep->SenseCount),
2654991554f2SKenneth D. Merry sizeof(struct scsi_sense_data));
2655991554f2SKenneth D. Merry if (returned_sense_len < csio->sense_len)
2656991554f2SKenneth D. Merry csio->sense_resid = csio->sense_len -
2657991554f2SKenneth D. Merry returned_sense_len;
2658991554f2SKenneth D. Merry else
2659991554f2SKenneth D. Merry csio->sense_resid = 0;
2660991554f2SKenneth D. Merry
2661991554f2SKenneth D. Merry sense_len = min(returned_sense_len,
2662991554f2SKenneth D. Merry csio->sense_len - csio->sense_resid);
2663991554f2SKenneth D. Merry bzero(&csio->sense_data, sizeof(csio->sense_data));
2664991554f2SKenneth D. Merry bcopy(cm->cm_sense, &csio->sense_data, sense_len);
2665991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_AUTOSNS_VALID;
2666991554f2SKenneth D. Merry }
2667991554f2SKenneth D. Merry
2668991554f2SKenneth D. Merry /*
2669991554f2SKenneth D. Merry * Check if this is an INQUIRY command. If it's a VPD inquiry,
2670991554f2SKenneth D. Merry * and it's page code 0 (Supported Page List), and there is
2671991554f2SKenneth D. Merry * inquiry data, and this is for a sequential access device, and
2672991554f2SKenneth D. Merry * the device is an SSP target, and TLR is supported by the
2673991554f2SKenneth D. Merry * controller, turn the TLR_bits value ON if page 0x90 is
2674991554f2SKenneth D. Merry * supported.
2675991554f2SKenneth D. Merry */
267667feec50SStephen McConnell if ((scsi_cdb[0] == INQUIRY) &&
267767feec50SStephen McConnell (scsi_cdb[1] & SI_EVPD) &&
267867feec50SStephen McConnell (scsi_cdb[2] == SVPD_SUPPORTED_PAGE_LIST) &&
2679991554f2SKenneth D. Merry ((csio->ccb_h.flags & CAM_DATA_MASK) == CAM_DATA_VADDR) &&
2680d2b4e18bSKenneth D. Merry (csio->data_ptr != NULL) &&
2681d2b4e18bSKenneth D. Merry ((csio->data_ptr[0] & 0x1f) == T_SEQUENTIAL) &&
2682d2b4e18bSKenneth D. Merry (sc->control_TLR) &&
2683a2c14879SStephen McConnell (sc->mapping_table[target_id].device_info &
2684991554f2SKenneth D. Merry MPI2_SAS_DEVICE_INFO_SSP_TARGET)) {
2685991554f2SKenneth D. Merry vpd_list = (struct scsi_vpd_supported_page_list *)
2686991554f2SKenneth D. Merry csio->data_ptr;
2687a2c14879SStephen McConnell TLR_bits = &sc->mapping_table[target_id].TLR_bits;
2688991554f2SKenneth D. Merry *TLR_bits = (u8)MPI2_SCSIIO_CONTROL_NO_TLR;
2689991554f2SKenneth D. Merry TLR_on = (u8)MPI2_SCSIIO_CONTROL_TLR_ON;
269067feec50SStephen McConnell alloc_len = ((u16)scsi_cdb[3] << 8) + scsi_cdb[4];
2691d2b4e18bSKenneth D. Merry alloc_len -= csio->resid;
2692991554f2SKenneth D. Merry for (i = 0; i < MIN(vpd_list->length, alloc_len); i++) {
2693991554f2SKenneth D. Merry if (vpd_list->list[i] == 0x90) {
2694991554f2SKenneth D. Merry *TLR_bits = TLR_on;
2695991554f2SKenneth D. Merry break;
2696991554f2SKenneth D. Merry }
2697991554f2SKenneth D. Merry }
2698991554f2SKenneth D. Merry }
2699a2c14879SStephen McConnell
2700a2c14879SStephen McConnell /*
2701a2c14879SStephen McConnell * If this is a SATA direct-access end device, mark it so that
2702a2c14879SStephen McConnell * a SCSI StartStopUnit command will be sent to it when the
2703a2c14879SStephen McConnell * driver is being shutdown.
2704a2c14879SStephen McConnell */
270567feec50SStephen McConnell if ((scsi_cdb[0] == INQUIRY) &&
2706fa699bb2SAlan Somers (csio->data_ptr != NULL) &&
2707a2c14879SStephen McConnell ((csio->data_ptr[0] & 0x1f) == T_DIRECT) &&
2708a2c14879SStephen McConnell (sc->mapping_table[target_id].device_info &
2709a2c14879SStephen McConnell MPI2_SAS_DEVICE_INFO_SATA_DEVICE) &&
2710a2c14879SStephen McConnell ((sc->mapping_table[target_id].device_info &
2711a2c14879SStephen McConnell MPI2_SAS_DEVICE_INFO_MASK_DEVICE_TYPE) ==
2712a2c14879SStephen McConnell MPI2_SAS_DEVICE_INFO_END_DEVICE)) {
2713a2c14879SStephen McConnell target = &sassc->targets[target_id];
2714a2c14879SStephen McConnell target->supports_SSU = TRUE;
2715a2c14879SStephen McConnell mpr_dprint(sc, MPR_XINFO, "Target %d supports SSU\n",
2716a2c14879SStephen McConnell target_id);
2717a2c14879SStephen McConnell }
2718991554f2SKenneth D. Merry break;
2719991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_INVALID_DEVHANDLE:
2720991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_DEVICE_NOT_THERE:
2721991554f2SKenneth D. Merry /*
2722991554f2SKenneth D. Merry * If devinfo is 0 this will be a volume. In that case don't
2723991554f2SKenneth D. Merry * tell CAM that the volume is not there. We want volumes to
2724991554f2SKenneth D. Merry * be enumerated until they are deleted/removed, not just
2725991554f2SKenneth D. Merry * failed.
2726991554f2SKenneth D. Merry */
2727991554f2SKenneth D. Merry if (cm->cm_targ->devinfo == 0)
2728a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
2729991554f2SKenneth D. Merry else
2730a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
2731991554f2SKenneth D. Merry break;
2732991554f2SKenneth D. Merry case MPI2_IOCSTATUS_INVALID_SGL:
2733991554f2SKenneth D. Merry mpr_print_scsiio_cmd(sc, cm);
2734a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_UNREC_HBA_ERROR);
2735991554f2SKenneth D. Merry break;
2736991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_TASK_TERMINATED:
2737991554f2SKenneth D. Merry /*
2738991554f2SKenneth D. Merry * This is one of the responses that comes back when an I/O
2739991554f2SKenneth D. Merry * has been aborted. If it is because of a timeout that we
2740991554f2SKenneth D. Merry * initiated, just set the status to CAM_CMD_TIMEOUT.
2741991554f2SKenneth D. Merry * Otherwise set it to CAM_REQ_ABORTED. The effect on the
2742991554f2SKenneth D. Merry * command is the same (it gets retried, subject to the
2743991554f2SKenneth D. Merry * retry counter), the only difference is what gets printed
2744991554f2SKenneth D. Merry * on the console.
2745991554f2SKenneth D. Merry */
27468fe7bf06SWarner Losh if (cm->cm_flags & MPR_CM_FLAGS_TIMEDOUT)
2747a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_CMD_TIMEOUT);
2748991554f2SKenneth D. Merry else
2749a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_ABORTED);
2750991554f2SKenneth D. Merry break;
2751991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_DATA_OVERRUN:
2752991554f2SKenneth D. Merry /* resid is ignored for this condition */
2753991554f2SKenneth D. Merry csio->resid = 0;
2754a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DATA_RUN_ERR);
2755991554f2SKenneth D. Merry break;
2756991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_IOC_TERMINATED:
2757991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_EXT_TERMINATED:
2758991554f2SKenneth D. Merry /*
27596adfa7edSAlan Somers * These can sometimes be transient transport-related
27606adfa7edSAlan Somers * errors, and sometimes persistent drive-related errors.
27616adfa7edSAlan Somers * We used to retry these without decrementing the retry
27626adfa7edSAlan Somers * count by returning CAM_REQUEUE_REQ. Unfortunately, if
27636adfa7edSAlan Somers * we hit a persistent drive problem that returns one of
27646adfa7edSAlan Somers * these error codes, we would retry indefinitely. So,
276559dc489aSWarner Losh * return CAM_REQ_CMP_ERR so that we decrement the retry
27666adfa7edSAlan Somers * count and avoid infinite retries. We're taking the
27676adfa7edSAlan Somers * potential risk of flagging false failures in the event
27686adfa7edSAlan Somers * of a topology-related error (e.g. a SAS expander problem
27696adfa7edSAlan Somers * causes a command addressed to a drive to fail), but
27704c1cdd4aSWarner Losh * avoiding getting into an infinite retry loop. However,
277159dc489aSWarner Losh * if we get them while were removing a device, we should
27724c1cdd4aSWarner Losh * fail the request as 'not there' because the device
27734c1cdd4aSWarner Losh * is effectively gone.
2774991554f2SKenneth D. Merry */
27754c1cdd4aSWarner Losh if (cm->cm_targ->flags & MPRSAS_TARGET_INREMOVAL)
27764c1cdd4aSWarner Losh mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
27774c1cdd4aSWarner Losh else
27786adfa7edSAlan Somers mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
27796eea4f46SScott Long mpr_dprint(sc, MPR_INFO,
27804c1cdd4aSWarner Losh "Controller reported %s tgt %u SMID %u loginfo %x%s\n",
27812bf620cbSScott Long mpr_describe_table(mpr_iocstatus_string,
27822bf620cbSScott Long le16toh(rep->IOCStatus) & MPI2_IOCSTATUS_MASK),
27832bf620cbSScott Long target_id, cm->cm_desc.Default.SMID,
27844c1cdd4aSWarner Losh le32toh(rep->IOCLogInfo),
27854c1cdd4aSWarner Losh (cm->cm_targ->flags & MPRSAS_TARGET_INREMOVAL) ? " departing" : "");
27866eea4f46SScott Long mpr_dprint(sc, MPR_XINFO,
27876eea4f46SScott Long "SCSIStatus %x SCSIState %x xfercount %u\n",
2788694cb8b8SScott Long rep->SCSIStatus, rep->SCSIState,
2789991554f2SKenneth D. Merry le32toh(rep->TransferCount));
2790991554f2SKenneth D. Merry break;
2791991554f2SKenneth D. Merry case MPI2_IOCSTATUS_INVALID_FUNCTION:
2792991554f2SKenneth D. Merry case MPI2_IOCSTATUS_INTERNAL_ERROR:
2793991554f2SKenneth D. Merry case MPI2_IOCSTATUS_INVALID_VPID:
2794991554f2SKenneth D. Merry case MPI2_IOCSTATUS_INVALID_FIELD:
2795991554f2SKenneth D. Merry case MPI2_IOCSTATUS_INVALID_STATE:
2796991554f2SKenneth D. Merry case MPI2_IOCSTATUS_OP_STATE_NOT_SUPPORTED:
2797991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_IO_DATA_ERROR:
2798991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_PROTOCOL_ERROR:
2799991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_RESIDUAL_MISMATCH:
2800991554f2SKenneth D. Merry case MPI2_IOCSTATUS_SCSI_TASK_MGMT_FAILED:
2801991554f2SKenneth D. Merry default:
2802991554f2SKenneth D. Merry mprsas_log_command(cm, MPR_XINFO,
2803694cb8b8SScott Long "completed ioc %x loginfo %x scsi %x state %x xfer %u\n",
2804694cb8b8SScott Long le16toh(rep->IOCStatus), le32toh(rep->IOCLogInfo),
2805694cb8b8SScott Long rep->SCSIStatus, rep->SCSIState,
2806991554f2SKenneth D. Merry le32toh(rep->TransferCount));
2807991554f2SKenneth D. Merry csio->resid = cm->cm_length;
280867feec50SStephen McConnell
280967feec50SStephen McConnell if (scsi_cdb[0] == UNMAP &&
281067feec50SStephen McConnell target->is_nvme &&
281167feec50SStephen McConnell (csio->ccb_h.flags & CAM_DATA_MASK) == CAM_DATA_VADDR)
281267feec50SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
281367feec50SStephen McConnell else
2814a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
281567feec50SStephen McConnell
2816991554f2SKenneth D. Merry break;
2817991554f2SKenneth D. Merry }
2818991554f2SKenneth D. Merry
2819991554f2SKenneth D. Merry mpr_sc_failed_io_info(sc, csio, rep, cm->cm_targ);
2820991554f2SKenneth D. Merry
2821991554f2SKenneth D. Merry if (sassc->flags & MPRSAS_QUEUE_FROZEN) {
2822991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_RELEASE_SIMQ;
2823991554f2SKenneth D. Merry sassc->flags &= ~MPRSAS_QUEUE_FROZEN;
2824a8837c77SWarner Losh mpr_dprint(sc, MPR_INFO, "Command completed, unfreezing SIM "
2825991554f2SKenneth D. Merry "queue\n");
2826991554f2SKenneth D. Merry }
2827991554f2SKenneth D. Merry
2828a2c14879SStephen McConnell if (mprsas_get_ccbstatus(ccb) != CAM_REQ_CMP) {
2829991554f2SKenneth D. Merry ccb->ccb_h.status |= CAM_DEV_QFRZN;
2830991554f2SKenneth D. Merry xpt_freeze_devq(ccb->ccb_h.path, /*count*/ 1);
2831991554f2SKenneth D. Merry }
2832991554f2SKenneth D. Merry
28334c1cdd4aSWarner Losh /*
28344c1cdd4aSWarner Losh * Check to see if we're removing the device. If so, and this is the
28354c1cdd4aSWarner Losh * last command on the queue, proceed with the deferred removal of the
28364c1cdd4aSWarner Losh * device. Note, for removing a volume, this won't trigger because
28374c1cdd4aSWarner Losh * pending_remove_tm will be NULL.
28384c1cdd4aSWarner Losh */
28394c1cdd4aSWarner Losh if (cm->cm_targ->flags & MPRSAS_TARGET_INREMOVAL) {
28404c1cdd4aSWarner Losh if (TAILQ_FIRST(&cm->cm_targ->commands) == NULL &&
28414c1cdd4aSWarner Losh cm->cm_targ->pending_remove_tm != NULL) {
2842ca420b4eSWarner Losh mpr_dprint(sc, MPR_INFO,
2843ca420b4eSWarner Losh "Last pending command complete: starting remove_device target %u handle 0x%04x\n",
2844ca420b4eSWarner Losh cm->cm_targ->tid, cm->cm_targ->handle);
28454c1cdd4aSWarner Losh mpr_map_command(sc, cm->cm_targ->pending_remove_tm);
28464c1cdd4aSWarner Losh cm->cm_targ->pending_remove_tm = NULL;
28474c1cdd4aSWarner Losh }
28484c1cdd4aSWarner Losh }
28494c1cdd4aSWarner Losh
2850991554f2SKenneth D. Merry mpr_free_command(sc, cm);
2851991554f2SKenneth D. Merry xpt_done(ccb);
2852991554f2SKenneth D. Merry }
2853991554f2SKenneth D. Merry
2854991554f2SKenneth D. Merry static void
mprsas_smpio_complete(struct mpr_softc * sc,struct mpr_command * cm)2855991554f2SKenneth D. Merry mprsas_smpio_complete(struct mpr_softc *sc, struct mpr_command *cm)
2856991554f2SKenneth D. Merry {
2857991554f2SKenneth D. Merry MPI2_SMP_PASSTHROUGH_REPLY *rpl;
2858991554f2SKenneth D. Merry MPI2_SMP_PASSTHROUGH_REQUEST *req;
2859991554f2SKenneth D. Merry uint64_t sasaddr;
2860991554f2SKenneth D. Merry union ccb *ccb;
2861991554f2SKenneth D. Merry
2862991554f2SKenneth D. Merry ccb = cm->cm_complete_data;
2863991554f2SKenneth D. Merry
2864991554f2SKenneth D. Merry /*
2865991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
2866991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and SMP
2867991554f2SKenneth D. Merry * commands require two S/G elements only. That should be handled
2868991554f2SKenneth D. Merry * in the standard request size.
2869991554f2SKenneth D. Merry */
2870991554f2SKenneth D. Merry if ((cm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
2871a2c14879SStephen McConnell mpr_dprint(sc, MPR_ERROR, "%s: cm_flags = %#x on SMP "
2872a2c14879SStephen McConnell "request!\n", __func__, cm->cm_flags);
2873a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
2874991554f2SKenneth D. Merry goto bailout;
2875991554f2SKenneth D. Merry }
2876991554f2SKenneth D. Merry
2877991554f2SKenneth D. Merry rpl = (MPI2_SMP_PASSTHROUGH_REPLY *)cm->cm_reply;
2878991554f2SKenneth D. Merry if (rpl == NULL) {
2879991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: NULL cm_reply!\n", __func__);
2880a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
2881991554f2SKenneth D. Merry goto bailout;
2882991554f2SKenneth D. Merry }
2883991554f2SKenneth D. Merry
2884991554f2SKenneth D. Merry req = (MPI2_SMP_PASSTHROUGH_REQUEST *)cm->cm_req;
2885991554f2SKenneth D. Merry sasaddr = le32toh(req->SASAddress.Low);
2886991554f2SKenneth D. Merry sasaddr |= ((uint64_t)(le32toh(req->SASAddress.High))) << 32;
2887991554f2SKenneth D. Merry
2888991554f2SKenneth D. Merry if ((le16toh(rpl->IOCStatus) & MPI2_IOCSTATUS_MASK) !=
2889991554f2SKenneth D. Merry MPI2_IOCSTATUS_SUCCESS ||
2890991554f2SKenneth D. Merry rpl->SASStatus != MPI2_SASSTATUS_SUCCESS) {
2891991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "%s: IOCStatus %04x SASStatus %02x\n",
2892991554f2SKenneth D. Merry __func__, le16toh(rpl->IOCStatus), rpl->SASStatus);
2893a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
2894991554f2SKenneth D. Merry goto bailout;
2895991554f2SKenneth D. Merry }
2896991554f2SKenneth D. Merry
2897a2c14879SStephen McConnell mpr_dprint(sc, MPR_XINFO, "%s: SMP request to SAS address %#jx "
2898a2c14879SStephen McConnell "completed successfully\n", __func__, (uintmax_t)sasaddr);
2899991554f2SKenneth D. Merry
2900991554f2SKenneth D. Merry if (ccb->smpio.smp_response[2] == SMP_FR_ACCEPTED)
2901a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
2902991554f2SKenneth D. Merry else
2903a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_SMP_STATUS_ERROR);
2904991554f2SKenneth D. Merry
2905991554f2SKenneth D. Merry bailout:
2906991554f2SKenneth D. Merry /*
2907991554f2SKenneth D. Merry * We sync in both directions because we had DMAs in the S/G list
2908991554f2SKenneth D. Merry * in both directions.
2909991554f2SKenneth D. Merry */
2910991554f2SKenneth D. Merry bus_dmamap_sync(sc->buffer_dmat, cm->cm_dmamap,
2911991554f2SKenneth D. Merry BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
2912991554f2SKenneth D. Merry bus_dmamap_unload(sc->buffer_dmat, cm->cm_dmamap);
2913991554f2SKenneth D. Merry mpr_free_command(sc, cm);
2914991554f2SKenneth D. Merry xpt_done(ccb);
2915991554f2SKenneth D. Merry }
2916991554f2SKenneth D. Merry
2917991554f2SKenneth D. Merry static void
mprsas_send_smpcmd(struct mprsas_softc * sassc,union ccb * ccb,uint64_t sasaddr)29187a2a6a1aSStephen McConnell mprsas_send_smpcmd(struct mprsas_softc *sassc, union ccb *ccb, uint64_t sasaddr)
2919991554f2SKenneth D. Merry {
2920991554f2SKenneth D. Merry struct mpr_command *cm;
2921991554f2SKenneth D. Merry uint8_t *request, *response;
2922991554f2SKenneth D. Merry MPI2_SMP_PASSTHROUGH_REQUEST *req;
2923991554f2SKenneth D. Merry struct mpr_softc *sc;
2924991554f2SKenneth D. Merry int error;
2925991554f2SKenneth D. Merry
2926991554f2SKenneth D. Merry sc = sassc->sc;
2927991554f2SKenneth D. Merry error = 0;
2928991554f2SKenneth D. Merry
2929991554f2SKenneth D. Merry switch (ccb->ccb_h.flags & CAM_DATA_MASK) {
2930991554f2SKenneth D. Merry case CAM_DATA_PADDR:
2931991554f2SKenneth D. Merry case CAM_DATA_SG_PADDR:
2932991554f2SKenneth D. Merry /*
2933991554f2SKenneth D. Merry * XXX We don't yet support physical addresses here.
2934991554f2SKenneth D. Merry */
2935991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: physical addresses not "
2936991554f2SKenneth D. Merry "supported\n", __func__);
2937a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_INVALID);
2938991554f2SKenneth D. Merry xpt_done(ccb);
2939991554f2SKenneth D. Merry return;
2940991554f2SKenneth D. Merry case CAM_DATA_SG:
2941991554f2SKenneth D. Merry /*
2942991554f2SKenneth D. Merry * The chip does not support more than one buffer for the
2943991554f2SKenneth D. Merry * request or response.
2944991554f2SKenneth D. Merry */
2945991554f2SKenneth D. Merry if ((ccb->smpio.smp_request_sglist_cnt > 1)
2946991554f2SKenneth D. Merry || (ccb->smpio.smp_response_sglist_cnt > 1)) {
29477a2a6a1aSStephen McConnell mpr_dprint(sc, MPR_ERROR, "%s: multiple request or "
29487a2a6a1aSStephen McConnell "response buffer segments not supported for SMP\n",
29497a2a6a1aSStephen McConnell __func__);
2950a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_INVALID);
2951991554f2SKenneth D. Merry xpt_done(ccb);
2952991554f2SKenneth D. Merry return;
2953991554f2SKenneth D. Merry }
2954991554f2SKenneth D. Merry
2955991554f2SKenneth D. Merry /*
2956991554f2SKenneth D. Merry * The CAM_SCATTER_VALID flag was originally implemented
2957991554f2SKenneth D. Merry * for the XPT_SCSI_IO CCB, which only has one data pointer.
2958991554f2SKenneth D. Merry * We have two. So, just take that flag to mean that we
2959991554f2SKenneth D. Merry * might have S/G lists, and look at the S/G segment count
2960991554f2SKenneth D. Merry * to figure out whether that is the case for each individual
2961991554f2SKenneth D. Merry * buffer.
2962991554f2SKenneth D. Merry */
2963991554f2SKenneth D. Merry if (ccb->smpio.smp_request_sglist_cnt != 0) {
2964991554f2SKenneth D. Merry bus_dma_segment_t *req_sg;
2965991554f2SKenneth D. Merry
2966991554f2SKenneth D. Merry req_sg = (bus_dma_segment_t *)ccb->smpio.smp_request;
2967991554f2SKenneth D. Merry request = (uint8_t *)(uintptr_t)req_sg[0].ds_addr;
2968991554f2SKenneth D. Merry } else
2969991554f2SKenneth D. Merry request = ccb->smpio.smp_request;
2970991554f2SKenneth D. Merry
2971991554f2SKenneth D. Merry if (ccb->smpio.smp_response_sglist_cnt != 0) {
2972991554f2SKenneth D. Merry bus_dma_segment_t *rsp_sg;
2973991554f2SKenneth D. Merry
2974991554f2SKenneth D. Merry rsp_sg = (bus_dma_segment_t *)ccb->smpio.smp_response;
2975991554f2SKenneth D. Merry response = (uint8_t *)(uintptr_t)rsp_sg[0].ds_addr;
2976991554f2SKenneth D. Merry } else
2977991554f2SKenneth D. Merry response = ccb->smpio.smp_response;
2978991554f2SKenneth D. Merry break;
2979991554f2SKenneth D. Merry case CAM_DATA_VADDR:
2980991554f2SKenneth D. Merry request = ccb->smpio.smp_request;
2981991554f2SKenneth D. Merry response = ccb->smpio.smp_response;
2982991554f2SKenneth D. Merry break;
2983991554f2SKenneth D. Merry default:
2984a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_INVALID);
2985991554f2SKenneth D. Merry xpt_done(ccb);
2986991554f2SKenneth D. Merry return;
2987991554f2SKenneth D. Merry }
2988991554f2SKenneth D. Merry
2989991554f2SKenneth D. Merry cm = mpr_alloc_command(sc);
2990991554f2SKenneth D. Merry if (cm == NULL) {
29917a2a6a1aSStephen McConnell mpr_dprint(sc, MPR_ERROR, "%s: cannot allocate command\n",
29927a2a6a1aSStephen McConnell __func__);
2993a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_RESRC_UNAVAIL);
2994991554f2SKenneth D. Merry xpt_done(ccb);
2995991554f2SKenneth D. Merry return;
2996991554f2SKenneth D. Merry }
2997991554f2SKenneth D. Merry
2998991554f2SKenneth D. Merry req = (MPI2_SMP_PASSTHROUGH_REQUEST *)cm->cm_req;
2999991554f2SKenneth D. Merry bzero(req, sizeof(*req));
3000991554f2SKenneth D. Merry req->Function = MPI2_FUNCTION_SMP_PASSTHROUGH;
3001991554f2SKenneth D. Merry
3002991554f2SKenneth D. Merry /* Allow the chip to use any route to this SAS address. */
3003991554f2SKenneth D. Merry req->PhysicalPort = 0xff;
3004991554f2SKenneth D. Merry
3005991554f2SKenneth D. Merry req->RequestDataLength = htole16(ccb->smpio.smp_request_len);
3006991554f2SKenneth D. Merry req->SGLFlags =
3007991554f2SKenneth D. Merry MPI2_SGLFLAGS_SYSTEM_ADDRESS_SPACE | MPI2_SGLFLAGS_SGL_TYPE_MPI;
3008991554f2SKenneth D. Merry
3009991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO, "%s: sending SMP request to SAS address "
3010991554f2SKenneth D. Merry "%#jx\n", __func__, (uintmax_t)sasaddr);
3011991554f2SKenneth D. Merry
3012991554f2SKenneth D. Merry mpr_init_sge(cm, req, &req->SGL);
3013991554f2SKenneth D. Merry
3014991554f2SKenneth D. Merry /*
3015991554f2SKenneth D. Merry * Set up a uio to pass into mpr_map_command(). This allows us to
3016991554f2SKenneth D. Merry * do one map command, and one busdma call in there.
3017991554f2SKenneth D. Merry */
3018991554f2SKenneth D. Merry cm->cm_uio.uio_iov = cm->cm_iovec;
3019991554f2SKenneth D. Merry cm->cm_uio.uio_iovcnt = 2;
3020991554f2SKenneth D. Merry cm->cm_uio.uio_segflg = UIO_SYSSPACE;
3021991554f2SKenneth D. Merry
3022991554f2SKenneth D. Merry /*
3023991554f2SKenneth D. Merry * The read/write flag isn't used by busdma, but set it just in
3024991554f2SKenneth D. Merry * case. This isn't exactly accurate, either, since we're going in
3025991554f2SKenneth D. Merry * both directions.
3026991554f2SKenneth D. Merry */
3027991554f2SKenneth D. Merry cm->cm_uio.uio_rw = UIO_WRITE;
3028991554f2SKenneth D. Merry
3029991554f2SKenneth D. Merry cm->cm_iovec[0].iov_base = request;
3030991554f2SKenneth D. Merry cm->cm_iovec[0].iov_len = le16toh(req->RequestDataLength);
3031991554f2SKenneth D. Merry cm->cm_iovec[1].iov_base = response;
3032991554f2SKenneth D. Merry cm->cm_iovec[1].iov_len = ccb->smpio.smp_response_len;
3033991554f2SKenneth D. Merry
3034991554f2SKenneth D. Merry cm->cm_uio.uio_resid = cm->cm_iovec[0].iov_len +
3035991554f2SKenneth D. Merry cm->cm_iovec[1].iov_len;
3036991554f2SKenneth D. Merry
3037991554f2SKenneth D. Merry /*
3038991554f2SKenneth D. Merry * Trigger a warning message in mpr_data_cb() for the user if we
3039991554f2SKenneth D. Merry * wind up exceeding two S/G segments. The chip expects one
3040991554f2SKenneth D. Merry * segment for the request and another for the response.
3041991554f2SKenneth D. Merry */
3042991554f2SKenneth D. Merry cm->cm_max_segs = 2;
3043991554f2SKenneth D. Merry
3044991554f2SKenneth D. Merry cm->cm_desc.Default.RequestFlags = MPI2_REQ_DESCRIPT_FLAGS_DEFAULT_TYPE;
3045991554f2SKenneth D. Merry cm->cm_complete = mprsas_smpio_complete;
3046991554f2SKenneth D. Merry cm->cm_complete_data = ccb;
3047991554f2SKenneth D. Merry
3048991554f2SKenneth D. Merry /*
3049991554f2SKenneth D. Merry * Tell the mapping code that we're using a uio, and that this is
3050991554f2SKenneth D. Merry * an SMP passthrough request. There is a little special-case
3051991554f2SKenneth D. Merry * logic there (in mpr_data_cb()) to handle the bidirectional
3052991554f2SKenneth D. Merry * transfer.
3053991554f2SKenneth D. Merry */
3054991554f2SKenneth D. Merry cm->cm_flags |= MPR_CM_FLAGS_USE_UIO | MPR_CM_FLAGS_SMP_PASS |
3055991554f2SKenneth D. Merry MPR_CM_FLAGS_DATAIN | MPR_CM_FLAGS_DATAOUT;
3056991554f2SKenneth D. Merry
3057991554f2SKenneth D. Merry /* The chip data format is little endian. */
3058991554f2SKenneth D. Merry req->SASAddress.High = htole32(sasaddr >> 32);
3059991554f2SKenneth D. Merry req->SASAddress.Low = htole32(sasaddr);
3060991554f2SKenneth D. Merry
3061991554f2SKenneth D. Merry /*
3062991554f2SKenneth D. Merry * XXX Note that we don't have a timeout/abort mechanism here.
3063991554f2SKenneth D. Merry * From the manual, it looks like task management requests only
3064991554f2SKenneth D. Merry * work for SCSI IO and SATA passthrough requests. We may need to
3065991554f2SKenneth D. Merry * have a mechanism to retry requests in the event of a chip reset
3066991554f2SKenneth D. Merry * at least. Hopefully the chip will insure that any errors short
3067991554f2SKenneth D. Merry * of that are relayed back to the driver.
3068991554f2SKenneth D. Merry */
3069991554f2SKenneth D. Merry error = mpr_map_command(sc, cm);
3070991554f2SKenneth D. Merry if ((error != 0) && (error != EINPROGRESS)) {
3071991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: error %d returned from "
3072991554f2SKenneth D. Merry "mpr_map_command()\n", __func__, error);
3073991554f2SKenneth D. Merry goto bailout_error;
3074991554f2SKenneth D. Merry }
3075991554f2SKenneth D. Merry
3076991554f2SKenneth D. Merry return;
3077991554f2SKenneth D. Merry
3078991554f2SKenneth D. Merry bailout_error:
3079991554f2SKenneth D. Merry mpr_free_command(sc, cm);
3080a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_RESRC_UNAVAIL);
3081991554f2SKenneth D. Merry xpt_done(ccb);
3082991554f2SKenneth D. Merry return;
3083991554f2SKenneth D. Merry }
3084991554f2SKenneth D. Merry
3085991554f2SKenneth D. Merry static void
mprsas_action_smpio(struct mprsas_softc * sassc,union ccb * ccb)3086991554f2SKenneth D. Merry mprsas_action_smpio(struct mprsas_softc *sassc, union ccb *ccb)
3087991554f2SKenneth D. Merry {
3088991554f2SKenneth D. Merry struct mpr_softc *sc;
3089991554f2SKenneth D. Merry struct mprsas_target *targ;
3090991554f2SKenneth D. Merry uint64_t sasaddr = 0;
3091991554f2SKenneth D. Merry
3092991554f2SKenneth D. Merry sc = sassc->sc;
3093991554f2SKenneth D. Merry
3094991554f2SKenneth D. Merry /*
3095991554f2SKenneth D. Merry * Make sure the target exists.
3096991554f2SKenneth D. Merry */
3097991554f2SKenneth D. Merry KASSERT(ccb->ccb_h.target_id < sassc->maxtargets,
3098991554f2SKenneth D. Merry ("Target %d out of bounds in XPT_SMP_IO\n", ccb->ccb_h.target_id));
3099991554f2SKenneth D. Merry targ = &sassc->targets[ccb->ccb_h.target_id];
3100991554f2SKenneth D. Merry if (targ->handle == 0x0) {
3101991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: target %d does not exist!\n",
3102991554f2SKenneth D. Merry __func__, ccb->ccb_h.target_id);
3103a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_SEL_TIMEOUT);
3104991554f2SKenneth D. Merry xpt_done(ccb);
3105991554f2SKenneth D. Merry return;
3106991554f2SKenneth D. Merry }
3107991554f2SKenneth D. Merry
3108991554f2SKenneth D. Merry /*
3109991554f2SKenneth D. Merry * If this device has an embedded SMP target, we'll talk to it
3110991554f2SKenneth D. Merry * directly.
3111991554f2SKenneth D. Merry * figure out what the expander's address is.
3112991554f2SKenneth D. Merry */
3113991554f2SKenneth D. Merry if ((targ->devinfo & MPI2_SAS_DEVICE_INFO_SMP_TARGET) != 0)
3114991554f2SKenneth D. Merry sasaddr = targ->sasaddr;
3115991554f2SKenneth D. Merry
3116991554f2SKenneth D. Merry /*
3117991554f2SKenneth D. Merry * If we don't have a SAS address for the expander yet, try
3118991554f2SKenneth D. Merry * grabbing it from the page 0x83 information cached in the
3119991554f2SKenneth D. Merry * transport layer for this target. LSI expanders report the
3120991554f2SKenneth D. Merry * expander SAS address as the port-associated SAS address in
3121991554f2SKenneth D. Merry * Inquiry VPD page 0x83. Maxim expanders don't report it in page
3122991554f2SKenneth D. Merry * 0x83.
3123991554f2SKenneth D. Merry *
3124991554f2SKenneth D. Merry * XXX KDM disable this for now, but leave it commented out so that
3125991554f2SKenneth D. Merry * it is obvious that this is another possible way to get the SAS
3126991554f2SKenneth D. Merry * address.
3127991554f2SKenneth D. Merry *
3128991554f2SKenneth D. Merry * The parent handle method below is a little more reliable, and
3129991554f2SKenneth D. Merry * the other benefit is that it works for devices other than SES
3130991554f2SKenneth D. Merry * devices. So you can send a SMP request to a da(4) device and it
3131991554f2SKenneth D. Merry * will get routed to the expander that device is attached to.
3132991554f2SKenneth D. Merry * (Assuming the da(4) device doesn't contain an SMP target...)
3133991554f2SKenneth D. Merry */
3134991554f2SKenneth D. Merry #if 0
3135991554f2SKenneth D. Merry if (sasaddr == 0)
3136991554f2SKenneth D. Merry sasaddr = xpt_path_sas_addr(ccb->ccb_h.path);
3137991554f2SKenneth D. Merry #endif
3138991554f2SKenneth D. Merry
3139991554f2SKenneth D. Merry /*
3140991554f2SKenneth D. Merry * If we still don't have a SAS address for the expander, look for
3141991554f2SKenneth D. Merry * the parent device of this device, which is probably the expander.
3142991554f2SKenneth D. Merry */
3143991554f2SKenneth D. Merry if (sasaddr == 0) {
3144991554f2SKenneth D. Merry #ifdef OLD_MPR_PROBE
3145991554f2SKenneth D. Merry struct mprsas_target *parent_target;
3146991554f2SKenneth D. Merry #endif
3147991554f2SKenneth D. Merry
3148991554f2SKenneth D. Merry if (targ->parent_handle == 0x0) {
3149991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: handle %d does not have "
3150991554f2SKenneth D. Merry "a valid parent handle!\n", __func__, targ->handle);
3151a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
3152991554f2SKenneth D. Merry goto bailout;
3153991554f2SKenneth D. Merry }
3154991554f2SKenneth D. Merry #ifdef OLD_MPR_PROBE
3155991554f2SKenneth D. Merry parent_target = mprsas_find_target_by_handle(sassc, 0,
3156991554f2SKenneth D. Merry targ->parent_handle);
3157991554f2SKenneth D. Merry
3158991554f2SKenneth D. Merry if (parent_target == NULL) {
3159991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: handle %d does not have "
3160991554f2SKenneth D. Merry "a valid parent target!\n", __func__, targ->handle);
3161a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
3162991554f2SKenneth D. Merry goto bailout;
3163991554f2SKenneth D. Merry }
3164991554f2SKenneth D. Merry
3165991554f2SKenneth D. Merry if ((parent_target->devinfo &
3166991554f2SKenneth D. Merry MPI2_SAS_DEVICE_INFO_SMP_TARGET) == 0) {
3167991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: handle %d parent %d "
3168991554f2SKenneth D. Merry "does not have an SMP target!\n", __func__,
3169991554f2SKenneth D. Merry targ->handle, parent_target->handle);
3170a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
3171991554f2SKenneth D. Merry goto bailout;
3172991554f2SKenneth D. Merry }
3173991554f2SKenneth D. Merry
3174991554f2SKenneth D. Merry sasaddr = parent_target->sasaddr;
3175991554f2SKenneth D. Merry #else /* OLD_MPR_PROBE */
3176991554f2SKenneth D. Merry if ((targ->parent_devinfo &
3177991554f2SKenneth D. Merry MPI2_SAS_DEVICE_INFO_SMP_TARGET) == 0) {
3178991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: handle %d parent %d "
3179991554f2SKenneth D. Merry "does not have an SMP target!\n", __func__,
3180991554f2SKenneth D. Merry targ->handle, targ->parent_handle);
3181a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
3182991554f2SKenneth D. Merry goto bailout;
3183991554f2SKenneth D. Merry }
3184991554f2SKenneth D. Merry if (targ->parent_sasaddr == 0x0) {
3185991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: handle %d parent handle "
3186991554f2SKenneth D. Merry "%d does not have a valid SAS address!\n", __func__,
3187991554f2SKenneth D. Merry targ->handle, targ->parent_handle);
3188a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
3189991554f2SKenneth D. Merry goto bailout;
3190991554f2SKenneth D. Merry }
3191991554f2SKenneth D. Merry
3192991554f2SKenneth D. Merry sasaddr = targ->parent_sasaddr;
3193991554f2SKenneth D. Merry #endif /* OLD_MPR_PROBE */
3194991554f2SKenneth D. Merry }
3195991554f2SKenneth D. Merry
3196991554f2SKenneth D. Merry if (sasaddr == 0) {
3197991554f2SKenneth D. Merry mpr_dprint(sc, MPR_INFO, "%s: unable to find SAS address for "
3198991554f2SKenneth D. Merry "handle %d\n", __func__, targ->handle);
3199a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_DEV_NOT_THERE);
3200991554f2SKenneth D. Merry goto bailout;
3201991554f2SKenneth D. Merry }
3202991554f2SKenneth D. Merry mprsas_send_smpcmd(sassc, ccb, sasaddr);
3203991554f2SKenneth D. Merry
3204991554f2SKenneth D. Merry return;
3205991554f2SKenneth D. Merry
3206991554f2SKenneth D. Merry bailout:
3207991554f2SKenneth D. Merry xpt_done(ccb);
3208991554f2SKenneth D. Merry
3209991554f2SKenneth D. Merry }
3210991554f2SKenneth D. Merry
3211991554f2SKenneth D. Merry static void
mprsas_action_resetdev(struct mprsas_softc * sassc,union ccb * ccb)3212991554f2SKenneth D. Merry mprsas_action_resetdev(struct mprsas_softc *sassc, union ccb *ccb)
3213991554f2SKenneth D. Merry {
3214991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
3215991554f2SKenneth D. Merry struct mpr_softc *sc;
3216991554f2SKenneth D. Merry struct mpr_command *tm;
3217991554f2SKenneth D. Merry struct mprsas_target *targ;
3218991554f2SKenneth D. Merry
3219991554f2SKenneth D. Merry MPR_FUNCTRACE(sassc->sc);
3220991554f2SKenneth D. Merry mtx_assert(&sassc->sc->mpr_mtx, MA_OWNED);
3221991554f2SKenneth D. Merry
32227a2a6a1aSStephen McConnell KASSERT(ccb->ccb_h.target_id < sassc->maxtargets, ("Target %d out of "
32237a2a6a1aSStephen McConnell "bounds in XPT_RESET_DEV\n", ccb->ccb_h.target_id));
3224991554f2SKenneth D. Merry sc = sassc->sc;
32253921a9f7SScott Long tm = mprsas_alloc_tm(sc);
3226991554f2SKenneth D. Merry if (tm == NULL) {
32277a2a6a1aSStephen McConnell mpr_dprint(sc, MPR_ERROR, "command alloc failure in "
32287a2a6a1aSStephen McConnell "mprsas_action_resetdev\n");
3229a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_RESRC_UNAVAIL);
3230991554f2SKenneth D. Merry xpt_done(ccb);
3231991554f2SKenneth D. Merry return;
3232991554f2SKenneth D. Merry }
3233991554f2SKenneth D. Merry
3234991554f2SKenneth D. Merry targ = &sassc->targets[ccb->ccb_h.target_id];
3235991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
3236991554f2SKenneth D. Merry req->DevHandle = htole16(targ->handle);
3237991554f2SKenneth D. Merry req->TaskType = MPI2_SCSITASKMGMT_TASKTYPE_TARGET_RESET;
3238991554f2SKenneth D. Merry
323989d1c21fSKashyap D Desai if (!targ->is_nvme || sc->custom_nvme_tm_handling) {
3240991554f2SKenneth D. Merry /* SAS Hard Link Reset / SATA Link Reset */
3241991554f2SKenneth D. Merry req->MsgFlags = MPI2_SCSITASKMGMT_MSGFLAGS_LINK_RESET;
324289d1c21fSKashyap D Desai } else {
324389d1c21fSKashyap D Desai /* PCIe Protocol Level Reset*/
324489d1c21fSKashyap D Desai req->MsgFlags =
324589d1c21fSKashyap D Desai MPI26_SCSITASKMGMT_MSGFLAGS_PROTOCOL_LVL_RST_PCIE;
324689d1c21fSKashyap D Desai }
3247991554f2SKenneth D. Merry
3248991554f2SKenneth D. Merry tm->cm_data = NULL;
3249991554f2SKenneth D. Merry tm->cm_complete = mprsas_resetdev_complete;
3250991554f2SKenneth D. Merry tm->cm_complete_data = ccb;
3251a2c14879SStephen McConnell
3252a2c14879SStephen McConnell mpr_dprint(sc, MPR_INFO, "%s: Sending reset for target ID %d\n",
3253a2c14879SStephen McConnell __func__, targ->tid);
3254991554f2SKenneth D. Merry tm->cm_targ = targ;
3255a2c14879SStephen McConnell
325646b9415fSScott Long mprsas_prepare_for_tm(sc, tm, targ, CAM_LUN_WILDCARD);
3257991554f2SKenneth D. Merry mpr_map_command(sc, tm);
3258991554f2SKenneth D. Merry }
3259991554f2SKenneth D. Merry
3260991554f2SKenneth D. Merry static void
mprsas_resetdev_complete(struct mpr_softc * sc,struct mpr_command * tm)3261991554f2SKenneth D. Merry mprsas_resetdev_complete(struct mpr_softc *sc, struct mpr_command *tm)
3262991554f2SKenneth D. Merry {
3263991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REPLY *resp;
3264991554f2SKenneth D. Merry union ccb *ccb;
3265991554f2SKenneth D. Merry
3266991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
3267991554f2SKenneth D. Merry mtx_assert(&sc->mpr_mtx, MA_OWNED);
3268991554f2SKenneth D. Merry
3269991554f2SKenneth D. Merry resp = (MPI2_SCSI_TASK_MANAGE_REPLY *)tm->cm_reply;
3270991554f2SKenneth D. Merry ccb = tm->cm_complete_data;
3271991554f2SKenneth D. Merry
3272991554f2SKenneth D. Merry /*
3273991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
3274991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
3275991554f2SKenneth D. Merry * task management commands don't have S/G lists.
3276991554f2SKenneth D. Merry */
3277991554f2SKenneth D. Merry if ((tm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
3278991554f2SKenneth D. Merry MPI2_SCSI_TASK_MANAGE_REQUEST *req;
3279991554f2SKenneth D. Merry
3280991554f2SKenneth D. Merry req = (MPI2_SCSI_TASK_MANAGE_REQUEST *)tm->cm_req;
3281991554f2SKenneth D. Merry
3282991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: cm_flags = %#x for reset of "
3283991554f2SKenneth D. Merry "handle %#04x! This should not happen!\n", __func__,
3284991554f2SKenneth D. Merry tm->cm_flags, req->DevHandle);
3285a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
3286991554f2SKenneth D. Merry goto bailout;
3287991554f2SKenneth D. Merry }
3288991554f2SKenneth D. Merry
32897a2a6a1aSStephen McConnell mpr_dprint(sc, MPR_XINFO, "%s: IOCStatus = 0x%x ResponseCode = 0x%x\n",
32907a2a6a1aSStephen McConnell __func__, le16toh(resp->IOCStatus), le32toh(resp->ResponseCode));
3291991554f2SKenneth D. Merry
3292991554f2SKenneth D. Merry if (le32toh(resp->ResponseCode) == MPI2_SCSITASKMGMT_RSP_TM_COMPLETE) {
3293a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP);
3294991554f2SKenneth D. Merry mprsas_announce_reset(sc, AC_SENT_BDR, tm->cm_targ->tid,
3295991554f2SKenneth D. Merry CAM_LUN_WILDCARD);
3296991554f2SKenneth D. Merry }
3297991554f2SKenneth D. Merry else
3298a2c14879SStephen McConnell mprsas_set_ccbstatus(ccb, CAM_REQ_CMP_ERR);
3299991554f2SKenneth D. Merry
3300991554f2SKenneth D. Merry bailout:
3301991554f2SKenneth D. Merry
3302991554f2SKenneth D. Merry mprsas_free_tm(sc, tm);
3303991554f2SKenneth D. Merry xpt_done(ccb);
3304991554f2SKenneth D. Merry }
3305991554f2SKenneth D. Merry
3306991554f2SKenneth D. Merry static void
mprsas_poll(struct cam_sim * sim)3307991554f2SKenneth D. Merry mprsas_poll(struct cam_sim *sim)
3308991554f2SKenneth D. Merry {
3309991554f2SKenneth D. Merry struct mprsas_softc *sassc;
3310991554f2SKenneth D. Merry
3311991554f2SKenneth D. Merry sassc = cam_sim_softc(sim);
3312991554f2SKenneth D. Merry
3313991554f2SKenneth D. Merry if (sassc->sc->mpr_debug & MPR_TRACE) {
3314991554f2SKenneth D. Merry /* frequent debug messages during a panic just slow
3315991554f2SKenneth D. Merry * everything down too much.
3316991554f2SKenneth D. Merry */
3317a2c14879SStephen McConnell mpr_dprint(sassc->sc, MPR_XINFO, "%s clearing MPR_TRACE\n",
3318a2c14879SStephen McConnell __func__);
3319991554f2SKenneth D. Merry sassc->sc->mpr_debug &= ~MPR_TRACE;
3320991554f2SKenneth D. Merry }
3321991554f2SKenneth D. Merry
3322991554f2SKenneth D. Merry mpr_intr_locked(sassc->sc);
3323991554f2SKenneth D. Merry }
3324991554f2SKenneth D. Merry
3325991554f2SKenneth D. Merry static void
mprsas_async(void * callback_arg,uint32_t code,struct cam_path * path,void * arg)3326991554f2SKenneth D. Merry mprsas_async(void *callback_arg, uint32_t code, struct cam_path *path,
3327991554f2SKenneth D. Merry void *arg)
3328991554f2SKenneth D. Merry {
3329991554f2SKenneth D. Merry struct mpr_softc *sc;
3330991554f2SKenneth D. Merry
3331991554f2SKenneth D. Merry sc = (struct mpr_softc *)callback_arg;
3332991554f2SKenneth D. Merry
333302d81940SAlexander Motin mpr_lock(sc);
3334991554f2SKenneth D. Merry switch (code) {
3335991554f2SKenneth D. Merry case AC_ADVINFO_CHANGED: {
3336991554f2SKenneth D. Merry struct mprsas_target *target;
3337991554f2SKenneth D. Merry struct mprsas_softc *sassc;
3338991554f2SKenneth D. Merry struct scsi_read_capacity_data_long rcap_buf;
3339991554f2SKenneth D. Merry struct ccb_dev_advinfo cdai;
3340991554f2SKenneth D. Merry struct mprsas_lun *lun;
3341991554f2SKenneth D. Merry lun_id_t lunid;
3342991554f2SKenneth D. Merry int found_lun;
3343991554f2SKenneth D. Merry uintptr_t buftype;
3344991554f2SKenneth D. Merry
3345991554f2SKenneth D. Merry buftype = (uintptr_t)arg;
3346991554f2SKenneth D. Merry
3347991554f2SKenneth D. Merry found_lun = 0;
3348991554f2SKenneth D. Merry sassc = sc->sassc;
3349991554f2SKenneth D. Merry
3350991554f2SKenneth D. Merry /*
3351991554f2SKenneth D. Merry * We're only interested in read capacity data changes.
3352991554f2SKenneth D. Merry */
3353991554f2SKenneth D. Merry if (buftype != CDAI_TYPE_RCAPLONG)
3354991554f2SKenneth D. Merry break;
3355991554f2SKenneth D. Merry
3356991554f2SKenneth D. Merry /*
3357991554f2SKenneth D. Merry * We should have a handle for this, but check to make sure.
3358991554f2SKenneth D. Merry */
3359991554f2SKenneth D. Merry KASSERT(xpt_path_target_id(path) < sassc->maxtargets,
3360991554f2SKenneth D. Merry ("Target %d out of bounds in mprsas_async\n",
3361991554f2SKenneth D. Merry xpt_path_target_id(path)));
3362991554f2SKenneth D. Merry target = &sassc->targets[xpt_path_target_id(path)];
3363991554f2SKenneth D. Merry if (target->handle == 0)
3364991554f2SKenneth D. Merry break;
3365991554f2SKenneth D. Merry
3366991554f2SKenneth D. Merry lunid = xpt_path_lun_id(path);
3367991554f2SKenneth D. Merry
3368991554f2SKenneth D. Merry SLIST_FOREACH(lun, &target->luns, lun_link) {
3369991554f2SKenneth D. Merry if (lun->lun_id == lunid) {
3370991554f2SKenneth D. Merry found_lun = 1;
3371991554f2SKenneth D. Merry break;
3372991554f2SKenneth D. Merry }
3373991554f2SKenneth D. Merry }
3374991554f2SKenneth D. Merry
3375991554f2SKenneth D. Merry if (found_lun == 0) {
3376991554f2SKenneth D. Merry lun = malloc(sizeof(struct mprsas_lun), M_MPR,
3377991554f2SKenneth D. Merry M_NOWAIT | M_ZERO);
3378991554f2SKenneth D. Merry if (lun == NULL) {
3379991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "Unable to alloc "
3380991554f2SKenneth D. Merry "LUN for EEDP support.\n");
3381991554f2SKenneth D. Merry break;
3382991554f2SKenneth D. Merry }
3383991554f2SKenneth D. Merry lun->lun_id = lunid;
3384991554f2SKenneth D. Merry SLIST_INSERT_HEAD(&target->luns, lun, lun_link);
3385991554f2SKenneth D. Merry }
3386991554f2SKenneth D. Merry
3387991554f2SKenneth D. Merry bzero(&rcap_buf, sizeof(rcap_buf));
33887608b98cSEdward Tomasz Napierala bzero(&cdai, sizeof(cdai));
3389991554f2SKenneth D. Merry xpt_setup_ccb(&cdai.ccb_h, path, CAM_PRIORITY_NORMAL);
3390991554f2SKenneth D. Merry cdai.ccb_h.func_code = XPT_DEV_ADVINFO;
3391991554f2SKenneth D. Merry cdai.ccb_h.flags = CAM_DIR_IN;
3392991554f2SKenneth D. Merry cdai.buftype = CDAI_TYPE_RCAPLONG;
3393e8577fb4SKenneth D. Merry cdai.flags = CDAI_FLAG_NONE;
3394991554f2SKenneth D. Merry cdai.bufsiz = sizeof(rcap_buf);
3395991554f2SKenneth D. Merry cdai.buf = (uint8_t *)&rcap_buf;
3396991554f2SKenneth D. Merry xpt_action((union ccb *)&cdai);
3397991554f2SKenneth D. Merry if ((cdai.ccb_h.status & CAM_DEV_QFRZN) != 0)
3398991554f2SKenneth D. Merry cam_release_devq(cdai.ccb_h.path, 0, 0, 0, FALSE);
3399991554f2SKenneth D. Merry
3400a2c14879SStephen McConnell if ((mprsas_get_ccbstatus((union ccb *)&cdai) == CAM_REQ_CMP)
3401991554f2SKenneth D. Merry && (rcap_buf.prot & SRC16_PROT_EN)) {
34028881681bSKenneth D. Merry switch (rcap_buf.prot & SRC16_P_TYPE) {
34038881681bSKenneth D. Merry case SRC16_PTYPE_1:
34048881681bSKenneth D. Merry case SRC16_PTYPE_3:
3405991554f2SKenneth D. Merry lun->eedp_formatted = TRUE;
34068881681bSKenneth D. Merry lun->eedp_block_size =
34078881681bSKenneth D. Merry scsi_4btoul(rcap_buf.length);
34088881681bSKenneth D. Merry break;
34098881681bSKenneth D. Merry case SRC16_PTYPE_2:
34108881681bSKenneth D. Merry default:
34118881681bSKenneth D. Merry lun->eedp_formatted = FALSE;
34128881681bSKenneth D. Merry lun->eedp_block_size = 0;
34138881681bSKenneth D. Merry break;
34148881681bSKenneth D. Merry }
3415991554f2SKenneth D. Merry } else {
3416991554f2SKenneth D. Merry lun->eedp_formatted = FALSE;
3417991554f2SKenneth D. Merry lun->eedp_block_size = 0;
3418991554f2SKenneth D. Merry }
3419991554f2SKenneth D. Merry break;
3420991554f2SKenneth D. Merry }
3421991554f2SKenneth D. Merry default:
3422991554f2SKenneth D. Merry break;
3423991554f2SKenneth D. Merry }
342402d81940SAlexander Motin mpr_unlock(sc);
3425991554f2SKenneth D. Merry }
3426991554f2SKenneth D. Merry
3427a2c14879SStephen McConnell /*
34289781c28cSAlexander Motin * Freeze the devq and set the INRESET flag so that no I/O will be sent to
34299781c28cSAlexander Motin * the target until the reset has completed. The CCB holds the path which
34309781c28cSAlexander Motin * is used to release the devq. The devq is released and the CCB is freed
3431a2c14879SStephen McConnell * when the TM completes.
3432a8837c77SWarner Losh * We only need to do this when we're entering reset, not at each time we
3433a8837c77SWarner Losh * need to send an abort (which will happen if multiple commands timeout
3434a8837c77SWarner Losh * while we're sending the abort). We do not release the queue for each
3435a8837c77SWarner Losh * command we complete (just at the end when we free the tm), so freezing
3436a8837c77SWarner Losh * it each time doesn't make sense.
3437a2c14879SStephen McConnell */
3438b7f1ee79SScott Long void
mprsas_prepare_for_tm(struct mpr_softc * sc,struct mpr_command * tm,struct mprsas_target * target,lun_id_t lun_id)3439b7f1ee79SScott Long mprsas_prepare_for_tm(struct mpr_softc *sc, struct mpr_command *tm,
3440b7f1ee79SScott Long struct mprsas_target *target, lun_id_t lun_id)
3441b7f1ee79SScott Long {
3442b7f1ee79SScott Long union ccb *ccb;
3443b7f1ee79SScott Long path_id_t path_id;
3444b7f1ee79SScott Long
3445a2c14879SStephen McConnell ccb = xpt_alloc_ccb_nowait();
3446a2c14879SStephen McConnell if (ccb) {
3447a2c14879SStephen McConnell path_id = cam_sim_path(sc->sassc->sim);
3448a2c14879SStephen McConnell if (xpt_create_path(&ccb->ccb_h.path, xpt_periph, path_id,
3449a2c14879SStephen McConnell target->tid, lun_id) != CAM_REQ_CMP) {
3450a2c14879SStephen McConnell xpt_free_ccb(ccb);
3451a2c14879SStephen McConnell } else {
3452a8837c77SWarner Losh tm->cm_ccb = ccb;
3453a8837c77SWarner Losh tm->cm_targ = target;
3454a8837c77SWarner Losh if ((target->flags & MPRSAS_TARGET_INRESET) == 0) {
3455a8837c77SWarner Losh mpr_dprint(sc, MPR_XINFO | MPR_RECOVERY,
34569781c28cSAlexander Motin "%s: Freezing devq for target ID %d\n",
34579781c28cSAlexander Motin __func__, target->tid);
34589781c28cSAlexander Motin xpt_freeze_devq(ccb->ccb_h.path, 1);
3459a2c14879SStephen McConnell target->flags |= MPRSAS_TARGET_INRESET;
3460a2c14879SStephen McConnell }
3461a2c14879SStephen McConnell }
3462a2c14879SStephen McConnell }
3463a8837c77SWarner Losh }
3464a2c14879SStephen McConnell
3465991554f2SKenneth D. Merry int
mprsas_startup(struct mpr_softc * sc)3466991554f2SKenneth D. Merry mprsas_startup(struct mpr_softc *sc)
3467991554f2SKenneth D. Merry {
3468991554f2SKenneth D. Merry /*
3469991554f2SKenneth D. Merry * Send the port enable message and set the wait_for_port_enable flag.
3470991554f2SKenneth D. Merry * This flag helps to keep the simq frozen until all discovery events
3471991554f2SKenneth D. Merry * are processed.
3472991554f2SKenneth D. Merry */
3473991554f2SKenneth D. Merry sc->wait_for_port_enable = 1;
3474991554f2SKenneth D. Merry mprsas_send_portenable(sc);
3475991554f2SKenneth D. Merry return (0);
3476991554f2SKenneth D. Merry }
3477991554f2SKenneth D. Merry
3478991554f2SKenneth D. Merry static int
mprsas_send_portenable(struct mpr_softc * sc)3479991554f2SKenneth D. Merry mprsas_send_portenable(struct mpr_softc *sc)
3480991554f2SKenneth D. Merry {
3481991554f2SKenneth D. Merry MPI2_PORT_ENABLE_REQUEST *request;
3482991554f2SKenneth D. Merry struct mpr_command *cm;
3483991554f2SKenneth D. Merry
3484991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
3485991554f2SKenneth D. Merry
3486991554f2SKenneth D. Merry if ((cm = mpr_alloc_command(sc)) == NULL)
3487991554f2SKenneth D. Merry return (EBUSY);
3488991554f2SKenneth D. Merry request = (MPI2_PORT_ENABLE_REQUEST *)cm->cm_req;
3489991554f2SKenneth D. Merry request->Function = MPI2_FUNCTION_PORT_ENABLE;
3490991554f2SKenneth D. Merry request->MsgFlags = 0;
3491991554f2SKenneth D. Merry request->VP_ID = 0;
3492991554f2SKenneth D. Merry cm->cm_desc.Default.RequestFlags = MPI2_REQ_DESCRIPT_FLAGS_DEFAULT_TYPE;
3493991554f2SKenneth D. Merry cm->cm_complete = mprsas_portenable_complete;
3494991554f2SKenneth D. Merry cm->cm_data = NULL;
3495991554f2SKenneth D. Merry cm->cm_sge = NULL;
3496991554f2SKenneth D. Merry
3497991554f2SKenneth D. Merry mpr_map_command(sc, cm);
3498991554f2SKenneth D. Merry mpr_dprint(sc, MPR_XINFO,
3499991554f2SKenneth D. Merry "mpr_send_portenable finished cm %p req %p complete %p\n",
3500991554f2SKenneth D. Merry cm, cm->cm_req, cm->cm_complete);
3501991554f2SKenneth D. Merry return (0);
3502991554f2SKenneth D. Merry }
3503991554f2SKenneth D. Merry
3504991554f2SKenneth D. Merry static void
mprsas_portenable_complete(struct mpr_softc * sc,struct mpr_command * cm)3505991554f2SKenneth D. Merry mprsas_portenable_complete(struct mpr_softc *sc, struct mpr_command *cm)
3506991554f2SKenneth D. Merry {
3507991554f2SKenneth D. Merry MPI2_PORT_ENABLE_REPLY *reply;
3508991554f2SKenneth D. Merry struct mprsas_softc *sassc;
3509991554f2SKenneth D. Merry
3510991554f2SKenneth D. Merry MPR_FUNCTRACE(sc);
3511991554f2SKenneth D. Merry sassc = sc->sassc;
3512991554f2SKenneth D. Merry
3513991554f2SKenneth D. Merry /*
3514991554f2SKenneth D. Merry * Currently there should be no way we can hit this case. It only
3515991554f2SKenneth D. Merry * happens when we have a failure to allocate chain frames, and
3516991554f2SKenneth D. Merry * port enable commands don't have S/G lists.
3517991554f2SKenneth D. Merry */
3518991554f2SKenneth D. Merry if ((cm->cm_flags & MPR_CM_FLAGS_ERROR_MASK) != 0) {
3519991554f2SKenneth D. Merry mpr_dprint(sc, MPR_ERROR, "%s: cm_flags = %#x for port enable! "
3520991554f2SKenneth D. Merry "This should not happen!\n", __func__, cm->cm_flags);
3521991554f2SKenneth D. Merry }
3522991554f2SKenneth D. Merry
3523991554f2SKenneth D. Merry reply = (MPI2_PORT_ENABLE_REPLY *)cm->cm_reply;
3524991554f2SKenneth D. Merry if (reply == NULL)
3525991554f2SKenneth D. Merry mpr_dprint(sc, MPR_FAULT, "Portenable NULL reply\n");
352671900a79SAlfredo Dal'Ava Junior else if ((le16toh(reply->IOCStatus) & MPI2_IOCSTATUS_MASK) !=
3527991554f2SKenneth D. Merry MPI2_IOCSTATUS_SUCCESS)
3528991554f2SKenneth D. Merry mpr_dprint(sc, MPR_FAULT, "Portenable failed\n");
3529991554f2SKenneth D. Merry
3530991554f2SKenneth D. Merry mpr_free_command(sc, cm);
3531991554f2SKenneth D. Merry /*
3532991554f2SKenneth D. Merry * Done waiting for port enable to complete. Decrement the refcount.
3533991554f2SKenneth D. Merry * If refcount is 0, discovery is complete and a rescan of the bus can
3534991554f2SKenneth D. Merry * take place.
3535991554f2SKenneth D. Merry */
3536991554f2SKenneth D. Merry sc->wait_for_port_enable = 0;
3537991554f2SKenneth D. Merry sc->port_enable_complete = 1;
3538991554f2SKenneth D. Merry wakeup(&sc->port_enable_complete);
3539991554f2SKenneth D. Merry mprsas_startup_decrement(sassc);
3540991554f2SKenneth D. Merry }
3541991554f2SKenneth D. Merry
3542991554f2SKenneth D. Merry int
mprsas_check_id(struct mprsas_softc * sassc,int id)3543991554f2SKenneth D. Merry mprsas_check_id(struct mprsas_softc *sassc, int id)
3544991554f2SKenneth D. Merry {
3545991554f2SKenneth D. Merry struct mpr_softc *sc = sassc->sc;
3546991554f2SKenneth D. Merry char *ids;
3547991554f2SKenneth D. Merry char *name;
3548991554f2SKenneth D. Merry
3549991554f2SKenneth D. Merry ids = &sc->exclude_ids[0];
3550991554f2SKenneth D. Merry while((name = strsep(&ids, ",")) != NULL) {
3551991554f2SKenneth D. Merry if (name[0] == '\0')
3552991554f2SKenneth D. Merry continue;
3553991554f2SKenneth D. Merry if (strtol(name, NULL, 0) == (long)id)
3554991554f2SKenneth D. Merry return (1);
3555991554f2SKenneth D. Merry }
3556991554f2SKenneth D. Merry
3557991554f2SKenneth D. Merry return (0);
3558991554f2SKenneth D. Merry }
3559a2c14879SStephen McConnell
3560a2c14879SStephen McConnell void
mprsas_realloc_targets(struct mpr_softc * sc,int maxtargets)3561a2c14879SStephen McConnell mprsas_realloc_targets(struct mpr_softc *sc, int maxtargets)
3562a2c14879SStephen McConnell {
3563a2c14879SStephen McConnell struct mprsas_softc *sassc;
3564a2c14879SStephen McConnell struct mprsas_lun *lun, *lun_tmp;
3565a2c14879SStephen McConnell struct mprsas_target *targ;
3566a2c14879SStephen McConnell int i;
3567a2c14879SStephen McConnell
3568a2c14879SStephen McConnell sassc = sc->sassc;
3569a2c14879SStephen McConnell /*
3570a2c14879SStephen McConnell * The number of targets is based on IOC Facts, so free all of
3571a2c14879SStephen McConnell * the allocated LUNs for each target and then the target buffer
3572a2c14879SStephen McConnell * itself.
3573a2c14879SStephen McConnell */
3574a2c14879SStephen McConnell for (i=0; i< maxtargets; i++) {
3575a2c14879SStephen McConnell targ = &sassc->targets[i];
3576a2c14879SStephen McConnell SLIST_FOREACH_SAFE(lun, &targ->luns, lun_link, lun_tmp) {
3577a2c14879SStephen McConnell free(lun, M_MPR);
3578a2c14879SStephen McConnell }
3579a2c14879SStephen McConnell }
3580a2c14879SStephen McConnell free(sassc->targets, M_MPR);
3581a2c14879SStephen McConnell
3582a2c14879SStephen McConnell sassc->targets = malloc(sizeof(struct mprsas_target) * maxtargets,
3583a2c14879SStephen McConnell M_MPR, M_WAITOK|M_ZERO);
3584a2c14879SStephen McConnell }
3585