From owner-freebsd-scsi@FreeBSD.ORG Tue Apr 1 14:07:00 2014 Return-Path: Delivered-To: scsi@freebsd.org Received: from mx1.freebsd.org (mx1.freebsd.org [8.8.178.115]) (using TLSv1 with cipher ADH-AES256-SHA (256/256 bits)) (No client certificate requested) by hub.freebsd.org (Postfix) with ESMTPS id C9922CB6 for ; Tue, 1 Apr 2014 14:07:00 +0000 (UTC) Received: from mail-we0-f169.google.com (mail-we0-f169.google.com [74.125.82.169]) (using TLSv1 with cipher ECDHE-RSA-RC4-SHA (128/128 bits)) (Client CN "smtp.gmail.com", Issuer "Google Internet Authority G2" (verified OK)) by mx1.freebsd.org (Postfix) with ESMTPS id 185C8849 for ; Tue, 1 Apr 2014 14:06:59 +0000 (UTC) Received: by mail-we0-f169.google.com with SMTP id w62so6403656wes.28 for ; Tue, 01 Apr 2014 07:06:52 -0700 (PDT) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20130820; h=x-gm-message-state:message-id:from:to:cc:subject:date:mime-version :content-type; bh=zZOV4m+s9sOGAjjiz5Suoxdvj+7/l8Qn7iPZM9ap/h0=; b=Svyf2VSjUpIE66nqcMBBiUuhMliABdIAm0rVj8yB0rda0+SFvYjyw/At8zVnAXQfWI FpdzZz0uAWaYBRuGabpFg3sCxawgC562lFRuqlp3K/wJ812W2UxnPyPj1RLGQ2ZJSl6G q5TJMRXQDLgwAZcbDXamGfbYB5Q/StqsaU3BqA4oMYAsmUEVzenC8LuVsAdk/uwiEyMF NUwGMIgxEmckScxBn82lN0jl6GEq3+9y/rhutdFZgVnEvmeiRSvDr9+l63g17MLV53bi S8SekKvtjE48O15UaPrh7WtRarCRZoM03EWLSW9pOsR/g+n1As4KpLP214riyQzz3eki a9mA== X-Gm-Message-State: ALoCoQmf6HIWiWlH1KIaMTOv57/M7fftC3wPScK5M4gE8sAIssnFCA2/uq9EVyhtG7xzY7Bc4wgr X-Received: by 10.194.84.144 with SMTP id z16mr22815161wjy.23.1396359629356; Tue, 01 Apr 2014 06:40:29 -0700 (PDT) Received: from r2d2 (82-69-141-170.dsl.in-addr.zen.co.uk. [82.69.141.170]) by mx.google.com with ESMTPSA id t10sm33231383wiw.19.2014.04.01.06.40.27 for (version=TLSv1 cipher=RC4-SHA bits=128/128); Tue, 01 Apr 2014 06:40:28 -0700 (PDT) Message-ID: <84D0D5ADFC374E3AB7EFB51DC941411A@multiplay.co.uk> From: "Steven Hartland" To: Subject: LSI mps phase16 -> phase 18 driver changes patch Date: Tue, 1 Apr 2014 14:40:24 +0100 MIME-Version: 1.0 Content-Type: multipart/mixed; boundary="----=_NextPart_000_094C_01CF4DB8.50E12F90" X-Priority: 3 X-MSMail-Priority: Normal X-Mailer: Microsoft Outlook Express 6.00.2900.5931 X-MimeOLE: Produced By Microsoft MimeOLE V6.00.2900.6157 Cc: Scott Long , Alexander Motin , "Kenneth D. Merry" X-BeenThere: freebsd-scsi@freebsd.org X-Mailman-Version: 2.1.17 Precedence: list List-Id: SCSI subsystem List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Tue, 01 Apr 2014 14:07:00 -0000 This is a multi-part message in MIME format. ------=_NextPart_000_094C_01CF4DB8.50E12F90 Content-Type: text/plain; format=flowed; charset="iso-8859-1"; reply-type=original Content-Transfer-Encoding: 7bit Hi guys attached is a patch which updates our mps with the changes LSI have implemented between phase 16 (our last sync up) and phase 18 (current latest LSI release) drivers. The main change seems to be the addition Start Stop Unit for SATA direct-attach devices in IR mode to avoid data corruption. Any objections to committing this to HEAD? Regards Steve ------=_NextPart_000_094C_01CF4DB8.50E12F90 Content-Type: application/octet-stream; name="mps-phase16-phase18.patch" Content-Transfer-Encoding: quoted-printable Content-Disposition: attachment; filename="mps-phase16-phase18.patch" Import LSI phase16 - phase18 changes=0A= * Implements Start Stop Unit for SATA direct-attach devices in IR mode = to avoid=0A= data corruption.=0A= --- sys/dev/mps/mps_sas.c.orig 2014-04-01 11:05:42.538799904 +0000=0A= +++ sys/dev/mps/mps_sas.c 2014-04-01 11:36:50.202671306 +0000=0A= @@ -188,6 +188,16 @@=0A= }=0A= =0A= void=0A= +mpssas_release_simq_reinit(struct mpssas_softc *sassc)=0A= +{=0A= + if (sassc->flags & MPSSAS_QUEUE_FROZEN) {=0A= + sassc->flags &=3D ~MPSSAS_QUEUE_FROZEN;=0A= + xpt_release_simq(sassc->sim, 1);=0A= + mps_dprint(sassc->sc, MPS_INFO, "Unfreezing SIM queue\n");=0A= + }=0A= +}=0A= +=0A= +void=0A= mpssas_startup_decrement(struct mpssas_softc *sassc)=0A= {=0A= MPS_FUNCTRACE(sassc->sc);=0A= @@ -995,7 +1005,7 @@=0A= =0A= targ =3D &sassc->targets[cts->ccb_h.target_id];=0A= if (targ->handle =3D=3D 0x0) {=0A= - cts->ccb_h.status =3D CAM_SEL_TIMEOUT;=0A= + cts->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= break;=0A= }=0A= =0A= @@ -1112,6 +1122,14 @@=0A= wakeup(cm);=0A= completed =3D 1;=0A= }=0A= +=0A= + if (cm->cm_sc->io_cmds_active !=3D 0) {=0A= + cm->cm_sc->io_cmds_active--;=0A= + } else {=0A= + mps_dprint(cm->cm_sc, MPS_INFO, "Warning: "=0A= + "io_cmds_active is out of sync - resynching to "=0A= + "0\n");=0A= + }=0A= =0A= if ((completed =3D=3D 0) && (cm->cm_state !=3D MPS_CM_STATE_FREE)) {=0A= /* this should never happen, but if it does, log */=0A= @@ -1643,14 +1661,14 @@=0A= if (targ->handle =3D=3D 0x0) {=0A= mps_dprint(sc, MPS_ERROR, "%s NULL handle for target %u\n", =0A= __func__, csio->ccb_h.target_id);=0A= - csio->ccb_h.status =3D CAM_SEL_TIMEOUT;=0A= + csio->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= xpt_done(ccb);=0A= return;=0A= }=0A= if (targ->flags & MPS_TARGET_FLAGS_RAID_COMPONENT) {=0A= mps_dprint(sc, MPS_ERROR, "%s Raid component no SCSI IO "=0A= "supported %u\n", __func__, csio->ccb_h.target_id);=0A= - csio->ccb_h.status =3D CAM_TID_INVALID;=0A= + csio->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= xpt_done(ccb);=0A= return;=0A= }=0A= @@ -1681,13 +1699,16 @@=0A= =0A= if ((sc->mps_flags & MPS_FLAGS_SHUTDOWN) !=3D 0) {=0A= mps_dprint(sc, MPS_INFO, "%s shutting down\n", __func__);=0A= - csio->ccb_h.status =3D CAM_TID_INVALID;=0A= + csio->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= xpt_done(ccb);=0A= return;=0A= }=0A= =0A= cm =3D mps_alloc_command(sc);=0A= - if (cm =3D=3D NULL) {=0A= + if (cm =3D=3D NULL || (sc->mps_flags & MPS_FLAGS_DIAGRESET)) {=0A= + if (cm !=3D NULL) {=0A= + mps_free_command(sc, cm);=0A= + }=0A= if ((sassc->flags & MPSSAS_QUEUE_FROZEN) =3D=3D 0) {=0A= xpt_freeze_simq(sassc->sim, 1);=0A= sassc->flags |=3D MPSSAS_QUEUE_FROZEN;=0A= @@ -2166,6 +2187,18 @@=0A= }=0A= }=0A= =0A= + /*=0A= + * If this is a Start Stop Unit command and it was issued by the driver=0A= + * during shutdown, decrement the refcount to account for all of the=0A= + * commands that were sent. All SSU commands should be completed = before=0A= + * shutdown completes, meaning SSU_refcount will be 0 after SSU_started=0A= + * is TRUE.=0A= + */=0A= + if (sc->SSU_started && (csio->cdb_io.cdb_bytes[0] =3D=3D = START_STOP_UNIT)) {=0A= + mps_dprint(sc, MPS_INFO, "Decrementing SSU count.\n");=0A= + sc->SSU_refcount--;=0A= + }=0A= +=0A= /* Take the fast path to completion */=0A= if (cm->cm_reply =3D=3D NULL) {=0A= if ((ccb->ccb_h.status & CAM_STATUS_MASK) =3D=3D CAM_REQ_INPROG) {=0A= @@ -2990,7 +3023,7 @@=0A= mps_dprint(sc, MPS_ERROR,=0A= "%s: handle %d does not have a valid "=0A= "parent handle!\n", __func__, targ->handle);=0A= - ccb->ccb_h.status =3D CAM_REQ_INVALID;=0A= + ccb->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= goto bailout;=0A= }=0A= #ifdef OLD_MPS_PROBE=0A= @@ -3001,7 +3034,7 @@=0A= mps_dprint(sc, MPS_ERROR,=0A= "%s: handle %d does not have a valid "=0A= "parent target!\n", __func__, targ->handle);=0A= - ccb->ccb_h.status =3D CAM_REQ_INVALID;=0A= + ccb->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= goto bailout;=0A= }=0A= =0A= @@ -3011,7 +3044,7 @@=0A= "%s: handle %d parent %d does not "=0A= "have an SMP target!\n", __func__,=0A= targ->handle, parent_target->handle);=0A= - ccb->ccb_h.status =3D CAM_REQ_INVALID;=0A= + ccb->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= goto bailout;=0A= =0A= }=0A= @@ -3024,7 +3057,7 @@=0A= "%s: handle %d parent %d does not "=0A= "have an SMP target!\n", __func__,=0A= targ->handle, targ->parent_handle);=0A= - ccb->ccb_h.status =3D CAM_REQ_INVALID;=0A= + ccb->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= goto bailout;=0A= =0A= }=0A= @@ -3033,7 +3066,7 @@=0A= "%s: handle %d parent handle %d does "=0A= "not have a valid SAS address!\n",=0A= __func__, targ->handle, targ->parent_handle);=0A= - ccb->ccb_h.status =3D CAM_REQ_INVALID;=0A= + ccb->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= goto bailout;=0A= }=0A= =0A= @@ -3046,7 +3079,7 @@=0A= mps_dprint(sc, MPS_INFO,=0A= "%s: unable to find SAS address for handle %d\n",=0A= __func__, targ->handle);=0A= - ccb->ccb_h.status =3D CAM_REQ_INVALID;=0A= + ccb->ccb_h.status =3D CAM_DEV_NOT_THERE;=0A= goto bailout;=0A= }=0A= mpssas_send_smpcmd(sassc, ccb, sasaddr);=0A= @@ -3348,6 +3381,20 @@=0A= }=0A= =0A= xpt_path_string(local_path, path_str, sizeof(path_str));=0A= +=0A= + /*=0A= + * If this is a SATA direct-access end device,=0A= + * mark it so that a SCSI StartStopUnit command=0A= + * will be sent to it when the driver is being=0A= + * shutdown.=0A= + */=0A= + if ((cgd.inq_data.device =3D=3D T_DIRECT) && =0A= + (target->devinfo & MPI2_SAS_DEVICE_INFO_SATA_DEVICE) &&=0A= + ((target->devinfo & MPI2_SAS_DEVICE_INFO_MASK_DEVICE_TYPE) =3D=3D=0A= + MPI2_SAS_DEVICE_INFO_END_DEVICE)) {=0A= + lun->stop_at_shutdown =3D TRUE;=0A= + }=0A= +=0A= mps_dprint(sc, MPS_INFO, "Sending read cap: path %s handle %d\n",=0A= path_str, target->handle);=0A= =0A= --- sys/dev/mps/mps_sas_lsi.c.orig 2014-03-26 17:49:01.000000000 +0000=0A= +++ sys/dev/mps/mps_sas_lsi.c 2014-04-01 11:06:42.158795951 +0000=0A= @@ -120,6 +120,9 @@=0A= u64 *sas_address, u16 handle, u32 device_info);=0A= static int mpssas_volume_add(struct mps_softc *sc,=0A= u16 handle);=0A= +static void mpssas_SSU_to_SATA_devices(struct mps_softc *sc);=0A= +static void mpssas_stop_unit_done(struct cam_periph *periph,=0A= + union ccb *done_ccb);=0A= =0A= void=0A= mpssas_evt_handler(struct mps_softc *sc, uintptr_t data,=0A= @@ -910,6 +913,138 @@=0A= }=0A= =0A= /**=0A= + * mpssas_SSU_to_SATA_devices =0A= + * @sc: per adapter object=0A= + *=0A= + * Looks through the target list and issues a StartStopUnit SCSI = command to each=0A= + * SATA direct-access device. This helps to ensure that data = corruption is=0A= + * avoided when the system is being shut down. This must be called = after the IR=0A= + * System Shutdown RAID Action is sent if in IR mode.=0A= + *=0A= + * Return nothing.=0A= + */=0A= +static void=0A= +mpssas_SSU_to_SATA_devices(struct mps_softc *sc)=0A= +{=0A= + struct mpssas_softc *sassc =3D sc->sassc;=0A= + union ccb *ccb;=0A= + path_id_t pathid =3D cam_sim_path(sassc->sim);=0A= + target_id_t targetid;=0A= + struct mpssas_target *target;=0A= + struct mpssas_lun *lun;=0A= + char path_str[64];=0A= + struct timeval cur_time, start_time;=0A= +=0A= + /*=0A= + * For each LUN of each target, issue a StartStopUnit command to stop=0A= + * the device.=0A= + */=0A= + sc->SSU_started =3D TRUE;=0A= + sc->SSU_refcount =3D 0;=0A= + for (targetid =3D 0; targetid < sc->facts->MaxTargets; targetid++) {=0A= + target =3D &sassc->targets[targetid];=0A= + if (target->handle =3D=3D 0x0) {=0A= + continue;=0A= + }=0A= +=0A= + SLIST_FOREACH(lun, &target->luns, lun_link) {=0A= + ccb =3D xpt_alloc_ccb_nowait();=0A= + if (ccb =3D=3D NULL) {=0A= + mps_dprint(sc, MPS_FAULT, "Unable to alloc CCB "=0A= + "to stop unit.\n");=0A= + return;=0A= + }=0A= +=0A= + /*=0A= + * The stop_at_shutdown flag will be set if this LUN is=0A= + * a SATA direct-access end device.=0A= + */=0A= + if (lun->stop_at_shutdown) {=0A= + if (xpt_create_path(&ccb->ccb_h.path,=0A= + xpt_periph, pathid, targetid,=0A= + lun->lun_id) !=3D CAM_REQ_CMP) {=0A= + mps_dprint(sc, MPS_FAULT, "Unable to "=0A= + "create LUN path to stop unit.\n");=0A= + xpt_free_ccb(ccb);=0A= + return;=0A= + }=0A= + xpt_path_string(ccb->ccb_h.path, path_str,=0A= + sizeof(path_str));=0A= +=0A= + mps_dprint(sc, MPS_INFO, "Sending StopUnit: "=0A= + "path %s handle %d\n", path_str,=0A= + target->handle);=0A= + =0A= + /*=0A= + * Issue a START STOP UNIT command for the LUN.=0A= + * Increment the SSU counter to be used to=0A= + * count the number of required replies.=0A= + */=0A= + mps_dprint(sc, MPS_INFO, "Incrementing SSU "=0A= + "count\n");=0A= + sc->SSU_refcount++;=0A= + ccb->ccb_h.target_id =3D=0A= + xpt_path_target_id(ccb->ccb_h.path);=0A= + ccb->ccb_h.target_lun =3D lun->lun_id;=0A= + ccb->ccb_h.ppriv_ptr1 =3D sassc;=0A= + scsi_start_stop(&ccb->csio,=0A= + /*retries*/0,=0A= + mpssas_stop_unit_done,=0A= + MSG_SIMPLE_Q_TAG,=0A= + /*start*/FALSE,=0A= + /*load/eject*/0,=0A= + /*immediate*/FALSE,=0A= + MPS_SENSE_LEN,=0A= + /*timeout*/10000);=0A= + xpt_action(ccb);=0A= + }=0A= + }=0A= + }=0A= +=0A= + /*=0A= + * Wait until all of the SSU commands have completed or time has=0A= + * expired (60 seconds). pause for 100ms each time through. If any=0A= + * command times out, the target will be reset in the SCSI command=0A= + * timeout routine.=0A= + */=0A= + getmicrotime(&start_time);=0A= + while (sc->SSU_refcount) {=0A= + pause("mpswait", hz/10);=0A= + =0A= + getmicrotime(&cur_time);=0A= + if ((cur_time.tv_sec - start_time.tv_sec) > 60) {=0A= + mps_dprint(sc, MPS_FAULT, "Time has expired waiting "=0A= + "for SSU commands to complete.\n");=0A= + break;=0A= + }=0A= + }=0A= +}=0A= +=0A= +static void=0A= +mpssas_stop_unit_done(struct cam_periph *periph, union ccb *done_ccb)=0A= +{=0A= + struct mpssas_softc *sassc;=0A= + char path_str[64];=0A= +=0A= + sassc =3D (struct mpssas_softc *)done_ccb->ccb_h.ppriv_ptr1;=0A= +=0A= + xpt_path_string(done_ccb->ccb_h.path, path_str, sizeof(path_str));=0A= + mps_dprint(sassc->sc, MPS_INFO, "Completing stop unit for %s\n",=0A= + path_str);=0A= +=0A= + if (done_ccb =3D=3D NULL)=0A= + return;=0A= +=0A= + /*=0A= + * Nothing more to do except free the CCB and path. If the command=0A= + * timed out, an abort reset, then target reset will be issued during=0A= + * the SCSI Command process.=0A= + */=0A= + xpt_free_path(done_ccb->ccb_h.path);=0A= + xpt_free_ccb(done_ccb);=0A= +}=0A= +=0A= +/**=0A= * mpssas_ir_shutdown - IR shutdown notification=0A= * @sc: per adapter object=0A= *=0A= @@ -933,7 +1068,7 @@=0A= =0A= /* is IR firmware build loaded? */=0A= if (!sc->ir_firmware)=0A= - return;=0A= + goto out;=0A= =0A= /* are there any volumes? Look at IR target IDs. */=0A= // TODO-later, this should be looked up in the RAID config structure=0A= @@ -958,11 +1093,11 @@=0A= }=0A= =0A= if (!found_volume)=0A= - return;=0A= + goto out;=0A= =0A= if ((cm =3D mps_alloc_command(sc)) =3D=3D NULL) {=0A= printf("%s: command alloc failed\n", __func__);=0A= - return;=0A= + goto out;=0A= }=0A= =0A= action =3D (MPI2_RAID_ACTION_REQUEST *)cm->cm_req;=0A= @@ -978,4 +1113,7 @@=0A= */=0A= if (cm)=0A= mps_free_command(sc, cm);=0A= +=0A= +out:=0A= + mpssas_SSU_to_SATA_devices(sc);=0A= }=0A= --- sys/dev/mps/mps_sas.h.orig 2014-03-26 17:49:01.000000000 +0000=0A= +++ sys/dev/mps/mps_sas.h 2014-04-01 11:06:42.157795822 +0000=0A= @@ -35,6 +35,7 @@=0A= lun_id_t lun_id;=0A= uint8_t eedp_formatted;=0A= uint32_t eedp_block_size;=0A= + uint8_t stop_at_shutdown;=0A= };=0A= =0A= struct mpssas_target {=0A= @@ -154,6 +155,7 @@=0A= void mpssas_discovery_end(struct mpssas_softc *sassc);=0A= void mpssas_startup_increment(struct mpssas_softc *sassc);=0A= void mpssas_startup_decrement(struct mpssas_softc *sassc);=0A= +void mpssas_release_simq_reinit(struct mpssas_softc *sassc);=0A= =0A= struct mps_command * mpssas_alloc_tm(struct mps_softc *sc);=0A= void mpssas_free_tm(struct mps_softc *sc, struct mps_command *tm);=0A= --- sys/dev/mps/mps.c.orig 2014-04-01 11:05:42.598799869 +0000=0A= +++ sys/dev/mps/mps.c 2014-04-01 11:51:15.083611560 +0000=0A= @@ -141,6 +141,7 @@=0A= {=0A= uint32_t reg;=0A= int i, error, tries =3D 0;=0A= + uint8_t first_wait_done =3D FALSE;=0A= =0A= mps_dprint(sc, MPS_TRACE, "%s\n", __func__);=0A= =0A= @@ -183,15 +184,32 @@=0A= =0A= /* Wait up to 300 seconds in 50ms intervals */=0A= error =3D ETIMEDOUT;=0A= - for (i =3D 0; i < 60000; i++) {=0A= - /* wait 50 msec */=0A= - if (mtx_owned(&sc->mps_mtx) && sleep_flag =3D=3D CAN_SLEEP)=0A= - msleep(&sc->msleep_fake_chan, &sc->mps_mtx, 0,=0A= - "mpsdiag", hz/20);=0A= - else if (sleep_flag =3D=3D CAN_SLEEP)=0A= - pause("mpsdiag", hz/20);=0A= - else=0A= - DELAY(50 * 1000);=0A= + for (i =3D 0; i < 6000; i++) {=0A= + /*=0A= + * Wait 50 msec. If this is the first time through, wait 256=0A= + * msec to satisfy Diag Reset timing requirements.=0A= + */=0A= + if (first_wait_done) {=0A= + if (mtx_owned(&sc->mps_mtx) && sleep_flag =3D=3D CAN_SLEEP)=0A= + msleep(&sc->msleep_fake_chan, &sc->mps_mtx, 0,=0A= + "mpsdiag", hz/20);=0A= + else if (sleep_flag =3D=3D CAN_SLEEP)=0A= + pause("mpsdiag", hz/20);=0A= + else=0A= + DELAY(50 * 1000);=0A= + } else {=0A= + DELAY(256 * 1000);=0A= + first_wait_done =3D TRUE;=0A= + }=0A= + /*=0A= + * Check for the RESET_ADAPTER bit to be cleared first, then=0A= + * wait for the RESET state to be cleared, which takes a little=0A= + * longer.=0A= + */=0A= + reg =3D mps_regread(sc, MPI2_HOST_DIAGNOSTIC_OFFSET);=0A= + if (reg & MPI2_DIAG_RESET_ADAPTER) {=0A= + continue;=0A= + }=0A= reg =3D mps_regread(sc, MPI2_DOORBELL_OFFSET);=0A= if ((reg & MPI2_IOC_STATE_MASK) !=3D MPI2_IOC_STATE_RESET) {=0A= error =3D 0;=0A= @@ -237,7 +255,7 @@=0A= sleep_flags =3D (sc->mps_flags & MPS_FLAGS_ATTACH_DONE)=0A= ? CAN_SLEEP:NO_SLEEP;=0A= error =3D 0;=0A= - while (tries++ < 5) {=0A= + while (tries++ < 1200) {=0A= reg =3D mps_regread(sc, MPI2_DOORBELL_OFFSET);=0A= mps_dprint(sc, MPS_INIT, "Doorbell=3D 0x%x\n", reg);=0A= =0A= @@ -679,6 +697,9 @@=0A= mps_reinit(struct mps_softc *sc)=0A= {=0A= int error;=0A= + struct mpssas_softc *sassc;=0A= +=0A= + sassc =3D sc->sassc;=0A= =0A= MPS_FUNCTRACE(sc);=0A= =0A= @@ -759,6 +780,8 @@=0A= mps_dprint(sc, MPS_INFO, "%s finished sc %p post %u free %u\n", =0A= __func__, sc, sc->replypostindex, sc->replyfreeindex);=0A= =0A= + mpssas_release_simq_reinit(sassc);=0A= +=0A= return 0;=0A= }=0A= =0A= @@ -2533,6 +2556,7 @@=0A= mps_request_polled(struct mps_softc *sc, struct mps_command *cm)=0A= {=0A= int error, timeout =3D 0, rc;=0A= + struct timeval cur_time, start_time;=0A= =0A= error =3D 0;=0A= =0A= @@ -2540,22 +2564,33 @@=0A= cm->cm_complete =3D NULL;=0A= mps_map_command(sc, cm);=0A= =0A= + getmicrotime(&start_time);=0A= while ((cm->cm_flags & MPS_CM_FLAGS_COMPLETE) =3D=3D 0) {=0A= mps_intr_locked(sc);=0A= =0A= - DELAY(50 * 1000);=0A= - if (timeout++ > 1000) {=0A= + if (mtx_owned(&sc->mps_mtx))=0A= + msleep(&sc->msleep_fake_chan, &sc->mps_mtx, 0,=0A= + "mpspoll", hz/20);=0A= + else=0A= + pause("mpsdiag", hz/20);=0A= +=0A= + /*=0A= + * Check for real-time timeout and fail if more than 60 seconds.=0A= + */=0A= + getmicrotime(&cur_time);=0A= + timeout =3D cur_time.tv_sec - start_time.tv_sec;=0A= + if (timeout > 60) {=0A= mps_dprint(sc, MPS_FAULT, "polling failed\n");=0A= error =3D ETIMEDOUT;=0A= break;=0A= }=0A= }=0A= - =0A= +=0A= if (error) {=0A= mps_dprint(sc, MPS_FAULT, "Calling Reinit from %s\n", __func__);=0A= rc =3D mps_reinit(sc);=0A= - mps_dprint(sc, MPS_FAULT, "Reinit %s\n", =0A= - (rc =3D=3D 0) ? "success" : "failed");=0A= + mps_dprint(sc, MPS_FAULT, "Reinit %s\n", (rc =3D=3D 0) ? "success" :=0A= + "failed");=0A= }=0A= =0A= return (error);=0A= --- sys/dev/mps/mpsvar.h.orig 2014-03-26 17:49:01.000000000 +0000=0A= +++ sys/dev/mps/mpsvar.h 2014-04-01 11:30:54.245695971 +0000=0A= @@ -32,7 +32,7 @@=0A= #ifndef _MPSVAR_H=0A= #define _MPSVAR_H=0A= =0A= -#define MPS_DRIVER_VERSION "16.00.00.00-fbsd"=0A= +#define MPS_DRIVER_VERSION "18.00.00.00-fbsd"=0A= =0A= #define MPS_DB_MAX_WAIT 2500=0A= =0A= @@ -417,6 +417,10 @@=0A= =0A= char exclude_ids[80];=0A= struct timeval lastfail;=0A= +=0A= + /* StartStopUnit command handling at shutdown */=0A= + uint32_t SSU_refcount;=0A= + uint8_t SSU_started;=0A= };=0A= =0A= struct mps_config_params {=0A= ------=_NextPart_000_094C_01CF4DB8.50E12F90--