2 * linux/drivers/message/fusion/mptsas.c
3 * For use with LSI PCI chip/adapter(s)
4 * running LSI Fusion MPT (Message Passing Technology) firmware.
6 * Copyright (c) 1999-2008 LSI Corporation
7 * (mailto:DL-MPTFusionLinux@lsi.com)
9 /*=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=*/
11 This program is free software; you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation; version 2 of the License.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
21 THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR
22 CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT
23 LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT,
24 MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is
25 solely responsible for determining the appropriateness of using and
26 distributing the Program and assumes all risks associated with its
27 exercise of rights under this Agreement, including but not limited to
28 the risks and costs of program errors, damage to or loss of data,
29 programs or equipment, and unavailability or interruption of operations.
31 DISCLAIMER OF LIABILITY
32 NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY
33 DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
34 DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND
35 ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
36 TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
37 USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED
38 HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES
40 You should have received a copy of the GNU General Public License
41 along with this program; if not, write to the Free Software
42 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
44 /*=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=*/
46 #include <linux/module.h>
47 #include <linux/kernel.h>
48 #include <linux/slab.h>
49 #include <linux/init.h>
50 #include <linux/errno.h>
51 #include <linux/jiffies.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_transport.h>
61 #include <scsi/scsi_dbg.h>
68 #define my_NAME "Fusion MPT SAS Host driver"
69 #define my_VERSION MPT_LINUX_VERSION_COMMON
70 #define MYNAM "mptsas"
73 * Reserved channel for integrated raid
75 #define MPTSAS_RAID_CHANNEL 1
77 #define SAS_CONFIG_PAGE_TIMEOUT 30
78 MODULE_AUTHOR(MODULEAUTHOR);
79 MODULE_DESCRIPTION(my_NAME);
80 MODULE_LICENSE("GPL");
81 MODULE_VERSION(my_VERSION);
83 static int mpt_pt_clear;
84 module_param(mpt_pt_clear, int, 0);
85 MODULE_PARM_DESC(mpt_pt_clear,
86 " Clear persistency table: enable=1 "
87 "(default=MPTSCSIH_PT_CLEAR=0)");
89 /* scsi-mid layer global parmeter is max_report_luns, which is 511 */
90 #define MPTSAS_MAX_LUN (16895)
91 static int max_lun = MPTSAS_MAX_LUN;
92 module_param(max_lun, int, 0);
93 MODULE_PARM_DESC(max_lun, " max lun, default=16895 ");
95 static int mpt_loadtime_max_sectors = 8192;
96 module_param(mpt_loadtime_max_sectors, int, 0);
97 MODULE_PARM_DESC(mpt_loadtime_max_sectors,
98 " Maximum sector define for Host Bus Adaptor.Range 64 to 8192 default=8192");
100 static u8 mptsasDoneCtx = MPT_MAX_PROTOCOL_DRIVERS;
101 static u8 mptsasTaskCtx = MPT_MAX_PROTOCOL_DRIVERS;
102 static u8 mptsasInternalCtx = MPT_MAX_PROTOCOL_DRIVERS; /* Used only for internal commands */
103 static u8 mptsasMgmtCtx = MPT_MAX_PROTOCOL_DRIVERS;
104 static u8 mptsasDeviceResetCtx = MPT_MAX_PROTOCOL_DRIVERS;
106 static void mptsas_firmware_event_work(struct work_struct *work);
107 static void mptsas_send_sas_event(struct fw_event_work *fw_event);
108 static void mptsas_send_raid_event(struct fw_event_work *fw_event);
109 static void mptsas_send_ir2_event(struct fw_event_work *fw_event);
110 static void mptsas_parse_device_info(struct sas_identify *identify,
111 struct mptsas_devinfo *device_info);
112 static inline void mptsas_set_rphy(MPT_ADAPTER *ioc,
113 struct mptsas_phyinfo *phy_info, struct sas_rphy *rphy);
114 static struct mptsas_phyinfo *mptsas_find_phyinfo_by_sas_address
115 (MPT_ADAPTER *ioc, u64 sas_address);
116 static int mptsas_sas_device_pg0(MPT_ADAPTER *ioc,
117 struct mptsas_devinfo *device_info, u32 form, u32 form_specific);
118 static int mptsas_sas_enclosure_pg0(MPT_ADAPTER *ioc,
119 struct mptsas_enclosure *enclosure, u32 form, u32 form_specific);
120 static int mptsas_add_end_device(MPT_ADAPTER *ioc,
121 struct mptsas_phyinfo *phy_info);
122 static void mptsas_del_end_device(MPT_ADAPTER *ioc,
123 struct mptsas_phyinfo *phy_info);
124 static void mptsas_send_link_status_event(struct fw_event_work *fw_event);
125 static struct mptsas_portinfo *mptsas_find_portinfo_by_sas_address
126 (MPT_ADAPTER *ioc, u64 sas_address);
127 static void mptsas_expander_delete(MPT_ADAPTER *ioc,
128 struct mptsas_portinfo *port_info, u8 force);
129 static void mptsas_send_expander_event(struct fw_event_work *fw_event);
130 static void mptsas_not_responding_devices(MPT_ADAPTER *ioc);
131 static void mptsas_scan_sas_topology(MPT_ADAPTER *ioc);
132 static void mptsas_broadcast_primative_work(struct fw_event_work *fw_event);
133 static void mptsas_handle_queue_full_event(struct fw_event_work *fw_event);
134 static void mptsas_volume_delete(MPT_ADAPTER *ioc, u8 id);
135 void mptsas_schedule_target_reset(void *ioc);
137 static void mptsas_print_phy_data(MPT_ADAPTER *ioc,
138 MPI_SAS_IO_UNIT0_PHY_DATA *phy_data)
140 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
141 "---- IO UNIT PAGE 0 ------------\n", ioc->name));
142 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Handle=0x%X\n",
143 ioc->name, le16_to_cpu(phy_data->AttachedDeviceHandle)));
144 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Controller Handle=0x%X\n",
145 ioc->name, le16_to_cpu(phy_data->ControllerDevHandle)));
146 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Port=0x%X\n",
147 ioc->name, phy_data->Port));
148 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Port Flags=0x%X\n",
149 ioc->name, phy_data->PortFlags));
150 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "PHY Flags=0x%X\n",
151 ioc->name, phy_data->PhyFlags));
152 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Negotiated Link Rate=0x%X\n",
153 ioc->name, phy_data->NegotiatedLinkRate));
154 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
155 "Controller PHY Device Info=0x%X\n", ioc->name,
156 le32_to_cpu(phy_data->ControllerPhyDeviceInfo)));
157 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "DiscoveryStatus=0x%X\n\n",
158 ioc->name, le32_to_cpu(phy_data->DiscoveryStatus)));
161 static void mptsas_print_phy_pg0(MPT_ADAPTER *ioc, SasPhyPage0_t *pg0)
165 memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
167 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
168 "---- SAS PHY PAGE 0 ------------\n", ioc->name));
169 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
170 "Attached Device Handle=0x%X\n", ioc->name,
171 le16_to_cpu(pg0->AttachedDevHandle)));
172 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "SAS Address=0x%llX\n",
173 ioc->name, (unsigned long long)le64_to_cpu(sas_address)));
174 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
175 "Attached PHY Identifier=0x%X\n", ioc->name,
176 pg0->AttachedPhyIdentifier));
177 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Attached Device Info=0x%X\n",
178 ioc->name, le32_to_cpu(pg0->AttachedDeviceInfo)));
179 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Programmed Link Rate=0x%X\n",
180 ioc->name, pg0->ProgrammedLinkRate));
181 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Change Count=0x%X\n",
182 ioc->name, pg0->ChangeCount));
183 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "PHY Info=0x%X\n\n",
184 ioc->name, le32_to_cpu(pg0->PhyInfo)));
187 static void mptsas_print_phy_pg1(MPT_ADAPTER *ioc, SasPhyPage1_t *pg1)
189 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
190 "---- SAS PHY PAGE 1 ------------\n", ioc->name));
191 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Invalid Dword Count=0x%x\n",
192 ioc->name, pg1->InvalidDwordCount));
193 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
194 "Running Disparity Error Count=0x%x\n", ioc->name,
195 pg1->RunningDisparityErrorCount));
196 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
197 "Loss Dword Synch Count=0x%x\n", ioc->name,
198 pg1->LossDwordSynchCount));
199 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
200 "PHY Reset Problem Count=0x%x\n\n", ioc->name,
201 pg1->PhyResetProblemCount));
204 static void mptsas_print_device_pg0(MPT_ADAPTER *ioc, SasDevicePage0_t *pg0)
208 memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
210 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
211 "---- SAS DEVICE PAGE 0 ---------\n", ioc->name));
212 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Handle=0x%X\n",
213 ioc->name, le16_to_cpu(pg0->DevHandle)));
214 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Parent Handle=0x%X\n",
215 ioc->name, le16_to_cpu(pg0->ParentDevHandle)));
216 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Enclosure Handle=0x%X\n",
217 ioc->name, le16_to_cpu(pg0->EnclosureHandle)));
218 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Slot=0x%X\n",
219 ioc->name, le16_to_cpu(pg0->Slot)));
220 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "SAS Address=0x%llX\n",
221 ioc->name, (unsigned long long)le64_to_cpu(sas_address)));
222 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Target ID=0x%X\n",
223 ioc->name, pg0->TargetID));
224 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Bus=0x%X\n",
225 ioc->name, pg0->Bus));
226 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Parent Phy Num=0x%X\n",
227 ioc->name, pg0->PhyNum));
228 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Access Status=0x%X\n",
229 ioc->name, le16_to_cpu(pg0->AccessStatus)));
230 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Device Info=0x%X\n",
231 ioc->name, le32_to_cpu(pg0->DeviceInfo)));
232 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Flags=0x%X\n",
233 ioc->name, le16_to_cpu(pg0->Flags)));
234 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Physical Port=0x%X\n\n",
235 ioc->name, pg0->PhysicalPort));
238 static void mptsas_print_expander_pg1(MPT_ADAPTER *ioc, SasExpanderPage1_t *pg1)
240 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
241 "---- SAS EXPANDER PAGE 1 ------------\n", ioc->name));
242 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Physical Port=0x%X\n",
243 ioc->name, pg1->PhysicalPort));
244 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "PHY Identifier=0x%X\n",
245 ioc->name, pg1->PhyIdentifier));
246 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Negotiated Link Rate=0x%X\n",
247 ioc->name, pg1->NegotiatedLinkRate));
248 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Programmed Link Rate=0x%X\n",
249 ioc->name, pg1->ProgrammedLinkRate));
250 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Hardware Link Rate=0x%X\n",
251 ioc->name, pg1->HwLinkRate));
252 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Owner Device Handle=0x%X\n",
253 ioc->name, le16_to_cpu(pg1->OwnerDevHandle)));
254 dsasprintk(ioc, printk(MYIOC_s_DEBUG_FMT
255 "Attached Device Handle=0x%X\n\n", ioc->name,
256 le16_to_cpu(pg1->AttachedDevHandle)));
259 /* inhibit sas firmware event handling */
261 mptsas_fw_event_off(MPT_ADAPTER *ioc)
265 spin_lock_irqsave(&ioc->fw_event_lock, flags);
266 ioc->fw_events_off = 1;
267 ioc->sas_discovery_quiesce_io = 0;
268 spin_unlock_irqrestore(&ioc->fw_event_lock, flags);
272 /* enable sas firmware event handling */
274 mptsas_fw_event_on(MPT_ADAPTER *ioc)
278 spin_lock_irqsave(&ioc->fw_event_lock, flags);
279 ioc->fw_events_off = 0;
280 spin_unlock_irqrestore(&ioc->fw_event_lock, flags);
283 /* queue a sas firmware event */
285 mptsas_add_fw_event(MPT_ADAPTER *ioc, struct fw_event_work *fw_event,
290 spin_lock_irqsave(&ioc->fw_event_lock, flags);
291 list_add_tail(&fw_event->list, &ioc->fw_event_list);
292 INIT_DELAYED_WORK(&fw_event->work, mptsas_firmware_event_work);
293 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "%s: add (fw_event=0x%p)"
294 "on cpuid %d\n", ioc->name, __func__,
295 fw_event, smp_processor_id()));
296 queue_delayed_work_on(smp_processor_id(), ioc->fw_event_q,
297 &fw_event->work, delay);
298 spin_unlock_irqrestore(&ioc->fw_event_lock, flags);
301 /* requeue a sas firmware event */
303 mptsas_requeue_fw_event(MPT_ADAPTER *ioc, struct fw_event_work *fw_event,
307 spin_lock_irqsave(&ioc->fw_event_lock, flags);
308 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "%s: reschedule task "
309 "(fw_event=0x%p)on cpuid %d\n", ioc->name, __func__,
310 fw_event, smp_processor_id()));
312 queue_delayed_work_on(smp_processor_id(), ioc->fw_event_q,
313 &fw_event->work, msecs_to_jiffies(delay));
314 spin_unlock_irqrestore(&ioc->fw_event_lock, flags);
317 /* free memory associated to a sas firmware event */
319 mptsas_free_fw_event(MPT_ADAPTER *ioc, struct fw_event_work *fw_event)
323 spin_lock_irqsave(&ioc->fw_event_lock, flags);
324 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "%s: kfree (fw_event=0x%p)\n",
325 ioc->name, __func__, fw_event));
326 list_del(&fw_event->list);
328 spin_unlock_irqrestore(&ioc->fw_event_lock, flags);
331 /* walk the firmware event queue, and either stop or wait for
332 * outstanding events to complete */
334 mptsas_cleanup_fw_event_q(MPT_ADAPTER *ioc)
336 struct fw_event_work *fw_event, *next;
337 struct mptsas_target_reset_event *target_reset_list, *n;
338 MPT_SCSI_HOST *hd = shost_priv(ioc->sh);
340 /* flush the target_reset_list */
341 if (!list_empty(&hd->target_reset_list)) {
342 list_for_each_entry_safe(target_reset_list, n,
343 &hd->target_reset_list, list) {
344 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
345 "%s: removing target reset for id=%d\n",
347 target_reset_list->sas_event_data.TargetID));
348 list_del(&target_reset_list->list);
349 kfree(target_reset_list);
353 if (list_empty(&ioc->fw_event_list) ||
354 !ioc->fw_event_q || in_interrupt())
357 list_for_each_entry_safe(fw_event, next, &ioc->fw_event_list, list) {
358 if (cancel_delayed_work(&fw_event->work))
359 mptsas_free_fw_event(ioc, fw_event);
364 static inline MPT_ADAPTER *phy_to_ioc(struct sas_phy *phy)
366 struct Scsi_Host *shost = dev_to_shost(phy->dev.parent);
367 return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
370 static inline MPT_ADAPTER *rphy_to_ioc(struct sas_rphy *rphy)
372 struct Scsi_Host *shost = dev_to_shost(rphy->dev.parent->parent);
373 return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
377 * mptsas_find_portinfo_by_handle
379 * This function should be called with the sas_topology_mutex already held
381 static struct mptsas_portinfo *
382 mptsas_find_portinfo_by_handle(MPT_ADAPTER *ioc, u16 handle)
384 struct mptsas_portinfo *port_info, *rc=NULL;
387 list_for_each_entry(port_info, &ioc->sas_topology, list)
388 for (i = 0; i < port_info->num_phys; i++)
389 if (port_info->phy_info[i].identify.handle == handle) {
398 * mptsas_find_portinfo_by_sas_address -
399 * @ioc: Pointer to MPT_ADAPTER structure
402 * This function should be called with the sas_topology_mutex already held
405 static struct mptsas_portinfo *
406 mptsas_find_portinfo_by_sas_address(MPT_ADAPTER *ioc, u64 sas_address)
408 struct mptsas_portinfo *port_info, *rc = NULL;
411 if (sas_address >= ioc->hba_port_sas_addr &&
412 sas_address < (ioc->hba_port_sas_addr +
413 ioc->hba_port_num_phy))
414 return ioc->hba_port_info;
416 mutex_lock(&ioc->sas_topology_mutex);
417 list_for_each_entry(port_info, &ioc->sas_topology, list)
418 for (i = 0; i < port_info->num_phys; i++)
419 if (port_info->phy_info[i].identify.sas_address ==
425 mutex_unlock(&ioc->sas_topology_mutex);
430 * Returns true if there is a scsi end device
433 mptsas_is_end_device(struct mptsas_devinfo * attached)
435 if ((attached->sas_address) &&
436 (attached->device_info &
437 MPI_SAS_DEVICE_INFO_END_DEVICE) &&
438 ((attached->device_info &
439 MPI_SAS_DEVICE_INFO_SSP_TARGET) |
440 (attached->device_info &
441 MPI_SAS_DEVICE_INFO_STP_TARGET) |
442 (attached->device_info &
443 MPI_SAS_DEVICE_INFO_SATA_DEVICE)))
451 mptsas_port_delete(MPT_ADAPTER *ioc, struct mptsas_portinfo_details * port_details)
453 struct mptsas_portinfo *port_info;
454 struct mptsas_phyinfo *phy_info;
460 port_info = port_details->port_info;
461 phy_info = port_info->phy_info;
463 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "%s: [%p]: num_phys=%02d "
464 "bitmask=0x%016llX\n", ioc->name, __func__, port_details,
465 port_details->num_phys, (unsigned long long)
466 port_details->phy_bitmask));
468 for (i = 0; i < port_info->num_phys; i++, phy_info++) {
469 if(phy_info->port_details != port_details)
471 memset(&phy_info->attached, 0, sizeof(struct mptsas_devinfo));
472 mptsas_set_rphy(ioc, phy_info, NULL);
473 phy_info->port_details = NULL;
478 static inline struct sas_rphy *
479 mptsas_get_rphy(struct mptsas_phyinfo *phy_info)
481 if (phy_info->port_details)
482 return phy_info->port_details->rphy;
488 mptsas_set_rphy(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info, struct sas_rphy *rphy)
490 if (phy_info->port_details) {
491 phy_info->port_details->rphy = rphy;
492 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "sas_rphy_add: rphy=%p\n",
497 dsaswideprintk(ioc, dev_printk(KERN_DEBUG,
498 &rphy->dev, MYIOC_s_FMT "add:", ioc->name));
499 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "rphy=%p release=%p\n",
500 ioc->name, rphy, rphy->dev.release));
504 static inline struct sas_port *
505 mptsas_get_port(struct mptsas_phyinfo *phy_info)
507 if (phy_info->port_details)
508 return phy_info->port_details->port;
514 mptsas_set_port(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info, struct sas_port *port)
516 if (phy_info->port_details)
517 phy_info->port_details->port = port;
520 dsaswideprintk(ioc, dev_printk(KERN_DEBUG,
521 &port->dev, MYIOC_s_FMT "add:", ioc->name));
522 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "port=%p release=%p\n",
523 ioc->name, port, port->dev.release));
527 static inline struct scsi_target *
528 mptsas_get_starget(struct mptsas_phyinfo *phy_info)
530 if (phy_info->port_details)
531 return phy_info->port_details->starget;
537 mptsas_set_starget(struct mptsas_phyinfo *phy_info, struct scsi_target *
540 if (phy_info->port_details)
541 phy_info->port_details->starget = starget;
545 * mptsas_add_device_component -
546 * @ioc: Pointer to MPT_ADAPTER structure
547 * @channel: fw mapped id's
554 mptsas_add_device_component(MPT_ADAPTER *ioc, u8 channel, u8 id,
555 u64 sas_address, u32 device_info, u16 slot, u64 enclosure_logical_id)
557 struct mptsas_device_info *sas_info, *next;
558 struct scsi_device *sdev;
559 struct scsi_target *starget;
560 struct sas_rphy *rphy;
563 * Delete all matching devices out of the list
565 mutex_lock(&ioc->sas_device_info_mutex);
566 list_for_each_entry_safe(sas_info, next, &ioc->sas_device_info_list,
568 if (!sas_info->is_logical_volume &&
569 (sas_info->sas_address == sas_address ||
570 (sas_info->fw.channel == channel &&
571 sas_info->fw.id == id))) {
572 list_del(&sas_info->list);
577 sas_info = kzalloc(sizeof(struct mptsas_device_info), GFP_KERNEL);
582 * Set Firmware mapping
584 sas_info->fw.id = id;
585 sas_info->fw.channel = channel;
587 sas_info->sas_address = sas_address;
588 sas_info->device_info = device_info;
589 sas_info->slot = slot;
590 sas_info->enclosure_logical_id = enclosure_logical_id;
591 INIT_LIST_HEAD(&sas_info->list);
592 list_add_tail(&sas_info->list, &ioc->sas_device_info_list);
597 shost_for_each_device(sdev, ioc->sh) {
598 starget = scsi_target(sdev);
599 rphy = dev_to_rphy(starget->dev.parent);
600 if (rphy->identify.sas_address == sas_address) {
601 sas_info->os.id = starget->id;
602 sas_info->os.channel = starget->channel;
607 mutex_unlock(&ioc->sas_device_info_mutex);
612 * mptsas_add_device_component_by_fw -
613 * @ioc: Pointer to MPT_ADAPTER structure
614 * @channel: fw mapped id's
619 mptsas_add_device_component_by_fw(MPT_ADAPTER *ioc, u8 channel, u8 id)
621 struct mptsas_devinfo sas_device;
622 struct mptsas_enclosure enclosure_info;
625 rc = mptsas_sas_device_pg0(ioc, &sas_device,
626 (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
627 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
628 (channel << 8) + id);
632 memset(&enclosure_info, 0, sizeof(struct mptsas_enclosure));
633 mptsas_sas_enclosure_pg0(ioc, &enclosure_info,
634 (MPI_SAS_ENCLOS_PGAD_FORM_HANDLE <<
635 MPI_SAS_ENCLOS_PGAD_FORM_SHIFT),
636 sas_device.handle_enclosure);
638 mptsas_add_device_component(ioc, sas_device.channel,
639 sas_device.id, sas_device.sas_address, sas_device.device_info,
640 sas_device.slot, enclosure_info.enclosure_logical_id);
644 * mptsas_add_device_component_starget_ir - Handle Integrated RAID, adding each individual device to list
645 * @ioc: Pointer to MPT_ADAPTER structure
646 * @channel: fw mapped id's
651 mptsas_add_device_component_starget_ir(MPT_ADAPTER *ioc,
652 struct scsi_target *starget)
655 ConfigPageHeader_t hdr;
656 dma_addr_t dma_handle;
657 pRaidVolumePage0_t buffer = NULL;
659 RaidPhysDiskPage0_t phys_disk;
660 struct mptsas_device_info *sas_info, *next;
662 memset(&cfg, 0 , sizeof(CONFIGPARMS));
663 memset(&hdr, 0 , sizeof(ConfigPageHeader_t));
664 hdr.PageType = MPI_CONFIG_PAGETYPE_RAID_VOLUME;
665 /* assumption that all volumes on channel = 0 */
666 cfg.pageAddr = starget->id;
667 cfg.cfghdr.hdr = &hdr;
668 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
669 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
671 if (mpt_config(ioc, &cfg) != 0)
677 buffer = pci_alloc_consistent(ioc->pcidev, hdr.PageLength * 4,
683 cfg.physAddr = dma_handle;
684 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
686 if (mpt_config(ioc, &cfg) != 0)
689 if (!buffer->NumPhysDisks)
693 * Adding entry for hidden components
695 for (i = 0; i < buffer->NumPhysDisks; i++) {
697 if (mpt_raid_phys_disk_pg0(ioc,
698 buffer->PhysDisk[i].PhysDiskNum, &phys_disk) != 0)
701 mptsas_add_device_component_by_fw(ioc, phys_disk.PhysDiskBus,
702 phys_disk.PhysDiskID);
704 mutex_lock(&ioc->sas_device_info_mutex);
705 list_for_each_entry(sas_info, &ioc->sas_device_info_list,
707 if (!sas_info->is_logical_volume &&
708 (sas_info->fw.channel == phys_disk.PhysDiskBus &&
709 sas_info->fw.id == phys_disk.PhysDiskID)) {
710 sas_info->is_hidden_raid_component = 1;
711 sas_info->volume_id = starget->id;
714 mutex_unlock(&ioc->sas_device_info_mutex);
719 * Delete all matching devices out of the list
721 mutex_lock(&ioc->sas_device_info_mutex);
722 list_for_each_entry_safe(sas_info, next, &ioc->sas_device_info_list,
724 if (sas_info->is_logical_volume && sas_info->fw.id ==
726 list_del(&sas_info->list);
731 sas_info = kzalloc(sizeof(struct mptsas_device_info), GFP_KERNEL);
733 sas_info->fw.id = starget->id;
734 sas_info->os.id = starget->id;
735 sas_info->os.channel = starget->channel;
736 sas_info->is_logical_volume = 1;
737 INIT_LIST_HEAD(&sas_info->list);
738 list_add_tail(&sas_info->list, &ioc->sas_device_info_list);
740 mutex_unlock(&ioc->sas_device_info_mutex);
744 pci_free_consistent(ioc->pcidev, hdr.PageLength * 4, buffer,
749 * mptsas_add_device_component_starget -
750 * @ioc: Pointer to MPT_ADAPTER structure
755 mptsas_add_device_component_starget(MPT_ADAPTER *ioc,
756 struct scsi_target *starget)
759 struct sas_rphy *rphy;
760 struct mptsas_phyinfo *phy_info = NULL;
761 struct mptsas_enclosure enclosure_info;
763 rphy = dev_to_rphy(starget->dev.parent);
764 vtarget = starget->hostdata;
765 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
766 rphy->identify.sas_address);
770 memset(&enclosure_info, 0, sizeof(struct mptsas_enclosure));
771 mptsas_sas_enclosure_pg0(ioc, &enclosure_info,
772 (MPI_SAS_ENCLOS_PGAD_FORM_HANDLE <<
773 MPI_SAS_ENCLOS_PGAD_FORM_SHIFT),
774 phy_info->attached.handle_enclosure);
776 mptsas_add_device_component(ioc, phy_info->attached.channel,
777 phy_info->attached.id, phy_info->attached.sas_address,
778 phy_info->attached.device_info,
779 phy_info->attached.slot, enclosure_info.enclosure_logical_id);
783 * mptsas_del_device_component_by_os - Once a device has been removed, we mark the entry in the list as being cached
784 * @ioc: Pointer to MPT_ADAPTER structure
785 * @channel: os mapped id's
790 mptsas_del_device_component_by_os(MPT_ADAPTER *ioc, u8 channel, u8 id)
792 struct mptsas_device_info *sas_info, *next;
797 list_for_each_entry_safe(sas_info, next, &ioc->sas_device_info_list,
799 if (sas_info->os.channel == channel && sas_info->os.id == id)
800 sas_info->is_cached = 1;
805 * mptsas_del_device_components - Cleaning the list
806 * @ioc: Pointer to MPT_ADAPTER structure
810 mptsas_del_device_components(MPT_ADAPTER *ioc)
812 struct mptsas_device_info *sas_info, *next;
814 mutex_lock(&ioc->sas_device_info_mutex);
815 list_for_each_entry_safe(sas_info, next, &ioc->sas_device_info_list,
817 list_del(&sas_info->list);
820 mutex_unlock(&ioc->sas_device_info_mutex);
825 * mptsas_setup_wide_ports
827 * Updates for new and existing narrow/wide port configuration
828 * in the sas_topology
831 mptsas_setup_wide_ports(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
833 struct mptsas_portinfo_details * port_details;
834 struct mptsas_phyinfo *phy_info, *phy_info_cmp;
838 mutex_lock(&ioc->sas_topology_mutex);
840 phy_info = port_info->phy_info;
841 for (i = 0 ; i < port_info->num_phys ; i++, phy_info++) {
842 if (phy_info->attached.handle)
844 port_details = phy_info->port_details;
847 if (port_details->num_phys < 2)
850 * Removing a phy from a port, letting the last
851 * phy be removed by firmware events.
853 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT
854 "%s: [%p]: deleting phy = %d\n",
855 ioc->name, __func__, port_details, i));
856 port_details->num_phys--;
857 port_details->phy_bitmask &= ~ (1 << phy_info->phy_id);
858 memset(&phy_info->attached, 0, sizeof(struct mptsas_devinfo));
860 devtprintk(ioc, dev_printk(KERN_DEBUG,
861 &phy_info->phy->dev, MYIOC_s_FMT
862 "delete phy %d, phy-obj (0x%p)\n", ioc->name,
863 phy_info->phy_id, phy_info->phy));
864 sas_port_delete_phy(port_details->port, phy_info->phy);
866 phy_info->port_details = NULL;
870 * Populate and refresh the tree
872 phy_info = port_info->phy_info;
873 for (i = 0 ; i < port_info->num_phys ; i++, phy_info++) {
874 sas_address = phy_info->attached.sas_address;
875 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "phy_id=%d sas_address=0x%018llX\n",
876 ioc->name, i, (unsigned long long)sas_address));
879 port_details = phy_info->port_details;
884 port_details = kzalloc(sizeof(struct
885 mptsas_portinfo_details), GFP_KERNEL);
888 port_details->num_phys = 1;
889 port_details->port_info = port_info;
890 if (phy_info->phy_id < 64 )
891 port_details->phy_bitmask |=
892 (1 << phy_info->phy_id);
893 phy_info->sas_port_add_phy=1;
894 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "\t\tForming port\n\t\t"
895 "phy_id=%d sas_address=0x%018llX\n",
896 ioc->name, i, (unsigned long long)sas_address));
897 phy_info->port_details = port_details;
900 if (i == port_info->num_phys - 1)
902 phy_info_cmp = &port_info->phy_info[i + 1];
903 for (j = i + 1 ; j < port_info->num_phys ; j++,
905 if (!phy_info_cmp->attached.sas_address)
907 if (sas_address != phy_info_cmp->attached.sas_address)
909 if (phy_info_cmp->port_details == port_details )
911 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT
912 "\t\tphy_id=%d sas_address=0x%018llX\n",
913 ioc->name, j, (unsigned long long)
914 phy_info_cmp->attached.sas_address));
915 if (phy_info_cmp->port_details) {
917 mptsas_get_rphy(phy_info_cmp);
919 mptsas_get_port(phy_info_cmp);
920 port_details->starget =
921 mptsas_get_starget(phy_info_cmp);
922 port_details->num_phys =
923 phy_info_cmp->port_details->num_phys;
924 if (!phy_info_cmp->port_details->num_phys)
925 kfree(phy_info_cmp->port_details);
927 phy_info_cmp->sas_port_add_phy=1;
929 * Adding a phy to a port
931 phy_info_cmp->port_details = port_details;
932 if (phy_info_cmp->phy_id < 64 )
933 port_details->phy_bitmask |=
934 (1 << phy_info_cmp->phy_id);
935 port_details->num_phys++;
941 for (i = 0; i < port_info->num_phys; i++) {
942 port_details = port_info->phy_info[i].port_details;
945 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT
946 "%s: [%p]: phy_id=%02d num_phys=%02d "
947 "bitmask=0x%016llX\n", ioc->name, __func__,
948 port_details, i, port_details->num_phys,
949 (unsigned long long)port_details->phy_bitmask));
950 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT "\t\tport = %p rphy=%p\n",
951 ioc->name, port_details->port, port_details->rphy));
953 dsaswideprintk(ioc, printk("\n"));
954 mutex_unlock(&ioc->sas_topology_mutex);
958 * csmisas_find_vtarget
966 mptsas_find_vtarget(MPT_ADAPTER *ioc, u8 channel, u8 id)
968 struct scsi_device *sdev;
970 VirtTarget *vtarget = NULL;
972 shost_for_each_device(sdev, ioc->sh) {
973 vdevice = sdev->hostdata;
974 if ((vdevice == NULL) ||
975 (vdevice->vtarget == NULL))
977 if ((vdevice->vtarget->tflags &
978 MPT_TARGET_FLAGS_RAID_COMPONENT ||
979 vdevice->vtarget->raidVolume))
981 if (vdevice->vtarget->id == id &&
982 vdevice->vtarget->channel == channel)
983 vtarget = vdevice->vtarget;
989 mptsas_queue_device_delete(MPT_ADAPTER *ioc,
990 MpiEventDataSasDeviceStatusChange_t *sas_event_data)
992 struct fw_event_work *fw_event;
995 sz = offsetof(struct fw_event_work, event_data) +
996 sizeof(MpiEventDataSasDeviceStatusChange_t);
997 fw_event = kzalloc(sz, GFP_ATOMIC);
999 printk(MYIOC_s_WARN_FMT "%s: failed at (line=%d)\n",
1000 ioc->name, __func__, __LINE__);
1003 memcpy(fw_event->event_data, sas_event_data,
1004 sizeof(MpiEventDataSasDeviceStatusChange_t));
1005 fw_event->event = MPI_EVENT_SAS_DEVICE_STATUS_CHANGE;
1006 fw_event->ioc = ioc;
1007 mptsas_add_fw_event(ioc, fw_event, msecs_to_jiffies(1));
1011 mptsas_queue_rescan(MPT_ADAPTER *ioc)
1013 struct fw_event_work *fw_event;
1016 sz = offsetof(struct fw_event_work, event_data);
1017 fw_event = kzalloc(sz, GFP_ATOMIC);
1019 printk(MYIOC_s_WARN_FMT "%s: failed at (line=%d)\n",
1020 ioc->name, __func__, __LINE__);
1023 fw_event->event = -1;
1024 fw_event->ioc = ioc;
1025 mptsas_add_fw_event(ioc, fw_event, msecs_to_jiffies(1));
1030 * mptsas_target_reset
1032 * Issues TARGET_RESET to end device using handshaking method
1038 * Returns (1) success
1043 mptsas_target_reset(MPT_ADAPTER *ioc, u8 channel, u8 id)
1046 SCSITaskMgmt_t *pScsiTm;
1047 if (mpt_set_taskmgmt_in_progress_flag(ioc) != 0)
1051 mf = mpt_get_msg_frame(mptsasDeviceResetCtx, ioc);
1053 dfailprintk(ioc, printk(MYIOC_s_WARN_FMT
1054 "%s, no msg frames @%d!!\n", ioc->name,
1055 __func__, __LINE__));
1059 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT "TaskMgmt request (mf=%p)\n",
1062 /* Format the Request
1064 pScsiTm = (SCSITaskMgmt_t *) mf;
1065 memset (pScsiTm, 0, sizeof(SCSITaskMgmt_t));
1066 pScsiTm->TargetID = id;
1067 pScsiTm->Bus = channel;
1068 pScsiTm->Function = MPI_FUNCTION_SCSI_TASK_MGMT;
1069 pScsiTm->TaskType = MPI_SCSITASKMGMT_TASKTYPE_TARGET_RESET;
1070 pScsiTm->MsgFlags = MPI_SCSITASKMGMT_MSGFLAGS_LIPRESET_RESET_OPTION;
1072 DBG_DUMP_TM_REQUEST_FRAME(ioc, (u32 *)mf);
1074 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1075 "TaskMgmt type=%d (sas device delete) fw_channel = %d fw_id = %d)\n",
1076 ioc->name, MPI_SCSITASKMGMT_TASKTYPE_TARGET_RESET, channel, id));
1078 mpt_put_msg_frame_hi_pri(mptsasDeviceResetCtx, ioc, mf);
1084 mpt_clear_taskmgmt_in_progress_flag(ioc);
1089 mptsas_block_io_sdev(struct scsi_device *sdev, void *data)
1091 scsi_device_set_state(sdev, SDEV_BLOCK);
1095 mptsas_block_io_starget(struct scsi_target *starget)
1098 starget_for_each_device(starget, NULL, mptsas_block_io_sdev);
1102 * mptsas_target_reset_queue
1104 * Receive request for TARGET_RESET after receiving an firmware
1105 * event NOT_RESPONDING_EVENT, then put command in link list
1106 * and queue if task_queue already in use.
1113 mptsas_target_reset_queue(MPT_ADAPTER *ioc,
1114 EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *sas_event_data)
1116 MPT_SCSI_HOST *hd = shost_priv(ioc->sh);
1117 VirtTarget *vtarget = NULL;
1118 struct mptsas_target_reset_event *target_reset_list;
1121 id = sas_event_data->TargetID;
1122 channel = sas_event_data->Bus;
1124 vtarget = mptsas_find_vtarget(ioc, channel, id);
1126 mptsas_block_io_starget(vtarget->starget);
1127 vtarget->deleted = 1; /* block IO */
1130 target_reset_list = kzalloc(sizeof(struct mptsas_target_reset_event),
1132 if (!target_reset_list) {
1133 dfailprintk(ioc, printk(MYIOC_s_WARN_FMT
1134 "%s, failed to allocate mem @%d..!!\n",
1135 ioc->name, __func__, __LINE__));
1139 memcpy(&target_reset_list->sas_event_data, sas_event_data,
1140 sizeof(*sas_event_data));
1141 list_add_tail(&target_reset_list->list, &hd->target_reset_list);
1143 target_reset_list->time_count = jiffies;
1145 if (mptsas_target_reset(ioc, channel, id)) {
1146 target_reset_list->target_reset_issued = 1;
1151 * mptsas_schedule_target_reset- send pending target reset
1152 * @iocp: per adapter object
1154 * This function will delete scheduled target reset from the list and
1155 * try to send next target reset. This will be called from completion
1156 * context of any Task management command.
1160 mptsas_schedule_target_reset(void *iocp)
1162 MPT_ADAPTER *ioc = (MPT_ADAPTER *)(iocp);
1163 MPT_SCSI_HOST *hd = shost_priv(ioc->sh);
1164 struct list_head *head = &hd->target_reset_list;
1165 struct mptsas_target_reset_event *target_reset_list;
1168 * issue target reset to next device in the queue
1171 head = &hd->target_reset_list;
1172 if (list_empty(head))
1175 target_reset_list = list_entry(head->next,
1176 struct mptsas_target_reset_event, list);
1178 id = target_reset_list->sas_event_data.TargetID;
1179 channel = target_reset_list->sas_event_data.Bus;
1180 target_reset_list->time_count = jiffies;
1182 if (mptsas_target_reset(ioc, channel, id))
1183 target_reset_list->target_reset_issued = 1;
1189 * mptsas_taskmgmt_complete - complete SAS task management function
1190 * @ioc: Pointer to MPT_ADAPTER structure
1192 * Completion for TARGET_RESET after NOT_RESPONDING_EVENT, enable work
1193 * queue to finish off removing device from upper layers. then send next
1194 * TARGET_RESET in the queue.
1197 mptsas_taskmgmt_complete(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *mr)
1199 MPT_SCSI_HOST *hd = shost_priv(ioc->sh);
1200 struct list_head *head = &hd->target_reset_list;
1202 struct mptsas_target_reset_event *target_reset_list;
1203 SCSITaskMgmtReply_t *pScsiTmReply;
1205 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT "TaskMgmt completed: "
1206 "(mf = %p, mr = %p)\n", ioc->name, mf, mr));
1208 pScsiTmReply = (SCSITaskMgmtReply_t *)mr;
1210 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1211 "\tTaskMgmt completed: fw_channel = %d, fw_id = %d,\n"
1212 "\ttask_type = 0x%02X, iocstatus = 0x%04X "
1213 "loginfo = 0x%08X,\n\tresponse_code = 0x%02X, "
1214 "term_cmnds = %d\n", ioc->name,
1215 pScsiTmReply->Bus, pScsiTmReply->TargetID,
1216 pScsiTmReply->TaskType,
1217 le16_to_cpu(pScsiTmReply->IOCStatus),
1218 le32_to_cpu(pScsiTmReply->IOCLogInfo),
1219 pScsiTmReply->ResponseCode,
1220 le32_to_cpu(pScsiTmReply->TerminationCount)));
1222 if (pScsiTmReply->ResponseCode)
1223 mptscsih_taskmgmt_response_code(ioc,
1224 pScsiTmReply->ResponseCode);
1227 if (pScsiTmReply && (pScsiTmReply->TaskType ==
1228 MPI_SCSITASKMGMT_TASKTYPE_QUERY_TASK || pScsiTmReply->TaskType ==
1229 MPI_SCSITASKMGMT_TASKTYPE_ABRT_TASK_SET)) {
1230 ioc->taskmgmt_cmds.status |= MPT_MGMT_STATUS_COMMAND_GOOD;
1231 ioc->taskmgmt_cmds.status |= MPT_MGMT_STATUS_RF_VALID;
1232 memcpy(ioc->taskmgmt_cmds.reply, mr,
1233 min(MPT_DEFAULT_FRAME_SIZE, 4 * mr->u.reply.MsgLength));
1234 if (ioc->taskmgmt_cmds.status & MPT_MGMT_STATUS_PENDING) {
1235 ioc->taskmgmt_cmds.status &= ~MPT_MGMT_STATUS_PENDING;
1236 complete(&ioc->taskmgmt_cmds.done);
1242 mpt_clear_taskmgmt_in_progress_flag(ioc);
1244 if (list_empty(head))
1247 target_reset_list = list_entry(head->next,
1248 struct mptsas_target_reset_event, list);
1250 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1251 "TaskMgmt: completed (%d seconds)\n",
1252 ioc->name, jiffies_to_msecs(jiffies -
1253 target_reset_list->time_count)/1000));
1255 id = pScsiTmReply->TargetID;
1256 channel = pScsiTmReply->Bus;
1257 target_reset_list->time_count = jiffies;
1260 * retry target reset
1262 if (!target_reset_list->target_reset_issued) {
1263 if (mptsas_target_reset(ioc, channel, id))
1264 target_reset_list->target_reset_issued = 1;
1269 * enable work queue to remove device from upper layers
1271 list_del(&target_reset_list->list);
1272 if (!ioc->fw_events_off)
1273 mptsas_queue_device_delete(ioc,
1274 &target_reset_list->sas_event_data);
1277 ioc->schedule_target_reset(ioc);
1283 * mptscsih_ioc_reset
1290 mptsas_ioc_reset(MPT_ADAPTER *ioc, int reset_phase)
1295 rc = mptscsih_ioc_reset(ioc, reset_phase);
1296 if ((ioc->bus_type != SAS) || (!rc))
1299 hd = shost_priv(ioc->sh);
1303 switch (reset_phase) {
1304 case MPT_IOC_SETUP_RESET:
1305 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1306 "%s: MPT_IOC_SETUP_RESET\n", ioc->name, __func__));
1307 mptsas_fw_event_off(ioc);
1309 case MPT_IOC_PRE_RESET:
1310 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1311 "%s: MPT_IOC_PRE_RESET\n", ioc->name, __func__));
1313 case MPT_IOC_POST_RESET:
1314 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1315 "%s: MPT_IOC_POST_RESET\n", ioc->name, __func__));
1316 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_PENDING) {
1317 ioc->sas_mgmt.status |= MPT_MGMT_STATUS_DID_IOCRESET;
1318 complete(&ioc->sas_mgmt.done);
1320 mptsas_cleanup_fw_event_q(ioc);
1321 mptsas_queue_rescan(ioc);
1333 * enum device_state -
1334 * @DEVICE_RETRY: need to retry the TUR
1335 * @DEVICE_ERROR: TUR return error, don't add device
1336 * @DEVICE_READY: device can be added
1346 mptsas_sas_enclosure_pg0(MPT_ADAPTER *ioc, struct mptsas_enclosure *enclosure,
1347 u32 form, u32 form_specific)
1349 ConfigExtendedPageHeader_t hdr;
1351 SasEnclosurePage0_t *buffer;
1352 dma_addr_t dma_handle;
1354 __le64 le_identifier;
1356 memset(&hdr, 0, sizeof(hdr));
1357 hdr.PageVersion = MPI_SASENCLOSURE0_PAGEVERSION;
1359 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
1360 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_ENCLOSURE;
1362 cfg.cfghdr.ehdr = &hdr;
1364 cfg.pageAddr = form + form_specific;
1365 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
1366 cfg.dir = 0; /* read */
1367 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
1369 error = mpt_config(ioc, &cfg);
1372 if (!hdr.ExtPageLength) {
1377 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
1384 cfg.physAddr = dma_handle;
1385 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
1387 error = mpt_config(ioc, &cfg);
1389 goto out_free_consistent;
1391 /* save config data */
1392 memcpy(&le_identifier, &buffer->EnclosureLogicalID, sizeof(__le64));
1393 enclosure->enclosure_logical_id = le64_to_cpu(le_identifier);
1394 enclosure->enclosure_handle = le16_to_cpu(buffer->EnclosureHandle);
1395 enclosure->flags = le16_to_cpu(buffer->Flags);
1396 enclosure->num_slot = le16_to_cpu(buffer->NumSlots);
1397 enclosure->start_slot = le16_to_cpu(buffer->StartSlot);
1398 enclosure->start_id = buffer->StartTargetID;
1399 enclosure->start_channel = buffer->StartBus;
1400 enclosure->sep_id = buffer->SEPTargetID;
1401 enclosure->sep_channel = buffer->SEPBus;
1403 out_free_consistent:
1404 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
1405 buffer, dma_handle);
1411 * mptsas_add_end_device - report a new end device to sas transport layer
1412 * @ioc: Pointer to MPT_ADAPTER structure
1413 * @phy_info: describes attached device
1415 * return (0) success (1) failure
1419 mptsas_add_end_device(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info)
1421 struct sas_rphy *rphy;
1422 struct sas_port *port;
1423 struct sas_identify identify;
1428 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1429 "%s: exit at line=%d\n", ioc->name,
1430 __func__, __LINE__));
1434 fw_id = phy_info->attached.id;
1436 if (mptsas_get_rphy(phy_info)) {
1437 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1438 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1439 __func__, fw_id, __LINE__));
1443 port = mptsas_get_port(phy_info);
1445 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1446 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1447 __func__, fw_id, __LINE__));
1451 if (phy_info->attached.device_info &
1452 MPI_SAS_DEVICE_INFO_SSP_TARGET)
1454 if (phy_info->attached.device_info &
1455 MPI_SAS_DEVICE_INFO_STP_TARGET)
1457 if (phy_info->attached.device_info &
1458 MPI_SAS_DEVICE_INFO_SATA_DEVICE)
1461 printk(MYIOC_s_INFO_FMT "attaching %s device: fw_channel %d, fw_id %d,"
1462 " phy %d, sas_addr 0x%llx\n", ioc->name, ds,
1463 phy_info->attached.channel, phy_info->attached.id,
1464 phy_info->attached.phy_id, (unsigned long long)
1465 phy_info->attached.sas_address);
1467 mptsas_parse_device_info(&identify, &phy_info->attached);
1468 rphy = sas_end_device_alloc(port);
1470 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1471 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1472 __func__, fw_id, __LINE__));
1473 return 5; /* non-fatal: an rphy can be added later */
1476 rphy->identify = identify;
1477 if (sas_rphy_add(rphy)) {
1478 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1479 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1480 __func__, fw_id, __LINE__));
1481 sas_rphy_free(rphy);
1484 mptsas_set_rphy(ioc, phy_info, rphy);
1489 * mptsas_del_end_device - report a deleted end device to sas transport layer
1490 * @ioc: Pointer to MPT_ADAPTER structure
1491 * @phy_info: describes attached device
1495 mptsas_del_end_device(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info)
1497 struct sas_rphy *rphy;
1498 struct sas_port *port;
1499 struct mptsas_portinfo *port_info;
1500 struct mptsas_phyinfo *phy_info_parent;
1509 fw_id = phy_info->attached.id;
1510 sas_address = phy_info->attached.sas_address;
1512 if (!phy_info->port_details) {
1513 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1514 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1515 __func__, fw_id, __LINE__));
1518 rphy = mptsas_get_rphy(phy_info);
1520 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1521 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1522 __func__, fw_id, __LINE__));
1526 if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SSP_INITIATOR
1527 || phy_info->attached.device_info
1528 & MPI_SAS_DEVICE_INFO_SMP_INITIATOR
1529 || phy_info->attached.device_info
1530 & MPI_SAS_DEVICE_INFO_STP_INITIATOR)
1532 if (phy_info->attached.device_info &
1533 MPI_SAS_DEVICE_INFO_SSP_TARGET)
1535 if (phy_info->attached.device_info &
1536 MPI_SAS_DEVICE_INFO_STP_TARGET)
1538 if (phy_info->attached.device_info &
1539 MPI_SAS_DEVICE_INFO_SATA_DEVICE)
1542 dev_printk(KERN_DEBUG, &rphy->dev, MYIOC_s_FMT
1543 "removing %s device: fw_channel %d, fw_id %d, phy %d,"
1544 "sas_addr 0x%llx\n", ioc->name, ds, phy_info->attached.channel,
1545 phy_info->attached.id, phy_info->attached.phy_id,
1546 (unsigned long long) sas_address);
1548 port = mptsas_get_port(phy_info);
1550 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
1551 "%s: fw_id=%d exit at line=%d\n", ioc->name,
1552 __func__, fw_id, __LINE__));
1555 port_info = phy_info->portinfo;
1556 phy_info_parent = port_info->phy_info;
1557 for (i = 0; i < port_info->num_phys; i++, phy_info_parent++) {
1558 if (!phy_info_parent->phy)
1560 if (phy_info_parent->attached.sas_address !=
1563 dev_printk(KERN_DEBUG, &phy_info_parent->phy->dev,
1564 MYIOC_s_FMT "delete phy %d, phy-obj (0x%p)\n",
1565 ioc->name, phy_info_parent->phy_id,
1566 phy_info_parent->phy);
1567 sas_port_delete_phy(port, phy_info_parent->phy);
1570 dev_printk(KERN_DEBUG, &port->dev, MYIOC_s_FMT
1571 "delete port %d, sas_addr (0x%llx)\n", ioc->name,
1572 port->port_identifier, (unsigned long long)sas_address);
1573 sas_port_delete(port);
1574 mptsas_set_port(ioc, phy_info, NULL);
1575 mptsas_port_delete(ioc, phy_info->port_details);
1578 struct mptsas_phyinfo *
1579 mptsas_refreshing_device_handles(MPT_ADAPTER *ioc,
1580 struct mptsas_devinfo *sas_device)
1582 struct mptsas_phyinfo *phy_info;
1583 struct mptsas_portinfo *port_info;
1586 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
1587 sas_device->sas_address);
1590 port_info = phy_info->portinfo;
1593 mutex_lock(&ioc->sas_topology_mutex);
1594 for (i = 0; i < port_info->num_phys; i++) {
1595 if (port_info->phy_info[i].attached.sas_address !=
1596 sas_device->sas_address)
1598 port_info->phy_info[i].attached.channel = sas_device->channel;
1599 port_info->phy_info[i].attached.id = sas_device->id;
1600 port_info->phy_info[i].attached.sas_address =
1601 sas_device->sas_address;
1602 port_info->phy_info[i].attached.handle = sas_device->handle;
1603 port_info->phy_info[i].attached.handle_parent =
1604 sas_device->handle_parent;
1605 port_info->phy_info[i].attached.handle_enclosure =
1606 sas_device->handle_enclosure;
1608 mutex_unlock(&ioc->sas_topology_mutex);
1614 * mptsas_firmware_event_work - work thread for processing fw events
1615 * @work: work queue payload containing info describing the event
1620 mptsas_firmware_event_work(struct work_struct *work)
1622 struct fw_event_work *fw_event =
1623 container_of(work, struct fw_event_work, work.work);
1624 MPT_ADAPTER *ioc = fw_event->ioc;
1626 /* special rescan topology handling */
1627 if (fw_event->event == -1) {
1628 if (ioc->in_rescan) {
1629 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
1630 "%s: rescan ignored as it is in progress\n",
1631 ioc->name, __func__));
1634 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "%s: rescan after "
1635 "reset\n", ioc->name, __func__));
1637 mptsas_not_responding_devices(ioc);
1638 mptsas_scan_sas_topology(ioc);
1640 mptsas_free_fw_event(ioc, fw_event);
1641 mptsas_fw_event_on(ioc);
1645 /* events handling turned off during host reset */
1646 if (ioc->fw_events_off) {
1647 mptsas_free_fw_event(ioc, fw_event);
1651 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "%s: fw_event=(0x%p), "
1652 "event = (0x%02x)\n", ioc->name, __func__, fw_event,
1653 (fw_event->event & 0xFF)));
1655 switch (fw_event->event) {
1656 case MPI_EVENT_SAS_DEVICE_STATUS_CHANGE:
1657 mptsas_send_sas_event(fw_event);
1659 case MPI_EVENT_INTEGRATED_RAID:
1660 mptsas_send_raid_event(fw_event);
1663 mptsas_send_ir2_event(fw_event);
1665 case MPI_EVENT_PERSISTENT_TABLE_FULL:
1666 mptbase_sas_persist_operation(ioc,
1667 MPI_SAS_OP_CLEAR_NOT_PRESENT);
1668 mptsas_free_fw_event(ioc, fw_event);
1670 case MPI_EVENT_SAS_BROADCAST_PRIMITIVE:
1671 mptsas_broadcast_primative_work(fw_event);
1673 case MPI_EVENT_SAS_EXPANDER_STATUS_CHANGE:
1674 mptsas_send_expander_event(fw_event);
1676 case MPI_EVENT_SAS_PHY_LINK_STATUS:
1677 mptsas_send_link_status_event(fw_event);
1679 case MPI_EVENT_QUEUE_FULL:
1680 mptsas_handle_queue_full_event(fw_event);
1688 mptsas_slave_configure(struct scsi_device *sdev)
1690 struct Scsi_Host *host = sdev->host;
1691 MPT_SCSI_HOST *hd = shost_priv(host);
1692 MPT_ADAPTER *ioc = hd->ioc;
1693 VirtDevice *vdevice = sdev->hostdata;
1695 if (vdevice->vtarget->deleted) {
1696 sdev_printk(KERN_INFO, sdev, "clearing deleted flag\n");
1697 vdevice->vtarget->deleted = 0;
1701 * RAID volumes placed beyond the last expected port.
1702 * Ignore sending sas mode pages in that case..
1704 if (sdev->channel == MPTSAS_RAID_CHANNEL) {
1705 mptsas_add_device_component_starget_ir(ioc, scsi_target(sdev));
1709 sas_read_port_mode_page(sdev);
1711 mptsas_add_device_component_starget(ioc, scsi_target(sdev));
1714 return mptscsih_slave_configure(sdev);
1718 mptsas_target_alloc(struct scsi_target *starget)
1720 struct Scsi_Host *host = dev_to_shost(&starget->dev);
1721 MPT_SCSI_HOST *hd = shost_priv(host);
1722 VirtTarget *vtarget;
1724 struct sas_rphy *rphy;
1725 struct mptsas_portinfo *p;
1727 MPT_ADAPTER *ioc = hd->ioc;
1729 vtarget = kzalloc(sizeof(VirtTarget), GFP_KERNEL);
1733 vtarget->starget = starget;
1734 vtarget->ioc_id = ioc->id;
1735 vtarget->tflags = MPT_TARGET_FLAGS_Q_YES;
1740 * RAID volumes placed beyond the last expected port.
1742 if (starget->channel == MPTSAS_RAID_CHANNEL) {
1743 if (!ioc->raid_data.pIocPg2) {
1747 for (i = 0; i < ioc->raid_data.pIocPg2->NumActiveVolumes; i++) {
1748 if (id == ioc->raid_data.pIocPg2->
1749 RaidVolume[i].VolumeID) {
1750 channel = ioc->raid_data.pIocPg2->
1751 RaidVolume[i].VolumeBus;
1754 vtarget->raidVolume = 1;
1758 rphy = dev_to_rphy(starget->dev.parent);
1759 mutex_lock(&ioc->sas_topology_mutex);
1760 list_for_each_entry(p, &ioc->sas_topology, list) {
1761 for (i = 0; i < p->num_phys; i++) {
1762 if (p->phy_info[i].attached.sas_address !=
1763 rphy->identify.sas_address)
1765 id = p->phy_info[i].attached.id;
1766 channel = p->phy_info[i].attached.channel;
1767 mptsas_set_starget(&p->phy_info[i], starget);
1770 * Exposing hidden raid components
1772 if (mptscsih_is_phys_disk(ioc, channel, id)) {
1773 id = mptscsih_raid_id_to_num(ioc,
1776 MPT_TARGET_FLAGS_RAID_COMPONENT;
1777 p->phy_info[i].attached.phys_disk_num = id;
1779 mutex_unlock(&ioc->sas_topology_mutex);
1783 mutex_unlock(&ioc->sas_topology_mutex);
1790 vtarget->channel = channel;
1791 starget->hostdata = vtarget;
1796 mptsas_target_destroy(struct scsi_target *starget)
1798 struct Scsi_Host *host = dev_to_shost(&starget->dev);
1799 MPT_SCSI_HOST *hd = shost_priv(host);
1800 struct sas_rphy *rphy;
1801 struct mptsas_portinfo *p;
1803 MPT_ADAPTER *ioc = hd->ioc;
1804 VirtTarget *vtarget;
1806 if (!starget->hostdata)
1809 vtarget = starget->hostdata;
1811 mptsas_del_device_component_by_os(ioc, starget->channel,
1815 if (starget->channel == MPTSAS_RAID_CHANNEL)
1818 rphy = dev_to_rphy(starget->dev.parent);
1819 list_for_each_entry(p, &ioc->sas_topology, list) {
1820 for (i = 0; i < p->num_phys; i++) {
1821 if (p->phy_info[i].attached.sas_address !=
1822 rphy->identify.sas_address)
1825 starget_printk(KERN_INFO, starget, MYIOC_s_FMT
1826 "delete device: fw_channel %d, fw_id %d, phy %d, "
1827 "sas_addr 0x%llx\n", ioc->name,
1828 p->phy_info[i].attached.channel,
1829 p->phy_info[i].attached.id,
1830 p->phy_info[i].attached.phy_id, (unsigned long long)
1831 p->phy_info[i].attached.sas_address);
1833 mptsas_set_starget(&p->phy_info[i], NULL);
1838 vtarget->starget = NULL;
1839 kfree(starget->hostdata);
1840 starget->hostdata = NULL;
1845 mptsas_slave_alloc(struct scsi_device *sdev)
1847 struct Scsi_Host *host = sdev->host;
1848 MPT_SCSI_HOST *hd = shost_priv(host);
1849 struct sas_rphy *rphy;
1850 struct mptsas_portinfo *p;
1851 VirtDevice *vdevice;
1852 struct scsi_target *starget;
1854 MPT_ADAPTER *ioc = hd->ioc;
1856 vdevice = kzalloc(sizeof(VirtDevice), GFP_KERNEL);
1858 printk(MYIOC_s_ERR_FMT "slave_alloc kzalloc(%zd) FAILED!\n",
1859 ioc->name, sizeof(VirtDevice));
1862 starget = scsi_target(sdev);
1863 vdevice->vtarget = starget->hostdata;
1865 if (sdev->channel == MPTSAS_RAID_CHANNEL)
1868 rphy = dev_to_rphy(sdev->sdev_target->dev.parent);
1869 mutex_lock(&ioc->sas_topology_mutex);
1870 list_for_each_entry(p, &ioc->sas_topology, list) {
1871 for (i = 0; i < p->num_phys; i++) {
1872 if (p->phy_info[i].attached.sas_address !=
1873 rphy->identify.sas_address)
1875 vdevice->lun = sdev->lun;
1877 * Exposing hidden raid components
1879 if (mptscsih_is_phys_disk(ioc,
1880 p->phy_info[i].attached.channel,
1881 p->phy_info[i].attached.id))
1882 sdev->no_uld_attach = 1;
1883 mutex_unlock(&ioc->sas_topology_mutex);
1887 mutex_unlock(&ioc->sas_topology_mutex);
1893 vdevice->vtarget->num_luns++;
1894 sdev->hostdata = vdevice;
1899 mptsas_qcmd_lck(struct scsi_cmnd *SCpnt, void (*done)(struct scsi_cmnd *))
1903 VirtDevice *vdevice = SCpnt->device->hostdata;
1905 if (!vdevice || !vdevice->vtarget || vdevice->vtarget->deleted) {
1906 SCpnt->result = DID_NO_CONNECT << 16;
1911 hd = shost_priv(SCpnt->device->host);
1914 if (ioc->sas_discovery_quiesce_io)
1915 return SCSI_MLQUEUE_HOST_BUSY;
1917 if (ioc->debug_level & MPT_DEBUG_SCSI)
1918 scsi_print_command(SCpnt);
1920 return mptscsih_qcmd(SCpnt,done);
1923 static DEF_SCSI_QCMD(mptsas_qcmd)
1926 * mptsas_mptsas_eh_timed_out - resets the scsi_cmnd timeout
1927 * if the device under question is currently in the
1928 * device removal delay.
1929 * @sc: scsi command that the midlayer is about to time out
1932 static enum blk_eh_timer_return mptsas_eh_timed_out(struct scsi_cmnd *sc)
1936 VirtDevice *vdevice;
1937 enum blk_eh_timer_return rc = BLK_EH_NOT_HANDLED;
1939 hd = shost_priv(sc->device->host);
1941 printk(KERN_ERR MYNAM ": %s: Can't locate host! (sc=%p)\n",
1947 if (ioc->bus_type != SAS) {
1948 printk(KERN_ERR MYNAM ": %s: Wrong bus type (sc=%p)\n",
1953 vdevice = sc->device->hostdata;
1954 if (vdevice && vdevice->vtarget && (vdevice->vtarget->inDMD
1955 || vdevice->vtarget->deleted)) {
1956 dtmprintk(ioc, printk(MYIOC_s_WARN_FMT ": %s: target removed "
1957 "or in device removal delay (sc=%p)\n",
1958 ioc->name, __func__, sc));
1959 rc = BLK_EH_RESET_TIMER;
1968 static struct scsi_host_template mptsas_driver_template = {
1969 .module = THIS_MODULE,
1970 .proc_name = "mptsas",
1971 .proc_info = mptscsih_proc_info,
1972 .name = "MPT SAS Host",
1973 .info = mptscsih_info,
1974 .queuecommand = mptsas_qcmd,
1975 .target_alloc = mptsas_target_alloc,
1976 .slave_alloc = mptsas_slave_alloc,
1977 .slave_configure = mptsas_slave_configure,
1978 .target_destroy = mptsas_target_destroy,
1979 .slave_destroy = mptscsih_slave_destroy,
1980 .change_queue_depth = mptscsih_change_queue_depth,
1981 .eh_abort_handler = mptscsih_abort,
1982 .eh_device_reset_handler = mptscsih_dev_reset,
1983 .eh_host_reset_handler = mptscsih_host_reset,
1984 .bios_param = mptscsih_bios_param,
1985 .can_queue = MPT_SAS_CAN_QUEUE,
1987 .sg_tablesize = MPT_SCSI_SG_DEPTH,
1988 .max_sectors = 8192,
1990 .use_clustering = ENABLE_CLUSTERING,
1991 .shost_attrs = mptscsih_host_attrs,
1994 static int mptsas_get_linkerrors(struct sas_phy *phy)
1996 MPT_ADAPTER *ioc = phy_to_ioc(phy);
1997 ConfigExtendedPageHeader_t hdr;
1999 SasPhyPage1_t *buffer;
2000 dma_addr_t dma_handle;
2003 /* FIXME: only have link errors on local phys */
2004 if (!scsi_is_sas_phy_local(phy))
2007 hdr.PageVersion = MPI_SASPHY1_PAGEVERSION;
2008 hdr.ExtPageLength = 0;
2009 hdr.PageNumber = 1 /* page number 1*/;
2012 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2013 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_PHY;
2015 cfg.cfghdr.ehdr = &hdr;
2017 cfg.pageAddr = phy->identify.phy_identifier;
2018 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2019 cfg.dir = 0; /* read */
2020 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2022 error = mpt_config(ioc, &cfg);
2025 if (!hdr.ExtPageLength)
2028 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2033 cfg.physAddr = dma_handle;
2034 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2036 error = mpt_config(ioc, &cfg);
2038 goto out_free_consistent;
2040 mptsas_print_phy_pg1(ioc, buffer);
2042 phy->invalid_dword_count = le32_to_cpu(buffer->InvalidDwordCount);
2043 phy->running_disparity_error_count =
2044 le32_to_cpu(buffer->RunningDisparityErrorCount);
2045 phy->loss_of_dword_sync_count =
2046 le32_to_cpu(buffer->LossDwordSynchCount);
2047 phy->phy_reset_problem_count =
2048 le32_to_cpu(buffer->PhyResetProblemCount);
2050 out_free_consistent:
2051 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2052 buffer, dma_handle);
2056 static int mptsas_mgmt_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *req,
2057 MPT_FRAME_HDR *reply)
2059 ioc->sas_mgmt.status |= MPT_MGMT_STATUS_COMMAND_GOOD;
2060 if (reply != NULL) {
2061 ioc->sas_mgmt.status |= MPT_MGMT_STATUS_RF_VALID;
2062 memcpy(ioc->sas_mgmt.reply, reply,
2063 min(ioc->reply_sz, 4 * reply->u.reply.MsgLength));
2066 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_PENDING) {
2067 ioc->sas_mgmt.status &= ~MPT_MGMT_STATUS_PENDING;
2068 complete(&ioc->sas_mgmt.done);
2074 static int mptsas_phy_reset(struct sas_phy *phy, int hard_reset)
2076 MPT_ADAPTER *ioc = phy_to_ioc(phy);
2077 SasIoUnitControlRequest_t *req;
2078 SasIoUnitControlReply_t *reply;
2081 unsigned long timeleft;
2082 int error = -ERESTARTSYS;
2084 /* FIXME: fusion doesn't allow non-local phy reset */
2085 if (!scsi_is_sas_phy_local(phy))
2088 /* not implemented for expanders */
2089 if (phy->identify.target_port_protocols & SAS_PROTOCOL_SMP)
2092 if (mutex_lock_interruptible(&ioc->sas_mgmt.mutex))
2095 mf = mpt_get_msg_frame(mptsasMgmtCtx, ioc);
2101 hdr = (MPIHeader_t *) mf;
2102 req = (SasIoUnitControlRequest_t *)mf;
2103 memset(req, 0, sizeof(SasIoUnitControlRequest_t));
2104 req->Function = MPI_FUNCTION_SAS_IO_UNIT_CONTROL;
2105 req->MsgContext = hdr->MsgContext;
2106 req->Operation = hard_reset ?
2107 MPI_SAS_OP_PHY_HARD_RESET : MPI_SAS_OP_PHY_LINK_RESET;
2108 req->PhyNum = phy->identify.phy_identifier;
2110 INITIALIZE_MGMT_STATUS(ioc->sas_mgmt.status)
2111 mpt_put_msg_frame(mptsasMgmtCtx, ioc, mf);
2113 timeleft = wait_for_completion_timeout(&ioc->sas_mgmt.done,
2115 if (!(ioc->sas_mgmt.status & MPT_MGMT_STATUS_COMMAND_GOOD)) {
2117 mpt_free_msg_frame(ioc, mf);
2118 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_DID_IOCRESET)
2121 mpt_Soft_Hard_ResetHandler(ioc, CAN_SLEEP);
2125 /* a reply frame is expected */
2126 if ((ioc->sas_mgmt.status &
2127 MPT_MGMT_STATUS_RF_VALID) == 0) {
2132 /* process the completed Reply Message Frame */
2133 reply = (SasIoUnitControlReply_t *)ioc->sas_mgmt.reply;
2134 if (reply->IOCStatus != MPI_IOCSTATUS_SUCCESS) {
2135 printk(MYIOC_s_INFO_FMT "%s: IOCStatus=0x%X IOCLogInfo=0x%X\n",
2136 ioc->name, __func__, reply->IOCStatus, reply->IOCLogInfo);
2144 CLEAR_MGMT_STATUS(ioc->sas_mgmt.status)
2145 mutex_unlock(&ioc->sas_mgmt.mutex);
2151 mptsas_get_enclosure_identifier(struct sas_rphy *rphy, u64 *identifier)
2153 MPT_ADAPTER *ioc = rphy_to_ioc(rphy);
2155 struct mptsas_portinfo *p;
2156 struct mptsas_enclosure enclosure_info;
2157 u64 enclosure_handle;
2159 mutex_lock(&ioc->sas_topology_mutex);
2160 list_for_each_entry(p, &ioc->sas_topology, list) {
2161 for (i = 0; i < p->num_phys; i++) {
2162 if (p->phy_info[i].attached.sas_address ==
2163 rphy->identify.sas_address) {
2164 enclosure_handle = p->phy_info[i].
2165 attached.handle_enclosure;
2170 mutex_unlock(&ioc->sas_topology_mutex);
2174 mutex_unlock(&ioc->sas_topology_mutex);
2175 memset(&enclosure_info, 0, sizeof(struct mptsas_enclosure));
2176 error = mptsas_sas_enclosure_pg0(ioc, &enclosure_info,
2177 (MPI_SAS_ENCLOS_PGAD_FORM_HANDLE <<
2178 MPI_SAS_ENCLOS_PGAD_FORM_SHIFT), enclosure_handle);
2180 *identifier = enclosure_info.enclosure_logical_id;
2185 mptsas_get_bay_identifier(struct sas_rphy *rphy)
2187 MPT_ADAPTER *ioc = rphy_to_ioc(rphy);
2188 struct mptsas_portinfo *p;
2191 mutex_lock(&ioc->sas_topology_mutex);
2192 list_for_each_entry(p, &ioc->sas_topology, list) {
2193 for (i = 0; i < p->num_phys; i++) {
2194 if (p->phy_info[i].attached.sas_address ==
2195 rphy->identify.sas_address) {
2196 rc = p->phy_info[i].attached.slot;
2203 mutex_unlock(&ioc->sas_topology_mutex);
2207 static int mptsas_smp_handler(struct Scsi_Host *shost, struct sas_rphy *rphy,
2208 struct request *req)
2210 MPT_ADAPTER *ioc = ((MPT_SCSI_HOST *) shost->hostdata)->ioc;
2212 SmpPassthroughRequest_t *smpreq;
2213 struct request *rsp = req->next_rq;
2216 unsigned long timeleft;
2218 dma_addr_t dma_addr_in = 0;
2219 dma_addr_t dma_addr_out = 0;
2220 u64 sas_address = 0;
2223 printk(MYIOC_s_ERR_FMT "%s: the smp response space is missing\n",
2224 ioc->name, __func__);
2228 /* do we need to support multiple segments? */
2229 if (req->bio->bi_vcnt > 1 || rsp->bio->bi_vcnt > 1) {
2230 printk(MYIOC_s_ERR_FMT "%s: multiple segments req %u %u, rsp %u %u\n",
2231 ioc->name, __func__, req->bio->bi_vcnt, blk_rq_bytes(req),
2232 rsp->bio->bi_vcnt, blk_rq_bytes(rsp));
2236 ret = mutex_lock_interruptible(&ioc->sas_mgmt.mutex);
2240 mf = mpt_get_msg_frame(mptsasMgmtCtx, ioc);
2246 smpreq = (SmpPassthroughRequest_t *)mf;
2247 memset(smpreq, 0, sizeof(*smpreq));
2249 smpreq->RequestDataLength = cpu_to_le16(blk_rq_bytes(req) - 4);
2250 smpreq->Function = MPI_FUNCTION_SMP_PASSTHROUGH;
2253 sas_address = rphy->identify.sas_address;
2255 struct mptsas_portinfo *port_info;
2257 mutex_lock(&ioc->sas_topology_mutex);
2258 port_info = ioc->hba_port_info;
2259 if (port_info && port_info->phy_info)
2261 port_info->phy_info[0].phy->identify.sas_address;
2262 mutex_unlock(&ioc->sas_topology_mutex);
2265 *((u64 *)&smpreq->SASAddress) = cpu_to_le64(sas_address);
2268 (((int *) mf) + (offsetof(SmpPassthroughRequest_t, SGL) / 4));
2271 flagsLength = (MPI_SGE_FLAGS_SIMPLE_ELEMENT |
2272 MPI_SGE_FLAGS_END_OF_BUFFER |
2273 MPI_SGE_FLAGS_DIRECTION)
2274 << MPI_SGE_FLAGS_SHIFT;
2275 flagsLength |= (blk_rq_bytes(req) - 4);
2277 dma_addr_out = pci_map_single(ioc->pcidev, bio_data(req->bio),
2278 blk_rq_bytes(req), PCI_DMA_BIDIRECTIONAL);
2281 ioc->add_sge(psge, flagsLength, dma_addr_out);
2282 psge += ioc->SGE_size;
2285 flagsLength = MPI_SGE_FLAGS_SIMPLE_ELEMENT |
2286 MPI_SGE_FLAGS_SYSTEM_ADDRESS |
2287 MPI_SGE_FLAGS_IOC_TO_HOST |
2288 MPI_SGE_FLAGS_END_OF_BUFFER;
2290 flagsLength = flagsLength << MPI_SGE_FLAGS_SHIFT;
2291 flagsLength |= blk_rq_bytes(rsp) + 4;
2292 dma_addr_in = pci_map_single(ioc->pcidev, bio_data(rsp->bio),
2293 blk_rq_bytes(rsp), PCI_DMA_BIDIRECTIONAL);
2296 ioc->add_sge(psge, flagsLength, dma_addr_in);
2298 INITIALIZE_MGMT_STATUS(ioc->sas_mgmt.status)
2299 mpt_put_msg_frame(mptsasMgmtCtx, ioc, mf);
2301 timeleft = wait_for_completion_timeout(&ioc->sas_mgmt.done, 10 * HZ);
2302 if (!(ioc->sas_mgmt.status & MPT_MGMT_STATUS_COMMAND_GOOD)) {
2304 mpt_free_msg_frame(ioc, mf);
2306 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_DID_IOCRESET)
2309 mpt_Soft_Hard_ResetHandler(ioc, CAN_SLEEP);
2314 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_RF_VALID) {
2315 SmpPassthroughReply_t *smprep;
2317 smprep = (SmpPassthroughReply_t *)ioc->sas_mgmt.reply;
2318 memcpy(req->sense, smprep, sizeof(*smprep));
2319 req->sense_len = sizeof(*smprep);
2321 rsp->resid_len -= smprep->ResponseDataLength;
2323 printk(MYIOC_s_ERR_FMT
2324 "%s: smp passthru reply failed to be returned\n",
2325 ioc->name, __func__);
2330 pci_unmap_single(ioc->pcidev, dma_addr_out, blk_rq_bytes(req),
2331 PCI_DMA_BIDIRECTIONAL);
2333 pci_unmap_single(ioc->pcidev, dma_addr_in, blk_rq_bytes(rsp),
2334 PCI_DMA_BIDIRECTIONAL);
2337 mpt_free_msg_frame(ioc, mf);
2339 CLEAR_MGMT_STATUS(ioc->sas_mgmt.status)
2340 mutex_unlock(&ioc->sas_mgmt.mutex);
2345 static struct sas_function_template mptsas_transport_functions = {
2346 .get_linkerrors = mptsas_get_linkerrors,
2347 .get_enclosure_identifier = mptsas_get_enclosure_identifier,
2348 .get_bay_identifier = mptsas_get_bay_identifier,
2349 .phy_reset = mptsas_phy_reset,
2350 .smp_handler = mptsas_smp_handler,
2353 static struct scsi_transport_template *mptsas_transport_template;
2356 mptsas_sas_io_unit_pg0(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
2358 ConfigExtendedPageHeader_t hdr;
2360 SasIOUnitPage0_t *buffer;
2361 dma_addr_t dma_handle;
2364 hdr.PageVersion = MPI_SASIOUNITPAGE0_PAGEVERSION;
2365 hdr.ExtPageLength = 0;
2369 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2370 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
2372 cfg.cfghdr.ehdr = &hdr;
2375 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2376 cfg.dir = 0; /* read */
2377 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2379 error = mpt_config(ioc, &cfg);
2382 if (!hdr.ExtPageLength) {
2387 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2394 cfg.physAddr = dma_handle;
2395 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2397 error = mpt_config(ioc, &cfg);
2399 goto out_free_consistent;
2401 port_info->num_phys = buffer->NumPhys;
2402 port_info->phy_info = kcalloc(port_info->num_phys,
2403 sizeof(struct mptsas_phyinfo), GFP_KERNEL);
2404 if (!port_info->phy_info) {
2406 goto out_free_consistent;
2409 ioc->nvdata_version_persistent =
2410 le16_to_cpu(buffer->NvdataVersionPersistent);
2411 ioc->nvdata_version_default =
2412 le16_to_cpu(buffer->NvdataVersionDefault);
2414 for (i = 0; i < port_info->num_phys; i++) {
2415 mptsas_print_phy_data(ioc, &buffer->PhyData[i]);
2416 port_info->phy_info[i].phy_id = i;
2417 port_info->phy_info[i].port_id =
2418 buffer->PhyData[i].Port;
2419 port_info->phy_info[i].negotiated_link_rate =
2420 buffer->PhyData[i].NegotiatedLinkRate;
2421 port_info->phy_info[i].portinfo = port_info;
2422 port_info->phy_info[i].handle =
2423 le16_to_cpu(buffer->PhyData[i].ControllerDevHandle);
2426 out_free_consistent:
2427 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2428 buffer, dma_handle);
2434 mptsas_sas_io_unit_pg1(MPT_ADAPTER *ioc)
2436 ConfigExtendedPageHeader_t hdr;
2438 SasIOUnitPage1_t *buffer;
2439 dma_addr_t dma_handle;
2441 u8 device_missing_delay;
2443 memset(&hdr, 0, sizeof(ConfigExtendedPageHeader_t));
2444 memset(&cfg, 0, sizeof(CONFIGPARMS));
2446 cfg.cfghdr.ehdr = &hdr;
2447 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2448 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2449 cfg.cfghdr.ehdr->PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2450 cfg.cfghdr.ehdr->ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
2451 cfg.cfghdr.ehdr->PageVersion = MPI_SASIOUNITPAGE1_PAGEVERSION;
2452 cfg.cfghdr.ehdr->PageNumber = 1;
2454 error = mpt_config(ioc, &cfg);
2457 if (!hdr.ExtPageLength) {
2462 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2469 cfg.physAddr = dma_handle;
2470 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2472 error = mpt_config(ioc, &cfg);
2474 goto out_free_consistent;
2476 ioc->io_missing_delay =
2477 le16_to_cpu(buffer->IODeviceMissingDelay);
2478 device_missing_delay = buffer->ReportDeviceMissingDelay;
2479 ioc->device_missing_delay = (device_missing_delay & MPI_SAS_IOUNIT1_REPORT_MISSING_UNIT_16) ?
2480 (device_missing_delay & MPI_SAS_IOUNIT1_REPORT_MISSING_TIMEOUT_MASK) * 16 :
2481 device_missing_delay & MPI_SAS_IOUNIT1_REPORT_MISSING_TIMEOUT_MASK;
2483 out_free_consistent:
2484 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2485 buffer, dma_handle);
2491 mptsas_sas_phy_pg0(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info,
2492 u32 form, u32 form_specific)
2494 ConfigExtendedPageHeader_t hdr;
2496 SasPhyPage0_t *buffer;
2497 dma_addr_t dma_handle;
2500 hdr.PageVersion = MPI_SASPHY0_PAGEVERSION;
2501 hdr.ExtPageLength = 0;
2505 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2506 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_PHY;
2508 cfg.cfghdr.ehdr = &hdr;
2509 cfg.dir = 0; /* read */
2510 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2512 /* Get Phy Pg 0 for each Phy. */
2514 cfg.pageAddr = form + form_specific;
2515 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2517 error = mpt_config(ioc, &cfg);
2521 if (!hdr.ExtPageLength) {
2526 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2533 cfg.physAddr = dma_handle;
2534 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2536 error = mpt_config(ioc, &cfg);
2538 goto out_free_consistent;
2540 mptsas_print_phy_pg0(ioc, buffer);
2542 phy_info->hw_link_rate = buffer->HwLinkRate;
2543 phy_info->programmed_link_rate = buffer->ProgrammedLinkRate;
2544 phy_info->identify.handle = le16_to_cpu(buffer->OwnerDevHandle);
2545 phy_info->attached.handle = le16_to_cpu(buffer->AttachedDevHandle);
2547 out_free_consistent:
2548 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2549 buffer, dma_handle);
2555 mptsas_sas_device_pg0(MPT_ADAPTER *ioc, struct mptsas_devinfo *device_info,
2556 u32 form, u32 form_specific)
2558 ConfigExtendedPageHeader_t hdr;
2560 SasDevicePage0_t *buffer;
2561 dma_addr_t dma_handle;
2565 hdr.PageVersion = MPI_SASDEVICE0_PAGEVERSION;
2566 hdr.ExtPageLength = 0;
2570 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2571 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_DEVICE;
2573 cfg.cfghdr.ehdr = &hdr;
2574 cfg.pageAddr = form + form_specific;
2576 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2577 cfg.dir = 0; /* read */
2578 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2580 memset(device_info, 0, sizeof(struct mptsas_devinfo));
2581 error = mpt_config(ioc, &cfg);
2584 if (!hdr.ExtPageLength) {
2589 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2596 cfg.physAddr = dma_handle;
2597 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2599 error = mpt_config(ioc, &cfg);
2601 if (error == MPI_IOCSTATUS_CONFIG_INVALID_PAGE) {
2603 goto out_free_consistent;
2607 goto out_free_consistent;
2609 mptsas_print_device_pg0(ioc, buffer);
2611 memset(device_info, 0, sizeof(struct mptsas_devinfo));
2612 device_info->handle = le16_to_cpu(buffer->DevHandle);
2613 device_info->handle_parent = le16_to_cpu(buffer->ParentDevHandle);
2614 device_info->handle_enclosure =
2615 le16_to_cpu(buffer->EnclosureHandle);
2616 device_info->slot = le16_to_cpu(buffer->Slot);
2617 device_info->phy_id = buffer->PhyNum;
2618 device_info->port_id = buffer->PhysicalPort;
2619 device_info->id = buffer->TargetID;
2620 device_info->phys_disk_num = ~0;
2621 device_info->channel = buffer->Bus;
2622 memcpy(&sas_address, &buffer->SASAddress, sizeof(__le64));
2623 device_info->sas_address = le64_to_cpu(sas_address);
2624 device_info->device_info =
2625 le32_to_cpu(buffer->DeviceInfo);
2626 device_info->flags = le16_to_cpu(buffer->Flags);
2628 out_free_consistent:
2629 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2630 buffer, dma_handle);
2636 mptsas_sas_expander_pg0(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info,
2637 u32 form, u32 form_specific)
2639 ConfigExtendedPageHeader_t hdr;
2641 SasExpanderPage0_t *buffer;
2642 dma_addr_t dma_handle;
2646 memset(port_info, 0, sizeof(struct mptsas_portinfo));
2647 hdr.PageVersion = MPI_SASEXPANDER0_PAGEVERSION;
2648 hdr.ExtPageLength = 0;
2652 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2653 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_EXPANDER;
2655 cfg.cfghdr.ehdr = &hdr;
2657 cfg.pageAddr = form + form_specific;
2658 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2659 cfg.dir = 0; /* read */
2660 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2662 memset(port_info, 0, sizeof(struct mptsas_portinfo));
2663 error = mpt_config(ioc, &cfg);
2667 if (!hdr.ExtPageLength) {
2672 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2679 cfg.physAddr = dma_handle;
2680 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2682 error = mpt_config(ioc, &cfg);
2683 if (error == MPI_IOCSTATUS_CONFIG_INVALID_PAGE) {
2685 goto out_free_consistent;
2689 goto out_free_consistent;
2691 /* save config data */
2692 port_info->num_phys = (buffer->NumPhys) ? buffer->NumPhys : 1;
2693 port_info->phy_info = kcalloc(port_info->num_phys,
2694 sizeof(struct mptsas_phyinfo), GFP_KERNEL);
2695 if (!port_info->phy_info) {
2697 goto out_free_consistent;
2700 memcpy(&sas_address, &buffer->SASAddress, sizeof(__le64));
2701 for (i = 0; i < port_info->num_phys; i++) {
2702 port_info->phy_info[i].portinfo = port_info;
2703 port_info->phy_info[i].handle =
2704 le16_to_cpu(buffer->DevHandle);
2705 port_info->phy_info[i].identify.sas_address =
2706 le64_to_cpu(sas_address);
2707 port_info->phy_info[i].identify.handle_parent =
2708 le16_to_cpu(buffer->ParentDevHandle);
2711 out_free_consistent:
2712 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2713 buffer, dma_handle);
2719 mptsas_sas_expander_pg1(MPT_ADAPTER *ioc, struct mptsas_phyinfo *phy_info,
2720 u32 form, u32 form_specific)
2722 ConfigExtendedPageHeader_t hdr;
2724 SasExpanderPage1_t *buffer;
2725 dma_addr_t dma_handle;
2728 hdr.PageVersion = MPI_SASEXPANDER1_PAGEVERSION;
2729 hdr.ExtPageLength = 0;
2733 hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
2734 hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_EXPANDER;
2736 cfg.cfghdr.ehdr = &hdr;
2738 cfg.pageAddr = form + form_specific;
2739 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
2740 cfg.dir = 0; /* read */
2741 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
2743 error = mpt_config(ioc, &cfg);
2747 if (!hdr.ExtPageLength) {
2752 buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2759 cfg.physAddr = dma_handle;
2760 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
2762 error = mpt_config(ioc, &cfg);
2764 if (error == MPI_IOCSTATUS_CONFIG_INVALID_PAGE) {
2766 goto out_free_consistent;
2770 goto out_free_consistent;
2773 mptsas_print_expander_pg1(ioc, buffer);
2775 /* save config data */
2776 phy_info->phy_id = buffer->PhyIdentifier;
2777 phy_info->port_id = buffer->PhysicalPort;
2778 phy_info->negotiated_link_rate = buffer->NegotiatedLinkRate;
2779 phy_info->programmed_link_rate = buffer->ProgrammedLinkRate;
2780 phy_info->hw_link_rate = buffer->HwLinkRate;
2781 phy_info->identify.handle = le16_to_cpu(buffer->OwnerDevHandle);
2782 phy_info->attached.handle = le16_to_cpu(buffer->AttachedDevHandle);
2784 out_free_consistent:
2785 pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
2786 buffer, dma_handle);
2791 struct rep_manu_request{
2798 struct rep_manu_reply{
2799 u8 smp_frame_type; /* 0x41 */
2800 u8 function; /* 0x01 */
2803 u16 expander_change_count;
2808 u8 vendor_id[SAS_EXPANDER_VENDOR_ID_LEN];
2809 u8 product_id[SAS_EXPANDER_PRODUCT_ID_LEN];
2810 u8 product_rev[SAS_EXPANDER_PRODUCT_REV_LEN];
2811 u8 component_vendor_id[SAS_EXPANDER_COMPONENT_VENDOR_ID_LEN];
2813 u8 component_revision_id;
2815 u8 vendor_specific[8];
2819 * mptsas_exp_repmanufacture_info -
2820 * @ioc: per adapter object
2821 * @sas_address: expander sas address
2822 * @edev: the sas_expander_device object
2824 * Fills in the sas_expander_device object when SMP port is created.
2826 * Returns 0 for success, non-zero for failure.
2829 mptsas_exp_repmanufacture_info(MPT_ADAPTER *ioc,
2830 u64 sas_address, struct sas_expander_device *edev)
2833 SmpPassthroughRequest_t *smpreq;
2834 SmpPassthroughReply_t *smprep;
2835 struct rep_manu_reply *manufacture_reply;
2836 struct rep_manu_request *manufacture_request;
2839 unsigned long timeleft;
2841 unsigned long flags;
2842 void *data_out = NULL;
2843 dma_addr_t data_out_dma = 0;
2846 spin_lock_irqsave(&ioc->taskmgmt_lock, flags);
2847 if (ioc->ioc_reset_in_progress) {
2848 spin_unlock_irqrestore(&ioc->taskmgmt_lock, flags);
2849 printk(MYIOC_s_INFO_FMT "%s: host reset in progress!\n",
2850 __func__, ioc->name);
2853 spin_unlock_irqrestore(&ioc->taskmgmt_lock, flags);
2855 ret = mutex_lock_interruptible(&ioc->sas_mgmt.mutex);
2859 mf = mpt_get_msg_frame(mptsasMgmtCtx, ioc);
2865 smpreq = (SmpPassthroughRequest_t *)mf;
2866 memset(smpreq, 0, sizeof(*smpreq));
2868 sz = sizeof(struct rep_manu_request) + sizeof(struct rep_manu_reply);
2870 data_out = pci_alloc_consistent(ioc->pcidev, sz, &data_out_dma);
2872 printk(KERN_ERR "Memory allocation failure at %s:%d/%s()!\n",
2873 __FILE__, __LINE__, __func__);
2878 manufacture_request = data_out;
2879 manufacture_request->smp_frame_type = 0x40;
2880 manufacture_request->function = 1;
2881 manufacture_request->reserved = 0;
2882 manufacture_request->request_length = 0;
2884 smpreq->Function = MPI_FUNCTION_SMP_PASSTHROUGH;
2885 smpreq->PhysicalPort = 0xFF;
2886 *((u64 *)&smpreq->SASAddress) = cpu_to_le64(sas_address);
2887 smpreq->RequestDataLength = sizeof(struct rep_manu_request);
2890 (((int *) mf) + (offsetof(SmpPassthroughRequest_t, SGL) / 4));
2892 flagsLength = MPI_SGE_FLAGS_SIMPLE_ELEMENT |
2893 MPI_SGE_FLAGS_SYSTEM_ADDRESS |
2894 MPI_SGE_FLAGS_HOST_TO_IOC |
2895 MPI_SGE_FLAGS_END_OF_BUFFER;
2896 flagsLength = flagsLength << MPI_SGE_FLAGS_SHIFT;
2897 flagsLength |= sizeof(struct rep_manu_request);
2899 ioc->add_sge(psge, flagsLength, data_out_dma);
2900 psge += ioc->SGE_size;
2902 flagsLength = MPI_SGE_FLAGS_SIMPLE_ELEMENT |
2903 MPI_SGE_FLAGS_SYSTEM_ADDRESS |
2904 MPI_SGE_FLAGS_IOC_TO_HOST |
2905 MPI_SGE_FLAGS_END_OF_BUFFER;
2906 flagsLength = flagsLength << MPI_SGE_FLAGS_SHIFT;
2907 flagsLength |= sizeof(struct rep_manu_reply);
2908 ioc->add_sge(psge, flagsLength, data_out_dma +
2909 sizeof(struct rep_manu_request));
2911 INITIALIZE_MGMT_STATUS(ioc->sas_mgmt.status)
2912 mpt_put_msg_frame(mptsasMgmtCtx, ioc, mf);
2914 timeleft = wait_for_completion_timeout(&ioc->sas_mgmt.done, 10 * HZ);
2915 if (!(ioc->sas_mgmt.status & MPT_MGMT_STATUS_COMMAND_GOOD)) {
2917 mpt_free_msg_frame(ioc, mf);
2919 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_DID_IOCRESET)
2922 mpt_Soft_Hard_ResetHandler(ioc, CAN_SLEEP);
2928 if (ioc->sas_mgmt.status & MPT_MGMT_STATUS_RF_VALID) {
2931 smprep = (SmpPassthroughReply_t *)ioc->sas_mgmt.reply;
2932 if (le16_to_cpu(smprep->ResponseDataLength) !=
2933 sizeof(struct rep_manu_reply))
2936 manufacture_reply = data_out + sizeof(struct rep_manu_request);
2937 strncpy(edev->vendor_id, manufacture_reply->vendor_id,
2938 SAS_EXPANDER_VENDOR_ID_LEN);
2939 strncpy(edev->product_id, manufacture_reply->product_id,
2940 SAS_EXPANDER_PRODUCT_ID_LEN);
2941 strncpy(edev->product_rev, manufacture_reply->product_rev,
2942 SAS_EXPANDER_PRODUCT_REV_LEN);
2943 edev->level = manufacture_reply->sas_format;
2944 if (manufacture_reply->sas_format) {
2945 strncpy(edev->component_vendor_id,
2946 manufacture_reply->component_vendor_id,
2947 SAS_EXPANDER_COMPONENT_VENDOR_ID_LEN);
2948 tmp = (u8 *)&manufacture_reply->component_id;
2949 edev->component_id = tmp[0] << 8 | tmp[1];
2950 edev->component_revision_id =
2951 manufacture_reply->component_revision_id;
2954 printk(MYIOC_s_ERR_FMT
2955 "%s: smp passthru reply failed to be returned\n",
2956 ioc->name, __func__);
2961 pci_free_consistent(ioc->pcidev, sz, data_out, data_out_dma);
2964 mpt_free_msg_frame(ioc, mf);
2966 CLEAR_MGMT_STATUS(ioc->sas_mgmt.status)
2967 mutex_unlock(&ioc->sas_mgmt.mutex);
2973 mptsas_parse_device_info(struct sas_identify *identify,
2974 struct mptsas_devinfo *device_info)
2978 identify->sas_address = device_info->sas_address;
2979 identify->phy_identifier = device_info->phy_id;
2982 * Fill in Phy Initiator Port Protocol.
2983 * Bits 6:3, more than one bit can be set, fall through cases.
2985 protocols = device_info->device_info & 0x78;
2986 identify->initiator_port_protocols = 0;
2987 if (protocols & MPI_SAS_DEVICE_INFO_SSP_INITIATOR)
2988 identify->initiator_port_protocols |= SAS_PROTOCOL_SSP;
2989 if (protocols & MPI_SAS_DEVICE_INFO_STP_INITIATOR)
2990 identify->initiator_port_protocols |= SAS_PROTOCOL_STP;
2991 if (protocols & MPI_SAS_DEVICE_INFO_SMP_INITIATOR)
2992 identify->initiator_port_protocols |= SAS_PROTOCOL_SMP;
2993 if (protocols & MPI_SAS_DEVICE_INFO_SATA_HOST)
2994 identify->initiator_port_protocols |= SAS_PROTOCOL_SATA;
2997 * Fill in Phy Target Port Protocol.
2998 * Bits 10:7, more than one bit can be set, fall through cases.
3000 protocols = device_info->device_info & 0x780;
3001 identify->target_port_protocols = 0;
3002 if (protocols & MPI_SAS_DEVICE_INFO_SSP_TARGET)
3003 identify->target_port_protocols |= SAS_PROTOCOL_SSP;
3004 if (protocols & MPI_SAS_DEVICE_INFO_STP_TARGET)
3005 identify->target_port_protocols |= SAS_PROTOCOL_STP;
3006 if (protocols & MPI_SAS_DEVICE_INFO_SMP_TARGET)
3007 identify->target_port_protocols |= SAS_PROTOCOL_SMP;
3008 if (protocols & MPI_SAS_DEVICE_INFO_SATA_DEVICE)
3009 identify->target_port_protocols |= SAS_PROTOCOL_SATA;
3012 * Fill in Attached device type.
3014 switch (device_info->device_info &
3015 MPI_SAS_DEVICE_INFO_MASK_DEVICE_TYPE) {
3016 case MPI_SAS_DEVICE_INFO_NO_DEVICE:
3017 identify->device_type = SAS_PHY_UNUSED;
3019 case MPI_SAS_DEVICE_INFO_END_DEVICE:
3020 identify->device_type = SAS_END_DEVICE;
3022 case MPI_SAS_DEVICE_INFO_EDGE_EXPANDER:
3023 identify->device_type = SAS_EDGE_EXPANDER_DEVICE;
3025 case MPI_SAS_DEVICE_INFO_FANOUT_EXPANDER:
3026 identify->device_type = SAS_FANOUT_EXPANDER_DEVICE;
3031 static int mptsas_probe_one_phy(struct device *dev,
3032 struct mptsas_phyinfo *phy_info, int index, int local)
3035 struct sas_phy *phy;
3036 struct sas_port *port;
3038 VirtTarget *vtarget;
3045 if (!phy_info->phy) {
3046 phy = sas_phy_alloc(dev, index);
3052 phy = phy_info->phy;
3054 mptsas_parse_device_info(&phy->identify, &phy_info->identify);
3057 * Set Negotiated link rate.
3059 switch (phy_info->negotiated_link_rate) {
3060 case MPI_SAS_IOUNIT0_RATE_PHY_DISABLED:
3061 phy->negotiated_linkrate = SAS_PHY_DISABLED;
3063 case MPI_SAS_IOUNIT0_RATE_FAILED_SPEED_NEGOTIATION:
3064 phy->negotiated_linkrate = SAS_LINK_RATE_FAILED;
3066 case MPI_SAS_IOUNIT0_RATE_1_5:
3067 phy->negotiated_linkrate = SAS_LINK_RATE_1_5_GBPS;
3069 case MPI_SAS_IOUNIT0_RATE_3_0:
3070 phy->negotiated_linkrate = SAS_LINK_RATE_3_0_GBPS;
3072 case MPI_SAS_IOUNIT0_RATE_6_0:
3073 phy->negotiated_linkrate = SAS_LINK_RATE_6_0_GBPS;
3075 case MPI_SAS_IOUNIT0_RATE_SATA_OOB_COMPLETE:
3076 case MPI_SAS_IOUNIT0_RATE_UNKNOWN:
3078 phy->negotiated_linkrate = SAS_LINK_RATE_UNKNOWN;
3083 * Set Max hardware link rate.
3085 switch (phy_info->hw_link_rate & MPI_SAS_PHY0_PRATE_MAX_RATE_MASK) {
3086 case MPI_SAS_PHY0_HWRATE_MAX_RATE_1_5:
3087 phy->maximum_linkrate_hw = SAS_LINK_RATE_1_5_GBPS;
3089 case MPI_SAS_PHY0_PRATE_MAX_RATE_3_0:
3090 phy->maximum_linkrate_hw = SAS_LINK_RATE_3_0_GBPS;
3097 * Set Max programmed link rate.
3099 switch (phy_info->programmed_link_rate &
3100 MPI_SAS_PHY0_PRATE_MAX_RATE_MASK) {
3101 case MPI_SAS_PHY0_PRATE_MAX_RATE_1_5:
3102 phy->maximum_linkrate = SAS_LINK_RATE_1_5_GBPS;
3104 case MPI_SAS_PHY0_PRATE_MAX_RATE_3_0:
3105 phy->maximum_linkrate = SAS_LINK_RATE_3_0_GBPS;
3112 * Set Min hardware link rate.
3114 switch (phy_info->hw_link_rate & MPI_SAS_PHY0_HWRATE_MIN_RATE_MASK) {
3115 case MPI_SAS_PHY0_HWRATE_MIN_RATE_1_5:
3116 phy->minimum_linkrate_hw = SAS_LINK_RATE_1_5_GBPS;
3118 case MPI_SAS_PHY0_PRATE_MIN_RATE_3_0:
3119 phy->minimum_linkrate_hw = SAS_LINK_RATE_3_0_GBPS;
3126 * Set Min programmed link rate.
3128 switch (phy_info->programmed_link_rate &
3129 MPI_SAS_PHY0_PRATE_MIN_RATE_MASK) {
3130 case MPI_SAS_PHY0_PRATE_MIN_RATE_1_5:
3131 phy->minimum_linkrate = SAS_LINK_RATE_1_5_GBPS;
3133 case MPI_SAS_PHY0_PRATE_MIN_RATE_3_0:
3134 phy->minimum_linkrate = SAS_LINK_RATE_3_0_GBPS;
3140 if (!phy_info->phy) {
3142 error = sas_phy_add(phy);
3147 phy_info->phy = phy;
3150 if (!phy_info->attached.handle ||
3151 !phy_info->port_details)
3154 port = mptsas_get_port(phy_info);
3155 ioc = phy_to_ioc(phy_info->phy);
3157 if (phy_info->sas_port_add_phy) {
3160 port = sas_port_alloc_num(dev);
3165 error = sas_port_add(port);
3167 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
3168 "%s: exit at line=%d\n", ioc->name,
3169 __func__, __LINE__));
3172 mptsas_set_port(ioc, phy_info, port);
3173 devtprintk(ioc, dev_printk(KERN_DEBUG, &port->dev,
3174 MYIOC_s_FMT "add port %d, sas_addr (0x%llx)\n",
3175 ioc->name, port->port_identifier,
3176 (unsigned long long)phy_info->
3177 attached.sas_address));
3179 dsaswideprintk(ioc, printk(MYIOC_s_DEBUG_FMT
3180 "sas_port_add_phy: phy_id=%d\n",
3181 ioc->name, phy_info->phy_id));
3182 sas_port_add_phy(port, phy_info->phy);
3183 phy_info->sas_port_add_phy = 0;
3184 devtprintk(ioc, dev_printk(KERN_DEBUG, &phy_info->phy->dev,
3185 MYIOC_s_FMT "add phy %d, phy-obj (0x%p)\n", ioc->name,
3186 phy_info->phy_id, phy_info->phy));
3188 if (!mptsas_get_rphy(phy_info) && port && !port->rphy) {
3190 struct sas_rphy *rphy;
3191 struct device *parent;
3192 struct sas_identify identify;
3194 parent = dev->parent->parent;
3196 * Let the hotplug_work thread handle processing
3197 * the adding/removing of devices that occur
3198 * after start of day.
3200 if (mptsas_is_end_device(&phy_info->attached) &&
3201 phy_info->attached.handle_parent) {
3205 mptsas_parse_device_info(&identify, &phy_info->attached);
3206 if (scsi_is_host_device(parent)) {
3207 struct mptsas_portinfo *port_info;
3210 port_info = ioc->hba_port_info;
3212 for (i = 0; i < port_info->num_phys; i++)
3213 if (port_info->phy_info[i].identify.sas_address ==
3214 identify.sas_address) {
3215 sas_port_mark_backlink(port);
3219 } else if (scsi_is_sas_rphy(parent)) {
3220 struct sas_rphy *parent_rphy = dev_to_rphy(parent);
3221 if (identify.sas_address ==
3222 parent_rphy->identify.sas_address) {
3223 sas_port_mark_backlink(port);
3228 switch (identify.device_type) {
3229 case SAS_END_DEVICE:
3230 rphy = sas_end_device_alloc(port);
3232 case SAS_EDGE_EXPANDER_DEVICE:
3233 case SAS_FANOUT_EXPANDER_DEVICE:
3234 rphy = sas_expander_alloc(port, identify.device_type);
3241 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
3242 "%s: exit at line=%d\n", ioc->name,
3243 __func__, __LINE__));
3247 rphy->identify = identify;
3248 error = sas_rphy_add(rphy);
3250 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
3251 "%s: exit at line=%d\n", ioc->name,
3252 __func__, __LINE__));
3253 sas_rphy_free(rphy);
3256 mptsas_set_rphy(ioc, phy_info, rphy);
3257 if (identify.device_type == SAS_EDGE_EXPANDER_DEVICE ||
3258 identify.device_type == SAS_FANOUT_EXPANDER_DEVICE)
3259 mptsas_exp_repmanufacture_info(ioc,
3260 identify.sas_address,
3261 rphy_to_expander_device(rphy));
3264 /* If the device exists,verify it wasn't previously flagged
3265 as a missing device. If so, clear it */
3266 vtarget = mptsas_find_vtarget(ioc,
3267 phy_info->attached.channel,
3268 phy_info->attached.id);
3269 if (vtarget && vtarget->inDMD) {
3270 printk(KERN_INFO "Device returned, unsetting inDMD\n");
3279 mptsas_probe_hba_phys(MPT_ADAPTER *ioc)
3281 struct mptsas_portinfo *port_info, *hba;
3282 int error = -ENOMEM, i;
3284 hba = kzalloc(sizeof(struct mptsas_portinfo), GFP_KERNEL);
3288 error = mptsas_sas_io_unit_pg0(ioc, hba);
3290 goto out_free_port_info;
3292 mptsas_sas_io_unit_pg1(ioc);
3293 mutex_lock(&ioc->sas_topology_mutex);
3294 port_info = ioc->hba_port_info;
3296 ioc->hba_port_info = port_info = hba;
3297 ioc->hba_port_num_phy = port_info->num_phys;
3298 list_add_tail(&port_info->list, &ioc->sas_topology);
3300 for (i = 0; i < hba->num_phys; i++) {
3301 port_info->phy_info[i].negotiated_link_rate =
3302 hba->phy_info[i].negotiated_link_rate;
3303 port_info->phy_info[i].handle =
3304 hba->phy_info[i].handle;
3305 port_info->phy_info[i].port_id =
3306 hba->phy_info[i].port_id;
3308 kfree(hba->phy_info);
3312 mutex_unlock(&ioc->sas_topology_mutex);
3313 #if defined(CPQ_CIM)
3314 ioc->num_ports = port_info->num_phys;
3316 for (i = 0; i < port_info->num_phys; i++) {
3317 mptsas_sas_phy_pg0(ioc, &port_info->phy_info[i],
3318 (MPI_SAS_PHY_PGAD_FORM_PHY_NUMBER <<
3319 MPI_SAS_PHY_PGAD_FORM_SHIFT), i);
3320 port_info->phy_info[i].identify.handle =
3321 port_info->phy_info[i].handle;
3322 mptsas_sas_device_pg0(ioc, &port_info->phy_info[i].identify,
3323 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
3324 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
3325 port_info->phy_info[i].identify.handle);
3326 if (!ioc->hba_port_sas_addr)
3327 ioc->hba_port_sas_addr =
3328 port_info->phy_info[i].identify.sas_address;
3329 port_info->phy_info[i].identify.phy_id =
3330 port_info->phy_info[i].phy_id = i;
3331 if (port_info->phy_info[i].attached.handle)
3332 mptsas_sas_device_pg0(ioc,
3333 &port_info->phy_info[i].attached,
3334 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
3335 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
3336 port_info->phy_info[i].attached.handle);
3339 mptsas_setup_wide_ports(ioc, port_info);
3341 for (i = 0; i < port_info->num_phys; i++, ioc->sas_index++)
3342 mptsas_probe_one_phy(&ioc->sh->shost_gendev,
3343 &port_info->phy_info[i], ioc->sas_index, 1);
3354 mptsas_expander_refresh(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
3356 struct mptsas_portinfo *parent;
3357 struct device *parent_dev;
3358 struct sas_rphy *rphy;
3360 u64 sas_address; /* expander sas address */
3363 handle = port_info->phy_info[0].handle;
3364 sas_address = port_info->phy_info[0].identify.sas_address;
3365 for (i = 0; i < port_info->num_phys; i++) {
3366 mptsas_sas_expander_pg1(ioc, &port_info->phy_info[i],
3367 (MPI_SAS_EXPAND_PGAD_FORM_HANDLE_PHY_NUM <<
3368 MPI_SAS_EXPAND_PGAD_FORM_SHIFT), (i << 16) + handle);
3370 mptsas_sas_device_pg0(ioc,
3371 &port_info->phy_info[i].identify,
3372 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
3373 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
3374 port_info->phy_info[i].identify.handle);
3375 port_info->phy_info[i].identify.phy_id =
3376 port_info->phy_info[i].phy_id;
3378 if (port_info->phy_info[i].attached.handle) {
3379 mptsas_sas_device_pg0(ioc,
3380 &port_info->phy_info[i].attached,
3381 (MPI_SAS_DEVICE_PGAD_FORM_HANDLE <<
3382 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
3383 port_info->phy_info[i].attached.handle);
3384 port_info->phy_info[i].attached.phy_id =
3385 port_info->phy_info[i].phy_id;
3389 mutex_lock(&ioc->sas_topology_mutex);
3390 parent = mptsas_find_portinfo_by_handle(ioc,
3391 port_info->phy_info[0].identify.handle_parent);
3393 mutex_unlock(&ioc->sas_topology_mutex);
3396 for (i = 0, parent_dev = NULL; i < parent->num_phys && !parent_dev;
3398 if (parent->phy_info[i].attached.sas_address == sas_address) {
3399 rphy = mptsas_get_rphy(&parent->phy_info[i]);
3400 parent_dev = &rphy->dev;
3403 mutex_unlock(&ioc->sas_topology_mutex);
3405 mptsas_setup_wide_ports(ioc, port_info);
3406 for (i = 0; i < port_info->num_phys; i++, ioc->sas_index++)
3407 mptsas_probe_one_phy(parent_dev, &port_info->phy_info[i],
3412 mptsas_expander_event_add(MPT_ADAPTER *ioc,
3413 MpiEventDataSasExpanderStatusChange_t *expander_data)
3415 struct mptsas_portinfo *port_info;
3419 port_info = kzalloc(sizeof(struct mptsas_portinfo), GFP_KERNEL);
3422 port_info->num_phys = (expander_data->NumPhys) ?
3423 expander_data->NumPhys : 1;
3424 port_info->phy_info = kcalloc(port_info->num_phys,
3425 sizeof(struct mptsas_phyinfo), GFP_KERNEL);
3426 if (!port_info->phy_info)
3428 memcpy(&sas_address, &expander_data->SASAddress, sizeof(__le64));
3429 for (i = 0; i < port_info->num_phys; i++) {
3430 port_info->phy_info[i].portinfo = port_info;
3431 port_info->phy_info[i].handle =
3432 le16_to_cpu(expander_data->DevHandle);
3433 port_info->phy_info[i].identify.sas_address =
3434 le64_to_cpu(sas_address);
3435 port_info->phy_info[i].identify.handle_parent =
3436 le16_to_cpu(expander_data->ParentDevHandle);
3439 mutex_lock(&ioc->sas_topology_mutex);
3440 list_add_tail(&port_info->list, &ioc->sas_topology);
3441 mutex_unlock(&ioc->sas_topology_mutex);
3443 printk(MYIOC_s_INFO_FMT "add expander: num_phys %d, "
3444 "sas_addr (0x%llx)\n", ioc->name, port_info->num_phys,
3445 (unsigned long long)sas_address);
3447 mptsas_expander_refresh(ioc, port_info);
3451 * mptsas_delete_expander_siblings - remove siblings attached to expander
3452 * @ioc: Pointer to MPT_ADAPTER structure
3453 * @parent: the parent port_info object
3454 * @expander: the expander port_info object
3457 mptsas_delete_expander_siblings(MPT_ADAPTER *ioc, struct mptsas_portinfo
3458 *parent, struct mptsas_portinfo *expander)
3460 struct mptsas_phyinfo *phy_info;
3461 struct mptsas_portinfo *port_info;
3462 struct sas_rphy *rphy;
3465 phy_info = expander->phy_info;
3466 for (i = 0; i < expander->num_phys; i++, phy_info++) {
3467 rphy = mptsas_get_rphy(phy_info);
3470 if (rphy->identify.device_type == SAS_END_DEVICE)
3471 mptsas_del_end_device(ioc, phy_info);
3474 phy_info = expander->phy_info;
3475 for (i = 0; i < expander->num_phys; i++, phy_info++) {
3476 rphy = mptsas_get_rphy(phy_info);
3479 if (rphy->identify.device_type ==
3480 MPI_SAS_DEVICE_INFO_EDGE_EXPANDER ||
3481 rphy->identify.device_type ==
3482 MPI_SAS_DEVICE_INFO_FANOUT_EXPANDER) {
3483 port_info = mptsas_find_portinfo_by_sas_address(ioc,
3484 rphy->identify.sas_address);
3487 if (port_info == parent) /* backlink rphy */
3490 Delete this expander even if the expdevpage is exists
3491 because the parent expander is already deleted
3493 mptsas_expander_delete(ioc, port_info, 1);
3500 * mptsas_expander_delete - remove this expander
3501 * @ioc: Pointer to MPT_ADAPTER structure
3502 * @port_info: expander port_info struct
3503 * @force: Flag to forcefully delete the expander
3507 static void mptsas_expander_delete(MPT_ADAPTER *ioc,
3508 struct mptsas_portinfo *port_info, u8 force)
3511 struct mptsas_portinfo *parent;
3513 u64 expander_sas_address;
3514 struct mptsas_phyinfo *phy_info;
3515 struct mptsas_portinfo buffer;
3516 struct mptsas_portinfo_details *port_details;
3517 struct sas_port *port;
3522 /* see if expander is still there before deleting */
3523 mptsas_sas_expander_pg0(ioc, &buffer,
3524 (MPI_SAS_EXPAND_PGAD_FORM_HANDLE <<
3525 MPI_SAS_EXPAND_PGAD_FORM_SHIFT),
3526 port_info->phy_info[0].identify.handle);
3528 if (buffer.num_phys) {
3529 kfree(buffer.phy_info);
3536 * Obtain the port_info instance to the parent port
3538 port_details = NULL;
3539 expander_sas_address =
3540 port_info->phy_info[0].identify.sas_address;
3541 parent = mptsas_find_portinfo_by_handle(ioc,
3542 port_info->phy_info[0].identify.handle_parent);
3543 mptsas_delete_expander_siblings(ioc, parent, port_info);
3548 * Delete rphys in the parent that point
3551 phy_info = parent->phy_info;
3553 for (i = 0; i < parent->num_phys; i++, phy_info++) {
3556 if (phy_info->attached.sas_address !=
3557 expander_sas_address)
3560 port = mptsas_get_port(phy_info);
3561 port_details = phy_info->port_details;
3563 dev_printk(KERN_DEBUG, &phy_info->phy->dev,
3564 MYIOC_s_FMT "delete phy %d, phy-obj (0x%p)\n", ioc->name,
3565 phy_info->phy_id, phy_info->phy);
3566 sas_port_delete_phy(port, phy_info->phy);
3569 dev_printk(KERN_DEBUG, &port->dev,
3570 MYIOC_s_FMT "delete port %d, sas_addr (0x%llx)\n",
3571 ioc->name, port->port_identifier,
3572 (unsigned long long)expander_sas_address);
3573 sas_port_delete(port);
3574 mptsas_port_delete(ioc, port_details);
3578 printk(MYIOC_s_INFO_FMT "delete expander: num_phys %d, "
3579 "sas_addr (0x%llx)\n", ioc->name, port_info->num_phys,
3580 (unsigned long long)expander_sas_address);
3585 list_del(&port_info->list);
3586 kfree(port_info->phy_info);
3592 * mptsas_send_expander_event - expanders events
3593 * @ioc: Pointer to MPT_ADAPTER structure
3594 * @expander_data: event data
3597 * This function handles adding, removing, and refreshing
3598 * device handles within the expander objects.
3601 mptsas_send_expander_event(struct fw_event_work *fw_event)
3604 MpiEventDataSasExpanderStatusChange_t *expander_data;
3605 struct mptsas_portinfo *port_info;
3609 ioc = fw_event->ioc;
3610 expander_data = (MpiEventDataSasExpanderStatusChange_t *)
3611 fw_event->event_data;
3612 memcpy(&sas_address, &expander_data->SASAddress, sizeof(__le64));
3613 sas_address = le64_to_cpu(sas_address);
3614 port_info = mptsas_find_portinfo_by_sas_address(ioc, sas_address);
3616 if (expander_data->ReasonCode == MPI_EVENT_SAS_EXP_RC_ADDED) {
3618 for (i = 0; i < port_info->num_phys; i++) {
3619 port_info->phy_info[i].portinfo = port_info;
3620 port_info->phy_info[i].handle =
3621 le16_to_cpu(expander_data->DevHandle);
3622 port_info->phy_info[i].identify.sas_address =
3623 le64_to_cpu(sas_address);
3624 port_info->phy_info[i].identify.handle_parent =
3625 le16_to_cpu(expander_data->ParentDevHandle);
3627 mptsas_expander_refresh(ioc, port_info);
3628 } else if (!port_info && expander_data->NumPhys)
3629 mptsas_expander_event_add(ioc, expander_data);
3630 } else if (expander_data->ReasonCode ==
3631 MPI_EVENT_SAS_EXP_RC_NOT_RESPONDING)
3632 mptsas_expander_delete(ioc, port_info, 0);
3634 mptsas_free_fw_event(ioc, fw_event);
3639 * mptsas_expander_add -
3640 * @ioc: Pointer to MPT_ADAPTER structure
3644 struct mptsas_portinfo *
3645 mptsas_expander_add(MPT_ADAPTER *ioc, u16 handle)
3647 struct mptsas_portinfo buffer, *port_info;
3650 if ((mptsas_sas_expander_pg0(ioc, &buffer,
3651 (MPI_SAS_EXPAND_PGAD_FORM_HANDLE <<
3652 MPI_SAS_EXPAND_PGAD_FORM_SHIFT), handle)))
3655 port_info = kzalloc(sizeof(struct mptsas_portinfo), GFP_ATOMIC);
3657 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
3658 "%s: exit at line=%d\n", ioc->name,
3659 __func__, __LINE__));
3662 port_info->num_phys = buffer.num_phys;
3663 port_info->phy_info = buffer.phy_info;
3664 for (i = 0; i < port_info->num_phys; i++)
3665 port_info->phy_info[i].portinfo = port_info;
3666 mutex_lock(&ioc->sas_topology_mutex);
3667 list_add_tail(&port_info->list, &ioc->sas_topology);
3668 mutex_unlock(&ioc->sas_topology_mutex);
3669 printk(MYIOC_s_INFO_FMT "add expander: num_phys %d, "
3670 "sas_addr (0x%llx)\n", ioc->name, port_info->num_phys,
3671 (unsigned long long)buffer.phy_info[0].identify.sas_address);
3672 mptsas_expander_refresh(ioc, port_info);
3677 mptsas_send_link_status_event(struct fw_event_work *fw_event)
3680 MpiEventDataSasPhyLinkStatus_t *link_data;
3681 struct mptsas_portinfo *port_info;
3682 struct mptsas_phyinfo *phy_info = NULL;
3687 ioc = fw_event->ioc;
3688 link_data = (MpiEventDataSasPhyLinkStatus_t *)fw_event->event_data;
3690 memcpy(&sas_address, &link_data->SASAddress, sizeof(__le64));
3691 sas_address = le64_to_cpu(sas_address);
3692 link_rate = link_data->LinkRates >> 4;
3693 phy_num = link_data->PhyNum;
3695 port_info = mptsas_find_portinfo_by_sas_address(ioc, sas_address);
3697 phy_info = &port_info->phy_info[phy_num];
3699 phy_info->negotiated_link_rate = link_rate;
3702 if (link_rate == MPI_SAS_IOUNIT0_RATE_1_5 ||
3703 link_rate == MPI_SAS_IOUNIT0_RATE_3_0 ||
3704 link_rate == MPI_SAS_IOUNIT0_RATE_6_0) {
3707 if (ioc->old_sas_discovery_protocal) {
3708 port_info = mptsas_expander_add(ioc,
3709 le16_to_cpu(link_data->DevHandle));
3716 if (port_info == ioc->hba_port_info)
3717 mptsas_probe_hba_phys(ioc);
3719 mptsas_expander_refresh(ioc, port_info);
3720 } else if (phy_info && phy_info->phy) {
3721 if (link_rate == MPI_SAS_IOUNIT0_RATE_PHY_DISABLED)
3722 phy_info->phy->negotiated_linkrate =
3724 else if (link_rate ==
3725 MPI_SAS_IOUNIT0_RATE_FAILED_SPEED_NEGOTIATION)
3726 phy_info->phy->negotiated_linkrate =
3727 SAS_LINK_RATE_FAILED;
3729 phy_info->phy->negotiated_linkrate =
3730 SAS_LINK_RATE_UNKNOWN;
3731 if (ioc->device_missing_delay &&
3732 mptsas_is_end_device(&phy_info->attached)) {
3733 struct scsi_device *sdev;
3734 VirtDevice *vdevice;
3736 id = phy_info->attached.id;
3737 channel = phy_info->attached.channel;
3738 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
3739 "Link down for fw_id %d:fw_channel %d\n",
3740 ioc->name, phy_info->attached.id,
3741 phy_info->attached.channel));
3743 shost_for_each_device(sdev, ioc->sh) {
3744 vdevice = sdev->hostdata;
3745 if ((vdevice == NULL) ||
3746 (vdevice->vtarget == NULL))
3748 if ((vdevice->vtarget->tflags &
3749 MPT_TARGET_FLAGS_RAID_COMPONENT ||
3750 vdevice->vtarget->raidVolume))
3752 if (vdevice->vtarget->id == id &&
3753 vdevice->vtarget->channel ==
3756 printk(MYIOC_s_DEBUG_FMT
3757 "SDEV OUTSTANDING CMDS"
3759 sdev->device_busy));
3766 mptsas_free_fw_event(ioc, fw_event);
3770 mptsas_not_responding_devices(MPT_ADAPTER *ioc)
3772 struct mptsas_portinfo buffer, *port_info;
3773 struct mptsas_device_info *sas_info;
3774 struct mptsas_devinfo sas_device;
3776 VirtTarget *vtarget = NULL;
3777 struct mptsas_phyinfo *phy_info;
3779 int retval, retry_count;
3780 unsigned long flags;
3782 mpt_findImVolumes(ioc);
3784 spin_lock_irqsave(&ioc->taskmgmt_lock, flags);
3785 if (ioc->ioc_reset_in_progress) {
3786 dfailprintk(ioc, printk(MYIOC_s_DEBUG_FMT
3787 "%s: exiting due to a parallel reset \n", ioc->name,
3789 spin_unlock_irqrestore(&ioc->taskmgmt_lock, flags);
3792 spin_unlock_irqrestore(&ioc->taskmgmt_lock, flags);
3794 /* devices, logical volumes */
3795 mutex_lock(&ioc->sas_device_info_mutex);
3797 list_for_each_entry(sas_info, &ioc->sas_device_info_list, list) {
3798 if (sas_info->is_cached)
3800 if (!sas_info->is_logical_volume) {
3801 sas_device.handle = 0;
3804 retval = mptsas_sas_device_pg0(ioc, &sas_device,
3805 (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID
3806 << MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
3807 (sas_info->fw.channel << 8) +
3810 if (sas_device.handle)
3812 if (retval == -EBUSY) {
3813 spin_lock_irqsave(&ioc->taskmgmt_lock, flags);
3814 if (ioc->ioc_reset_in_progress) {
3816 printk(MYIOC_s_DEBUG_FMT
3817 "%s: exiting due to reset\n",
3818 ioc->name, __func__));
3819 spin_unlock_irqrestore
3820 (&ioc->taskmgmt_lock, flags);
3822 sas_device_info_mutex);
3825 spin_unlock_irqrestore(&ioc->taskmgmt_lock,
3829 if (retval && (retval != -ENODEV)) {
3830 if (retry_count < 10) {
3834 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
3835 "%s: Config page retry exceeded retry "
3836 "count deleting device 0x%llx\n",
3837 ioc->name, __func__,
3838 sas_info->sas_address));
3843 vtarget = mptsas_find_vtarget(ioc,
3844 sas_info->fw.channel, sas_info->fw.id);
3847 vtarget->deleted = 1;
3849 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
3850 sas_info->sas_address);
3853 mptsas_del_end_device(ioc, phy_info);
3854 goto redo_device_scan;
3857 mptsas_volume_delete(ioc, sas_info->fw.id);
3859 mutex_unlock(&ioc->sas_device_info_mutex);
3862 mutex_lock(&ioc->sas_topology_mutex);
3864 list_for_each_entry(port_info, &ioc->sas_topology, list) {
3866 if (port_info->phy_info &&
3867 (!(port_info->phy_info[0].identify.device_info &
3868 MPI_SAS_DEVICE_INFO_SMP_TARGET)))
3872 while (!mptsas_sas_expander_pg0(ioc, &buffer,
3873 (MPI_SAS_EXPAND_PGAD_FORM_GET_NEXT_HANDLE <<
3874 MPI_SAS_EXPAND_PGAD_FORM_SHIFT), handle) &&
3877 handle = buffer.phy_info[0].handle;
3878 if (buffer.phy_info[0].identify.sas_address ==
3879 port_info->phy_info[0].identify.sas_address) {
3882 kfree(buffer.phy_info);
3885 if (!found_expander) {
3886 mptsas_expander_delete(ioc, port_info, 0);
3887 goto redo_expander_scan;
3890 mutex_unlock(&ioc->sas_topology_mutex);
3894 * mptsas_probe_expanders - adding expanders
3895 * @ioc: Pointer to MPT_ADAPTER structure
3899 mptsas_probe_expanders(MPT_ADAPTER *ioc)
3901 struct mptsas_portinfo buffer, *port_info;
3906 while (!mptsas_sas_expander_pg0(ioc, &buffer,
3907 (MPI_SAS_EXPAND_PGAD_FORM_GET_NEXT_HANDLE <<
3908 MPI_SAS_EXPAND_PGAD_FORM_SHIFT), handle)) {
3910 handle = buffer.phy_info[0].handle;
3911 port_info = mptsas_find_portinfo_by_sas_address(ioc,
3912 buffer.phy_info[0].identify.sas_address);
3915 /* refreshing handles */
3916 for (i = 0; i < buffer.num_phys; i++) {
3917 port_info->phy_info[i].handle = handle;
3918 port_info->phy_info[i].identify.handle_parent =
3919 buffer.phy_info[0].identify.handle_parent;
3921 mptsas_expander_refresh(ioc, port_info);
3922 kfree(buffer.phy_info);
3926 port_info = kzalloc(sizeof(struct mptsas_portinfo), GFP_KERNEL);
3928 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
3929 "%s: exit at line=%d\n", ioc->name,
3930 __func__, __LINE__));
3933 port_info->num_phys = buffer.num_phys;
3934 port_info->phy_info = buffer.phy_info;
3935 for (i = 0; i < port_info->num_phys; i++)
3936 port_info->phy_info[i].portinfo = port_info;
3937 mutex_lock(&ioc->sas_topology_mutex);
3938 list_add_tail(&port_info->list, &ioc->sas_topology);
3939 mutex_unlock(&ioc->sas_topology_mutex);
3940 printk(MYIOC_s_INFO_FMT "add expander: num_phys %d, "
3941 "sas_addr (0x%llx)\n", ioc->name, port_info->num_phys,
3942 (unsigned long long)buffer.phy_info[0].identify.sas_address);
3943 mptsas_expander_refresh(ioc, port_info);
3948 mptsas_probe_devices(MPT_ADAPTER *ioc)
3951 struct mptsas_devinfo sas_device;
3952 struct mptsas_phyinfo *phy_info;
3955 while (!(mptsas_sas_device_pg0(ioc, &sas_device,
3956 MPI_SAS_DEVICE_PGAD_FORM_GET_NEXT_HANDLE, handle))) {
3958 handle = sas_device.handle;
3960 if ((sas_device.device_info &
3961 (MPI_SAS_DEVICE_INFO_SSP_TARGET |
3962 MPI_SAS_DEVICE_INFO_STP_TARGET |
3963 MPI_SAS_DEVICE_INFO_SATA_DEVICE)) == 0)
3966 /* If there is no FW B_T mapping for this device then continue
3968 if (!(sas_device.flags & MPI_SAS_DEVICE0_FLAGS_DEVICE_PRESENT)
3969 || !(sas_device.flags &
3970 MPI_SAS_DEVICE0_FLAGS_DEVICE_MAPPED))
3973 phy_info = mptsas_refreshing_device_handles(ioc, &sas_device);
3977 if (mptsas_get_rphy(phy_info))
3980 mptsas_add_end_device(ioc, phy_info);
3985 * mptsas_scan_sas_topology -
3986 * @ioc: Pointer to MPT_ADAPTER structure
3991 mptsas_scan_sas_topology(MPT_ADAPTER *ioc)
3993 struct scsi_device *sdev;
3996 mptsas_probe_hba_phys(ioc);
3997 mptsas_probe_expanders(ioc);
3998 mptsas_probe_devices(ioc);
4001 Reporting RAID volumes.
4003 if (!ioc->ir_firmware || !ioc->raid_data.pIocPg2 ||
4004 !ioc->raid_data.pIocPg2->NumActiveVolumes)
4006 for (i = 0; i < ioc->raid_data.pIocPg2->NumActiveVolumes; i++) {
4007 sdev = scsi_device_lookup(ioc->sh, MPTSAS_RAID_CHANNEL,
4008 ioc->raid_data.pIocPg2->RaidVolume[i].VolumeID, 0);
4010 scsi_device_put(sdev);
4013 printk(MYIOC_s_INFO_FMT "attaching raid volume, channel %d, "
4014 "id %d\n", ioc->name, MPTSAS_RAID_CHANNEL,
4015 ioc->raid_data.pIocPg2->RaidVolume[i].VolumeID);
4016 scsi_add_device(ioc->sh, MPTSAS_RAID_CHANNEL,
4017 ioc->raid_data.pIocPg2->RaidVolume[i].VolumeID, 0);
4023 mptsas_handle_queue_full_event(struct fw_event_work *fw_event)
4026 EventDataQueueFull_t *qfull_data;
4027 struct mptsas_device_info *sas_info;
4028 struct scsi_device *sdev;
4032 int fw_id, fw_channel;
4036 ioc = fw_event->ioc;
4037 qfull_data = (EventDataQueueFull_t *)fw_event->event_data;
4038 fw_id = qfull_data->TargetID;
4039 fw_channel = qfull_data->Bus;
4040 current_depth = le16_to_cpu(qfull_data->CurrentDepth);
4042 /* if hidden raid component, look for the volume id */
4043 mutex_lock(&ioc->sas_device_info_mutex);
4044 if (mptscsih_is_phys_disk(ioc, fw_channel, fw_id)) {
4045 list_for_each_entry(sas_info, &ioc->sas_device_info_list,
4047 if (sas_info->is_cached ||
4048 sas_info->is_logical_volume)
4050 if (sas_info->is_hidden_raid_component &&
4051 (sas_info->fw.channel == fw_channel &&
4052 sas_info->fw.id == fw_id)) {
4053 id = sas_info->volume_id;
4054 channel = MPTSAS_RAID_CHANNEL;
4059 list_for_each_entry(sas_info, &ioc->sas_device_info_list,
4061 if (sas_info->is_cached ||
4062 sas_info->is_hidden_raid_component ||
4063 sas_info->is_logical_volume)
4065 if (sas_info->fw.channel == fw_channel &&
4066 sas_info->fw.id == fw_id) {
4067 id = sas_info->os.id;
4068 channel = sas_info->os.channel;
4076 mutex_unlock(&ioc->sas_device_info_mutex);
4079 shost_for_each_device(sdev, ioc->sh) {
4080 if (sdev->id == id && sdev->channel == channel) {
4081 if (current_depth > sdev->queue_depth) {
4082 sdev_printk(KERN_INFO, sdev,
4083 "strange observation, the queue "
4084 "depth is (%d) meanwhile fw queue "
4085 "depth (%d)\n", sdev->queue_depth,
4089 depth = scsi_track_queue_full(sdev,
4092 sdev_printk(KERN_INFO, sdev,
4093 "Queue depth reduced to (%d)\n",
4096 sdev_printk(KERN_INFO, sdev,
4097 "Tagged Command Queueing is being "
4099 else if (depth == 0)
4100 sdev_printk(KERN_INFO, sdev,
4101 "Queue depth not changed yet\n");
4106 mptsas_free_fw_event(ioc, fw_event);
4110 static struct mptsas_phyinfo *
4111 mptsas_find_phyinfo_by_sas_address(MPT_ADAPTER *ioc, u64 sas_address)
4113 struct mptsas_portinfo *port_info;
4114 struct mptsas_phyinfo *phy_info = NULL;
4117 mutex_lock(&ioc->sas_topology_mutex);
4118 list_for_each_entry(port_info, &ioc->sas_topology, list) {
4119 for (i = 0; i < port_info->num_phys; i++) {
4120 if (!mptsas_is_end_device(
4121 &port_info->phy_info[i].attached))
4123 if (port_info->phy_info[i].attached.sas_address
4126 phy_info = &port_info->phy_info[i];
4130 mutex_unlock(&ioc->sas_topology_mutex);
4135 * mptsas_find_phyinfo_by_phys_disk_num -
4136 * @ioc: Pointer to MPT_ADAPTER structure
4142 static struct mptsas_phyinfo *
4143 mptsas_find_phyinfo_by_phys_disk_num(MPT_ADAPTER *ioc, u8 phys_disk_num,
4146 struct mptsas_phyinfo *phy_info = NULL;
4147 struct mptsas_portinfo *port_info;
4148 RaidPhysDiskPage1_t *phys_disk = NULL;
4150 u64 sas_address = 0;
4154 if (!ioc->raid_data.pIocPg3)
4156 /* dual port support */
4157 num_paths = mpt_raid_phys_disk_get_num_paths(ioc, phys_disk_num);
4160 phys_disk = kzalloc(offsetof(RaidPhysDiskPage1_t, Path) +
4161 (num_paths * sizeof(RAID_PHYS_DISK1_PATH)), GFP_KERNEL);
4164 mpt_raid_phys_disk_pg1(ioc, phys_disk_num, phys_disk);
4165 for (i = 0; i < num_paths; i++) {
4166 if ((phys_disk->Path[i].Flags & 1) != 0)
4167 /* entry no longer valid */
4169 if ((id == phys_disk->Path[i].PhysDiskID) &&
4170 (channel == phys_disk->Path[i].PhysDiskBus)) {
4171 memcpy(&sas_address, &phys_disk->Path[i].WWID,
4173 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
4185 * Extra code to handle RAID0 case, where the sas_address is not updated
4186 * in phys_disk_page_1 when hotswapped
4188 mutex_lock(&ioc->sas_topology_mutex);
4189 list_for_each_entry(port_info, &ioc->sas_topology, list) {
4190 for (i = 0; i < port_info->num_phys && !phy_info; i++) {
4191 if (!mptsas_is_end_device(
4192 &port_info->phy_info[i].attached))
4194 if (port_info->phy_info[i].attached.phys_disk_num == ~0)
4196 if ((port_info->phy_info[i].attached.phys_disk_num ==
4198 (port_info->phy_info[i].attached.id == id) &&
4199 (port_info->phy_info[i].attached.channel ==
4201 phy_info = &port_info->phy_info[i];
4204 mutex_unlock(&ioc->sas_topology_mutex);
4209 mptsas_reprobe_lun(struct scsi_device *sdev, void *data)
4213 sdev->no_uld_attach = data ? 1 : 0;
4214 rc = scsi_device_reprobe(sdev);
4218 mptsas_reprobe_target(struct scsi_target *starget, int uld_attach)
4220 starget_for_each_device(starget, uld_attach ? (void *)1 : NULL,
4221 mptsas_reprobe_lun);
4225 mptsas_adding_inactive_raid_components(MPT_ADAPTER *ioc, u8 channel, u8 id)
4228 ConfigPageHeader_t hdr;
4229 dma_addr_t dma_handle;
4230 pRaidVolumePage0_t buffer = NULL;
4231 RaidPhysDiskPage0_t phys_disk;
4233 struct mptsas_phyinfo *phy_info;
4234 struct mptsas_devinfo sas_device;
4236 memset(&cfg, 0 , sizeof(CONFIGPARMS));
4237 memset(&hdr, 0 , sizeof(ConfigPageHeader_t));
4238 hdr.PageType = MPI_CONFIG_PAGETYPE_RAID_VOLUME;
4239 cfg.pageAddr = (channel << 8) + id;
4240 cfg.cfghdr.hdr = &hdr;
4241 cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
4242 cfg.timeout = SAS_CONFIG_PAGE_TIMEOUT;
4244 if (mpt_config(ioc, &cfg) != 0)
4247 if (!hdr.PageLength)
4250 buffer = pci_alloc_consistent(ioc->pcidev, hdr.PageLength * 4,
4256 cfg.physAddr = dma_handle;
4257 cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
4259 if (mpt_config(ioc, &cfg) != 0)
4262 if (!(buffer->VolumeStatus.Flags &
4263 MPI_RAIDVOL0_STATUS_FLAG_VOLUME_INACTIVE))
4266 if (!buffer->NumPhysDisks)
4269 for (i = 0; i < buffer->NumPhysDisks; i++) {
4271 if (mpt_raid_phys_disk_pg0(ioc,
4272 buffer->PhysDisk[i].PhysDiskNum, &phys_disk) != 0)
4275 if (mptsas_sas_device_pg0(ioc, &sas_device,
4276 (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
4277 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
4278 (phys_disk.PhysDiskBus << 8) +
4279 phys_disk.PhysDiskID))
4282 /* If there is no FW B_T mapping for this device then continue
4284 if (!(sas_device.flags & MPI_SAS_DEVICE0_FLAGS_DEVICE_PRESENT)
4285 || !(sas_device.flags &
4286 MPI_SAS_DEVICE0_FLAGS_DEVICE_MAPPED))
4290 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
4291 sas_device.sas_address);
4292 mptsas_add_end_device(ioc, phy_info);
4297 pci_free_consistent(ioc->pcidev, hdr.PageLength * 4, buffer,
4301 * Work queue thread to handle SAS hotplug events
4304 mptsas_hotplug_work(MPT_ADAPTER *ioc, struct fw_event_work *fw_event,
4305 struct mptsas_hotplug_event *hot_plug_info)
4307 struct mptsas_phyinfo *phy_info;
4308 struct scsi_target * starget;
4309 struct mptsas_devinfo sas_device;
4310 VirtTarget *vtarget;
4312 struct mptsas_portinfo *port_info;
4314 switch (hot_plug_info->event_type) {
4316 case MPTSAS_ADD_PHYSDISK:
4318 if (!ioc->raid_data.pIocPg2)
4321 for (i = 0; i < ioc->raid_data.pIocPg2->NumActiveVolumes; i++) {
4322 if (ioc->raid_data.pIocPg2->RaidVolume[i].VolumeID ==
4323 hot_plug_info->id) {
4324 printk(MYIOC_s_WARN_FMT "firmware bug: unable "
4325 "to add hidden disk - target_id matchs "
4326 "volume_id\n", ioc->name);
4327 mptsas_free_fw_event(ioc, fw_event);
4331 mpt_findImVolumes(ioc);
4333 case MPTSAS_ADD_DEVICE:
4334 memset(&sas_device, 0, sizeof(struct mptsas_devinfo));
4335 mptsas_sas_device_pg0(ioc, &sas_device,
4336 (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
4337 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
4338 (hot_plug_info->channel << 8) +
4341 /* If there is no FW B_T mapping for this device then break
4343 if (!(sas_device.flags & MPI_SAS_DEVICE0_FLAGS_DEVICE_PRESENT)
4344 || !(sas_device.flags &
4345 MPI_SAS_DEVICE0_FLAGS_DEVICE_MAPPED))
4348 if (!sas_device.handle)
4351 phy_info = mptsas_refreshing_device_handles(ioc, &sas_device);
4352 /* Only For SATA Device ADD */
4353 if (!phy_info && (sas_device.device_info &
4354 MPI_SAS_DEVICE_INFO_SATA_DEVICE)) {
4355 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
4356 "%s %d SATA HOT PLUG: "
4357 "parent handle of device %x\n", ioc->name,
4358 __func__, __LINE__, sas_device.handle_parent));
4359 port_info = mptsas_find_portinfo_by_handle(ioc,
4360 sas_device.handle_parent);
4362 if (port_info == ioc->hba_port_info)
4363 mptsas_probe_hba_phys(ioc);
4365 mptsas_expander_refresh(ioc, port_info);
4367 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4368 "%s %d port info is NULL\n",
4369 ioc->name, __func__, __LINE__));
4372 phy_info = mptsas_refreshing_device_handles
4377 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4378 "%s %d phy info is NULL\n",
4379 ioc->name, __func__, __LINE__));
4383 if (mptsas_get_rphy(phy_info))
4386 mptsas_add_end_device(ioc, phy_info);
4389 case MPTSAS_DEL_DEVICE:
4390 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
4391 hot_plug_info->sas_address);
4392 mptsas_del_end_device(ioc, phy_info);
4395 case MPTSAS_DEL_PHYSDISK:
4397 mpt_findImVolumes(ioc);
4399 phy_info = mptsas_find_phyinfo_by_phys_disk_num(
4400 ioc, hot_plug_info->phys_disk_num,
4401 hot_plug_info->channel,
4403 mptsas_del_end_device(ioc, phy_info);
4406 case MPTSAS_ADD_PHYSDISK_REPROBE:
4408 if (mptsas_sas_device_pg0(ioc, &sas_device,
4409 (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
4410 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
4411 (hot_plug_info->channel << 8) + hot_plug_info->id)) {
4412 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4413 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4414 __func__, hot_plug_info->id, __LINE__));
4418 /* If there is no FW B_T mapping for this device then break
4420 if (!(sas_device.flags & MPI_SAS_DEVICE0_FLAGS_DEVICE_PRESENT)
4421 || !(sas_device.flags &
4422 MPI_SAS_DEVICE0_FLAGS_DEVICE_MAPPED))
4425 phy_info = mptsas_find_phyinfo_by_sas_address(
4426 ioc, sas_device.sas_address);
4429 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4430 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4431 __func__, hot_plug_info->id, __LINE__));
4435 starget = mptsas_get_starget(phy_info);
4437 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4438 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4439 __func__, hot_plug_info->id, __LINE__));
4443 vtarget = starget->hostdata;
4445 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4446 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4447 __func__, hot_plug_info->id, __LINE__));
4451 mpt_findImVolumes(ioc);
4453 starget_printk(KERN_INFO, starget, MYIOC_s_FMT "RAID Hidding: "
4454 "fw_channel=%d, fw_id=%d, physdsk %d, sas_addr 0x%llx\n",
4455 ioc->name, hot_plug_info->channel, hot_plug_info->id,
4456 hot_plug_info->phys_disk_num, (unsigned long long)
4457 sas_device.sas_address);
4459 vtarget->id = hot_plug_info->phys_disk_num;
4460 vtarget->tflags |= MPT_TARGET_FLAGS_RAID_COMPONENT;
4461 phy_info->attached.phys_disk_num = hot_plug_info->phys_disk_num;
4462 mptsas_reprobe_target(starget, 1);
4465 case MPTSAS_DEL_PHYSDISK_REPROBE:
4467 if (mptsas_sas_device_pg0(ioc, &sas_device,
4468 (MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
4469 MPI_SAS_DEVICE_PGAD_FORM_SHIFT),
4470 (hot_plug_info->channel << 8) + hot_plug_info->id)) {
4471 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4472 "%s: fw_id=%d exit at line=%d\n",
4473 ioc->name, __func__,
4474 hot_plug_info->id, __LINE__));
4478 /* If there is no FW B_T mapping for this device then break
4480 if (!(sas_device.flags & MPI_SAS_DEVICE0_FLAGS_DEVICE_PRESENT)
4481 || !(sas_device.flags &
4482 MPI_SAS_DEVICE0_FLAGS_DEVICE_MAPPED))
4485 phy_info = mptsas_find_phyinfo_by_sas_address(ioc,
4486 sas_device.sas_address);
4488 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4489 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4490 __func__, hot_plug_info->id, __LINE__));
4494 starget = mptsas_get_starget(phy_info);
4496 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4497 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4498 __func__, hot_plug_info->id, __LINE__));
4502 vtarget = starget->hostdata;
4504 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4505 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4506 __func__, hot_plug_info->id, __LINE__));
4510 if (!(vtarget->tflags & MPT_TARGET_FLAGS_RAID_COMPONENT)) {
4511 dfailprintk(ioc, printk(MYIOC_s_ERR_FMT
4512 "%s: fw_id=%d exit at line=%d\n", ioc->name,
4513 __func__, hot_plug_info->id, __LINE__));
4517 mpt_findImVolumes(ioc);
4519 starget_printk(KERN_INFO, starget, MYIOC_s_FMT "RAID Exposing:"
4520 " fw_channel=%d, fw_id=%d, physdsk %d, sas_addr 0x%llx\n",
4521 ioc->name, hot_plug_info->channel, hot_plug_info->id,
4522 hot_plug_info->phys_disk_num, (unsigned long long)
4523 sas_device.sas_address);
4525 vtarget->tflags &= ~MPT_TARGET_FLAGS_RAID_COMPONENT;
4526 vtarget->id = hot_plug_info->id;
4527 phy_info->attached.phys_disk_num = ~0;
4528 mptsas_reprobe_target(starget, 0);
4529 mptsas_add_device_component_by_fw(ioc,
4530 hot_plug_info->channel, hot_plug_info->id);
4533 case MPTSAS_ADD_RAID:
4535 mpt_findImVolumes(ioc);
4536 printk(MYIOC_s_INFO_FMT "attaching raid volume, channel %d, "
4537 "id %d\n", ioc->name, MPTSAS_RAID_CHANNEL,
4539 scsi_add_device(ioc->sh, MPTSAS_RAID_CHANNEL,
4540 hot_plug_info->id, 0);
4543 case MPTSAS_DEL_RAID:
4545 mpt_findImVolumes(ioc);
4546 printk(MYIOC_s_INFO_FMT "removing raid volume, channel %d, "
4547 "id %d\n", ioc->name, MPTSAS_RAID_CHANNEL,
4549 scsi_remove_device(hot_plug_info->sdev);
4550 scsi_device_put(hot_plug_info->sdev);
4553 case MPTSAS_ADD_INACTIVE_VOLUME:
4555 mpt_findImVolumes(ioc);
4556 mptsas_adding_inactive_raid_components(ioc,
4557 hot_plug_info->channel, hot_plug_info->id);
4564 mptsas_free_fw_event(ioc, fw_event);
4568 mptsas_send_sas_event(struct fw_event_work *fw_event)
4571 struct mptsas_hotplug_event hot_plug_info;
4572 EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *sas_event_data;
4576 ioc = fw_event->ioc;
4577 sas_event_data = (EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *)
4578 fw_event->event_data;
4579 device_info = le32_to_cpu(sas_event_data->DeviceInfo);
4582 (MPI_SAS_DEVICE_INFO_SSP_TARGET |
4583 MPI_SAS_DEVICE_INFO_STP_TARGET |
4584 MPI_SAS_DEVICE_INFO_SATA_DEVICE)) == 0) {
4585 mptsas_free_fw_event(ioc, fw_event);
4589 if (sas_event_data->ReasonCode ==
4590 MPI_EVENT_SAS_DEV_STAT_RC_NO_PERSIST_ADDED) {
4591 mptbase_sas_persist_operation(ioc,
4592 MPI_SAS_OP_CLEAR_NOT_PRESENT);
4593 mptsas_free_fw_event(ioc, fw_event);
4597 switch (sas_event_data->ReasonCode) {
4598 case MPI_EVENT_SAS_DEV_STAT_RC_NOT_RESPONDING:
4599 case MPI_EVENT_SAS_DEV_STAT_RC_ADDED:
4600 memset(&hot_plug_info, 0, sizeof(struct mptsas_hotplug_event));
4601 hot_plug_info.handle = le16_to_cpu(sas_event_data->DevHandle);
4602 hot_plug_info.channel = sas_event_data->Bus;
4603 hot_plug_info.id = sas_event_data->TargetID;
4604 hot_plug_info.phy_id = sas_event_data->PhyNum;
4605 memcpy(&sas_address, &sas_event_data->SASAddress,
4607 hot_plug_info.sas_address = le64_to_cpu(sas_address);
4608 hot_plug_info.device_info = device_info;
4609 if (sas_event_data->ReasonCode &
4610 MPI_EVENT_SAS_DEV_STAT_RC_ADDED)
4611 hot_plug_info.event_type = MPTSAS_ADD_DEVICE;
4613 hot_plug_info.event_type = MPTSAS_DEL_DEVICE;
4614 mptsas_hotplug_work(ioc, fw_event, &hot_plug_info);
4617 case MPI_EVENT_SAS_DEV_STAT_RC_NO_PERSIST_ADDED:
4618 mptbase_sas_persist_operation(ioc,
4619 MPI_SAS_OP_CLEAR_NOT_PRESENT);
4620 mptsas_free_fw_event(ioc, fw_event);
4623 case MPI_EVENT_SAS_DEV_STAT_RC_SMART_DATA:
4625 case MPI_EVENT_SAS_DEV_STAT_RC_INTERNAL_DEVICE_RESET:
4628 mptsas_free_fw_event(ioc, fw_event);
4634 mptsas_send_raid_event(struct fw_event_work *fw_event)
4637 EVENT_DATA_RAID *raid_event_data;
4638 struct mptsas_hotplug_event hot_plug_info;
4641 struct scsi_device *sdev = NULL;
4642 VirtDevice *vdevice = NULL;
4643 RaidPhysDiskPage0_t phys_disk;
4645 ioc = fw_event->ioc;
4646 raid_event_data = (EVENT_DATA_RAID *)fw_event->event_data;
4647 status = le32_to_cpu(raid_event_data->SettingsStatus);
4648 state = (status >> 8) & 0xff;
4650 memset(&hot_plug_info, 0, sizeof(struct mptsas_hotplug_event));
4651 hot_plug_info.id = raid_event_data->VolumeID;
4652 hot_plug_info.channel = raid_event_data->VolumeBus;
4653 hot_plug_info.phys_disk_num = raid_event_data->PhysDiskNum;
4655 if (raid_event_data->ReasonCode == MPI_EVENT_RAID_RC_VOLUME_DELETED ||
4656 raid_event_data->ReasonCode == MPI_EVENT_RAID_RC_VOLUME_CREATED ||
4657 raid_event_data->ReasonCode ==
4658 MPI_EVENT_RAID_RC_VOLUME_STATUS_CHANGED) {
4659 sdev = scsi_device_lookup(ioc->sh, MPTSAS_RAID_CHANNEL,
4660 hot_plug_info.id, 0);
4661 hot_plug_info.sdev = sdev;
4663 vdevice = sdev->hostdata;
4666 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Entering %s: "
4667 "ReasonCode=%02x\n", ioc->name, __func__,
4668 raid_event_data->ReasonCode));
4670 switch (raid_event_data->ReasonCode) {
4671 case MPI_EVENT_RAID_RC_PHYSDISK_DELETED:
4672 hot_plug_info.event_type = MPTSAS_DEL_PHYSDISK_REPROBE;
4674 case MPI_EVENT_RAID_RC_PHYSDISK_CREATED:
4675 hot_plug_info.event_type = MPTSAS_ADD_PHYSDISK_REPROBE;
4677 case MPI_EVENT_RAID_RC_PHYSDISK_STATUS_CHANGED:
4679 case MPI_PD_STATE_ONLINE:
4680 case MPI_PD_STATE_NOT_COMPATIBLE:
4681 mpt_raid_phys_disk_pg0(ioc,
4682 raid_event_data->PhysDiskNum, &phys_disk);
4683 hot_plug_info.id = phys_disk.PhysDiskID;
4684 hot_plug_info.channel = phys_disk.PhysDiskBus;
4685 hot_plug_info.event_type = MPTSAS_ADD_PHYSDISK;
4687 case MPI_PD_STATE_FAILED:
4688 case MPI_PD_STATE_MISSING:
4689 case MPI_PD_STATE_OFFLINE_AT_HOST_REQUEST:
4690 case MPI_PD_STATE_FAILED_AT_HOST_REQUEST:
4691 case MPI_PD_STATE_OFFLINE_FOR_ANOTHER_REASON:
4692 hot_plug_info.event_type = MPTSAS_DEL_PHYSDISK;
4698 case MPI_EVENT_RAID_RC_VOLUME_DELETED:
4701 vdevice->vtarget->deleted = 1; /* block IO */
4702 hot_plug_info.event_type = MPTSAS_DEL_RAID;
4704 case MPI_EVENT_RAID_RC_VOLUME_CREATED:
4706 scsi_device_put(sdev);
4709 hot_plug_info.event_type = MPTSAS_ADD_RAID;
4711 case MPI_EVENT_RAID_RC_VOLUME_STATUS_CHANGED:
4712 if (!(status & MPI_RAIDVOL0_STATUS_FLAG_ENABLED)) {
4715 vdevice->vtarget->deleted = 1; /* block IO */
4716 hot_plug_info.event_type = MPTSAS_DEL_RAID;
4720 case MPI_RAIDVOL0_STATUS_STATE_FAILED:
4721 case MPI_RAIDVOL0_STATUS_STATE_MISSING:
4724 vdevice->vtarget->deleted = 1; /* block IO */
4725 hot_plug_info.event_type = MPTSAS_DEL_RAID;
4727 case MPI_RAIDVOL0_STATUS_STATE_OPTIMAL:
4728 case MPI_RAIDVOL0_STATUS_STATE_DEGRADED:
4730 scsi_device_put(sdev);
4733 hot_plug_info.event_type = MPTSAS_ADD_RAID;
4743 if (hot_plug_info.event_type != MPTSAS_IGNORE_EVENT)
4744 mptsas_hotplug_work(ioc, fw_event, &hot_plug_info);
4746 mptsas_free_fw_event(ioc, fw_event);
4750 * mptsas_issue_tm - send mptsas internal tm request
4751 * @ioc: Pointer to MPT_ADAPTER structure
4752 * @type: Task Management type
4753 * @channel: channel number for task management
4754 * @id: Logical Target ID for reset (if appropriate)
4755 * @lun: Logical unit for reset (if appropriate)
4756 * @task_context: Context for the task to be aborted
4757 * @timeout: timeout for task management control
4759 * return 0 on success and -1 on failure:
4763 mptsas_issue_tm(MPT_ADAPTER *ioc, u8 type, u8 channel, u8 id, u64 lun,
4764 int task_context, ulong timeout, u8 *issue_reset)
4767 SCSITaskMgmt_t *pScsiTm;
4769 unsigned long timeleft;
4772 mf = mpt_get_msg_frame(mptsasDeviceResetCtx, ioc);
4774 retval = -1; /* return failure */
4775 dtmprintk(ioc, printk(MYIOC_s_WARN_FMT "TaskMgmt request: no "
4776 "msg frames!!\n", ioc->name));
4780 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT "TaskMgmt request: mr = %p, "
4781 "task_type = 0x%02X,\n\t timeout = %ld, fw_channel = %d, "
4782 "fw_id = %d, lun = %lld,\n\t task_context = 0x%x\n", ioc->name, mf,
4783 type, timeout, channel, id, (unsigned long long)lun,
4786 pScsiTm = (SCSITaskMgmt_t *) mf;
4787 memset(pScsiTm, 0, sizeof(SCSITaskMgmt_t));
4788 pScsiTm->Function = MPI_FUNCTION_SCSI_TASK_MGMT;
4789 pScsiTm->TaskType = type;
4790 pScsiTm->MsgFlags = 0;
4791 pScsiTm->TargetID = id;
4792 pScsiTm->Bus = channel;
4793 pScsiTm->ChainOffset = 0;
4794 pScsiTm->Reserved = 0;
4795 pScsiTm->Reserved1 = 0;
4796 pScsiTm->TaskMsgContext = task_context;
4797 int_to_scsilun(lun, (struct scsi_lun *)pScsiTm->LUN);
4799 INITIALIZE_MGMT_STATUS(ioc->taskmgmt_cmds.status)
4800 CLEAR_MGMT_STATUS(ioc->internal_cmds.status)
4802 mpt_put_msg_frame_hi_pri(mptsasDeviceResetCtx, ioc, mf);
4804 /* Now wait for the command to complete */
4805 timeleft = wait_for_completion_timeout(&ioc->taskmgmt_cmds.done,
4807 if (!(ioc->taskmgmt_cmds.status & MPT_MGMT_STATUS_COMMAND_GOOD)) {
4808 retval = -1; /* return failure */
4809 dtmprintk(ioc, printk(MYIOC_s_ERR_FMT
4810 "TaskMgmt request: TIMED OUT!(mr=%p)\n", ioc->name, mf));
4811 mpt_free_msg_frame(ioc, mf);
4812 if (ioc->taskmgmt_cmds.status & MPT_MGMT_STATUS_DID_IOCRESET)
4818 if (!(ioc->taskmgmt_cmds.status & MPT_MGMT_STATUS_RF_VALID)) {
4819 retval = -1; /* return failure */
4820 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
4821 "TaskMgmt request: failed with no reply\n", ioc->name));
4826 CLEAR_MGMT_STATUS(ioc->taskmgmt_cmds.status)
4831 * mptsas_broadcast_primative_work - Handle broadcast primitives
4832 * @work: work queue payload containing info describing the event
4834 * this will be handled in workqueue context.
4837 mptsas_broadcast_primative_work(struct fw_event_work *fw_event)
4839 MPT_ADAPTER *ioc = fw_event->ioc;
4841 VirtDevice *vdevice;
4843 struct scsi_cmnd *sc;
4844 SCSITaskMgmtReply_t *pScsiTmReply;
4849 u32 termination_count;
4852 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
4853 "%s - enter\n", ioc->name, __func__));
4855 mutex_lock(&ioc->taskmgmt_cmds.mutex);
4856 if (mpt_set_taskmgmt_in_progress_flag(ioc) != 0) {
4857 mutex_unlock(&ioc->taskmgmt_cmds.mutex);
4858 mptsas_requeue_fw_event(ioc, fw_event, 1000);
4863 termination_count = 0;
4865 mpt_findImVolumes(ioc);
4866 pScsiTmReply = (SCSITaskMgmtReply_t *) ioc->taskmgmt_cmds.reply;
4868 for (ii = 0; ii < ioc->req_depth; ii++) {
4869 if (ioc->fw_events_off)
4871 sc = mptscsih_get_scsi_lookup(ioc, ii);
4874 mf = MPT_INDEX_2_MFPTR(ioc, ii);
4877 task_context = mf->u.frame.hwhdr.msgctxu.MsgContext;
4878 vdevice = sc->device->hostdata;
4879 if (!vdevice || !vdevice->vtarget)
4881 if (vdevice->vtarget->tflags & MPT_TARGET_FLAGS_RAID_COMPONENT)
4882 continue; /* skip hidden raid components */
4883 if (vdevice->vtarget->raidVolume)
4884 continue; /* skip hidden raid components */
4885 channel = vdevice->vtarget->channel;
4886 id = vdevice->vtarget->id;
4888 if (mptsas_issue_tm(ioc, MPI_SCSITASKMGMT_TASKTYPE_QUERY_TASK,
4889 channel, id, (u64)lun, task_context, 30, &issue_reset))
4892 termination_count +=
4893 le32_to_cpu(pScsiTmReply->TerminationCount);
4894 if ((pScsiTmReply->IOCStatus == MPI_IOCSTATUS_SUCCESS) &&
4895 (pScsiTmReply->ResponseCode ==
4896 MPI_SCSITASKMGMT_RSP_TM_SUCCEEDED ||
4897 pScsiTmReply->ResponseCode ==
4898 MPI_SCSITASKMGMT_RSP_IO_QUEUED_ON_IOC))
4900 if (mptsas_issue_tm(ioc,
4901 MPI_SCSITASKMGMT_TASKTYPE_ABRT_TASK_SET,
4902 channel, id, (u64)lun, 0, 30, &issue_reset))
4904 termination_count +=
4905 le32_to_cpu(pScsiTmReply->TerminationCount);
4909 dtmprintk(ioc, printk(MYIOC_s_DEBUG_FMT
4910 "%s - exit, query_count = %d termination_count = %d\n",
4911 ioc->name, __func__, query_count, termination_count));
4913 ioc->broadcast_aen_busy = 0;
4914 mpt_clear_taskmgmt_in_progress_flag(ioc);
4915 mutex_unlock(&ioc->taskmgmt_cmds.mutex);
4918 printk(MYIOC_s_WARN_FMT
4919 "Issuing Reset from %s!! doorbell=0x%08x\n",
4920 ioc->name, __func__, mpt_GetIocState(ioc, 0));
4921 mpt_Soft_Hard_ResetHandler(ioc, CAN_SLEEP);
4923 mptsas_free_fw_event(ioc, fw_event);
4927 * mptsas_send_ir2_event - handle exposing hidden disk when
4928 * an inactive raid volume is added
4930 * @ioc: Pointer to MPT_ADAPTER structure
4935 mptsas_send_ir2_event(struct fw_event_work *fw_event)
4938 struct mptsas_hotplug_event hot_plug_info;
4939 MPI_EVENT_DATA_IR2 *ir2_data;
4941 RaidPhysDiskPage0_t phys_disk;
4943 ioc = fw_event->ioc;
4944 ir2_data = (MPI_EVENT_DATA_IR2 *)fw_event->event_data;
4945 reasonCode = ir2_data->ReasonCode;
4947 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT "Entering %s: "
4948 "ReasonCode=%02x\n", ioc->name, __func__, reasonCode));
4950 memset(&hot_plug_info, 0, sizeof(struct mptsas_hotplug_event));
4951 hot_plug_info.id = ir2_data->TargetID;
4952 hot_plug_info.channel = ir2_data->Bus;
4953 switch (reasonCode) {
4954 case MPI_EVENT_IR2_RC_FOREIGN_CFG_DETECTED:
4955 hot_plug_info.event_type = MPTSAS_ADD_INACTIVE_VOLUME;
4957 case MPI_EVENT_IR2_RC_DUAL_PORT_REMOVED:
4958 hot_plug_info.phys_disk_num = ir2_data->PhysDiskNum;
4959 hot_plug_info.event_type = MPTSAS_DEL_PHYSDISK;
4961 case MPI_EVENT_IR2_RC_DUAL_PORT_ADDED:
4962 hot_plug_info.phys_disk_num = ir2_data->PhysDiskNum;
4963 mpt_raid_phys_disk_pg0(ioc,
4964 ir2_data->PhysDiskNum, &phys_disk);
4965 hot_plug_info.id = phys_disk.PhysDiskID;
4966 hot_plug_info.event_type = MPTSAS_ADD_PHYSDISK;
4969 mptsas_free_fw_event(ioc, fw_event);
4972 mptsas_hotplug_work(ioc, fw_event, &hot_plug_info);
4976 mptsas_event_process(MPT_ADAPTER *ioc, EventNotificationReply_t *reply)
4978 u32 event = le32_to_cpu(reply->Event);
4979 int sz, event_data_sz;
4980 struct fw_event_work *fw_event;
4981 unsigned long delay;
4983 if (ioc->bus_type != SAS)
4986 /* events turned off due to host reset or driver unloading */
4987 if (ioc->fw_events_off)
4990 delay = msecs_to_jiffies(1);
4992 case MPI_EVENT_SAS_BROADCAST_PRIMITIVE:
4994 EVENT_DATA_SAS_BROADCAST_PRIMITIVE *broadcast_event_data =
4995 (EVENT_DATA_SAS_BROADCAST_PRIMITIVE *)reply->Data;
4996 if (broadcast_event_data->Primitive !=
4997 MPI_EVENT_PRIMITIVE_ASYNCHRONOUS_EVENT)
4999 if (ioc->broadcast_aen_busy)
5001 ioc->broadcast_aen_busy = 1;
5004 case MPI_EVENT_SAS_DEVICE_STATUS_CHANGE:
5006 EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *sas_event_data =
5007 (EVENT_DATA_SAS_DEVICE_STATUS_CHANGE *)reply->Data;
5009 ioc_stat = le16_to_cpu(reply->IOCStatus);
5011 if (sas_event_data->ReasonCode ==
5012 MPI_EVENT_SAS_DEV_STAT_RC_NOT_RESPONDING) {
5013 mptsas_target_reset_queue(ioc, sas_event_data);
5016 if (sas_event_data->ReasonCode ==
5017 MPI_EVENT_SAS_DEV_STAT_RC_INTERNAL_DEVICE_RESET &&
5018 ioc->device_missing_delay &&
5019 (ioc_stat & MPI_IOCSTATUS_FLAG_LOG_INFO_AVAILABLE)) {
5020 VirtTarget *vtarget = NULL;
5023 id = sas_event_data->TargetID;
5024 channel = sas_event_data->Bus;
5026 vtarget = mptsas_find_vtarget(ioc, channel, id);
5028 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
5029 "LogInfo (0x%x) available for "
5030 "INTERNAL_DEVICE_RESET"
5031 "fw_id %d fw_channel %d\n", ioc->name,
5032 le32_to_cpu(reply->IOCLogInfo),
5034 if (vtarget->raidVolume) {
5035 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
5036 "Skipping Raid Volume for inDMD\n",
5039 devtprintk(ioc, printk(MYIOC_s_DEBUG_FMT
5040 "Setting device flag inDMD\n",
5051 case MPI_EVENT_SAS_EXPANDER_STATUS_CHANGE:
5053 MpiEventDataSasExpanderStatusChange_t *expander_data =
5054 (MpiEventDataSasExpanderStatusChange_t *)reply->Data;
5056 if (ioc->old_sas_discovery_protocal)
5059 if (expander_data->ReasonCode ==
5060 MPI_EVENT_SAS_EXP_RC_NOT_RESPONDING &&
5061 ioc->device_missing_delay)
5062 delay = HZ * ioc->device_missing_delay;
5065 case MPI_EVENT_SAS_DISCOVERY:
5067 u32 discovery_status;
5068 EventDataSasDiscovery_t *discovery_data =
5069 (EventDataSasDiscovery_t *)reply->Data;
5071 discovery_status = le32_to_cpu(discovery_data->DiscoveryStatus);
5072 ioc->sas_discovery_quiesce_io = discovery_status ? 1 : 0;
5073 if (ioc->old_sas_discovery_protocal && !discovery_status)
5074 mptsas_queue_rescan(ioc);
5077 case MPI_EVENT_INTEGRATED_RAID:
5078 case MPI_EVENT_PERSISTENT_TABLE_FULL:
5080 case MPI_EVENT_SAS_PHY_LINK_STATUS:
5081 case MPI_EVENT_QUEUE_FULL:
5087 event_data_sz = ((reply->MsgLength * 4) -
5088 offsetof(EventNotificationReply_t, Data));
5089 sz = offsetof(struct fw_event_work, event_data) + event_data_sz;
5090 fw_event = kzalloc(sz, GFP_ATOMIC);
5092 printk(MYIOC_s_WARN_FMT "%s: failed at (line=%d)\n", ioc->name,
5093 __func__, __LINE__);
5096 memcpy(fw_event->event_data, reply->Data, event_data_sz);
5097 fw_event->event = event;
5098 fw_event->ioc = ioc;
5099 mptsas_add_fw_event(ioc, fw_event, delay);
5103 /* Delete a volume when no longer listed in ioc pg2
5105 static void mptsas_volume_delete(MPT_ADAPTER *ioc, u8 id)
5107 struct scsi_device *sdev;
5110 sdev = scsi_device_lookup(ioc->sh, MPTSAS_RAID_CHANNEL, id, 0);
5113 if (!ioc->raid_data.pIocPg2)
5115 if (!ioc->raid_data.pIocPg2->NumActiveVolumes)
5117 for (i = 0; i < ioc->raid_data.pIocPg2->NumActiveVolumes; i++)
5118 if (ioc->raid_data.pIocPg2->RaidVolume[i].VolumeID == id)
5121 printk(MYIOC_s_INFO_FMT "removing raid volume, channel %d, "
5122 "id %d\n", ioc->name, MPTSAS_RAID_CHANNEL, id);
5123 scsi_remove_device(sdev);
5125 scsi_device_put(sdev);
5129 mptsas_probe(struct pci_dev *pdev, const struct pci_device_id *id)
5131 struct Scsi_Host *sh;
5134 unsigned long flags;
5142 r = mpt_attach(pdev,id);
5146 ioc = pci_get_drvdata(pdev);
5147 mptsas_fw_event_off(ioc);
5148 ioc->DoneCtx = mptsasDoneCtx;
5149 ioc->TaskCtx = mptsasTaskCtx;
5150 ioc->InternalCtx = mptsasInternalCtx;
5151 ioc->schedule_target_reset = &mptsas_schedule_target_reset;
5152 ioc->schedule_dead_ioc_flush_running_cmds =
5153 &mptscsih_flush_running_cmds;
5154 /* Added sanity check on readiness of the MPT adapter.
5156 if (ioc->last_state != MPI_IOC_STATE_OPERATIONAL) {
5157 printk(MYIOC_s_WARN_FMT
5158 "Skipping because it's not operational!\n",
5161 goto out_mptsas_probe;
5165 printk(MYIOC_s_WARN_FMT "Skipping because it's disabled!\n",
5168 goto out_mptsas_probe;
5171 /* Sanity check - ensure at least 1 port is INITIATOR capable
5174 for (ii = 0; ii < ioc->facts.NumberOfPorts; ii++) {
5175 if (ioc->pfacts[ii].ProtocolFlags &
5176 MPI_PORTFACTS_PROTOCOL_INITIATOR)
5181 printk(MYIOC_s_WARN_FMT
5182 "Skipping ioc=%p because SCSI Initiator mode "
5183 "is NOT enabled!\n", ioc->name, ioc);
5187 sh = scsi_host_alloc(&mptsas_driver_template, sizeof(MPT_SCSI_HOST));
5189 printk(MYIOC_s_WARN_FMT
5190 "Unable to register controller with SCSI subsystem\n",
5193 goto out_mptsas_probe;
5196 spin_lock_irqsave(&ioc->FreeQlock, flags);
5198 /* Attach the SCSI Host to the IOC structure
5206 /* set 16 byte cdb's */
5207 sh->max_cmd_len = 16;
5208 sh->can_queue = min_t(int, ioc->req_depth - 10, sh->can_queue);
5210 sh->max_lun = max_lun;
5211 sh->transportt = mptsas_transport_template;
5215 sh->unique_id = ioc->id;
5217 INIT_LIST_HEAD(&ioc->sas_topology);
5218 mutex_init(&ioc->sas_topology_mutex);
5219 mutex_init(&ioc->sas_discovery_mutex);
5220 mutex_init(&ioc->sas_mgmt.mutex);
5221 init_completion(&ioc->sas_mgmt.done);
5223 /* Verify that we won't exceed the maximum
5224 * number of chain buffers
5225 * We can optimize: ZZ = req_sz/sizeof(SGE)
5227 * numSGE = 1 + (ZZ-1)*(maxChain -1) + ZZ
5228 * + (req_sz - 64)/sizeof(SGE)
5229 * A slightly different algorithm is required for
5232 scale = ioc->req_sz/ioc->SGE_size;
5233 if (ioc->sg_addr_size == sizeof(u64)) {
5234 numSGE = (scale - 1) *
5235 (ioc->facts.MaxChainDepth-1) + scale +
5236 (ioc->req_sz - 60) / ioc->SGE_size;
5238 numSGE = 1 + (scale - 1) *
5239 (ioc->facts.MaxChainDepth-1) + scale +
5240 (ioc->req_sz - 64) / ioc->SGE_size;
5243 if (numSGE < sh->sg_tablesize) {
5244 /* Reset this value */
5245 dprintk(ioc, printk(MYIOC_s_DEBUG_FMT
5246 "Resetting sg_tablesize to %d from %d\n",
5247 ioc->name, numSGE, sh->sg_tablesize));
5248 sh->sg_tablesize = numSGE;
5251 if (mpt_loadtime_max_sectors) {
5252 if (mpt_loadtime_max_sectors < 64 ||
5253 mpt_loadtime_max_sectors > 8192) {
5254 printk(MYIOC_s_INFO_FMT "Invalid value passed for"
5255 "mpt_loadtime_max_sectors %d."
5256 "Range from 64 to 8192\n", ioc->name,
5257 mpt_loadtime_max_sectors);
5259 mpt_loadtime_max_sectors &= 0xFFFFFFFE;
5260 dprintk(ioc, printk(MYIOC_s_DEBUG_FMT
5261 "Resetting max sector to %d from %d\n",
5262 ioc->name, mpt_loadtime_max_sectors, sh->max_sectors));
5263 sh->max_sectors = mpt_loadtime_max_sectors;
5266 hd = shost_priv(sh);
5269 /* SCSI needs scsi_cmnd lookup table!
5270 * (with size equal to req_depth*PtrSz!)
5272 ioc->ScsiLookup = kcalloc(ioc->req_depth, sizeof(void *), GFP_ATOMIC);
5273 if (!ioc->ScsiLookup) {
5275 spin_unlock_irqrestore(&ioc->FreeQlock, flags);
5276 goto out_mptsas_probe;
5278 spin_lock_init(&ioc->scsi_lookup_lock);
5280 dprintk(ioc, printk(MYIOC_s_DEBUG_FMT "ScsiLookup @ %p\n",
5281 ioc->name, ioc->ScsiLookup));
5283 ioc->sas_data.ptClear = mpt_pt_clear;
5285 hd->last_queue_full = 0;
5286 INIT_LIST_HEAD(&hd->target_reset_list);
5287 INIT_LIST_HEAD(&ioc->sas_device_info_list);
5288 mutex_init(&ioc->sas_device_info_mutex);
5290 spin_unlock_irqrestore(&ioc->FreeQlock, flags);
5292 if (ioc->sas_data.ptClear==1) {
5293 mptbase_sas_persist_operation(
5294 ioc, MPI_SAS_OP_CLEAR_ALL_PERSISTENT);
5297 error = scsi_add_host(sh, &ioc->pcidev->dev);
5299 dprintk(ioc, printk(MYIOC_s_ERR_FMT
5300 "scsi_add_host failed\n", ioc->name));
5301 goto out_mptsas_probe;
5304 /* older firmware doesn't support expander events */
5305 if ((ioc->facts.HeaderVersion >> 8) < 0xE)
5306 ioc->old_sas_discovery_protocal = 1;
5307 mptsas_scan_sas_topology(ioc);
5308 mptsas_fw_event_on(ioc);
5313 mptscsih_remove(pdev);
5318 mptsas_shutdown(struct pci_dev *pdev)
5320 MPT_ADAPTER *ioc = pci_get_drvdata(pdev);
5322 mptsas_fw_event_off(ioc);
5323 mptsas_cleanup_fw_event_q(ioc);
5326 static void __devexit mptsas_remove(struct pci_dev *pdev)
5328 MPT_ADAPTER *ioc = pci_get_drvdata(pdev);
5329 struct mptsas_portinfo *p, *n;
5333 printk(MYIOC_s_INFO_FMT "IOC is in Target mode\n", ioc->name);
5338 mptsas_shutdown(pdev);
5340 mptsas_del_device_components(ioc);
5342 ioc->sas_discovery_ignore_events = 1;
5343 sas_remove_host(ioc->sh);
5345 mutex_lock(&ioc->sas_topology_mutex);
5346 list_for_each_entry_safe(p, n, &ioc->sas_topology, list) {
5348 for (i = 0 ; i < p->num_phys ; i++)
5349 mptsas_port_delete(ioc, p->phy_info[i].port_details);
5354 mutex_unlock(&ioc->sas_topology_mutex);
5355 ioc->hba_port_info = NULL;
5356 mptscsih_remove(pdev);
5359 static struct pci_device_id mptsas_pci_table[] = {
5360 { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1064,
5361 PCI_ANY_ID, PCI_ANY_ID },
5362 { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1068,
5363 PCI_ANY_ID, PCI_ANY_ID },
5364 { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1064E,
5365 PCI_ANY_ID, PCI_ANY_ID },
5366 { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1068E,
5367 PCI_ANY_ID, PCI_ANY_ID },
5368 { PCI_VENDOR_ID_LSI_LOGIC, MPI_MANUFACTPAGE_DEVID_SAS1078,
5369 PCI_ANY_ID, PCI_ANY_ID },
5370 {0} /* Terminating entry */
5372 MODULE_DEVICE_TABLE(pci, mptsas_pci_table);
5375 static struct pci_driver mptsas_driver = {
5377 .id_table = mptsas_pci_table,
5378 .probe = mptsas_probe,
5379 .remove = __devexit_p(mptsas_remove),
5380 .shutdown = mptsas_shutdown,
5382 .suspend = mptscsih_suspend,
5383 .resume = mptscsih_resume,
5392 show_mptmod_ver(my_NAME, my_VERSION);
5394 mptsas_transport_template =
5395 sas_attach_transport(&mptsas_transport_functions);
5396 if (!mptsas_transport_template)
5398 mptsas_transport_template->eh_timed_out = mptsas_eh_timed_out;
5400 mptsasDoneCtx = mpt_register(mptscsih_io_done, MPTSAS_DRIVER,
5401 "mptscsih_io_done");
5402 mptsasTaskCtx = mpt_register(mptscsih_taskmgmt_complete, MPTSAS_DRIVER,
5403 "mptscsih_taskmgmt_complete");
5405 mpt_register(mptscsih_scandv_complete, MPTSAS_DRIVER,
5406 "mptscsih_scandv_complete");
5407 mptsasMgmtCtx = mpt_register(mptsas_mgmt_done, MPTSAS_DRIVER,
5408 "mptsas_mgmt_done");
5409 mptsasDeviceResetCtx =
5410 mpt_register(mptsas_taskmgmt_complete, MPTSAS_DRIVER,
5411 "mptsas_taskmgmt_complete");
5413 mpt_event_register(mptsasDoneCtx, mptsas_event_process);
5414 mpt_reset_register(mptsasDoneCtx, mptsas_ioc_reset);
5416 error = pci_register_driver(&mptsas_driver);
5418 sas_release_transport(mptsas_transport_template);
5426 pci_unregister_driver(&mptsas_driver);
5427 sas_release_transport(mptsas_transport_template);
5429 mpt_reset_deregister(mptsasDoneCtx);
5430 mpt_event_deregister(mptsasDoneCtx);
5432 mpt_deregister(mptsasMgmtCtx);
5433 mpt_deregister(mptsasInternalCtx);
5434 mpt_deregister(mptsasTaskCtx);
5435 mpt_deregister(mptsasDoneCtx);
5436 mpt_deregister(mptsasDeviceResetCtx);
5439 module_init(mptsas_init);
5440 module_exit(mptsas_exit);