2  *  linux/drivers/message/fusion/mptsas.c
 
   3  *      For use with LSI Logic PCI chip/adapter(s)
 
   4  *      running LSI Logic Fusion MPT (Message Passing Technology) firmware.
 
   6  *  Copyright (c) 1999-2005 LSI Logic Corporation
 
   7  *  (mailto:mpt_linux_developer@lsil.com)
 
   8  *  Copyright (c) 2005-2006 Dell
 
  10 /*=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=*/
 
  12     This program is free software; you can redistribute it and/or modify
 
  13     it under the terms of the GNU General Public License as published by
 
  14     the Free Software Foundation; version 2 of the License.
 
  16     This program is distributed in the hope that it will be useful,
 
  17     but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  18     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
  19     GNU General Public License for more details.
 
  22     THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR
 
  23     CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT
 
  24     LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT,
 
  25     MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is
 
  26     solely responsible for determining the appropriateness of using and
 
  27     distributing the Program and assumes all risks associated with its
 
  28     exercise of rights under this Agreement, including but not limited to
 
  29     the risks and costs of program errors, damage to or loss of data,
 
  30     programs or equipment, and unavailability or interruption of operations.
 
  32     DISCLAIMER OF LIABILITY
 
  33     NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY
 
  34     DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 
  35     DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND
 
  36     ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
 
  37     TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
 
  38     USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED
 
  39     HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES
 
  41     You should have received a copy of the GNU General Public License
 
  42     along with this program; if not, write to the Free Software
 
  43     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
  45 /*=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=*/
 
  47 #include <linux/module.h>
 
  48 #include <linux/kernel.h>
 
  49 #include <linux/init.h>
 
  50 #include <linux/errno.h>
 
  51 #include <linux/sched.h>
 
  52 #include <linux/workqueue.h>
 
  53 #include <linux/delay.h>        /* for mdelay */
 
  55 #include <scsi/scsi.h>
 
  56 #include <scsi/scsi_cmnd.h>
 
  57 #include <scsi/scsi_device.h>
 
  58 #include <scsi/scsi_host.h>
 
  59 #include <scsi/scsi_transport_sas.h>
 
  60 #include <scsi/scsi_dbg.h>
 
  66 #define my_NAME         "Fusion MPT SAS Host driver"
 
  67 #define my_VERSION      MPT_LINUX_VERSION_COMMON
 
  68 #define MYNAM           "mptsas"
 
  71  * Reserved channel for integrated raid
 
  73 #define MPTSAS_RAID_CHANNEL     1
 
  75 MODULE_AUTHOR(MODULEAUTHOR);
 
  76 MODULE_DESCRIPTION(my_NAME);
 
  77 MODULE_LICENSE("GPL");
 
  79 static int mpt_pt_clear;
 
  80 module_param(mpt_pt_clear, int, 0);
 
  81 MODULE_PARM_DESC(mpt_pt_clear,
 
  82                 " Clear persistency table: enable=1  "
 
  83                 "(default=MPTSCSIH_PT_CLEAR=0)");
 
  85 static int      mptsasDoneCtx = -1;
 
  86 static int      mptsasTaskCtx = -1;
 
  87 static int      mptsasInternalCtx = -1; /* Used only for internal commands */
 
  88 static int      mptsasMgmtCtx = -1;
 
  91 enum mptsas_hotplug_action {
 
  99 struct mptsas_hotplug_event {
 
 100         struct work_struct      work;
 
 102         enum mptsas_hotplug_action event_type;
 
 111         u8                      phys_disk_num_valid;
 
 114 struct mptsas_discovery_event {
 
 115         struct work_struct      work;
 
 120  * SAS topology structures
 
 122  * The MPT Fusion firmware interface spreads information about the
 
 123  * SAS topology over many manufacture pages, thus we need some data
 
 124  * structure to collect it and process it for the SAS transport class.
 
 127 struct mptsas_devinfo {
 
 128         u16     handle;         /* unique id to address this device */
 
 129         u16     handle_parent;  /* unique id to address parent device */
 
 130         u16     handle_enclosure; /* enclosure identifier of the enclosure */
 
 131         u16     slot;           /* physical slot in enclosure */
 
 132         u8      phy_id;         /* phy number of parent device */
 
 133         u8      port_id;        /* sas physical port this device
 
 135         u8      id;             /* logical target id of this device */
 
 136         u8      channel;        /* logical bus number of this device */
 
 137         u64     sas_address;    /* WWN of this device,
 
 138                                    SATA is assigned by HBA,expander */
 
 139         u32     device_info;    /* bitfield detailed info about this device */
 
 143  * Specific details on ports, wide/narrow
 
 145 struct mptsas_portinfo_details{
 
 146         u16     num_phys;       /* number of phys belong to this port */
 
 147         u64     phy_bitmask;    /* TODO, extend support for 255 phys */
 
 148         struct sas_rphy *rphy;  /* transport layer rphy object */
 
 149         struct sas_port *port;  /* transport layer port object */
 
 150         struct scsi_target *starget;
 
 151         struct mptsas_portinfo *port_info;
 
 154 struct mptsas_phyinfo {
 
 155         u8      phy_id;                 /* phy index */
 
 156         u8      port_id;                /* firmware port identifier */
 
 157         u8      negotiated_link_rate;   /* nego'd link rate for this phy */
 
 158         u8      hw_link_rate;           /* hardware max/min phys link rate */
 
 159         u8      programmed_link_rate;   /* programmed max/min phy link rate */
 
 160         u8      sas_port_add_phy;       /* flag to request sas_port_add_phy*/
 
 161         struct mptsas_devinfo identify; /* point to phy device info */
 
 162         struct mptsas_devinfo attached; /* point to attached device info */
 
 163         struct sas_phy *phy;            /* transport layer phy object */
 
 164         struct mptsas_portinfo *portinfo;
 
 165         struct mptsas_portinfo_details * port_details;
 
 168 struct mptsas_portinfo {
 
 169         struct list_head list;
 
 170         u16             handle;         /* unique id to address this */
 
 171         u16             num_phys;       /* number of phys */
 
 172         struct mptsas_phyinfo *phy_info;
 
 175 struct mptsas_enclosure {
 
 176         u64     enclosure_logical_id;   /* The WWN for the enclosure */
 
 177         u16     enclosure_handle;       /* unique id to address this */
 
 178         u16     flags;                  /* details enclosure management */
 
 179         u16     num_slot;               /* num slots */
 
 180         u16     start_slot;             /* first slot */
 
 181         u8      start_id;               /* starting logical target id */
 
 182         u8      start_channel;          /* starting logical channel id */
 
 183         u8      sep_id;                 /* SEP device logical target id */
 
 184         u8      sep_channel;            /* SEP channel logical channel id */
 
 188 static void mptsas_print_phy_data(MPI_SAS_IO_UNIT0_PHY_DATA *phy_data)
 
 190         printk("---- IO UNIT PAGE 0 ------------\n");
 
 191         printk("Handle=0x%X\n",
 
 192                 le16_to_cpu(phy_data->AttachedDeviceHandle));
 
 193         printk("Controller Handle=0x%X\n",
 
 194                 le16_to_cpu(phy_data->ControllerDevHandle));
 
 195         printk("Port=0x%X\n", phy_data->Port);
 
 196         printk("Port Flags=0x%X\n", phy_data->PortFlags);
 
 197         printk("PHY Flags=0x%X\n", phy_data->PhyFlags);
 
 198         printk("Negotiated Link Rate=0x%X\n", phy_data->NegotiatedLinkRate);
 
 199         printk("Controller PHY Device Info=0x%X\n",
 
 200                 le32_to_cpu(phy_data->ControllerPhyDeviceInfo));
 
 201         printk("DiscoveryStatus=0x%X\n",
 
 202                 le32_to_cpu(phy_data->DiscoveryStatus));
 
 206 static void mptsas_print_phy_pg0(SasPhyPage0_t *pg0)
 
 210         memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
 
 212         printk("---- SAS PHY PAGE 0 ------------\n");
 
 213         printk("Attached Device Handle=0x%X\n",
 
 214                         le16_to_cpu(pg0->AttachedDevHandle));
 
 215         printk("SAS Address=0x%llX\n",
 
 216                         (unsigned long long)le64_to_cpu(sas_address));
 
 217         printk("Attached PHY Identifier=0x%X\n", pg0->AttachedPhyIdentifier);
 
 218         printk("Attached Device Info=0x%X\n",
 
 219                         le32_to_cpu(pg0->AttachedDeviceInfo));
 
 220         printk("Programmed Link Rate=0x%X\n", pg0->ProgrammedLinkRate);
 
 221         printk("Change Count=0x%X\n", pg0->ChangeCount);
 
 222         printk("PHY Info=0x%X\n", le32_to_cpu(pg0->PhyInfo));
 
 226 static void mptsas_print_phy_pg1(SasPhyPage1_t *pg1)
 
 228         printk("---- SAS PHY PAGE 1 ------------\n");
 
 229         printk("Invalid Dword Count=0x%x\n", pg1->InvalidDwordCount);
 
 230         printk("Running Disparity Error Count=0x%x\n",
 
 231                         pg1->RunningDisparityErrorCount);
 
 232         printk("Loss Dword Synch Count=0x%x\n", pg1->LossDwordSynchCount);
 
 233         printk("PHY Reset Problem Count=0x%x\n", pg1->PhyResetProblemCount);
 
 237 static void mptsas_print_device_pg0(SasDevicePage0_t *pg0)
 
 241         memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
 
 243         printk("---- SAS DEVICE PAGE 0 ---------\n");
 
 244         printk("Handle=0x%X\n" ,le16_to_cpu(pg0->DevHandle));
 
 245         printk("Parent Handle=0x%X\n" ,le16_to_cpu(pg0->ParentDevHandle));
 
 246         printk("Enclosure Handle=0x%X\n", le16_to_cpu(pg0->EnclosureHandle));
 
 247         printk("Slot=0x%X\n", le16_to_cpu(pg0->Slot));
 
 248         printk("SAS Address=0x%llX\n", le64_to_cpu(sas_address));
 
 249         printk("Target ID=0x%X\n", pg0->TargetID);
 
 250         printk("Bus=0x%X\n", pg0->Bus);
 
 251         /* The PhyNum field specifies the PHY number of the parent
 
 252          * device this device is linked to
 
 254         printk("Parent Phy Num=0x%X\n", pg0->PhyNum);
 
 255         printk("Access Status=0x%X\n", le16_to_cpu(pg0->AccessStatus));
 
 256         printk("Device Info=0x%X\n", le32_to_cpu(pg0->DeviceInfo));
 
 257         printk("Flags=0x%X\n", le16_to_cpu(pg0->Flags));
 
 258         printk("Physical Port=0x%X\n", pg0->PhysicalPort);
 
 262 static void mptsas_print_expander_pg1(SasExpanderPage1_t *pg1)
 
 264         printk("---- SAS EXPANDER PAGE 1 ------------\n");
 
 266         printk("Physical Port=0x%X\n", pg1->PhysicalPort);
 
 267         printk("PHY Identifier=0x%X\n", pg1->PhyIdentifier);
 
 268         printk("Negotiated Link Rate=0x%X\n", pg1->NegotiatedLinkRate);
 
 269         printk("Programmed Link Rate=0x%X\n", pg1->ProgrammedLinkRate);
 
 270         printk("Hardware Link Rate=0x%X\n", pg1->HwLinkRate);
 
 271         printk("Owner Device Handle=0x%X\n",
 
 272                         le16_to_cpu(pg1->OwnerDevHandle));
 
 273         printk("Attached Device Handle=0x%X\n",
 
 274                         le16_to_cpu(pg1->AttachedDevHandle));
 
 277 #define mptsas_print_phy_data(phy_data)         do { } while (0)
 
 278 #define mptsas_print_phy_pg0(pg0)               do { } while (0)
 
 279 #define mptsas_print_phy_pg1(pg1)               do { } while (0)
 
 280 #define mptsas_print_device_pg0(pg0)            do { } while (0)
 
 281 #define mptsas_print_expander_pg1(pg1)          do { } while (0)
 
 284 static inline MPT_ADAPTER *phy_to_ioc(struct sas_phy *phy)
 
 286         struct Scsi_Host *shost = dev_to_shost(phy->dev.parent);
 
 287         return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
 
 290 static inline MPT_ADAPTER *rphy_to_ioc(struct sas_rphy *rphy)
 
 292         struct Scsi_Host *shost = dev_to_shost(rphy->dev.parent->parent);
 
 293         return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
 
 297  * mptsas_find_portinfo_by_handle
 
 299  * This function should be called with the sas_topology_mutex already held
 
 301 static struct mptsas_portinfo *
 
 302 mptsas_find_portinfo_by_handle(MPT_ADAPTER *ioc, u16 handle)
 
 304         struct mptsas_portinfo *port_info, *rc=NULL;
 
 307         list_for_each_entry(port_info, &ioc->sas_topology, list)
 
 308                 for (i = 0; i < port_info->num_phys; i++)
 
 309                         if (port_info->phy_info[i].identify.handle == handle) {
 
 318  * Returns true if there is a scsi end device
 
 321 mptsas_is_end_device(struct mptsas_devinfo * attached)
 
 323         if ((attached->sas_address) &&
 
 324             (attached->device_info &
 
 325             MPI_SAS_DEVICE_INFO_END_DEVICE) &&
 
 326             ((attached->device_info &
 
 327             MPI_SAS_DEVICE_INFO_SSP_TARGET) |
 
 328             (attached->device_info &
 
 329             MPI_SAS_DEVICE_INFO_STP_TARGET) |
 
 330             (attached->device_info &
 
 331             MPI_SAS_DEVICE_INFO_SATA_DEVICE)))
 
 339 mptsas_port_delete(struct mptsas_portinfo_details * port_details)
 
 341         struct mptsas_portinfo *port_info;
 
 342         struct mptsas_phyinfo *phy_info;
 
 348         port_info = port_details->port_info;
 
 349         phy_info = port_info->phy_info;
 
 351         dsaswideprintk((KERN_DEBUG "%s: [%p]: num_phys=%02d "
 
 352                 "bitmask=0x%016llX\n",
 
 353                 __FUNCTION__, port_details, port_details->num_phys,
 
 354                     port_details->phy_bitmask));
 
 356         for (i = 0; i < port_info->num_phys; i++, phy_info++) {
 
 357                 if(phy_info->port_details != port_details)
 
 359                 memset(&phy_info->attached, 0, sizeof(struct mptsas_devinfo));
 
 360                 phy_info->port_details = NULL;
 
 365 static inline struct sas_rphy *
 
 366 mptsas_get_rphy(struct mptsas_phyinfo *phy_info)
 
 368         if (phy_info->port_details)
 
 369                 return phy_info->port_details->rphy;
 
 375 mptsas_set_rphy(struct mptsas_phyinfo *phy_info, struct sas_rphy *rphy)
 
 377         if (phy_info->port_details) {
 
 378                 phy_info->port_details->rphy = rphy;
 
 379                 dsaswideprintk((KERN_DEBUG "sas_rphy_add: rphy=%p\n", rphy));
 
 382 #ifdef MPT_DEBUG_SAS_WIDE
 
 384                 dev_printk(KERN_DEBUG, &rphy->dev, "add:");
 
 385                 printk("rphy=%p release=%p\n",
 
 386                         rphy, rphy->dev.release);
 
 391 static inline struct sas_port *
 
 392 mptsas_get_port(struct mptsas_phyinfo *phy_info)
 
 394         if (phy_info->port_details)
 
 395                 return phy_info->port_details->port;
 
 401 mptsas_set_port(struct mptsas_phyinfo *phy_info, struct sas_port *port)
 
 403         if (phy_info->port_details)
 
 404                 phy_info->port_details->port = port;
 
 406 #ifdef MPT_DEBUG_SAS_WIDE
 
 408                 dev_printk(KERN_DEBUG, &port->dev, "add: ");
 
 409                 printk("port=%p release=%p\n",
 
 410                         port, port->dev.release);
 
 415 static inline struct scsi_target *
 
 416 mptsas_get_starget(struct mptsas_phyinfo *phy_info)
 
 418         if (phy_info->port_details)
 
 419                 return phy_info->port_details->starget;
 
 425 mptsas_set_starget(struct mptsas_phyinfo *phy_info, struct scsi_target *
 
 428         if (phy_info->port_details)
 
 429                 phy_info->port_details->starget = starget;
 
 434  * mptsas_setup_wide_ports
 
 436  * Updates for new and existing narrow/wide port configuration
 
 437  * in the sas_topology
 
 440 mptsas_setup_wide_ports(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
 
 442         struct mptsas_portinfo_details * port_details;
 
 443         struct mptsas_phyinfo *phy_info, *phy_info_cmp;
 
 447         mutex_lock(&ioc->sas_topology_mutex);
 
 449         phy_info = port_info->phy_info;
 
 450         for (i = 0 ; i < port_info->num_phys ; i++, phy_info++) {
 
 451                 if (phy_info->attached.handle)
 
 453                 port_details = phy_info->port_details;
 
 456                 if (port_details->num_phys < 2)
 
 459                  * Removing a phy from a port, letting the last
 
 460                  * phy be removed by firmware events.
 
 462                 dsaswideprintk((KERN_DEBUG
 
 463                         "%s: [%p]: deleting phy = %d\n",
 
 464                         __FUNCTION__, port_details, i));
 
 465                 port_details->num_phys--;
 
 466                 port_details->phy_bitmask &= ~ (1 << phy_info->phy_id);
 
 467                 memset(&phy_info->attached, 0, sizeof(struct mptsas_devinfo));
 
 468                 sas_port_delete_phy(port_details->port, phy_info->phy);
 
 469                 phy_info->port_details = NULL;
 
 473          * Populate and refresh the tree
 
 475         phy_info = port_info->phy_info;
 
 476         for (i = 0 ; i < port_info->num_phys ; i++, phy_info++) {
 
 477                 sas_address = phy_info->attached.sas_address;
 
 478                 dsaswideprintk((KERN_DEBUG "phy_id=%d sas_address=0x%018llX\n",
 
 482                 port_details = phy_info->port_details;
 
 487                         port_details = kzalloc(sizeof(*port_details),
 
 491                         port_details->num_phys = 1;
 
 492                         port_details->port_info = port_info;
 
 493                         if (phy_info->phy_id < 64 )
 
 494                                 port_details->phy_bitmask |=
 
 495                                     (1 << phy_info->phy_id);
 
 496                         phy_info->sas_port_add_phy=1;
 
 497                         dsaswideprintk((KERN_DEBUG "\t\tForming port\n\t\t"
 
 498                                 "phy_id=%d sas_address=0x%018llX\n",
 
 500                         phy_info->port_details = port_details;
 
 503                 if (i == port_info->num_phys - 1)
 
 505                 phy_info_cmp = &port_info->phy_info[i + 1];
 
 506                 for (j = i + 1 ; j < port_info->num_phys ; j++,
 
 508                         if (!phy_info_cmp->attached.sas_address)
 
 510                         if (sas_address != phy_info_cmp->attached.sas_address)
 
 512                         if (phy_info_cmp->port_details == port_details )
 
 514                         dsaswideprintk((KERN_DEBUG
 
 515                                 "\t\tphy_id=%d sas_address=0x%018llX\n",
 
 516                                 j, phy_info_cmp->attached.sas_address));
 
 517                         if (phy_info_cmp->port_details) {
 
 519                                     mptsas_get_rphy(phy_info_cmp);
 
 521                                     mptsas_get_port(phy_info_cmp);
 
 522                                 port_details->starget =
 
 523                                     mptsas_get_starget(phy_info_cmp);
 
 524                                 port_details->num_phys =
 
 525                                         phy_info_cmp->port_details->num_phys;
 
 526                                 if (!phy_info_cmp->port_details->num_phys)
 
 527                                         kfree(phy_info_cmp->port_details);
 
 529                                 phy_info_cmp->sas_port_add_phy=1;
 
 531                          * Adding a phy to a port
 
 533                         phy_info_cmp->port_details = port_details;
 
 534                         if (phy_info_cmp->phy_id < 64 )
 
 535                                 port_details->phy_bitmask |=
 
 536                                 (1 << phy_info_cmp->phy_id);
 
 537                         port_details->num_phys++;
 
 543 #ifdef MPT_DEBUG_SAS_WIDE
 
 544         for (i = 0; i < port_info->num_phys; i++) {
 
 545                 port_details = port_info->phy_info[i].port_details;
 
 548                 dsaswideprintk((KERN_DEBUG
 
 549                         "%s: [%p]: phy_id=%02d num_phys=%02d "
 
 550                         "bitmask=0x%016llX\n",
 
 552                         port_details, i, port_details->num_phys,
 
 553                         port_details->phy_bitmask));
 
 554                 dsaswideprintk((KERN_DEBUG"\t\tport = %p rphy=%p\n",
 
 555                         port_details->port, port_details->rphy));
 
 557         dsaswideprintk((KERN_DEBUG"\n"));
 
 559         mutex_unlock(&ioc->sas_topology_mutex);
 
 563 mptsas_target_reset(MPT_ADAPTER *ioc, VirtTarget * vtarget)
 
 565         MPT_SCSI_HOST           *hd = (MPT_SCSI_HOST *)ioc->sh->hostdata;
 
 567         if (mptscsih_TMHandler(hd,
 
 568              MPI_SCSITASKMGMT_TASKTYPE_TARGET_RESET,
 
 569              vtarget->bus_id, vtarget->target_id, 0, 0, 5) < 0) {
 
 571                 hd->tmState = TM_STATE_NONE;
 
 572                 printk(MYIOC_s_WARN_FMT
 
 573                "Error processing TaskMgmt id=%d TARGET_RESET\n",
 
 574                         ioc->name, vtarget->target_id);
 
 579 mptsas_sas_enclosure_pg0(MPT_ADAPTER *ioc, struct mptsas_enclosure *enclosure,
 
 580                 u32 form, u32 form_specific)
 
 582         ConfigExtendedPageHeader_t hdr;
 
 584         SasEnclosurePage0_t *buffer;
 
 585         dma_addr_t dma_handle;
 
 587         __le64 le_identifier;
 
 589         memset(&hdr, 0, sizeof(hdr));
 
 590         hdr.PageVersion = MPI_SASENCLOSURE0_PAGEVERSION;
 
 592         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
 593         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_ENCLOSURE;
 
 595         cfg.cfghdr.ehdr = &hdr;
 
 597         cfg.pageAddr = form + form_specific;
 
 598         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
 599         cfg.dir = 0;    /* read */
 
 602         error = mpt_config(ioc, &cfg);
 
 605         if (!hdr.ExtPageLength) {
 
 610         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
 617         cfg.physAddr = dma_handle;
 
 618         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
 620         error = mpt_config(ioc, &cfg);
 
 622                 goto out_free_consistent;
 
 624         /* save config data */
 
 625         memcpy(&le_identifier, &buffer->EnclosureLogicalID, sizeof(__le64));
 
 626         enclosure->enclosure_logical_id = le64_to_cpu(le_identifier);
 
 627         enclosure->enclosure_handle = le16_to_cpu(buffer->EnclosureHandle);
 
 628         enclosure->flags = le16_to_cpu(buffer->Flags);
 
 629         enclosure->num_slot = le16_to_cpu(buffer->NumSlots);
 
 630         enclosure->start_slot = le16_to_cpu(buffer->StartSlot);
 
 631         enclosure->start_id = buffer->StartTargetID;
 
 632         enclosure->start_channel = buffer->StartBus;
 
 633         enclosure->sep_id = buffer->SEPTargetID;
 
 634         enclosure->sep_channel = buffer->SEPBus;
 
 637         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
 644 mptsas_slave_configure(struct scsi_device *sdev)
 
 647         if (sdev->channel == MPTSAS_RAID_CHANNEL)
 
 650         sas_read_port_mode_page(sdev);
 
 653         return mptscsih_slave_configure(sdev);
 
 657 mptsas_target_alloc(struct scsi_target *starget)
 
 659         struct Scsi_Host *host = dev_to_shost(&starget->dev);
 
 660         MPT_SCSI_HOST           *hd = (MPT_SCSI_HOST *)host->hostdata;
 
 664         struct sas_rphy         *rphy;
 
 665         struct mptsas_portinfo  *p;
 
 668         vtarget = kzalloc(sizeof(VirtTarget), GFP_KERNEL);
 
 672         vtarget->starget = starget;
 
 673         vtarget->ioc_id = hd->ioc->id;
 
 674         vtarget->tflags = MPT_TARGET_FLAGS_Q_YES|MPT_TARGET_FLAGS_VALID_INQUIRY;
 
 676         target_id = starget->id;
 
 679         hd->Targets[target_id] = vtarget;
 
 681         if (starget->channel == MPTSAS_RAID_CHANNEL)
 
 684         rphy = dev_to_rphy(starget->dev.parent);
 
 685         mutex_lock(&hd->ioc->sas_topology_mutex);
 
 686         list_for_each_entry(p, &hd->ioc->sas_topology, list) {
 
 687                 for (i = 0; i < p->num_phys; i++) {
 
 688                         if (p->phy_info[i].attached.sas_address !=
 
 689                                         rphy->identify.sas_address)
 
 691                         target_id = p->phy_info[i].attached.id;
 
 692                         channel = p->phy_info[i].attached.channel;
 
 693                         mptsas_set_starget(&p->phy_info[i], starget);
 
 696                          * Exposing hidden raid components
 
 698                         if (mptscsih_is_phys_disk(hd->ioc, target_id)) {
 
 699                                 target_id = mptscsih_raid_id_to_num(hd,
 
 702                                     MPT_TARGET_FLAGS_RAID_COMPONENT;
 
 704                         mutex_unlock(&hd->ioc->sas_topology_mutex);
 
 708         mutex_unlock(&hd->ioc->sas_topology_mutex);
 
 714         vtarget->target_id = target_id;
 
 715         vtarget->bus_id = channel;
 
 716         starget->hostdata = vtarget;
 
 721 mptsas_target_destroy(struct scsi_target *starget)
 
 723         struct Scsi_Host *host = dev_to_shost(&starget->dev);
 
 724         MPT_SCSI_HOST           *hd = (MPT_SCSI_HOST *)host->hostdata;
 
 725         struct sas_rphy         *rphy;
 
 726         struct mptsas_portinfo  *p;
 
 729         if (!starget->hostdata)
 
 732         if (starget->channel == MPTSAS_RAID_CHANNEL)
 
 735         rphy = dev_to_rphy(starget->dev.parent);
 
 736         list_for_each_entry(p, &hd->ioc->sas_topology, list) {
 
 737                 for (i = 0; i < p->num_phys; i++) {
 
 738                         if (p->phy_info[i].attached.sas_address !=
 
 739                                         rphy->identify.sas_address)
 
 741                         mptsas_set_starget(&p->phy_info[i], NULL);
 
 747         kfree(starget->hostdata);
 
 748         starget->hostdata = NULL;
 
 753 mptsas_slave_alloc(struct scsi_device *sdev)
 
 755         struct Scsi_Host        *host = sdev->host;
 
 756         MPT_SCSI_HOST           *hd = (MPT_SCSI_HOST *)host->hostdata;
 
 757         struct sas_rphy         *rphy;
 
 758         struct mptsas_portinfo  *p;
 
 760         struct scsi_target      *starget;
 
 763         vdev = kzalloc(sizeof(VirtDevice), GFP_KERNEL);
 
 765                 printk(MYIOC_s_ERR_FMT "slave_alloc kzalloc(%zd) FAILED!\n",
 
 766                                 hd->ioc->name, sizeof(VirtDevice));
 
 769         starget = scsi_target(sdev);
 
 770         vdev->vtarget = starget->hostdata;
 
 772         if (sdev->channel == MPTSAS_RAID_CHANNEL)
 
 775         rphy = dev_to_rphy(sdev->sdev_target->dev.parent);
 
 776         mutex_lock(&hd->ioc->sas_topology_mutex);
 
 777         list_for_each_entry(p, &hd->ioc->sas_topology, list) {
 
 778                 for (i = 0; i < p->num_phys; i++) {
 
 779                         if (p->phy_info[i].attached.sas_address !=
 
 780                                         rphy->identify.sas_address)
 
 782                         vdev->lun = sdev->lun;
 
 784                          * Exposing hidden raid components
 
 786                         if (mptscsih_is_phys_disk(hd->ioc,
 
 787                                         p->phy_info[i].attached.id))
 
 788                                 sdev->no_uld_attach = 1;
 
 789                         mutex_unlock(&hd->ioc->sas_topology_mutex);
 
 793         mutex_unlock(&hd->ioc->sas_topology_mutex);
 
 799         vdev->vtarget->num_luns++;
 
 800         sdev->hostdata = vdev;
 
 805 mptsas_qcmd(struct scsi_cmnd *SCpnt, void (*done)(struct scsi_cmnd *))
 
 807         VirtDevice      *vdev = SCpnt->device->hostdata;
 
 809 //      scsi_print_command(SCpnt);
 
 810         if (vdev->vtarget->deleted) {
 
 811                 SCpnt->result = DID_NO_CONNECT << 16;
 
 816         return mptscsih_qcmd(SCpnt,done);
 
 820 static struct scsi_host_template mptsas_driver_template = {
 
 821         .module                         = THIS_MODULE,
 
 822         .proc_name                      = "mptsas",
 
 823         .proc_info                      = mptscsih_proc_info,
 
 824         .name                           = "MPT SPI Host",
 
 825         .info                           = mptscsih_info,
 
 826         .queuecommand                   = mptsas_qcmd,
 
 827         .target_alloc                   = mptsas_target_alloc,
 
 828         .slave_alloc                    = mptsas_slave_alloc,
 
 829         .slave_configure                = mptsas_slave_configure,
 
 830         .target_destroy                 = mptsas_target_destroy,
 
 831         .slave_destroy                  = mptscsih_slave_destroy,
 
 832         .change_queue_depth             = mptscsih_change_queue_depth,
 
 833         .eh_abort_handler               = mptscsih_abort,
 
 834         .eh_device_reset_handler        = mptscsih_dev_reset,
 
 835         .eh_bus_reset_handler           = mptscsih_bus_reset,
 
 836         .eh_host_reset_handler          = mptscsih_host_reset,
 
 837         .bios_param                     = mptscsih_bios_param,
 
 838         .can_queue                      = MPT_FC_CAN_QUEUE,
 
 840         .sg_tablesize                   = MPT_SCSI_SG_DEPTH,
 
 843         .use_clustering                 = ENABLE_CLUSTERING,
 
 846 static int mptsas_get_linkerrors(struct sas_phy *phy)
 
 848         MPT_ADAPTER *ioc = phy_to_ioc(phy);
 
 849         ConfigExtendedPageHeader_t hdr;
 
 851         SasPhyPage1_t *buffer;
 
 852         dma_addr_t dma_handle;
 
 855         /* FIXME: only have link errors on local phys */
 
 856         if (!scsi_is_sas_phy_local(phy))
 
 859         hdr.PageVersion = MPI_SASPHY1_PAGEVERSION;
 
 860         hdr.ExtPageLength = 0;
 
 861         hdr.PageNumber = 1 /* page number 1*/;
 
 864         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
 865         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_PHY;
 
 867         cfg.cfghdr.ehdr = &hdr;
 
 869         cfg.pageAddr = phy->identify.phy_identifier;
 
 870         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
 871         cfg.dir = 0;    /* read */
 
 874         error = mpt_config(ioc, &cfg);
 
 877         if (!hdr.ExtPageLength)
 
 880         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
 885         cfg.physAddr = dma_handle;
 
 886         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
 888         error = mpt_config(ioc, &cfg);
 
 890                 goto out_free_consistent;
 
 892         mptsas_print_phy_pg1(buffer);
 
 894         phy->invalid_dword_count = le32_to_cpu(buffer->InvalidDwordCount);
 
 895         phy->running_disparity_error_count =
 
 896                 le32_to_cpu(buffer->RunningDisparityErrorCount);
 
 897         phy->loss_of_dword_sync_count =
 
 898                 le32_to_cpu(buffer->LossDwordSynchCount);
 
 899         phy->phy_reset_problem_count =
 
 900                 le32_to_cpu(buffer->PhyResetProblemCount);
 
 903         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
 908 static int mptsas_mgmt_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *req,
 
 909                 MPT_FRAME_HDR *reply)
 
 911         ioc->sas_mgmt.status |= MPT_SAS_MGMT_STATUS_COMMAND_GOOD;
 
 913                 ioc->sas_mgmt.status |= MPT_SAS_MGMT_STATUS_RF_VALID;
 
 914                 memcpy(ioc->sas_mgmt.reply, reply,
 
 915                     min(ioc->reply_sz, 4 * reply->u.reply.MsgLength));
 
 917         complete(&ioc->sas_mgmt.done);
 
 921 static int mptsas_phy_reset(struct sas_phy *phy, int hard_reset)
 
 923         MPT_ADAPTER *ioc = phy_to_ioc(phy);
 
 924         SasIoUnitControlRequest_t *req;
 
 925         SasIoUnitControlReply_t *reply;
 
 928         unsigned long timeleft;
 
 929         int error = -ERESTARTSYS;
 
 931         /* FIXME: fusion doesn't allow non-local phy reset */
 
 932         if (!scsi_is_sas_phy_local(phy))
 
 935         /* not implemented for expanders */
 
 936         if (phy->identify.target_port_protocols & SAS_PROTOCOL_SMP)
 
 939         if (mutex_lock_interruptible(&ioc->sas_mgmt.mutex))
 
 942         mf = mpt_get_msg_frame(mptsasMgmtCtx, ioc);
 
 948         hdr = (MPIHeader_t *) mf;
 
 949         req = (SasIoUnitControlRequest_t *)mf;
 
 950         memset(req, 0, sizeof(SasIoUnitControlRequest_t));
 
 951         req->Function = MPI_FUNCTION_SAS_IO_UNIT_CONTROL;
 
 952         req->MsgContext = hdr->MsgContext;
 
 953         req->Operation = hard_reset ?
 
 954                 MPI_SAS_OP_PHY_HARD_RESET : MPI_SAS_OP_PHY_LINK_RESET;
 
 955         req->PhyNum = phy->identify.phy_identifier;
 
 957         mpt_put_msg_frame(mptsasMgmtCtx, ioc, mf);
 
 959         timeleft = wait_for_completion_timeout(&ioc->sas_mgmt.done,
 
 962                 /* On timeout reset the board */
 
 963                 mpt_free_msg_frame(ioc, mf);
 
 964                 mpt_HardResetHandler(ioc, CAN_SLEEP);
 
 969         /* a reply frame is expected */
 
 970         if ((ioc->sas_mgmt.status &
 
 971             MPT_IOCTL_STATUS_RF_VALID) == 0) {
 
 976         /* process the completed Reply Message Frame */
 
 977         reply = (SasIoUnitControlReply_t *)ioc->sas_mgmt.reply;
 
 978         if (reply->IOCStatus != MPI_IOCSTATUS_SUCCESS) {
 
 979                 printk("%s: IOCStatus=0x%X IOCLogInfo=0x%X\n",
 
 990         mutex_unlock(&ioc->sas_mgmt.mutex);
 
 996 mptsas_get_enclosure_identifier(struct sas_rphy *rphy, u64 *identifier)
 
 998         MPT_ADAPTER *ioc = rphy_to_ioc(rphy);
 
1000         struct mptsas_portinfo *p;
 
1001         struct mptsas_enclosure enclosure_info;
 
1002         u64 enclosure_handle;
 
1004         mutex_lock(&ioc->sas_topology_mutex);
 
1005         list_for_each_entry(p, &ioc->sas_topology, list) {
 
1006                 for (i = 0; i < p->num_phys; i++) {
 
1007                         if (p->phy_info[i].attached.sas_address ==
 
1008                             rphy->identify.sas_address) {
 
1009                                 enclosure_handle = p->phy_info[i].
 
1010                                         attached.handle_enclosure;
 
1015         mutex_unlock(&ioc->sas_topology_mutex);
 
1019         mutex_unlock(&ioc->sas_topology_mutex);
 
1020         memset(&enclosure_info, 0, sizeof(struct mptsas_enclosure));
 
1021         error = mptsas_sas_enclosure_pg0(ioc, &enclosure_info,
 
1022                         (MPI_SAS_ENCLOS_PGAD_FORM_HANDLE <<
 
1023                          MPI_SAS_ENCLOS_PGAD_FORM_SHIFT), enclosure_handle);
 
1025                 *identifier = enclosure_info.enclosure_logical_id;
 
1030 mptsas_get_bay_identifier(struct sas_rphy *rphy)
 
1032         MPT_ADAPTER *ioc = rphy_to_ioc(rphy);
 
1033         struct mptsas_portinfo *p;
 
1036         mutex_lock(&ioc->sas_topology_mutex);
 
1037         list_for_each_entry(p, &ioc->sas_topology, list) {
 
1038                 for (i = 0; i < p->num_phys; i++) {
 
1039                         if (p->phy_info[i].attached.sas_address ==
 
1040                             rphy->identify.sas_address) {
 
1041                                 rc = p->phy_info[i].attached.slot;
 
1048         mutex_unlock(&ioc->sas_topology_mutex);
 
1052 static struct sas_function_template mptsas_transport_functions = {
 
1053         .get_linkerrors         = mptsas_get_linkerrors,
 
1054         .get_enclosure_identifier = mptsas_get_enclosure_identifier,
 
1055         .get_bay_identifier     = mptsas_get_bay_identifier,
 
1056         .phy_reset              = mptsas_phy_reset,
 
1059 static struct scsi_transport_template *mptsas_transport_template;
 
1062 mptsas_sas_io_unit_pg0(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
 
1064         ConfigExtendedPageHeader_t hdr;
 
1066         SasIOUnitPage0_t *buffer;
 
1067         dma_addr_t dma_handle;
 
1070         hdr.PageVersion = MPI_SASIOUNITPAGE0_PAGEVERSION;
 
1071         hdr.ExtPageLength = 0;
 
1075         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
1076         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
 
1078         cfg.cfghdr.ehdr = &hdr;
 
1081         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
1082         cfg.dir = 0;    /* read */
 
1085         error = mpt_config(ioc, &cfg);
 
1088         if (!hdr.ExtPageLength) {
 
1093         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1100         cfg.physAddr = dma_handle;
 
1101         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
1103         error = mpt_config(ioc, &cfg);
 
1105                 goto out_free_consistent;
 
1107         port_info->num_phys = buffer->NumPhys;
 
1108         port_info->phy_info = kcalloc(port_info->num_phys,
 
1109                 sizeof(*port_info->phy_info),GFP_KERNEL);
 
1110         if (!port_info->phy_info) {
 
1112                 goto out_free_consistent;
 
1115         if (port_info->num_phys)
 
1117                     le16_to_cpu(buffer->PhyData[0].ControllerDevHandle);
 
1118         for (i = 0; i < port_info->num_phys; i++) {
 
1119                 mptsas_print_phy_data(&buffer->PhyData[i]);
 
1120                 port_info->phy_info[i].phy_id = i;
 
1121                 port_info->phy_info[i].port_id =
 
1122                     buffer->PhyData[i].Port;
 
1123                 port_info->phy_info[i].negotiated_link_rate =
 
1124                     buffer->PhyData[i].NegotiatedLinkRate;
 
1125                 port_info->phy_info[i].portinfo = port_info;
 
1128  out_free_consistent:
 
1129         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1130                             buffer, dma_handle);
 
1136 mptsas_sas_phy_pg0(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info,
 
1137                 u32 form, u32 form_specific)
 
1139         ConfigExtendedPageHeader_t hdr;
 
1141         SasPhyPage0_t *buffer;
 
1142         dma_addr_t dma_handle;
 
1145         hdr.PageVersion = MPI_SASPHY0_PAGEVERSION;
 
1146         hdr.ExtPageLength = 0;
 
1150         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
1151         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_PHY;
 
1153         cfg.cfghdr.ehdr = &hdr;
 
1154         cfg.dir = 0;    /* read */
 
1157         /* Get Phy Pg 0 for each Phy. */
 
1159         cfg.pageAddr = form + form_specific;
 
1160         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
1162         error = mpt_config(ioc, &cfg);
 
1166         if (!hdr.ExtPageLength) {
 
1171         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1178         cfg.physAddr = dma_handle;
 
1179         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
1181         error = mpt_config(ioc, &cfg);
 
1183                 goto out_free_consistent;
 
1185         mptsas_print_phy_pg0(buffer);
 
1187         phy_info->hw_link_rate = buffer->HwLinkRate;
 
1188         phy_info->programmed_link_rate = buffer->ProgrammedLinkRate;
 
1189         phy_info->identify.handle = le16_to_cpu(buffer->OwnerDevHandle);
 
1190         phy_info->attached.handle = le16_to_cpu(buffer->AttachedDevHandle);
 
1192  out_free_consistent:
 
1193         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1194                             buffer, dma_handle);
 
1200 mptsas_sas_device_pg0(MPT_ADAPTER *ioc, struct mptsas_devinfo *device_info,
 
1201                 u32 form, u32 form_specific)
 
1203         ConfigExtendedPageHeader_t hdr;
 
1205         SasDevicePage0_t *buffer;
 
1206         dma_addr_t dma_handle;
 
1210         if (ioc->sas_discovery_runtime &&
 
1211                 mptsas_is_end_device(device_info))
 
1214         hdr.PageVersion = MPI_SASDEVICE0_PAGEVERSION;
 
1215         hdr.ExtPageLength = 0;
 
1219         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
1220         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_DEVICE;
 
1222         cfg.cfghdr.ehdr = &hdr;
 
1223         cfg.pageAddr = form + form_specific;
 
1225         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
1226         cfg.dir = 0;    /* read */
 
1229         memset(device_info, 0, sizeof(struct mptsas_devinfo));
 
1230         error = mpt_config(ioc, &cfg);
 
1233         if (!hdr.ExtPageLength) {
 
1238         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1245         cfg.physAddr = dma_handle;
 
1246         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
1248         error = mpt_config(ioc, &cfg);
 
1250                 goto out_free_consistent;
 
1252         mptsas_print_device_pg0(buffer);
 
1254         device_info->handle = le16_to_cpu(buffer->DevHandle);
 
1255         device_info->handle_parent = le16_to_cpu(buffer->ParentDevHandle);
 
1256         device_info->handle_enclosure =
 
1257             le16_to_cpu(buffer->EnclosureHandle);
 
1258         device_info->slot = le16_to_cpu(buffer->Slot);
 
1259         device_info->phy_id = buffer->PhyNum;
 
1260         device_info->port_id = buffer->PhysicalPort;
 
1261         device_info->id = buffer->TargetID;
 
1262         device_info->channel = buffer->Bus;
 
1263         memcpy(&sas_address, &buffer->SASAddress, sizeof(__le64));
 
1264         device_info->sas_address = le64_to_cpu(sas_address);
 
1265         device_info->device_info =
 
1266             le32_to_cpu(buffer->DeviceInfo);
 
1268  out_free_consistent:
 
1269         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1270                             buffer, dma_handle);
 
1276 mptsas_sas_expander_pg0(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info,
 
1277                 u32 form, u32 form_specific)
 
1279         ConfigExtendedPageHeader_t hdr;
 
1281         SasExpanderPage0_t *buffer;
 
1282         dma_addr_t dma_handle;
 
1285         hdr.PageVersion = MPI_SASEXPANDER0_PAGEVERSION;
 
1286         hdr.ExtPageLength = 0;
 
1290         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
1291         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_EXPANDER;
 
1293         cfg.cfghdr.ehdr = &hdr;
 
1295         cfg.pageAddr = form + form_specific;
 
1296         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
1297         cfg.dir = 0;    /* read */
 
1300         memset(port_info, 0, sizeof(struct mptsas_portinfo));
 
1301         error = mpt_config(ioc, &cfg);
 
1305         if (!hdr.ExtPageLength) {
 
1310         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1317         cfg.physAddr = dma_handle;
 
1318         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
1320         error = mpt_config(ioc, &cfg);
 
1322                 goto out_free_consistent;
 
1324         /* save config data */
 
1325         port_info->num_phys = buffer->NumPhys;
 
1326         port_info->handle = le16_to_cpu(buffer->DevHandle);
 
1327         port_info->phy_info = kcalloc(port_info->num_phys,
 
1328                 sizeof(*port_info->phy_info),GFP_KERNEL);
 
1329         if (!port_info->phy_info) {
 
1331                 goto out_free_consistent;
 
1334         for (i = 0; i < port_info->num_phys; i++)
 
1335                 port_info->phy_info[i].portinfo = port_info;
 
1337  out_free_consistent:
 
1338         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1339                             buffer, dma_handle);
 
1345 mptsas_sas_expander_pg1(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info,
 
1346                 u32 form, u32 form_specific)
 
1348         ConfigExtendedPageHeader_t hdr;
 
1350         SasExpanderPage1_t *buffer;
 
1351         dma_addr_t dma_handle;
 
1354         if (ioc->sas_discovery_runtime &&
 
1355                 mptsas_is_end_device(&phy_info->attached))
 
1358         hdr.PageVersion = MPI_SASEXPANDER0_PAGEVERSION;
 
1359         hdr.ExtPageLength = 0;
 
1363         hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
 
1364         hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_EXPANDER;
 
1366         cfg.cfghdr.ehdr = &hdr;
 
1368         cfg.pageAddr = form + form_specific;
 
1369         cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
 
1370         cfg.dir = 0;    /* read */
 
1373         error = mpt_config(ioc, &cfg);
 
1377         if (!hdr.ExtPageLength) {
 
1382         buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1389         cfg.physAddr = dma_handle;
 
1390         cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
 
1392         error = mpt_config(ioc, &cfg);
 
1394                 goto out_free_consistent;
 
1397         mptsas_print_expander_pg1(buffer);
 
1399         /* save config data */
 
1400         phy_info->phy_id = buffer->PhyIdentifier;
 
1401         phy_info->port_id = buffer->PhysicalPort;
 
1402         phy_info->negotiated_link_rate = buffer->NegotiatedLinkRate;
 
1403         phy_info->programmed_link_rate = buffer->ProgrammedLinkRate;
 
1404         phy_info->hw_link_rate = buffer->HwLinkRate;
 
1405         phy_info->identify.handle = le16_to_cpu(buffer->OwnerDevHandle);
 
1406         phy_info->attached.handle = le16_to_cpu(buffer->AttachedDevHandle);
 
1408  out_free_consistent:
 
1409         pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
 
1410                             buffer, dma_handle);
 
1416 mptsas_parse_device_info(struct sas_identify *identify,
 
1417                 struct mptsas_devinfo *device_info)
 
1421         identify->sas_address = device_info->sas_address;
 
1422         identify->phy_identifier = device_info->phy_id;
 
1425          * Fill in Phy Initiator Port Protocol.
 
1426          * Bits 6:3, more than one bit can be set, fall through cases.
 
1428         protocols = device_info->device_info & 0x78;
 
1429         identify->initiator_port_protocols = 0;
 
1430         if (protocols & MPI_SAS_DEVICE_INFO_SSP_INITIATOR)
 
1431                 identify->initiator_port_protocols |= SAS_PROTOCOL_SSP;
 
1432         if (protocols & MPI_SAS_DEVICE_INFO_STP_INITIATOR)
 
1433                 identify->initiator_port_protocols |= SAS_PROTOCOL_STP;
 
1434         if (protocols & MPI_SAS_DEVICE_INFO_SMP_INITIATOR)
 
1435                 identify->initiator_port_protocols |= SAS_PROTOCOL_SMP;
 
1436         if (protocols & MPI_SAS_DEVICE_INFO_SATA_HOST)
 
1437                 identify->initiator_port_protocols |= SAS_PROTOCOL_SATA;
 
1440          * Fill in Phy Target Port Protocol.
 
1441          * Bits 10:7, more than one bit can be set, fall through cases.
 
1443         protocols = device_info->device_info & 0x780;
 
1444         identify->target_port_protocols = 0;
 
1445         if (protocols & MPI_SAS_DEVICE_INFO_SSP_TARGET)
 
1446                 identify->target_port_protocols |= SAS_PROTOCOL_SSP;
 
1447         if (protocols & MPI_SAS_DEVICE_INFO_STP_TARGET)
 
1448                 identify->target_port_protocols |= SAS_PROTOCOL_STP;
 
1449         if (protocols & MPI_SAS_DEVICE_INFO_SMP_TARGET)
 
1450                 identify->target_port_protocols |= SAS_PROTOCOL_SMP;
 
1451         if (protocols & MPI_SAS_DEVICE_INFO_SATA_DEVICE)
 
1452                 identify->target_port_protocols |= SAS_PROTOCOL_SATA;
 
1455          * Fill in Attached device type.
 
1457         switch (device_info->device_info &
 
1458                         MPI_SAS_DEVICE_INFO_MASK_DEVICE_TYPE) {
 
1459         case MPI_SAS_DEVICE_INFO_NO_DEVICE:
 
1460                 identify->device_type = SAS_PHY_UNUSED;
 
1462         case MPI_SAS_DEVICE_INFO_END_DEVICE:
 
1463                 identify->device_type = SAS_END_DEVICE;
 
1465         case MPI_SAS_DEVICE_INFO_EDGE_EXPANDER:
 
1466                 identify->device_type = SAS_EDGE_EXPANDER_DEVICE;
 
1468         case MPI_SAS_DEVICE_INFO_FANOUT_EXPANDER:
 
1469                 identify->device_type = SAS_FANOUT_EXPANDER_DEVICE;
 
1474 static int mptsas_probe_one_phy(struct device *dev,
 
1475                 struct mptsas_phyinfo *phy_info, int index, int local)
 
1478         struct sas_phy *phy;
 
1479         struct sas_port *port;
 
1487         if (!phy_info->phy) {
 
1488                 phy = sas_phy_alloc(dev, index);
 
1494                 phy = phy_info->phy;
 
1496         mptsas_parse_device_info(&phy->identify, &phy_info->identify);
 
1499          * Set Negotiated link rate.
 
1501         switch (phy_info->negotiated_link_rate) {
 
1502         case MPI_SAS_IOUNIT0_RATE_PHY_DISABLED:
 
1503                 phy->negotiated_linkrate = SAS_PHY_DISABLED;
 
1505         case MPI_SAS_IOUNIT0_RATE_FAILED_SPEED_NEGOTIATION:
 
1506                 phy->negotiated_linkrate = SAS_LINK_RATE_FAILED;
 
1508         case MPI_SAS_IOUNIT0_RATE_1_5:
 
1509                 phy->negotiated_linkrate = SAS_LINK_RATE_1_5_GBPS;
 
1511         case MPI_SAS_IOUNIT0_RATE_3_0:
 
1512                 phy->negotiated_linkrate = SAS_LINK_RATE_3_0_GBPS;
 
1514         case MPI_SAS_IOUNIT0_RATE_SATA_OOB_COMPLETE:
 
1515         case MPI_SAS_IOUNIT0_RATE_UNKNOWN:
 
1517                 phy->negotiated_linkrate = SAS_LINK_RATE_UNKNOWN;
 
1522          * Set Max hardware link rate.
 
1524         switch (phy_info->hw_link_rate & MPI_SAS_PHY0_PRATE_MAX_RATE_MASK) {
 
1525         case MPI_SAS_PHY0_HWRATE_MAX_RATE_1_5:
 
1526                 phy->maximum_linkrate_hw = SAS_LINK_RATE_1_5_GBPS;
 
1528         case MPI_SAS_PHY0_PRATE_MAX_RATE_3_0:
 
1529                 phy->maximum_linkrate_hw = SAS_LINK_RATE_3_0_GBPS;
 
1536          * Set Max programmed link rate.
 
1538         switch (phy_info->programmed_link_rate &
 
1539                         MPI_SAS_PHY0_PRATE_MAX_RATE_MASK) {
 
1540         case MPI_SAS_PHY0_PRATE_MAX_RATE_1_5:
 
1541                 phy->maximum_linkrate = SAS_LINK_RATE_1_5_GBPS;
 
1543         case MPI_SAS_PHY0_PRATE_MAX_RATE_3_0:
 
1544                 phy->maximum_linkrate = SAS_LINK_RATE_3_0_GBPS;
 
1551          * Set Min hardware link rate.
 
1553         switch (phy_info->hw_link_rate & MPI_SAS_PHY0_HWRATE_MIN_RATE_MASK) {
 
1554         case MPI_SAS_PHY0_HWRATE_MIN_RATE_1_5:
 
1555                 phy->minimum_linkrate_hw = SAS_LINK_RATE_1_5_GBPS;
 
1557         case MPI_SAS_PHY0_PRATE_MIN_RATE_3_0:
 
1558                 phy->minimum_linkrate_hw = SAS_LINK_RATE_3_0_GBPS;
 
1565          * Set Min programmed link rate.
 
1567         switch (phy_info->programmed_link_rate &
 
1568                         MPI_SAS_PHY0_PRATE_MIN_RATE_MASK) {
 
1569         case MPI_SAS_PHY0_PRATE_MIN_RATE_1_5:
 
1570                 phy->minimum_linkrate = SAS_LINK_RATE_1_5_GBPS;
 
1572         case MPI_SAS_PHY0_PRATE_MIN_RATE_3_0:
 
1573                 phy->minimum_linkrate = SAS_LINK_RATE_3_0_GBPS;
 
1579         if (!phy_info->phy) {
 
1581                 error = sas_phy_add(phy);
 
1586                 phy_info->phy = phy;
 
1589         if (!phy_info->attached.handle ||
 
1590                         !phy_info->port_details)
 
1593         port = mptsas_get_port(phy_info);
 
1594         ioc = phy_to_ioc(phy_info->phy);
 
1596         if (phy_info->sas_port_add_phy) {
 
1599                         port = sas_port_alloc_num(dev);
 
1604                         error = sas_port_add(port);
 
1606                                 dfailprintk((MYIOC_s_ERR_FMT
 
1607                                         "%s: exit at line=%d\n", ioc->name,
 
1608                                         __FUNCTION__, __LINE__));
 
1611                         mptsas_set_port(phy_info, port);
 
1612                         dsaswideprintk((KERN_DEBUG
 
1613                             "sas_port_alloc: port=%p dev=%p port_id=%d\n",
 
1614                             port, dev, port->port_identifier));
 
1616                 dsaswideprintk((KERN_DEBUG "sas_port_add_phy: phy_id=%d\n",
 
1618                 sas_port_add_phy(port, phy_info->phy);
 
1619                 phy_info->sas_port_add_phy = 0;
 
1622         if (!mptsas_get_rphy(phy_info) && port && !port->rphy) {
 
1624                 struct sas_rphy *rphy;
 
1625                 struct device *parent;
 
1626                 struct sas_identify identify;
 
1628                 parent = dev->parent->parent;
 
1630                  * Let the hotplug_work thread handle processing
 
1631                  * the adding/removing of devices that occur
 
1632                  * after start of day.
 
1634                 if (ioc->sas_discovery_runtime &&
 
1635                         mptsas_is_end_device(&phy_info->attached))
 
1638                 mptsas_parse_device_info(&identify, &phy_info->attached);
 
1639                 if (scsi_is_host_device(parent)) {
 
1640                         struct mptsas_portinfo *port_info;
 
1643                         mutex_lock(&ioc->sas_topology_mutex);
 
1644                         port_info = mptsas_find_portinfo_by_handle(ioc,
 
1646                         mutex_unlock(&ioc->sas_topology_mutex);
 
1648                         for (i = 0; i < port_info->num_phys; i++)
 
1649                                 if (port_info->phy_info[i].identify.sas_address ==
 
1650                                     identify.sas_address) {
 
1651                                         sas_port_mark_backlink(port);
 
1655                 } else if (scsi_is_sas_rphy(parent)) {
 
1656                         struct sas_rphy *parent_rphy = dev_to_rphy(parent);
 
1657                         if (identify.sas_address ==
 
1658                             parent_rphy->identify.sas_address) {
 
1659                                 sas_port_mark_backlink(port);
 
1664                 switch (identify.device_type) {
 
1665                 case SAS_END_DEVICE:
 
1666                         rphy = sas_end_device_alloc(port);
 
1668                 case SAS_EDGE_EXPANDER_DEVICE:
 
1669                 case SAS_FANOUT_EXPANDER_DEVICE:
 
1670                         rphy = sas_expander_alloc(port, identify.device_type);
 
1677                         dfailprintk((MYIOC_s_ERR_FMT
 
1678                                 "%s: exit at line=%d\n", ioc->name,
 
1679                                 __FUNCTION__, __LINE__));
 
1683                 rphy->identify = identify;
 
1684                 error = sas_rphy_add(rphy);
 
1686                         dfailprintk((MYIOC_s_ERR_FMT
 
1687                                 "%s: exit at line=%d\n", ioc->name,
 
1688                                 __FUNCTION__, __LINE__));
 
1689                         sas_rphy_free(rphy);
 
1692                 mptsas_set_rphy(phy_info, rphy);
 
1700 mptsas_probe_hba_phys(MPT_ADAPTER *ioc)
 
1702         struct mptsas_portinfo *port_info, *hba;
 
1703         u32 handle = 0xFFFF;
 
1704         int error = -ENOMEM, i;
 
1706         hba = kzalloc(sizeof(*port_info), GFP_KERNEL);
 
1710         error = mptsas_sas_io_unit_pg0(ioc, hba);
 
1712                 goto out_free_port_info;
 
1714         mutex_lock(&ioc->sas_topology_mutex);
 
1715         ioc->handle = hba->handle;
 
1716         port_info = mptsas_find_portinfo_by_handle(ioc, hba->handle);
 
1719                 list_add_tail(&port_info->list, &ioc->sas_topology);
 
1721                 port_info->handle = hba->handle;
 
1722                 for (i = 0; i < hba->num_phys; i++)
 
1723                         port_info->phy_info[i].negotiated_link_rate =
 
1724                                 hba->phy_info[i].negotiated_link_rate;
 
1725                 kfree(hba->phy_info);
 
1729         mutex_unlock(&ioc->sas_topology_mutex);
 
1731         for (i = 0; i < port_info->num_phys; i++) {
 
1732                 mptsas_sas_phy_pg0(ioc, &port_info->phy_info[i],
 
1733                         (MPI_SAS_PHY_PGAD_FORM_PHY_NUMBER <<
 
1734                          MPI_SAS_PHY_PGAD_FORM_SHIFT), i);
 
1736                 mptsas_sas_device_pg0(ioc, &port_info->phy_info[i].identify,
 
1737                         (MPI_SAS_DEVICE_PGAD_FORM_GET_NEXT_HANDLE <<
 
1738                          MPI_SAS_DEVICE_PGAD_FORM_SHIFT), handle);
 
1739                 port_info->phy_info[i].identify.phy_id =
 
1740                     port_info->phy_info[i].phy_id;
 
1741                 handle = port_info->phy_info[i].identify.handle;
 
1743                 if (port_info->phy_info[i].attached.handle)
 
1744                         mptsas_sas_device_pg0(ioc,
 
1745                                 &port_info->phy_info[i].attached,
 
1746                                 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
 
1747                                  MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
 
1748                                 port_info->phy_info[i].attached.handle);
 
1751         mptsas_setup_wide_ports(ioc, port_info);
 
1753         for (i = 0; i < port_info->num_phys; i++, ioc->sas_index++)
 
1754                 mptsas_probe_one_phy(&ioc->sh->shost_gendev,
 
1755                     &port_info->phy_info[i], ioc->sas_index, 1);
 
1766 mptsas_probe_expander_phys(MPT_ADAPTER *ioc, u32 *handle)
 
1768         struct mptsas_portinfo *port_info, *p, *ex;
 
1769         struct device *parent;
 
1770         struct sas_rphy *rphy;
 
1771         int error = -ENOMEM, i, j;
 
1773         ex = kzalloc(sizeof(*port_info), GFP_KERNEL);
 
1777         error = mptsas_sas_expander_pg0(ioc, ex,
 
1778                 (MPI_SAS_EXPAND_PGAD_FORM_GET_NEXT_HANDLE <<
 
1779                  MPI_SAS_EXPAND_PGAD_FORM_SHIFT), *handle);
 
1781                 goto out_free_port_info;
 
1783         *handle = ex->handle;
 
1785         mutex_lock(&ioc->sas_topology_mutex);
 
1786         port_info = mptsas_find_portinfo_by_handle(ioc, *handle);
 
1789                 list_add_tail(&port_info->list, &ioc->sas_topology);
 
1791                 port_info->handle = ex->handle;
 
1792                 kfree(ex->phy_info);
 
1796         mutex_unlock(&ioc->sas_topology_mutex);
 
1798         for (i = 0; i < port_info->num_phys; i++) {
 
1799                 mptsas_sas_expander_pg1(ioc, &port_info->phy_info[i],
 
1800                         (MPI_SAS_EXPAND_PGAD_FORM_HANDLE_PHY_NUM <<
 
1801                          MPI_SAS_EXPAND_PGAD_FORM_SHIFT), (i << 16) + *handle);
 
1803                 if (port_info->phy_info[i].identify.handle) {
 
1804                         mptsas_sas_device_pg0(ioc,
 
1805                                 &port_info->phy_info[i].identify,
 
1806                                 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
 
1807                                  MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
 
1808                                 port_info->phy_info[i].identify.handle);
 
1809                         port_info->phy_info[i].identify.phy_id =
 
1810                             port_info->phy_info[i].phy_id;
 
1813                 if (port_info->phy_info[i].attached.handle) {
 
1814                         mptsas_sas_device_pg0(ioc,
 
1815                                 &port_info->phy_info[i].attached,
 
1816                                 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
 
1817                                  MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
 
1818                                 port_info->phy_info[i].attached.handle);
 
1819                         port_info->phy_info[i].attached.phy_id =
 
1820                             port_info->phy_info[i].phy_id;
 
1824         parent = &ioc->sh->shost_gendev;
 
1825         for (i = 0; i < port_info->num_phys; i++) {
 
1826                 mutex_lock(&ioc->sas_topology_mutex);
 
1827                 list_for_each_entry(p, &ioc->sas_topology, list) {
 
1828                         for (j = 0; j < p->num_phys; j++) {
 
1829                                 if (port_info->phy_info[i].identify.handle !=
 
1830                                                 p->phy_info[j].attached.handle)
 
1832                                 rphy = mptsas_get_rphy(&p->phy_info[j]);
 
1833                                 parent = &rphy->dev;
 
1836                 mutex_unlock(&ioc->sas_topology_mutex);
 
1839         mptsas_setup_wide_ports(ioc, port_info);
 
1841         for (i = 0; i < port_info->num_phys; i++, ioc->sas_index++)
 
1842                 mptsas_probe_one_phy(parent, &port_info->phy_info[i],
 
1849                 kfree(ex->phy_info);
 
1857  * mptsas_delete_expander_phys
 
1860  * This will traverse topology, and remove expanders
 
1861  * that are no longer present
 
1864 mptsas_delete_expander_phys(MPT_ADAPTER *ioc)
 
1866         struct mptsas_portinfo buffer;
 
1867         struct mptsas_portinfo *port_info, *n, *parent;
 
1868         struct mptsas_phyinfo *phy_info;
 
1869         struct scsi_target * starget;
 
1870         VirtTarget * vtarget;
 
1871         struct sas_port * port;
 
1873         u64     expander_sas_address;
 
1875         mutex_lock(&ioc->sas_topology_mutex);
 
1876         list_for_each_entry_safe(port_info, n, &ioc->sas_topology, list) {
 
1878                 if (port_info->phy_info &&
 
1879                     (!(port_info->phy_info[0].identify.device_info &
 
1880                     MPI_SAS_DEVICE_INFO_SMP_TARGET)))
 
1883                 if (mptsas_sas_expander_pg0(ioc, &buffer,
 
1884                      (MPI_SAS_EXPAND_PGAD_FORM_HANDLE <<
 
1885                      MPI_SAS_EXPAND_PGAD_FORM_SHIFT), port_info->handle)) {
 
1888                          * Issue target reset to all child end devices
 
1889                          * then mark them deleted to prevent further
 
1892                         phy_info = port_info->phy_info;
 
1893                         for (i = 0; i < port_info->num_phys; i++, phy_info++) {
 
1894                                 starget = mptsas_get_starget(phy_info);
 
1897                                 vtarget = starget->hostdata;
 
1898                                 if(vtarget->deleted)
 
1900                                 vtarget->deleted = 1;
 
1901                                 mptsas_target_reset(ioc, vtarget);
 
1902                                 sas_port_delete(mptsas_get_port(phy_info));
 
1903                                 mptsas_port_delete(phy_info->port_details);
 
1907                          * Obtain the port_info instance to the parent port
 
1909                         parent = mptsas_find_portinfo_by_handle(ioc,
 
1910                             port_info->phy_info[0].identify.handle_parent);
 
1915                         expander_sas_address =
 
1916                                 port_info->phy_info[0].identify.sas_address;
 
1919                          * Delete rphys in the parent that point
 
1920                          * to this expander.  The transport layer will
 
1921                          * cleanup all the children.
 
1923                         phy_info = parent->phy_info;
 
1924                         for (i = 0; i < parent->num_phys; i++, phy_info++) {
 
1925                                 port = mptsas_get_port(phy_info);
 
1928                                 if (phy_info->attached.sas_address !=
 
1929                                         expander_sas_address)
 
1931 #ifdef MPT_DEBUG_SAS_WIDE
 
1932                                 dev_printk(KERN_DEBUG, &port->dev,
 
1933                                     "delete port (%d)\n", port->port_identifier);
 
1935                                 sas_port_delete(port);
 
1936                                 mptsas_port_delete(phy_info->port_details);
 
1940                         phy_info = port_info->phy_info;
 
1941                         for (i = 0; i < port_info->num_phys; i++, phy_info++)
 
1942                                 mptsas_port_delete(phy_info->port_details);
 
1944                         list_del(&port_info->list);
 
1945                         kfree(port_info->phy_info);
 
1949                 * Free this memory allocated from inside
 
1950                 * mptsas_sas_expander_pg0
 
1952                 kfree(buffer.phy_info);
 
1954         mutex_unlock(&ioc->sas_topology_mutex);
 
1958  * Start of day discovery
 
1961 mptsas_scan_sas_topology(MPT_ADAPTER *ioc)
 
1963         u32 handle = 0xFFFF;
 
1966         mutex_lock(&ioc->sas_discovery_mutex);
 
1967         mptsas_probe_hba_phys(ioc);
 
1968         while (!mptsas_probe_expander_phys(ioc, &handle))
 
1971           Reporting RAID volumes.
 
1973         if (!ioc->raid_data.pIocPg2)
 
1975         if (!ioc->raid_data.pIocPg2->NumActiveVolumes)
 
1977         for (i=0; i<ioc->raid_data.pIocPg2->NumActiveVolumes; i++) {
 
1978                 scsi_add_device(ioc->sh, MPTSAS_RAID_CHANNEL,
 
1979                     ioc->raid_data.pIocPg2->RaidVolume[i].VolumeID, 0);
 
1982         mutex_unlock(&ioc->sas_discovery_mutex);
 
1986  * Work queue thread to handle Runtime discovery
 
1987  * Mere purpose is the hot add/delete of expanders
 
1991 __mptsas_discovery_work(MPT_ADAPTER *ioc)
 
1993         u32 handle = 0xFFFF;
 
1995         ioc->sas_discovery_runtime=1;
 
1996         mptsas_delete_expander_phys(ioc);
 
1997         mptsas_probe_hba_phys(ioc);
 
1998         while (!mptsas_probe_expander_phys(ioc, &handle))
 
2000         ioc->sas_discovery_runtime=0;
 
2004  * Work queue thread to handle Runtime discovery
 
2005  * Mere purpose is the hot add/delete of expanders
 
2009 mptsas_discovery_work(struct work_struct *work)
 
2011         struct mptsas_discovery_event *ev =
 
2012                 container_of(work, struct mptsas_discovery_event, work);
 
2013         MPT_ADAPTER *ioc = ev->ioc;
 
2015         mutex_lock(&ioc->sas_discovery_mutex);
 
2016         __mptsas_discovery_work(ioc);
 
2017         mutex_unlock(&ioc->sas_discovery_mutex);
 
2021 static struct mptsas_phyinfo *
 
2022 mptsas_find_phyinfo_by_sas_address(MPT_ADAPTER *ioc, u64 sas_address)
 
2024         struct mptsas_portinfo *port_info;
 
2025         struct mptsas_phyinfo *phy_info = NULL;
 
2028         mutex_lock(&ioc->sas_topology_mutex);
 
2029         list_for_each_entry(port_info, &ioc->sas_topology, list) {
 
2030                 for (i = 0; i < port_info->num_phys; i++) {
 
2031                         if (port_info->phy_info[i].attached.sas_address
 
2034                         if (!mptsas_is_end_device(
 
2035                                 &port_info->phy_info[i].attached))
 
2037                         phy_info = &port_info->phy_info[i];
 
2041         mutex_unlock(&ioc->sas_topology_mutex);
 
2045 static struct mptsas_phyinfo *
 
2046 mptsas_find_phyinfo_by_target(MPT_ADAPTER *ioc, u32 id)
 
2048         struct mptsas_portinfo *port_info;
 
2049         struct mptsas_phyinfo *phy_info = NULL;
 
2052         mutex_lock(&ioc->sas_topology_mutex);
 
2053         list_for_each_entry(port_info, &ioc->sas_topology, list) {
 
2054                 for (i = 0; i < port_info->num_phys; i++) {
 
2055                         if (port_info->phy_info[i].attached.id != id)
 
2057                         if (!mptsas_is_end_device(
 
2058                                 &port_info->phy_info[i].attached))
 
2060                         phy_info = &port_info->phy_info[i];
 
2064         mutex_unlock(&ioc->sas_topology_mutex);
 
2069  * Work queue thread to clear the persitency table
 
2072 mptsas_persist_clear_table(struct work_struct *work)
 
2074         MPT_ADAPTER *ioc = container_of(work, MPT_ADAPTER, sas_persist_task);
 
2076         mptbase_sas_persist_operation(ioc, MPI_SAS_OP_CLEAR_NOT_PRESENT);
 
2080 mptsas_reprobe_lun(struct scsi_device *sdev, void *data)
 
2082         sdev->no_uld_attach = data ? 1 : 0;
 
2083         scsi_device_reprobe(sdev);
 
2087 mptsas_reprobe_target(struct scsi_target *starget, int uld_attach)
 
2089         starget_for_each_device(starget, uld_attach ? (void *)1 : NULL,
 
2090                         mptsas_reprobe_lun);
 
2094  * Work queue thread to handle SAS hotplug events
 
2097 mptsas_hotplug_work(struct work_struct *work)
 
2099         struct mptsas_hotplug_event *ev =
 
2100                 container_of(work, struct mptsas_hotplug_event, work);
 
2101         MPT_ADAPTER *ioc = ev->ioc;
 
2102         struct mptsas_phyinfo *phy_info;
 
2103         struct sas_rphy *rphy;
 
2104         struct sas_port *port;
 
2105         struct scsi_device *sdev;
 
2106         struct scsi_target * starget;
 
2107         struct sas_identify identify;
 
2109         struct mptsas_devinfo sas_device;
 
2110         VirtTarget *vtarget;
 
2111         VirtDevice *vdevice;
 
2114         mutex_lock(&ioc->sas_discovery_mutex);
 
2115         switch (ev->event_type) {
 
2116         case MPTSAS_DEL_DEVICE:
 
2118                 phy_info = mptsas_find_phyinfo_by_target(ioc, ev->id);
 
2121                  * Sanity checks, for non-existing phys and remote rphys.
 
2123                 if (!phy_info || !phy_info->port_details) {
 
2124                         dfailprintk((MYIOC_s_ERR_FMT
 
2125                                 "%s: exit at line=%d\n", ioc->name,
 
2126                                 __FUNCTION__, __LINE__));
 
2129                 rphy = mptsas_get_rphy(phy_info);
 
2131                         dfailprintk((MYIOC_s_ERR_FMT
 
2132                                 "%s: exit at line=%d\n", ioc->name,
 
2133                                 __FUNCTION__, __LINE__));
 
2136                 port = mptsas_get_port(phy_info);
 
2138                         dfailprintk((MYIOC_s_ERR_FMT
 
2139                                 "%s: exit at line=%d\n", ioc->name,
 
2140                                 __FUNCTION__, __LINE__));
 
2144                 starget = mptsas_get_starget(phy_info);
 
2146                         vtarget = starget->hostdata;
 
2149                                 dfailprintk((MYIOC_s_ERR_FMT
 
2150                                         "%s: exit at line=%d\n", ioc->name,
 
2151                                         __FUNCTION__, __LINE__));
 
2156                          * Handling  RAID components
 
2158                         if (ev->phys_disk_num_valid) {
 
2159                                 vtarget->target_id = ev->phys_disk_num;
 
2160                                 vtarget->tflags |= MPT_TARGET_FLAGS_RAID_COMPONENT;
 
2161                                 mptsas_reprobe_target(starget, 1);
 
2165                         vtarget->deleted = 1;
 
2166                         mptsas_target_reset(ioc, vtarget);
 
2169                 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SSP_TARGET)
 
2171                 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_STP_TARGET)
 
2173                 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SATA_DEVICE)
 
2176                 printk(MYIOC_s_INFO_FMT
 
2177                        "removing %s device, channel %d, id %d, phy %d\n",
 
2178                        ioc->name, ds, ev->channel, ev->id, phy_info->phy_id);
 
2180 #ifdef MPT_DEBUG_SAS_WIDE
 
2181                 dev_printk(KERN_DEBUG, &port->dev,
 
2182                     "delete port (%d)\n", port->port_identifier);
 
2184                 sas_port_delete(port);
 
2185                 mptsas_port_delete(phy_info->port_details);
 
2187         case MPTSAS_ADD_DEVICE:
 
2189                 if (ev->phys_disk_num_valid)
 
2190                         mpt_findImVolumes(ioc);
 
2193                  * Refresh sas device pg0 data
 
2195                 if (mptsas_sas_device_pg0(ioc, &sas_device,
 
2196                     (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
 
2197                      MPI_SAS_DEVICE_PGAD_FORM_SHIFT), ev->id)) {
 
2198                                 dfailprintk((MYIOC_s_ERR_FMT
 
2199                                         "%s: exit at line=%d\n", ioc->name,
 
2200                                         __FUNCTION__, __LINE__));
 
2205                 __mptsas_discovery_work(ioc);
 
2207                 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
 
2208                                 sas_device.sas_address);
 
2210                 if (!phy_info || !phy_info->port_details) {
 
2211                         dfailprintk((MYIOC_s_ERR_FMT
 
2212                                 "%s: exit at line=%d\n", ioc->name,
 
2213                                 __FUNCTION__, __LINE__));
 
2217                 starget = mptsas_get_starget(phy_info);
 
2219                         vtarget = starget->hostdata;
 
2222                                 dfailprintk((MYIOC_s_ERR_FMT
 
2223                                         "%s: exit at line=%d\n", ioc->name,
 
2224                                         __FUNCTION__, __LINE__));
 
2228                          * Handling  RAID components
 
2230                         if (vtarget->tflags & MPT_TARGET_FLAGS_RAID_COMPONENT) {
 
2231                                 vtarget->tflags &= ~MPT_TARGET_FLAGS_RAID_COMPONENT;
 
2232                                 vtarget->target_id = ev->id;
 
2233                                 mptsas_reprobe_target(starget, 0);
 
2238                 if (mptsas_get_rphy(phy_info)) {
 
2239                         dfailprintk((MYIOC_s_ERR_FMT
 
2240                                 "%s: exit at line=%d\n", ioc->name,
 
2241                                 __FUNCTION__, __LINE__));
 
2244                 port = mptsas_get_port(phy_info);
 
2246                         dfailprintk((MYIOC_s_ERR_FMT
 
2247                                 "%s: exit at line=%d\n", ioc->name,
 
2248                                 __FUNCTION__, __LINE__));
 
2252                 memcpy(&phy_info->attached, &sas_device,
 
2253                     sizeof(struct mptsas_devinfo));
 
2255                 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SSP_TARGET)
 
2257                 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_STP_TARGET)
 
2259                 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SATA_DEVICE)
 
2262                 printk(MYIOC_s_INFO_FMT
 
2263                        "attaching %s device, channel %d, id %d, phy %d\n",
 
2264                        ioc->name, ds, ev->channel, ev->id, ev->phy_id);
 
2266                 mptsas_parse_device_info(&identify, &phy_info->attached);
 
2267                 rphy = sas_end_device_alloc(port);
 
2269                         dfailprintk((MYIOC_s_ERR_FMT
 
2270                                 "%s: exit at line=%d\n", ioc->name,
 
2271                                 __FUNCTION__, __LINE__));
 
2272                         break; /* non-fatal: an rphy can be added later */
 
2275                 rphy->identify = identify;
 
2276                 if (sas_rphy_add(rphy)) {
 
2277                         dfailprintk((MYIOC_s_ERR_FMT
 
2278                                 "%s: exit at line=%d\n", ioc->name,
 
2279                                 __FUNCTION__, __LINE__));
 
2280                         sas_rphy_free(rphy);
 
2283                 mptsas_set_rphy(phy_info, rphy);
 
2285         case MPTSAS_ADD_RAID:
 
2286                 sdev = scsi_device_lookup(ioc->sh, MPTSAS_RAID_CHANNEL,
 
2289                         scsi_device_put(sdev);
 
2292                 printk(MYIOC_s_INFO_FMT
 
2293                        "attaching raid volume, channel %d, id %d\n",
 
2294                        ioc->name, MPTSAS_RAID_CHANNEL, ev->id);
 
2295                 scsi_add_device(ioc->sh, MPTSAS_RAID_CHANNEL, ev->id, 0);
 
2296                 mpt_findImVolumes(ioc);
 
2298         case MPTSAS_DEL_RAID:
 
2299                 sdev = scsi_device_lookup(ioc->sh, MPTSAS_RAID_CHANNEL,
 
2303                 printk(MYIOC_s_INFO_FMT
 
2304                        "removing raid volume, channel %d, id %d\n",
 
2305                        ioc->name, MPTSAS_RAID_CHANNEL, ev->id);
 
2306                 vdevice = sdev->hostdata;
 
2307                 vdevice->vtarget->deleted = 1;
 
2308                 mptsas_target_reset(ioc, vdevice->vtarget);
 
2309                 scsi_remove_device(sdev);
 
2310                 scsi_device_put(sdev);
 
2311                 mpt_findImVolumes(ioc);
 
2313         case MPTSAS_IGNORE_EVENT:
 
2318         mutex_unlock(&ioc->sas_discovery_mutex);
 
2324 mptsas_send_sas_event(MPT_ADAPTER *ioc,
 
2325                 EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *sas_event_data)
 
2327         struct mptsas_hotplug_event *ev;
 
2328         u32 device_info = le32_to_cpu(sas_event_data->DeviceInfo);
 
2332              (MPI_SAS_DEVICE_INFO_SSP_TARGET |
 
2333               MPI_SAS_DEVICE_INFO_STP_TARGET |
 
2334               MPI_SAS_DEVICE_INFO_SATA_DEVICE )) == 0)
 
2337         switch (sas_event_data->ReasonCode) {
 
2338         case MPI_EVENT_SAS_DEV_STAT_RC_ADDED:
 
2339         case MPI_EVENT_SAS_DEV_STAT_RC_NOT_RESPONDING:
 
2340                 ev = kzalloc(sizeof(*ev), GFP_ATOMIC);
 
2342                         printk(KERN_WARNING "mptsas: lost hotplug event\n");
 
2346                 INIT_WORK(&ev->work, mptsas_hotplug_work);
 
2348                 ev->handle = le16_to_cpu(sas_event_data->DevHandle);
 
2350                     le16_to_cpu(sas_event_data->ParentDevHandle);
 
2351                 ev->channel = sas_event_data->Bus;
 
2352                 ev->id = sas_event_data->TargetID;
 
2353                 ev->phy_id = sas_event_data->PhyNum;
 
2354                 memcpy(&sas_address, &sas_event_data->SASAddress,
 
2356                 ev->sas_address = le64_to_cpu(sas_address);
 
2357                 ev->device_info = device_info;
 
2359                 if (sas_event_data->ReasonCode &
 
2360                     MPI_EVENT_SAS_DEV_STAT_RC_ADDED)
 
2361                         ev->event_type = MPTSAS_ADD_DEVICE;
 
2363                         ev->event_type = MPTSAS_DEL_DEVICE;
 
2364                 schedule_work(&ev->work);
 
2366         case MPI_EVENT_SAS_DEV_STAT_RC_NO_PERSIST_ADDED:
 
2368          * Persistent table is full.
 
2370                 INIT_WORK(&ioc->sas_persist_task,
 
2371                     mptsas_persist_clear_table);
 
2372                 schedule_work(&ioc->sas_persist_task);
 
2374         case MPI_EVENT_SAS_DEV_STAT_RC_SMART_DATA:
 
2376         case MPI_EVENT_SAS_DEV_STAT_RC_INTERNAL_DEVICE_RESET:
 
2384 mptsas_send_raid_event(MPT_ADAPTER *ioc,
 
2385                 EVENT_DATA_RAID *raid_event_data)
 
2387         struct mptsas_hotplug_event *ev;
 
2388         int status = le32_to_cpu(raid_event_data->SettingsStatus);
 
2389         int state = (status >> 8) & 0xff;
 
2391         if (ioc->bus_type != SAS)
 
2394         ev = kzalloc(sizeof(*ev), GFP_ATOMIC);
 
2396                 printk(KERN_WARNING "mptsas: lost hotplug event\n");
 
2400         INIT_WORK(&ev->work, mptsas_hotplug_work);
 
2402         ev->id = raid_event_data->VolumeID;
 
2403         ev->event_type = MPTSAS_IGNORE_EVENT;
 
2405         switch (raid_event_data->ReasonCode) {
 
2406         case MPI_EVENT_RAID_RC_PHYSDISK_DELETED:
 
2407                 ev->event_type = MPTSAS_ADD_DEVICE;
 
2409         case MPI_EVENT_RAID_RC_PHYSDISK_CREATED:
 
2410                 ioc->raid_data.isRaid = 1;
 
2411                 ev->phys_disk_num_valid = 1;
 
2412                 ev->phys_disk_num = raid_event_data->PhysDiskNum;
 
2413                 ev->event_type = MPTSAS_DEL_DEVICE;
 
2415         case MPI_EVENT_RAID_RC_PHYSDISK_STATUS_CHANGED:
 
2417                 case MPI_PD_STATE_ONLINE:
 
2418                         ioc->raid_data.isRaid = 1;
 
2419                         ev->phys_disk_num_valid = 1;
 
2420                         ev->phys_disk_num = raid_event_data->PhysDiskNum;
 
2421                         ev->event_type = MPTSAS_ADD_DEVICE;
 
2423                 case MPI_PD_STATE_MISSING:
 
2424                 case MPI_PD_STATE_NOT_COMPATIBLE:
 
2425                 case MPI_PD_STATE_OFFLINE_AT_HOST_REQUEST:
 
2426                 case MPI_PD_STATE_FAILED_AT_HOST_REQUEST:
 
2427                 case MPI_PD_STATE_OFFLINE_FOR_ANOTHER_REASON:
 
2428                         ev->event_type = MPTSAS_DEL_DEVICE;
 
2434         case MPI_EVENT_RAID_RC_VOLUME_DELETED:
 
2435                 ev->event_type = MPTSAS_DEL_RAID;
 
2437         case MPI_EVENT_RAID_RC_VOLUME_CREATED:
 
2438                 ev->event_type = MPTSAS_ADD_RAID;
 
2440         case MPI_EVENT_RAID_RC_VOLUME_STATUS_CHANGED:
 
2442                 case MPI_RAIDVOL0_STATUS_STATE_FAILED:
 
2443                 case MPI_RAIDVOL0_STATUS_STATE_MISSING:
 
2444                         ev->event_type = MPTSAS_DEL_RAID;
 
2446                 case MPI_RAIDVOL0_STATUS_STATE_OPTIMAL:
 
2447                 case MPI_RAIDVOL0_STATUS_STATE_DEGRADED:
 
2448                         ev->event_type = MPTSAS_ADD_RAID;
 
2457         schedule_work(&ev->work);
 
2461 mptsas_send_discovery_event(MPT_ADAPTER *ioc,
 
2462         EVENT_DATA_SAS_DISCOVERY *discovery_data)
 
2464         struct mptsas_discovery_event *ev;
 
2469          * This flag will be non-zero when firmware
 
2470          * kicks off discovery, and return to zero
 
2471          * once its completed.
 
2473         if (discovery_data->DiscoveryStatus)
 
2476         ev = kzalloc(sizeof(*ev), GFP_ATOMIC);
 
2479         INIT_WORK(&ev->work, mptsas_discovery_work);
 
2481         schedule_work(&ev->work);
 
2486 mptsas_event_process(MPT_ADAPTER *ioc, EventNotificationReply_t *reply)
 
2489         u8 event = le32_to_cpu(reply->Event) & 0xFF;
 
2495          * sas_discovery_ignore_events
 
2497          * This flag is to prevent anymore processing of
 
2498          * sas events once mptsas_remove function is called.
 
2500         if (ioc->sas_discovery_ignore_events) {
 
2501                 rc = mptscsih_event_process(ioc, reply);
 
2506         case MPI_EVENT_SAS_DEVICE_STATUS_CHANGE:
 
2507                 mptsas_send_sas_event(ioc,
 
2508                         (EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *)reply->Data);
 
2510         case MPI_EVENT_INTEGRATED_RAID:
 
2511                 mptsas_send_raid_event(ioc,
 
2512                         (EVENT_DATA_RAID *)reply->Data);
 
2514         case MPI_EVENT_PERSISTENT_TABLE_FULL:
 
2515                 INIT_WORK(&ioc->sas_persist_task,
 
2516                     mptsas_persist_clear_table);
 
2517                 schedule_work(&ioc->sas_persist_task);
 
2519          case MPI_EVENT_SAS_DISCOVERY:
 
2520                 mptsas_send_discovery_event(ioc,
 
2521                         (EVENT_DATA_SAS_DISCOVERY *)reply->Data);
 
2524                 rc = mptscsih_event_process(ioc, reply);
 
2533 mptsas_probe(struct pci_dev *pdev, const struct pci_device_id *id)
 
2535         struct Scsi_Host        *sh;
 
2538         unsigned long            flags;
 
2546         r = mpt_attach(pdev,id);
 
2550         ioc = pci_get_drvdata(pdev);
 
2551         ioc->DoneCtx = mptsasDoneCtx;
 
2552         ioc->TaskCtx = mptsasTaskCtx;
 
2553         ioc->InternalCtx = mptsasInternalCtx;
 
2555         /*  Added sanity check on readiness of the MPT adapter.
 
2557         if (ioc->last_state != MPI_IOC_STATE_OPERATIONAL) {
 
2558                 printk(MYIOC_s_WARN_FMT
 
2559                   "Skipping because it's not operational!\n",
 
2562                 goto out_mptsas_probe;
 
2566                 printk(MYIOC_s_WARN_FMT "Skipping because it's disabled!\n",
 
2569                 goto out_mptsas_probe;
 
2572         /*  Sanity check - ensure at least 1 port is INITIATOR capable
 
2575         for (ii = 0; ii < ioc->facts.NumberOfPorts; ii++) {
 
2576                 if (ioc->pfacts[ii].ProtocolFlags &
 
2577                                 MPI_PORTFACTS_PROTOCOL_INITIATOR)
 
2582                 printk(MYIOC_s_WARN_FMT
 
2583                         "Skipping ioc=%p because SCSI Initiator mode "
 
2584                         "is NOT enabled!\n", ioc->name, ioc);
 
2588         sh = scsi_host_alloc(&mptsas_driver_template, sizeof(MPT_SCSI_HOST));
 
2590                 printk(MYIOC_s_WARN_FMT
 
2591                         "Unable to register controller with SCSI subsystem\n",
 
2594                 goto out_mptsas_probe;
 
2597         spin_lock_irqsave(&ioc->FreeQlock, flags);
 
2599         /* Attach the SCSI Host to the IOC structure
 
2607         /* set 16 byte cdb's */
 
2608         sh->max_cmd_len = 16;
 
2610         sh->max_id = ioc->pfacts->MaxDevices + 1;
 
2612         sh->transportt = mptsas_transport_template;
 
2614         sh->max_lun = MPT_LAST_LUN + 1;
 
2615         sh->max_channel = 0;
 
2616         sh->this_id = ioc->pfacts[0].PortSCSIID;
 
2620         sh->unique_id = ioc->id;
 
2622         INIT_LIST_HEAD(&ioc->sas_topology);
 
2623         mutex_init(&ioc->sas_topology_mutex);
 
2624         mutex_init(&ioc->sas_discovery_mutex);
 
2625         mutex_init(&ioc->sas_mgmt.mutex);
 
2626         init_completion(&ioc->sas_mgmt.done);
 
2628         /* Verify that we won't exceed the maximum
 
2629          * number of chain buffers
 
2630          * We can optimize:  ZZ = req_sz/sizeof(SGE)
 
2632          *  numSGE = 1 + (ZZ-1)*(maxChain -1) + ZZ
 
2633          *               + (req_sz - 64)/sizeof(SGE)
 
2634          * A slightly different algorithm is required for
 
2637         scale = ioc->req_sz/(sizeof(dma_addr_t) + sizeof(u32));
 
2638         if (sizeof(dma_addr_t) == sizeof(u64)) {
 
2639                 numSGE = (scale - 1) *
 
2640                   (ioc->facts.MaxChainDepth-1) + scale +
 
2641                   (ioc->req_sz - 60) / (sizeof(dma_addr_t) +
 
2644                 numSGE = 1 + (scale - 1) *
 
2645                   (ioc->facts.MaxChainDepth-1) + scale +
 
2646                   (ioc->req_sz - 64) / (sizeof(dma_addr_t) +
 
2650         if (numSGE < sh->sg_tablesize) {
 
2651                 /* Reset this value */
 
2652                 dprintk((MYIOC_s_INFO_FMT
 
2653                   "Resetting sg_tablesize to %d from %d\n",
 
2654                   ioc->name, numSGE, sh->sg_tablesize));
 
2655                 sh->sg_tablesize = numSGE;
 
2658         spin_unlock_irqrestore(&ioc->FreeQlock, flags);
 
2660         hd = (MPT_SCSI_HOST *) sh->hostdata;
 
2663         /* SCSI needs scsi_cmnd lookup table!
 
2664          * (with size equal to req_depth*PtrSz!)
 
2666         hd->ScsiLookup = kcalloc(ioc->req_depth, sizeof(void *), GFP_ATOMIC);
 
2667         if (!hd->ScsiLookup) {
 
2669                 goto out_mptsas_probe;
 
2672         dprintk((MYIOC_s_INFO_FMT "ScsiLookup @ %p\n",
 
2673                  ioc->name, hd->ScsiLookup));
 
2675         /* Allocate memory for the device structures.
 
2676          * A non-Null pointer at an offset
 
2677          * indicates a device exists.
 
2678          * max_id = 1 + maximum id (hosts.h)
 
2680         hd->Targets = kcalloc(sh->max_id, sizeof(void *), GFP_ATOMIC);
 
2683                 goto out_mptsas_probe;
 
2686         dprintk((KERN_INFO "  vtarget @ %p\n", hd->Targets));
 
2688         /* Clear the TM flags
 
2691         hd->tmState = TM_STATE_NONE;
 
2692         hd->resetPending = 0;
 
2693         hd->abortSCpnt = NULL;
 
2695         /* Clear the pointer used to store
 
2696          * single-threaded commands, i.e., those
 
2697          * issued during a bus scan, dv and
 
2698          * configuration pages.
 
2702         /* Initialize this SCSI Hosts' timers
 
2703          * To use, set the timer expires field
 
2706         init_timer(&hd->timer);
 
2707         hd->timer.data = (unsigned long) hd;
 
2708         hd->timer.function = mptscsih_timer_expired;
 
2710         ioc->sas_data.ptClear = mpt_pt_clear;
 
2712         if (ioc->sas_data.ptClear==1) {
 
2713                 mptbase_sas_persist_operation(
 
2714                     ioc, MPI_SAS_OP_CLEAR_ALL_PERSISTENT);
 
2717         init_waitqueue_head(&hd->scandv_waitq);
 
2718         hd->scandv_wait_done = 0;
 
2719         hd->last_queue_full = 0;
 
2721         error = scsi_add_host(sh, &ioc->pcidev->dev);
 
2723                 dprintk((KERN_ERR MYNAM
 
2724                   "scsi_add_host failed\n"));
 
2725                 goto out_mptsas_probe;
 
2728         mptsas_scan_sas_topology(ioc);
 
2734         mptscsih_remove(pdev);
 
2738 static void __devexit mptsas_remove(struct pci_dev *pdev)
 
2740         MPT_ADAPTER *ioc = pci_get_drvdata(pdev);
 
2741         struct mptsas_portinfo *p, *n;
 
2744         ioc->sas_discovery_ignore_events=1;
 
2745         sas_remove_host(ioc->sh);
 
2747         mutex_lock(&ioc->sas_topology_mutex);
 
2748         list_for_each_entry_safe(p, n, &ioc->sas_topology, list) {
 
2750                 for (i = 0 ; i < p->num_phys ; i++)
 
2751                         mptsas_port_delete(p->phy_info[i].port_details);
 
2755         mutex_unlock(&ioc->sas_topology_mutex);
 
2757         mptscsih_remove(pdev);
 
2760 static struct pci_device_id mptsas_pci_table[] = {
 
2761         { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1064,
 
2762                 PCI_ANY_ID, PCI_ANY_ID },
 
2763         { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1068,
 
2764                 PCI_ANY_ID, PCI_ANY_ID },
 
2765         { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1064E,
 
2766                 PCI_ANY_ID, PCI_ANY_ID },
 
2767         { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1068E,
 
2768                 PCI_ANY_ID, PCI_ANY_ID },
 
2769         { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1078,
 
2770                 PCI_ANY_ID, PCI_ANY_ID },
 
2771         {0}     /* Terminating entry */
 
2773 MODULE_DEVICE_TABLE(pci, mptsas_pci_table);
 
2776 static struct pci_driver mptsas_driver = {
 
2778         .id_table       = mptsas_pci_table,
 
2779         .probe          = mptsas_probe,
 
2780         .remove         = __devexit_p(mptsas_remove),
 
2781         .shutdown       = mptscsih_shutdown,
 
2783         .suspend        = mptscsih_suspend,
 
2784         .resume         = mptscsih_resume,
 
2791         show_mptmod_ver(my_NAME, my_VERSION);
 
2793         mptsas_transport_template =
 
2794             sas_attach_transport(&mptsas_transport_functions);
 
2795         if (!mptsas_transport_template)
 
2798         mptsasDoneCtx = mpt_register(mptscsih_io_done, MPTSAS_DRIVER);
 
2799         mptsasTaskCtx = mpt_register(mptscsih_taskmgmt_complete, MPTSAS_DRIVER);
 
2801                 mpt_register(mptscsih_scandv_complete, MPTSAS_DRIVER);
 
2802         mptsasMgmtCtx = mpt_register(mptsas_mgmt_done, MPTSAS_DRIVER);
 
2804         if (mpt_event_register(mptsasDoneCtx, mptsas_event_process) == 0) {
 
2805                 devtverboseprintk((KERN_INFO MYNAM
 
2806                   ": Registered for IOC event notifications\n"));
 
2809         if (mpt_reset_register(mptsasDoneCtx, mptscsih_ioc_reset) == 0) {
 
2810                 dprintk((KERN_INFO MYNAM
 
2811                   ": Registered for IOC reset notifications\n"));
 
2814         return pci_register_driver(&mptsas_driver);
 
2820         pci_unregister_driver(&mptsas_driver);
 
2821         sas_release_transport(mptsas_transport_template);
 
2823         mpt_reset_deregister(mptsasDoneCtx);
 
2824         mpt_event_deregister(mptsasDoneCtx);
 
2826         mpt_deregister(mptsasMgmtCtx);
 
2827         mpt_deregister(mptsasInternalCtx);
 
2828         mpt_deregister(mptsasTaskCtx);
 
2829         mpt_deregister(mptsasDoneCtx);
 
2832 module_init(mptsas_init);
 
2833 module_exit(mptsas_exit);