| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * QLogic iSCSI HBA Driver | 
| Vikas Chaudhary | 7d01d06 | 2010-12-02 22:12:51 -0800 | [diff] [blame] | 3 | * Copyright (c)  2003-2010 QLogic Corporation | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 4 | * | 
|  | 5 | * See LICENSE.qla4xxx for copyright and licensing details. | 
|  | 6 | */ | 
|  | 7 |  | 
|  | 8 | #include "ql4_def.h" | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 9 | #include "ql4_glbl.h" | 
|  | 10 | #include "ql4_dbg.h" | 
|  | 11 | #include "ql4_inline.h" | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 12 |  | 
|  | 13 | /** | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 14 | * qla4xxx_copy_sense - copy sense data	into cmd sense buffer | 
|  | 15 | * @ha: Pointer to host adapter structure. | 
|  | 16 | * @sts_entry: Pointer to status entry structure. | 
|  | 17 | * @srb: Pointer to srb structure. | 
|  | 18 | **/ | 
|  | 19 | static void qla4xxx_copy_sense(struct scsi_qla_host *ha, | 
|  | 20 | struct status_entry *sts_entry, | 
|  | 21 | struct srb *srb) | 
|  | 22 | { | 
|  | 23 | struct scsi_cmnd *cmd = srb->cmd; | 
|  | 24 | uint16_t sense_len; | 
|  | 25 |  | 
|  | 26 | memset(cmd->sense_buffer, 0, SCSI_SENSE_BUFFERSIZE); | 
|  | 27 | sense_len = le16_to_cpu(sts_entry->senseDataByteCnt); | 
|  | 28 | if (sense_len == 0) | 
|  | 29 | return; | 
|  | 30 |  | 
|  | 31 | /* Save total available sense length, | 
|  | 32 | * not to exceed cmd's sense buffer size */ | 
|  | 33 | sense_len = min_t(uint16_t, sense_len, SCSI_SENSE_BUFFERSIZE); | 
|  | 34 | srb->req_sense_ptr = cmd->sense_buffer; | 
|  | 35 | srb->req_sense_len = sense_len; | 
|  | 36 |  | 
|  | 37 | /* Copy sense from sts_entry pkt */ | 
|  | 38 | sense_len = min_t(uint16_t, sense_len, IOCB_MAX_SENSEDATA_LEN); | 
|  | 39 | memcpy(cmd->sense_buffer, sts_entry->senseData, sense_len); | 
|  | 40 |  | 
|  | 41 | DEBUG2(printk(KERN_INFO "scsi%ld:%d:%d:%d: %s: sense key = %x, " | 
|  | 42 | "ASL= %02x, ASC/ASCQ = %02x/%02x\n", ha->host_no, | 
|  | 43 | cmd->device->channel, cmd->device->id, | 
|  | 44 | cmd->device->lun, __func__, | 
|  | 45 | sts_entry->senseData[2] & 0x0f, | 
|  | 46 | sts_entry->senseData[7], | 
|  | 47 | sts_entry->senseData[12], | 
|  | 48 | sts_entry->senseData[13])); | 
|  | 49 |  | 
|  | 50 | DEBUG5(qla4xxx_dump_buffer(cmd->sense_buffer, sense_len)); | 
|  | 51 | srb->flags |= SRB_GOT_SENSE; | 
|  | 52 |  | 
|  | 53 | /* Update srb, in case a sts_cont pkt follows */ | 
|  | 54 | srb->req_sense_ptr += sense_len; | 
|  | 55 | srb->req_sense_len -= sense_len; | 
|  | 56 | if (srb->req_sense_len != 0) | 
|  | 57 | ha->status_srb = srb; | 
|  | 58 | else | 
|  | 59 | ha->status_srb = NULL; | 
|  | 60 | } | 
|  | 61 |  | 
|  | 62 | /** | 
|  | 63 | * qla4xxx_status_cont_entry - Process a Status Continuations entry. | 
|  | 64 | * @ha: SCSI driver HA context | 
|  | 65 | * @sts_cont: Entry pointer | 
|  | 66 | * | 
|  | 67 | * Extended sense data. | 
|  | 68 | */ | 
|  | 69 | static void | 
|  | 70 | qla4xxx_status_cont_entry(struct scsi_qla_host *ha, | 
|  | 71 | struct status_cont_entry *sts_cont) | 
|  | 72 | { | 
|  | 73 | struct srb *srb = ha->status_srb; | 
|  | 74 | struct scsi_cmnd *cmd; | 
| Vikas Chaudhary | 735e415 | 2010-10-06 22:48:53 -0700 | [diff] [blame] | 75 | uint16_t sense_len; | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 76 |  | 
|  | 77 | if (srb == NULL) | 
|  | 78 | return; | 
|  | 79 |  | 
|  | 80 | cmd = srb->cmd; | 
|  | 81 | if (cmd == NULL) { | 
|  | 82 | DEBUG2(printk(KERN_INFO "scsi%ld: %s: Cmd already returned " | 
|  | 83 | "back to OS srb=%p srb->state:%d\n", ha->host_no, | 
|  | 84 | __func__, srb, srb->state)); | 
|  | 85 | ha->status_srb = NULL; | 
|  | 86 | return; | 
|  | 87 | } | 
|  | 88 |  | 
|  | 89 | /* Copy sense data. */ | 
|  | 90 | sense_len = min_t(uint16_t, srb->req_sense_len, | 
|  | 91 | IOCB_MAX_EXT_SENSEDATA_LEN); | 
|  | 92 | memcpy(srb->req_sense_ptr, sts_cont->ext_sense_data, sense_len); | 
|  | 93 | DEBUG5(qla4xxx_dump_buffer(srb->req_sense_ptr, sense_len)); | 
|  | 94 |  | 
|  | 95 | srb->req_sense_ptr += sense_len; | 
|  | 96 | srb->req_sense_len -= sense_len; | 
|  | 97 |  | 
|  | 98 | /* Place command on done queue. */ | 
|  | 99 | if (srb->req_sense_len == 0) { | 
| Vikas Chaudhary | 09a0f71 | 2010-04-28 11:42:24 +0530 | [diff] [blame] | 100 | kref_put(&srb->srb_ref, qla4xxx_srb_compl); | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 101 | ha->status_srb = NULL; | 
|  | 102 | } | 
|  | 103 | } | 
|  | 104 |  | 
|  | 105 | /** | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 106 | * qla4xxx_status_entry - processes status IOCBs | 
|  | 107 | * @ha: Pointer to host adapter structure. | 
|  | 108 | * @sts_entry: Pointer to status entry structure. | 
|  | 109 | **/ | 
|  | 110 | static void qla4xxx_status_entry(struct scsi_qla_host *ha, | 
|  | 111 | struct status_entry *sts_entry) | 
|  | 112 | { | 
|  | 113 | uint8_t scsi_status; | 
|  | 114 | struct scsi_cmnd *cmd; | 
|  | 115 | struct srb *srb; | 
|  | 116 | struct ddb_entry *ddb_entry; | 
|  | 117 | uint32_t residual; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 118 |  | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 119 | srb = qla4xxx_del_from_active_array(ha, le32_to_cpu(sts_entry->handle)); | 
|  | 120 | if (!srb) { | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 121 | DEBUG2(printk(KERN_WARNING "scsi%ld: %s: Status Entry invalid " | 
|  | 122 | "handle 0x%x, sp=%p. This cmd may have already " | 
|  | 123 | "been completed.\n", ha->host_no, __func__, | 
|  | 124 | le32_to_cpu(sts_entry->handle), srb)); | 
| Vikas Chaudhary | c2660df | 2010-07-10 14:51:02 +0530 | [diff] [blame] | 125 | ql4_printk(KERN_WARNING, ha, "%s invalid status entry:" | 
|  | 126 | " handle=0x%0x\n", __func__, sts_entry->handle); | 
| David C Somayajulu | 9d56291 | 2008-03-19 11:23:03 -0700 | [diff] [blame] | 127 | set_bit(DPC_RESET_HA, &ha->dpc_flags); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 128 | return; | 
|  | 129 | } | 
|  | 130 |  | 
|  | 131 | cmd = srb->cmd; | 
|  | 132 | if (cmd == NULL) { | 
|  | 133 | DEBUG2(printk("scsi%ld: %s: Command already returned back to " | 
|  | 134 | "OS pkt->handle=%d srb=%p srb->state:%d\n", | 
|  | 135 | ha->host_no, __func__, sts_entry->handle, | 
|  | 136 | srb, srb->state)); | 
| Vikas Chaudhary | c2660df | 2010-07-10 14:51:02 +0530 | [diff] [blame] | 137 | ql4_printk(KERN_WARNING, ha, "Command is NULL:" | 
|  | 138 | " already returned to OS (srb=%p)\n", srb); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 139 | return; | 
|  | 140 | } | 
|  | 141 |  | 
|  | 142 | ddb_entry = srb->ddb; | 
|  | 143 | if (ddb_entry == NULL) { | 
|  | 144 | cmd->result = DID_NO_CONNECT << 16; | 
|  | 145 | goto status_entry_exit; | 
|  | 146 | } | 
|  | 147 |  | 
|  | 148 | residual = le32_to_cpu(sts_entry->residualByteCnt); | 
|  | 149 |  | 
|  | 150 | /* Translate ISP error to a Linux SCSI error. */ | 
|  | 151 | scsi_status = sts_entry->scsiStatus; | 
|  | 152 | switch (sts_entry->completionStatus) { | 
|  | 153 | case SCS_COMPLETE: | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 154 |  | 
| David C Somayajulu | 6ea7e33 | 2007-07-09 12:44:36 -0700 | [diff] [blame] | 155 | if (sts_entry->iscsiFlags & ISCSI_FLAG_RESIDUAL_OVER) { | 
|  | 156 | cmd->result = DID_ERROR << 16; | 
|  | 157 | break; | 
|  | 158 | } | 
|  | 159 |  | 
|  | 160 | if (sts_entry->iscsiFlags &ISCSI_FLAG_RESIDUAL_UNDER) { | 
| FUJITA Tomonori | 5f7186c | 2007-05-26 14:08:20 +0900 | [diff] [blame] | 161 | scsi_set_resid(cmd, residual); | 
| David C Somayajulu | 9d56291 | 2008-03-19 11:23:03 -0700 | [diff] [blame] | 162 | if (!scsi_status && ((scsi_bufflen(cmd) - residual) < | 
|  | 163 | cmd->underflow)) { | 
| David C Somayajulu | 6ea7e33 | 2007-07-09 12:44:36 -0700 | [diff] [blame] | 164 |  | 
|  | 165 | cmd->result = DID_ERROR << 16; | 
|  | 166 |  | 
|  | 167 | DEBUG2(printk("scsi%ld:%d:%d:%d: %s: " | 
|  | 168 | "Mid-layer Data underrun0, " | 
|  | 169 | "xferlen = 0x%x, " | 
|  | 170 | "residual = 0x%x\n", ha->host_no, | 
|  | 171 | cmd->device->channel, | 
|  | 172 | cmd->device->id, | 
|  | 173 | cmd->device->lun, __func__, | 
|  | 174 | scsi_bufflen(cmd), residual)); | 
|  | 175 | break; | 
|  | 176 | } | 
|  | 177 | } | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 178 |  | 
|  | 179 | cmd->result = DID_OK << 16 | scsi_status; | 
|  | 180 |  | 
|  | 181 | if (scsi_status != SCSI_CHECK_CONDITION) | 
|  | 182 | break; | 
|  | 183 |  | 
|  | 184 | /* Copy Sense Data into sense buffer. */ | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 185 | qla4xxx_copy_sense(ha, sts_entry, srb); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 186 | break; | 
|  | 187 |  | 
|  | 188 | case SCS_INCOMPLETE: | 
|  | 189 | /* Always set the status to DID_ERROR, since | 
|  | 190 | * all conditions result in that status anyway */ | 
|  | 191 | cmd->result = DID_ERROR << 16; | 
|  | 192 | break; | 
|  | 193 |  | 
|  | 194 | case SCS_RESET_OCCURRED: | 
|  | 195 | DEBUG2(printk("scsi%ld:%d:%d:%d: %s: Device RESET occurred\n", | 
|  | 196 | ha->host_no, cmd->device->channel, | 
|  | 197 | cmd->device->id, cmd->device->lun, __func__)); | 
|  | 198 |  | 
|  | 199 | cmd->result = DID_RESET << 16; | 
|  | 200 | break; | 
|  | 201 |  | 
|  | 202 | case SCS_ABORTED: | 
|  | 203 | DEBUG2(printk("scsi%ld:%d:%d:%d: %s: Abort occurred\n", | 
|  | 204 | ha->host_no, cmd->device->channel, | 
|  | 205 | cmd->device->id, cmd->device->lun, __func__)); | 
|  | 206 |  | 
|  | 207 | cmd->result = DID_RESET << 16; | 
|  | 208 | break; | 
|  | 209 |  | 
|  | 210 | case SCS_TIMEOUT: | 
|  | 211 | DEBUG2(printk(KERN_INFO "scsi%ld:%d:%d:%d: Timeout\n", | 
|  | 212 | ha->host_no, cmd->device->channel, | 
|  | 213 | cmd->device->id, cmd->device->lun)); | 
|  | 214 |  | 
| Mike Christie | 56d7fcf | 2008-08-19 18:45:26 -0500 | [diff] [blame] | 215 | cmd->result = DID_TRANSPORT_DISRUPTED << 16; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 216 |  | 
|  | 217 | /* | 
|  | 218 | * Mark device missing so that we won't continue to send | 
|  | 219 | * I/O to this device.	We should get a ddb state change | 
|  | 220 | * AEN soon. | 
|  | 221 | */ | 
|  | 222 | if (atomic_read(&ddb_entry->state) == DDB_STATE_ONLINE) | 
|  | 223 | qla4xxx_mark_device_missing(ha, ddb_entry); | 
|  | 224 | break; | 
|  | 225 |  | 
|  | 226 | case SCS_DATA_UNDERRUN: | 
|  | 227 | case SCS_DATA_OVERRUN: | 
| David C Somayajulu | 6ea7e33 | 2007-07-09 12:44:36 -0700 | [diff] [blame] | 228 | if ((sts_entry->iscsiFlags & ISCSI_FLAG_RESIDUAL_OVER) || | 
| Mike Christie | b06fc73 | 2009-08-20 15:11:00 -0500 | [diff] [blame] | 229 | (sts_entry->completionStatus == SCS_DATA_OVERRUN)) { | 
|  | 230 | DEBUG2(printk("scsi%ld:%d:%d:%d: %s: " "Data overrun\n", | 
|  | 231 | ha->host_no, | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 232 | cmd->device->channel, cmd->device->id, | 
| Mike Christie | b06fc73 | 2009-08-20 15:11:00 -0500 | [diff] [blame] | 233 | cmd->device->lun, __func__)); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 234 |  | 
|  | 235 | cmd->result = DID_ERROR << 16; | 
|  | 236 | break; | 
|  | 237 | } | 
|  | 238 |  | 
| David C Somayajulu | 6ea7e33 | 2007-07-09 12:44:36 -0700 | [diff] [blame] | 239 | scsi_set_resid(cmd, residual); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 240 |  | 
|  | 241 | /* | 
|  | 242 | * If there is scsi_status, it takes precedense over | 
|  | 243 | * underflow condition. | 
|  | 244 | */ | 
|  | 245 | if (scsi_status != 0) { | 
|  | 246 | cmd->result = DID_OK << 16 | scsi_status; | 
|  | 247 |  | 
|  | 248 | if (scsi_status != SCSI_CHECK_CONDITION) | 
|  | 249 | break; | 
|  | 250 |  | 
|  | 251 | /* Copy Sense Data into sense buffer. */ | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 252 | qla4xxx_copy_sense(ha, sts_entry, srb); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 253 | } else { | 
|  | 254 | /* | 
|  | 255 | * If RISC reports underrun and target does not | 
|  | 256 | * report it then we must have a lost frame, so | 
|  | 257 | * tell upper layer to retry it by reporting a | 
|  | 258 | * bus busy. | 
|  | 259 | */ | 
|  | 260 | if ((sts_entry->iscsiFlags & | 
|  | 261 | ISCSI_FLAG_RESIDUAL_UNDER) == 0) { | 
|  | 262 | cmd->result = DID_BUS_BUSY << 16; | 
| FUJITA Tomonori | 5f7186c | 2007-05-26 14:08:20 +0900 | [diff] [blame] | 263 | } else if ((scsi_bufflen(cmd) - residual) < | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 264 | cmd->underflow) { | 
|  | 265 | /* | 
|  | 266 | * Handle mid-layer underflow??? | 
|  | 267 | * | 
|  | 268 | * For kernels less than 2.4, the driver must | 
|  | 269 | * return an error if an underflow is detected. | 
|  | 270 | * For kernels equal-to and above 2.4, the | 
|  | 271 | * mid-layer will appearantly handle the | 
|  | 272 | * underflow by detecting the residual count -- | 
|  | 273 | * unfortunately, we do not see where this is | 
|  | 274 | * actually being done.	 In the interim, we | 
|  | 275 | * will return DID_ERROR. | 
|  | 276 | */ | 
|  | 277 | DEBUG2(printk("scsi%ld:%d:%d:%d: %s: " | 
| David C Somayajulu | 6ea7e33 | 2007-07-09 12:44:36 -0700 | [diff] [blame] | 278 | "Mid-layer Data underrun1, " | 
|  | 279 | "xferlen = 0x%x, " | 
|  | 280 | "residual = 0x%x\n", ha->host_no, | 
|  | 281 | cmd->device->channel, | 
|  | 282 | cmd->device->id, | 
|  | 283 | cmd->device->lun, __func__, | 
|  | 284 | scsi_bufflen(cmd), residual)); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 285 |  | 
|  | 286 | cmd->result = DID_ERROR << 16; | 
|  | 287 | } else { | 
|  | 288 | cmd->result = DID_OK << 16; | 
|  | 289 | } | 
|  | 290 | } | 
|  | 291 | break; | 
|  | 292 |  | 
|  | 293 | case SCS_DEVICE_LOGGED_OUT: | 
|  | 294 | case SCS_DEVICE_UNAVAILABLE: | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 295 | DEBUG2(printk(KERN_INFO "scsi%ld:%d:%d:%d: SCS_DEVICE " | 
|  | 296 | "state: 0x%x\n", ha->host_no, | 
|  | 297 | cmd->device->channel, cmd->device->id, | 
|  | 298 | cmd->device->lun, sts_entry->completionStatus)); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 299 | /* | 
|  | 300 | * Mark device missing so that we won't continue to | 
|  | 301 | * send I/O to this device.  We should get a ddb | 
|  | 302 | * state change AEN soon. | 
|  | 303 | */ | 
|  | 304 | if (atomic_read(&ddb_entry->state) == DDB_STATE_ONLINE) | 
|  | 305 | qla4xxx_mark_device_missing(ha, ddb_entry); | 
|  | 306 |  | 
| Mike Christie | 56d7fcf | 2008-08-19 18:45:26 -0500 | [diff] [blame] | 307 | cmd->result = DID_TRANSPORT_DISRUPTED << 16; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 308 | break; | 
|  | 309 |  | 
|  | 310 | case SCS_QUEUE_FULL: | 
|  | 311 | /* | 
|  | 312 | * SCSI Mid-Layer handles device queue full | 
|  | 313 | */ | 
|  | 314 | cmd->result = DID_OK << 16 | sts_entry->scsiStatus; | 
|  | 315 | DEBUG2(printk("scsi%ld:%d:%d: %s: QUEUE FULL detected " | 
|  | 316 | "compl=%02x, scsi=%02x, state=%02x, iFlags=%02x," | 
|  | 317 | " iResp=%02x\n", ha->host_no, cmd->device->id, | 
|  | 318 | cmd->device->lun, __func__, | 
|  | 319 | sts_entry->completionStatus, | 
|  | 320 | sts_entry->scsiStatus, sts_entry->state_flags, | 
|  | 321 | sts_entry->iscsiFlags, | 
|  | 322 | sts_entry->iscsiResponse)); | 
|  | 323 | break; | 
|  | 324 |  | 
|  | 325 | default: | 
|  | 326 | cmd->result = DID_ERROR << 16; | 
|  | 327 | break; | 
|  | 328 | } | 
|  | 329 |  | 
|  | 330 | status_entry_exit: | 
|  | 331 |  | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 332 | /* complete the request, if not waiting for status_continuation pkt */ | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 333 | srb->cc_stat = sts_entry->completionStatus; | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 334 | if (ha->status_srb == NULL) | 
| Vikas Chaudhary | 09a0f71 | 2010-04-28 11:42:24 +0530 | [diff] [blame] | 335 | kref_put(&srb->srb_ref, qla4xxx_srb_compl); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 336 | } | 
|  | 337 |  | 
|  | 338 | /** | 
|  | 339 | * qla4xxx_process_response_queue - process response queue completions | 
|  | 340 | * @ha: Pointer to host adapter structure. | 
|  | 341 | * | 
|  | 342 | * This routine process response queue completions in interrupt context. | 
|  | 343 | * Hardware_lock locked upon entry | 
|  | 344 | **/ | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 345 | void qla4xxx_process_response_queue(struct scsi_qla_host *ha) | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 346 | { | 
|  | 347 | uint32_t count = 0; | 
|  | 348 | struct srb *srb = NULL; | 
|  | 349 | struct status_entry *sts_entry; | 
|  | 350 |  | 
|  | 351 | /* Process all responses from response queue */ | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 352 | while ((ha->response_ptr->signature != RESPONSE_PROCESSED)) { | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 353 | sts_entry = (struct status_entry *) ha->response_ptr; | 
|  | 354 | count++; | 
|  | 355 |  | 
|  | 356 | /* Advance pointers for next entry */ | 
|  | 357 | if (ha->response_out == (RESPONSE_QUEUE_DEPTH - 1)) { | 
|  | 358 | ha->response_out = 0; | 
|  | 359 | ha->response_ptr = ha->response_ring; | 
|  | 360 | } else { | 
|  | 361 | ha->response_out++; | 
|  | 362 | ha->response_ptr++; | 
|  | 363 | } | 
|  | 364 |  | 
|  | 365 | /* process entry */ | 
|  | 366 | switch (sts_entry->hdr.entryType) { | 
|  | 367 | case ET_STATUS: | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 368 | /* Common status */ | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 369 | qla4xxx_status_entry(ha, sts_entry); | 
|  | 370 | break; | 
|  | 371 |  | 
|  | 372 | case ET_PASSTHRU_STATUS: | 
|  | 373 | break; | 
|  | 374 |  | 
|  | 375 | case ET_STATUS_CONTINUATION: | 
| Karen Higgins | 94bced3 | 2009-07-15 15:02:58 -0500 | [diff] [blame] | 376 | qla4xxx_status_cont_entry(ha, | 
|  | 377 | (struct status_cont_entry *) sts_entry); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 378 | break; | 
|  | 379 |  | 
|  | 380 | case ET_COMMAND: | 
|  | 381 | /* ISP device queue is full. Command not | 
|  | 382 | * accepted by ISP.  Queue command for | 
|  | 383 | * later */ | 
|  | 384 |  | 
|  | 385 | srb = qla4xxx_del_from_active_array(ha, | 
|  | 386 | le32_to_cpu(sts_entry-> | 
|  | 387 | handle)); | 
|  | 388 | if (srb == NULL) | 
|  | 389 | goto exit_prq_invalid_handle; | 
|  | 390 |  | 
|  | 391 | DEBUG2(printk("scsi%ld: %s: FW device queue full, " | 
|  | 392 | "srb %p\n", ha->host_no, __func__, srb)); | 
|  | 393 |  | 
|  | 394 | /* ETRY normally by sending it back with | 
|  | 395 | * DID_BUS_BUSY */ | 
|  | 396 | srb->cmd->result = DID_BUS_BUSY << 16; | 
| Vikas Chaudhary | 09a0f71 | 2010-04-28 11:42:24 +0530 | [diff] [blame] | 397 | kref_put(&srb->srb_ref, qla4xxx_srb_compl); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 398 | break; | 
|  | 399 |  | 
|  | 400 | case ET_CONTINUE: | 
|  | 401 | /* Just throw away the continuation entries */ | 
|  | 402 | DEBUG2(printk("scsi%ld: %s: Continuation entry - " | 
|  | 403 | "ignoring\n", ha->host_no, __func__)); | 
|  | 404 | break; | 
|  | 405 |  | 
|  | 406 | default: | 
|  | 407 | /* | 
|  | 408 | * Invalid entry in response queue, reset RISC | 
|  | 409 | * firmware. | 
|  | 410 | */ | 
|  | 411 | DEBUG2(printk("scsi%ld: %s: Invalid entry %x in " | 
|  | 412 | "response queue \n", ha->host_no, | 
|  | 413 | __func__, | 
|  | 414 | sts_entry->hdr.entryType)); | 
|  | 415 | goto exit_prq_error; | 
|  | 416 | } | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 417 | ((struct response *)sts_entry)->signature = RESPONSE_PROCESSED; | 
|  | 418 | wmb(); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 419 | } | 
|  | 420 |  | 
|  | 421 | /* | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 422 | * Tell ISP we're done with response(s). This also clears the interrupt. | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 423 | */ | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 424 | ha->isp_ops->complete_iocb(ha); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 425 |  | 
|  | 426 | return; | 
|  | 427 |  | 
|  | 428 | exit_prq_invalid_handle: | 
|  | 429 | DEBUG2(printk("scsi%ld: %s: Invalid handle(srb)=%p type=%x IOCS=%x\n", | 
|  | 430 | ha->host_no, __func__, srb, sts_entry->hdr.entryType, | 
|  | 431 | sts_entry->completionStatus)); | 
|  | 432 |  | 
|  | 433 | exit_prq_error: | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 434 | ha->isp_ops->complete_iocb(ha); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 435 | set_bit(DPC_RESET_HA, &ha->dpc_flags); | 
|  | 436 | } | 
|  | 437 |  | 
|  | 438 | /** | 
|  | 439 | * qla4xxx_isr_decode_mailbox - decodes mailbox status | 
|  | 440 | * @ha: Pointer to host adapter structure. | 
|  | 441 | * @mailbox_status: Mailbox status. | 
|  | 442 | * | 
|  | 443 | * This routine decodes the mailbox status during the ISR. | 
|  | 444 | * Hardware_lock locked upon entry. runs in interrupt context. | 
|  | 445 | **/ | 
|  | 446 | static void qla4xxx_isr_decode_mailbox(struct scsi_qla_host * ha, | 
|  | 447 | uint32_t mbox_status) | 
|  | 448 | { | 
|  | 449 | int i; | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 450 | uint32_t mbox_sts[MBOX_AEN_REG_COUNT]; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 451 |  | 
|  | 452 | if ((mbox_status == MBOX_STS_BUSY) || | 
|  | 453 | (mbox_status == MBOX_STS_INTERMEDIATE_COMPLETION) || | 
|  | 454 | (mbox_status >> 12 == MBOX_COMPLETION_STATUS)) { | 
|  | 455 | ha->mbox_status[0] = mbox_status; | 
|  | 456 |  | 
|  | 457 | if (test_bit(AF_MBOX_COMMAND, &ha->flags)) { | 
|  | 458 | /* | 
|  | 459 | * Copy all mailbox registers to a temporary | 
|  | 460 | * location and set mailbox command done flag | 
|  | 461 | */ | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 462 | for (i = 0; i < ha->mbox_status_count; i++) | 
|  | 463 | ha->mbox_status[i] = is_qla8022(ha) | 
|  | 464 | ? readl(&ha->qla4_8xxx_reg->mailbox_out[i]) | 
|  | 465 | : readl(&ha->reg->mailbox[i]); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 466 |  | 
|  | 467 | set_bit(AF_MBOX_COMMAND_DONE, &ha->flags); | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 468 |  | 
|  | 469 | if (test_bit(AF_MBOX_COMMAND_NOPOLL, &ha->flags)) | 
|  | 470 | complete(&ha->mbx_intr_comp); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 471 | } | 
|  | 472 | } else if (mbox_status >> 12 == MBOX_ASYNC_EVENT_STATUS) { | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 473 | for (i = 0; i < MBOX_AEN_REG_COUNT; i++) | 
|  | 474 | mbox_sts[i] = is_qla8022(ha) | 
|  | 475 | ? readl(&ha->qla4_8xxx_reg->mailbox_out[i]) | 
|  | 476 | : readl(&ha->reg->mailbox[i]); | 
|  | 477 |  | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 478 | /* Immediately process the AENs that don't require much work. | 
|  | 479 | * Only queue the database_changed AENs */ | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 480 | if (ha->aen_log.count < MAX_AEN_ENTRIES) { | 
|  | 481 | for (i = 0; i < MBOX_AEN_REG_COUNT; i++) | 
|  | 482 | ha->aen_log.entry[ha->aen_log.count].mbox_sts[i] = | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 483 | mbox_sts[i]; | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 484 | ha->aen_log.count++; | 
|  | 485 | } | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 486 | switch (mbox_status) { | 
|  | 487 | case MBOX_ASTS_SYSTEM_ERROR: | 
|  | 488 | /* Log Mailbox registers */ | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 489 | ql4_printk(KERN_INFO, ha, "%s: System Err\n", __func__); | 
| Karen Higgins | 91a772a | 2010-10-06 22:50:21 -0700 | [diff] [blame] | 490 | qla4xxx_dump_registers(ha); | 
|  | 491 |  | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 492 | if (ql4xdontresethba) { | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 493 | DEBUG2(printk("scsi%ld: %s:Don't Reset HBA\n", | 
|  | 494 | ha->host_no, __func__)); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 495 | } else { | 
|  | 496 | set_bit(AF_GET_CRASH_RECORD, &ha->flags); | 
|  | 497 | set_bit(DPC_RESET_HA, &ha->dpc_flags); | 
|  | 498 | } | 
|  | 499 | break; | 
|  | 500 |  | 
|  | 501 | case MBOX_ASTS_REQUEST_TRANSFER_ERROR: | 
|  | 502 | case MBOX_ASTS_RESPONSE_TRANSFER_ERROR: | 
|  | 503 | case MBOX_ASTS_NVRAM_INVALID: | 
|  | 504 | case MBOX_ASTS_IP_ADDRESS_CHANGED: | 
|  | 505 | case MBOX_ASTS_DHCP_LEASE_EXPIRED: | 
|  | 506 | DEBUG2(printk("scsi%ld: AEN %04x, ERROR Status, " | 
|  | 507 | "Reset HA\n", ha->host_no, mbox_status)); | 
|  | 508 | set_bit(DPC_RESET_HA, &ha->dpc_flags); | 
|  | 509 | break; | 
|  | 510 |  | 
|  | 511 | case MBOX_ASTS_LINK_UP: | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 512 | set_bit(AF_LINK_UP, &ha->flags); | 
| Vikas Chaudhary | 065aa1b | 2010-04-28 11:38:11 +0530 | [diff] [blame] | 513 | if (test_bit(AF_INIT_DONE, &ha->flags)) | 
|  | 514 | set_bit(DPC_LINK_CHANGED, &ha->dpc_flags); | 
|  | 515 |  | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 516 | ql4_printk(KERN_INFO, ha, "%s: LINK UP\n", __func__); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 517 | break; | 
|  | 518 |  | 
|  | 519 | case MBOX_ASTS_LINK_DOWN: | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 520 | clear_bit(AF_LINK_UP, &ha->flags); | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 521 | if (test_bit(AF_INIT_DONE, &ha->flags)) | 
|  | 522 | set_bit(DPC_LINK_CHANGED, &ha->dpc_flags); | 
| Vikas Chaudhary | 065aa1b | 2010-04-28 11:38:11 +0530 | [diff] [blame] | 523 |  | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 524 | ql4_printk(KERN_INFO, ha, "%s: LINK DOWN\n", __func__); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 525 | break; | 
|  | 526 |  | 
|  | 527 | case MBOX_ASTS_HEARTBEAT: | 
|  | 528 | ha->seconds_since_last_heartbeat = 0; | 
|  | 529 | break; | 
|  | 530 |  | 
|  | 531 | case MBOX_ASTS_DHCP_LEASE_ACQUIRED: | 
|  | 532 | DEBUG2(printk("scsi%ld: AEN %04x DHCP LEASE " | 
|  | 533 | "ACQUIRED\n", ha->host_no, mbox_status)); | 
|  | 534 | set_bit(DPC_GET_DHCP_IP_ADDR, &ha->dpc_flags); | 
|  | 535 | break; | 
|  | 536 |  | 
|  | 537 | case MBOX_ASTS_PROTOCOL_STATISTIC_ALARM: | 
|  | 538 | case MBOX_ASTS_SCSI_COMMAND_PDU_REJECTED: /* Target | 
|  | 539 | * mode | 
|  | 540 | * only */ | 
|  | 541 | case MBOX_ASTS_UNSOLICITED_PDU_RECEIVED:  /* Connection mode */ | 
|  | 542 | case MBOX_ASTS_IPSEC_SYSTEM_FATAL_ERROR: | 
|  | 543 | case MBOX_ASTS_SUBNET_STATE_CHANGE: | 
| Prasanna Mumbai | 185f107 | 2011-05-17 23:17:03 -0700 | [diff] [blame^] | 544 | case MBOX_ASTS_DUPLICATE_IP: | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 545 | /* No action */ | 
|  | 546 | DEBUG2(printk("scsi%ld: AEN %04x\n", ha->host_no, | 
|  | 547 | mbox_status)); | 
|  | 548 | break; | 
|  | 549 |  | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 550 | case MBOX_ASTS_IP_ADDR_STATE_CHANGED: | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 551 | printk("scsi%ld: AEN %04x, mbox_sts[2]=%04x, " | 
|  | 552 | "mbox_sts[3]=%04x\n", ha->host_no, mbox_sts[0], | 
|  | 553 | mbox_sts[2], mbox_sts[3]); | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 554 |  | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 555 | /* mbox_sts[2] = Old ACB state | 
|  | 556 | * mbox_sts[3] = new ACB state */ | 
|  | 557 | if ((mbox_sts[3] == ACB_STATE_VALID) && | 
| Prasanna Mumbai | e128271 | 2010-12-02 22:12:43 -0800 | [diff] [blame] | 558 | ((mbox_sts[2] == ACB_STATE_TENTATIVE) || | 
|  | 559 | (mbox_sts[2] == ACB_STATE_ACQUIRING))) | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 560 | set_bit(DPC_GET_DHCP_IP_ADDR, &ha->dpc_flags); | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 561 | else if ((mbox_sts[3] == ACB_STATE_ACQUIRING) && | 
|  | 562 | (mbox_sts[2] == ACB_STATE_VALID)) | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 563 | set_bit(DPC_RESET_HA, &ha->dpc_flags); | 
|  | 564 | break; | 
|  | 565 |  | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 566 | case MBOX_ASTS_MAC_ADDRESS_CHANGED: | 
|  | 567 | case MBOX_ASTS_DNS: | 
|  | 568 | /* No action */ | 
|  | 569 | DEBUG2(printk(KERN_INFO "scsi%ld: AEN %04x, " | 
|  | 570 | "mbox_sts[1]=%04x, mbox_sts[2]=%04x\n", | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 571 | ha->host_no, mbox_sts[0], | 
|  | 572 | mbox_sts[1], mbox_sts[2])); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 573 | break; | 
|  | 574 |  | 
|  | 575 | case MBOX_ASTS_SELF_TEST_FAILED: | 
|  | 576 | case MBOX_ASTS_LOGIN_FAILED: | 
|  | 577 | /* No action */ | 
|  | 578 | DEBUG2(printk("scsi%ld: AEN %04x, mbox_sts[1]=%04x, " | 
|  | 579 | "mbox_sts[2]=%04x, mbox_sts[3]=%04x\n", | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 580 | ha->host_no, mbox_sts[0], mbox_sts[1], | 
|  | 581 | mbox_sts[2], mbox_sts[3])); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 582 | break; | 
|  | 583 |  | 
|  | 584 | case MBOX_ASTS_DATABASE_CHANGED: | 
|  | 585 | /* Queue AEN information and process it in the DPC | 
|  | 586 | * routine */ | 
|  | 587 | if (ha->aen_q_count > 0) { | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 588 |  | 
|  | 589 | /* decrement available counter */ | 
|  | 590 | ha->aen_q_count--; | 
|  | 591 |  | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 592 | for (i = 0; i < MBOX_AEN_REG_COUNT; i++) | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 593 | ha->aen_q[ha->aen_in].mbox_sts[i] = | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 594 | mbox_sts[i]; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 595 |  | 
|  | 596 | /* print debug message */ | 
| Prasanna Mumbai | 185f107 | 2011-05-17 23:17:03 -0700 | [diff] [blame^] | 597 | DEBUG2(printk("scsi%ld: AEN[%d] %04x queued " | 
|  | 598 | "mb1:0x%x mb2:0x%x mb3:0x%x " | 
|  | 599 | "mb4:0x%x mb5:0x%x\n", | 
|  | 600 | ha->host_no, ha->aen_in, | 
|  | 601 | mbox_sts[0], mbox_sts[1], | 
|  | 602 | mbox_sts[2], mbox_sts[3], | 
|  | 603 | mbox_sts[4], mbox_sts[5])); | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 604 |  | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 605 | /* advance pointer */ | 
|  | 606 | ha->aen_in++; | 
|  | 607 | if (ha->aen_in == MAX_AEN_ENTRIES) | 
|  | 608 | ha->aen_in = 0; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 609 |  | 
|  | 610 | /* The DPC routine will process the aen */ | 
|  | 611 | set_bit(DPC_AEN, &ha->dpc_flags); | 
|  | 612 | } else { | 
|  | 613 | DEBUG2(printk("scsi%ld: %s: aen %04x, queue " | 
|  | 614 | "overflowed!  AEN LOST!!\n", | 
|  | 615 | ha->host_no, __func__, | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 616 | mbox_sts[0])); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 617 |  | 
|  | 618 | DEBUG2(printk("scsi%ld: DUMP AEN QUEUE\n", | 
|  | 619 | ha->host_no)); | 
|  | 620 |  | 
|  | 621 | for (i = 0; i < MAX_AEN_ENTRIES; i++) { | 
|  | 622 | DEBUG2(printk("AEN[%d] %04x %04x %04x " | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 623 | "%04x\n", i, mbox_sts[0], | 
|  | 624 | mbox_sts[1], mbox_sts[2], | 
|  | 625 | mbox_sts[3])); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 626 | } | 
|  | 627 | } | 
|  | 628 | break; | 
|  | 629 |  | 
| Shyam Sundar | 6434080 | 2010-10-06 22:49:40 -0700 | [diff] [blame] | 630 | case MBOX_ASTS_TXSCVR_INSERTED: | 
|  | 631 | DEBUG2(printk(KERN_WARNING | 
|  | 632 | "scsi%ld: AEN %04x Transceiver" | 
|  | 633 | " inserted\n",  ha->host_no, mbox_sts[0])); | 
|  | 634 | break; | 
|  | 635 |  | 
|  | 636 | case MBOX_ASTS_TXSCVR_REMOVED: | 
|  | 637 | DEBUG2(printk(KERN_WARNING | 
|  | 638 | "scsi%ld: AEN %04x Transceiver" | 
|  | 639 | " removed\n",  ha->host_no, mbox_sts[0])); | 
|  | 640 | break; | 
|  | 641 |  | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 642 | default: | 
|  | 643 | DEBUG2(printk(KERN_WARNING | 
|  | 644 | "scsi%ld: AEN %04x UNKNOWN\n", | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 645 | ha->host_no, mbox_sts[0])); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 646 | break; | 
|  | 647 | } | 
|  | 648 | } else { | 
|  | 649 | DEBUG2(printk("scsi%ld: Unknown mailbox status %08X\n", | 
|  | 650 | ha->host_no, mbox_status)); | 
|  | 651 |  | 
|  | 652 | ha->mbox_status[0] = mbox_status; | 
|  | 653 | } | 
|  | 654 | } | 
|  | 655 |  | 
|  | 656 | /** | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 657 | * qla4_8xxx_interrupt_service_routine - isr | 
|  | 658 | * @ha: pointer to host adapter structure. | 
|  | 659 | * | 
|  | 660 | * This is the main interrupt service routine. | 
|  | 661 | * hardware_lock locked upon entry. runs in interrupt context. | 
|  | 662 | **/ | 
|  | 663 | void qla4_8xxx_interrupt_service_routine(struct scsi_qla_host *ha, | 
|  | 664 | uint32_t intr_status) | 
|  | 665 | { | 
|  | 666 | /* Process response queue interrupt. */ | 
|  | 667 | if (intr_status & HSRX_RISC_IOCB_INT) | 
|  | 668 | qla4xxx_process_response_queue(ha); | 
|  | 669 |  | 
|  | 670 | /* Process mailbox/asynch event interrupt.*/ | 
|  | 671 | if (intr_status & HSRX_RISC_MB_INT) | 
|  | 672 | qla4xxx_isr_decode_mailbox(ha, | 
|  | 673 | readl(&ha->qla4_8xxx_reg->mailbox_out[0])); | 
|  | 674 |  | 
|  | 675 | /* clear the interrupt */ | 
|  | 676 | writel(0, &ha->qla4_8xxx_reg->host_int); | 
|  | 677 | readl(&ha->qla4_8xxx_reg->host_int); | 
|  | 678 | } | 
|  | 679 |  | 
|  | 680 | /** | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 681 | * qla4xxx_interrupt_service_routine - isr | 
|  | 682 | * @ha: pointer to host adapter structure. | 
|  | 683 | * | 
|  | 684 | * This is the main interrupt service routine. | 
|  | 685 | * hardware_lock locked upon entry. runs in interrupt context. | 
|  | 686 | **/ | 
|  | 687 | void qla4xxx_interrupt_service_routine(struct scsi_qla_host * ha, | 
|  | 688 | uint32_t intr_status) | 
|  | 689 | { | 
|  | 690 | /* Process response queue interrupt. */ | 
|  | 691 | if (intr_status & CSR_SCSI_COMPLETION_INTR) | 
|  | 692 | qla4xxx_process_response_queue(ha); | 
|  | 693 |  | 
|  | 694 | /* Process mailbox/asynch event	 interrupt.*/ | 
|  | 695 | if (intr_status & CSR_SCSI_PROCESSOR_INTR) { | 
|  | 696 | qla4xxx_isr_decode_mailbox(ha, | 
|  | 697 | readl(&ha->reg->mailbox[0])); | 
|  | 698 |  | 
|  | 699 | /* Clear Mailbox Interrupt */ | 
|  | 700 | writel(set_rmask(CSR_SCSI_PROCESSOR_INTR), | 
|  | 701 | &ha->reg->ctrl_status); | 
|  | 702 | readl(&ha->reg->ctrl_status); | 
|  | 703 | } | 
|  | 704 | } | 
|  | 705 |  | 
|  | 706 | /** | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 707 | * qla4_8xxx_spurious_interrupt - processes spurious interrupt | 
|  | 708 | * @ha: pointer to host adapter structure. | 
|  | 709 | * @reqs_count: . | 
|  | 710 | * | 
|  | 711 | **/ | 
|  | 712 | static void qla4_8xxx_spurious_interrupt(struct scsi_qla_host *ha, | 
|  | 713 | uint8_t reqs_count) | 
|  | 714 | { | 
|  | 715 | if (reqs_count) | 
|  | 716 | return; | 
|  | 717 |  | 
|  | 718 | DEBUG2(ql4_printk(KERN_INFO, ha, "Spurious Interrupt\n")); | 
|  | 719 | if (is_qla8022(ha)) { | 
|  | 720 | writel(0, &ha->qla4_8xxx_reg->host_int); | 
|  | 721 | if (test_bit(AF_INTx_ENABLED, &ha->flags)) | 
|  | 722 | qla4_8xxx_wr_32(ha, ha->nx_legacy_intr.tgt_mask_reg, | 
|  | 723 | 0xfbff); | 
|  | 724 | } | 
|  | 725 | ha->spurious_int_count++; | 
|  | 726 | } | 
|  | 727 |  | 
|  | 728 | /** | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 729 | * qla4xxx_intr_handler - hardware interrupt handler. | 
|  | 730 | * @irq: Unused | 
|  | 731 | * @dev_id: Pointer to host adapter structure | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 732 | **/ | 
| David Howells | 7d12e78 | 2006-10-05 14:55:46 +0100 | [diff] [blame] | 733 | irqreturn_t qla4xxx_intr_handler(int irq, void *dev_id) | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 734 | { | 
|  | 735 | struct scsi_qla_host *ha; | 
|  | 736 | uint32_t intr_status; | 
|  | 737 | unsigned long flags = 0; | 
|  | 738 | uint8_t reqs_count = 0; | 
|  | 739 |  | 
|  | 740 | ha = (struct scsi_qla_host *) dev_id; | 
|  | 741 | if (!ha) { | 
|  | 742 | DEBUG2(printk(KERN_INFO | 
|  | 743 | "qla4xxx: Interrupt with NULL host ptr\n")); | 
|  | 744 | return IRQ_NONE; | 
|  | 745 | } | 
|  | 746 |  | 
|  | 747 | spin_lock_irqsave(&ha->hardware_lock, flags); | 
|  | 748 |  | 
| David C Somayajulu | d915058 | 2006-11-15 17:38:40 -0800 | [diff] [blame] | 749 | ha->isr_count++; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 750 | /* | 
|  | 751 | * Repeatedly service interrupts up to a maximum of | 
|  | 752 | * MAX_REQS_SERVICED_PER_INTR | 
|  | 753 | */ | 
|  | 754 | while (1) { | 
|  | 755 | /* | 
|  | 756 | * Read interrupt status | 
|  | 757 | */ | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 758 | if (ha->isp_ops->rd_shdw_rsp_q_in(ha) != | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 759 | ha->response_out) | 
|  | 760 | intr_status = CSR_SCSI_COMPLETION_INTR; | 
|  | 761 | else | 
|  | 762 | intr_status = readl(&ha->reg->ctrl_status); | 
|  | 763 |  | 
|  | 764 | if ((intr_status & | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 765 | (CSR_SCSI_RESET_INTR|CSR_FATAL_ERROR|INTR_PENDING)) == 0) { | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 766 | if (reqs_count == 0) | 
|  | 767 | ha->spurious_int_count++; | 
|  | 768 | break; | 
|  | 769 | } | 
|  | 770 |  | 
|  | 771 | if (intr_status & CSR_FATAL_ERROR) { | 
|  | 772 | DEBUG2(printk(KERN_INFO "scsi%ld: Fatal Error, " | 
|  | 773 | "Status 0x%04x\n", ha->host_no, | 
|  | 774 | readl(isp_port_error_status (ha)))); | 
|  | 775 |  | 
|  | 776 | /* Issue Soft Reset to clear this error condition. | 
|  | 777 | * This will prevent the RISC from repeatedly | 
|  | 778 | * interrupting the driver; thus, allowing the DPC to | 
|  | 779 | * get scheduled to continue error recovery. | 
|  | 780 | * NOTE: Disabling RISC interrupts does not work in | 
|  | 781 | * this case, as CSR_FATAL_ERROR overrides | 
|  | 782 | * CSR_SCSI_INTR_ENABLE */ | 
|  | 783 | if ((readl(&ha->reg->ctrl_status) & | 
|  | 784 | CSR_SCSI_RESET_INTR) == 0) { | 
|  | 785 | writel(set_rmask(CSR_SOFT_RESET), | 
|  | 786 | &ha->reg->ctrl_status); | 
|  | 787 | readl(&ha->reg->ctrl_status); | 
|  | 788 | } | 
|  | 789 |  | 
|  | 790 | writel(set_rmask(CSR_FATAL_ERROR), | 
|  | 791 | &ha->reg->ctrl_status); | 
|  | 792 | readl(&ha->reg->ctrl_status); | 
|  | 793 |  | 
|  | 794 | __qla4xxx_disable_intrs(ha); | 
|  | 795 |  | 
|  | 796 | set_bit(DPC_RESET_HA, &ha->dpc_flags); | 
|  | 797 |  | 
|  | 798 | break; | 
|  | 799 | } else if (intr_status & CSR_SCSI_RESET_INTR) { | 
|  | 800 | clear_bit(AF_ONLINE, &ha->flags); | 
|  | 801 | __qla4xxx_disable_intrs(ha); | 
|  | 802 |  | 
|  | 803 | writel(set_rmask(CSR_SCSI_RESET_INTR), | 
|  | 804 | &ha->reg->ctrl_status); | 
|  | 805 | readl(&ha->reg->ctrl_status); | 
|  | 806 |  | 
| Karen Higgins | 7eece5a | 2011-03-21 03:34:29 -0700 | [diff] [blame] | 807 | if (!test_bit(AF_HA_REMOVAL, &ha->flags)) | 
| David C Somayajulu | 477ffb9 | 2007-01-22 12:26:11 -0800 | [diff] [blame] | 808 | set_bit(DPC_RESET_HA_INTR, &ha->dpc_flags); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 809 |  | 
|  | 810 | break; | 
|  | 811 | } else if (intr_status & INTR_PENDING) { | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 812 | ha->isp_ops->interrupt_service_routine(ha, intr_status); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 813 | ha->total_io_count++; | 
|  | 814 | if (++reqs_count == MAX_REQS_SERVICED_PER_INTR) | 
|  | 815 | break; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 816 | } | 
|  | 817 | } | 
|  | 818 |  | 
|  | 819 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | 
|  | 820 |  | 
|  | 821 | return IRQ_HANDLED; | 
|  | 822 | } | 
|  | 823 |  | 
|  | 824 | /** | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 825 | * qla4_8xxx_intr_handler - hardware interrupt handler. | 
|  | 826 | * @irq: Unused | 
|  | 827 | * @dev_id: Pointer to host adapter structure | 
|  | 828 | **/ | 
|  | 829 | irqreturn_t qla4_8xxx_intr_handler(int irq, void *dev_id) | 
|  | 830 | { | 
|  | 831 | struct scsi_qla_host *ha = dev_id; | 
|  | 832 | uint32_t intr_status; | 
|  | 833 | uint32_t status; | 
|  | 834 | unsigned long flags = 0; | 
|  | 835 | uint8_t reqs_count = 0; | 
|  | 836 |  | 
| Lalit Chandivade | 2232be0 | 2010-07-30 14:38:47 +0530 | [diff] [blame] | 837 | if (unlikely(pci_channel_offline(ha->pdev))) | 
|  | 838 | return IRQ_HANDLED; | 
|  | 839 |  | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 840 | ha->isr_count++; | 
|  | 841 | status = qla4_8xxx_rd_32(ha, ISR_INT_VECTOR); | 
|  | 842 | if (!(status & ha->nx_legacy_intr.int_vec_bit)) | 
|  | 843 | return IRQ_NONE; | 
|  | 844 |  | 
|  | 845 | status = qla4_8xxx_rd_32(ha, ISR_INT_STATE_REG); | 
|  | 846 | if (!ISR_IS_LEGACY_INTR_TRIGGERED(status)) { | 
|  | 847 | DEBUG2(ql4_printk(KERN_INFO, ha, | 
|  | 848 | "%s legacy Int not triggered\n", __func__)); | 
|  | 849 | return IRQ_NONE; | 
|  | 850 | } | 
|  | 851 |  | 
|  | 852 | /* clear the interrupt */ | 
|  | 853 | qla4_8xxx_wr_32(ha, ha->nx_legacy_intr.tgt_status_reg, 0xffffffff); | 
|  | 854 |  | 
|  | 855 | /* read twice to ensure write is flushed */ | 
|  | 856 | qla4_8xxx_rd_32(ha, ISR_INT_VECTOR); | 
|  | 857 | qla4_8xxx_rd_32(ha, ISR_INT_VECTOR); | 
|  | 858 |  | 
|  | 859 | spin_lock_irqsave(&ha->hardware_lock, flags); | 
|  | 860 | while (1) { | 
|  | 861 | if (!(readl(&ha->qla4_8xxx_reg->host_int) & | 
|  | 862 | ISRX_82XX_RISC_INT)) { | 
|  | 863 | qla4_8xxx_spurious_interrupt(ha, reqs_count); | 
|  | 864 | break; | 
|  | 865 | } | 
|  | 866 | intr_status =  readl(&ha->qla4_8xxx_reg->host_status); | 
|  | 867 | if ((intr_status & | 
|  | 868 | (HSRX_RISC_MB_INT | HSRX_RISC_IOCB_INT)) == 0)  { | 
|  | 869 | qla4_8xxx_spurious_interrupt(ha, reqs_count); | 
|  | 870 | break; | 
|  | 871 | } | 
|  | 872 |  | 
|  | 873 | ha->isp_ops->interrupt_service_routine(ha, intr_status); | 
|  | 874 |  | 
|  | 875 | /* Enable Interrupt */ | 
|  | 876 | qla4_8xxx_wr_32(ha, ha->nx_legacy_intr.tgt_mask_reg, 0xfbff); | 
|  | 877 |  | 
|  | 878 | if (++reqs_count == MAX_REQS_SERVICED_PER_INTR) | 
|  | 879 | break; | 
|  | 880 | } | 
|  | 881 |  | 
|  | 882 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | 
|  | 883 | return IRQ_HANDLED; | 
|  | 884 | } | 
|  | 885 |  | 
|  | 886 | irqreturn_t | 
|  | 887 | qla4_8xxx_msi_handler(int irq, void *dev_id) | 
|  | 888 | { | 
|  | 889 | struct scsi_qla_host *ha; | 
|  | 890 |  | 
|  | 891 | ha = (struct scsi_qla_host *) dev_id; | 
|  | 892 | if (!ha) { | 
|  | 893 | DEBUG2(printk(KERN_INFO | 
|  | 894 | "qla4xxx: MSIX: Interrupt with NULL host ptr\n")); | 
|  | 895 | return IRQ_NONE; | 
|  | 896 | } | 
|  | 897 |  | 
|  | 898 | ha->isr_count++; | 
|  | 899 | /* clear the interrupt */ | 
|  | 900 | qla4_8xxx_wr_32(ha, ha->nx_legacy_intr.tgt_status_reg, 0xffffffff); | 
|  | 901 |  | 
|  | 902 | /* read twice to ensure write is flushed */ | 
|  | 903 | qla4_8xxx_rd_32(ha, ISR_INT_VECTOR); | 
|  | 904 | qla4_8xxx_rd_32(ha, ISR_INT_VECTOR); | 
|  | 905 |  | 
|  | 906 | return qla4_8xxx_default_intr_handler(irq, dev_id); | 
|  | 907 | } | 
|  | 908 |  | 
|  | 909 | /** | 
|  | 910 | * qla4_8xxx_default_intr_handler - hardware interrupt handler. | 
|  | 911 | * @irq: Unused | 
|  | 912 | * @dev_id: Pointer to host adapter structure | 
|  | 913 | * | 
|  | 914 | * This interrupt handler is called directly for MSI-X, and | 
|  | 915 | * called indirectly for MSI. | 
|  | 916 | **/ | 
|  | 917 | irqreturn_t | 
|  | 918 | qla4_8xxx_default_intr_handler(int irq, void *dev_id) | 
|  | 919 | { | 
|  | 920 | struct scsi_qla_host *ha = dev_id; | 
|  | 921 | unsigned long   flags; | 
|  | 922 | uint32_t intr_status; | 
|  | 923 | uint8_t reqs_count = 0; | 
|  | 924 |  | 
|  | 925 | spin_lock_irqsave(&ha->hardware_lock, flags); | 
|  | 926 | while (1) { | 
|  | 927 | if (!(readl(&ha->qla4_8xxx_reg->host_int) & | 
|  | 928 | ISRX_82XX_RISC_INT)) { | 
|  | 929 | qla4_8xxx_spurious_interrupt(ha, reqs_count); | 
|  | 930 | break; | 
|  | 931 | } | 
|  | 932 |  | 
|  | 933 | intr_status =  readl(&ha->qla4_8xxx_reg->host_status); | 
|  | 934 | if ((intr_status & | 
|  | 935 | (HSRX_RISC_MB_INT | HSRX_RISC_IOCB_INT)) == 0) { | 
|  | 936 | qla4_8xxx_spurious_interrupt(ha, reqs_count); | 
|  | 937 | break; | 
|  | 938 | } | 
|  | 939 |  | 
|  | 940 | ha->isp_ops->interrupt_service_routine(ha, intr_status); | 
|  | 941 |  | 
|  | 942 | if (++reqs_count == MAX_REQS_SERVICED_PER_INTR) | 
|  | 943 | break; | 
|  | 944 | } | 
|  | 945 |  | 
|  | 946 | ha->isr_count++; | 
|  | 947 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | 
|  | 948 | return IRQ_HANDLED; | 
|  | 949 | } | 
|  | 950 |  | 
|  | 951 | irqreturn_t | 
|  | 952 | qla4_8xxx_msix_rsp_q(int irq, void *dev_id) | 
|  | 953 | { | 
|  | 954 | struct scsi_qla_host *ha = dev_id; | 
|  | 955 | unsigned long flags; | 
|  | 956 |  | 
|  | 957 | spin_lock_irqsave(&ha->hardware_lock, flags); | 
|  | 958 | qla4xxx_process_response_queue(ha); | 
|  | 959 | writel(0, &ha->qla4_8xxx_reg->host_int); | 
|  | 960 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | 
|  | 961 |  | 
|  | 962 | ha->isr_count++; | 
|  | 963 | return IRQ_HANDLED; | 
|  | 964 | } | 
|  | 965 |  | 
|  | 966 | /** | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 967 | * qla4xxx_process_aen - processes AENs generated by firmware | 
|  | 968 | * @ha: pointer to host adapter structure. | 
|  | 969 | * @process_aen: type of AENs to process | 
|  | 970 | * | 
|  | 971 | * Processes specific types of Asynchronous Events generated by firmware. | 
|  | 972 | * The type of AENs to process is specified by process_aen and can be | 
|  | 973 | *	PROCESS_ALL_AENS	 0 | 
|  | 974 | *	FLUSH_DDB_CHANGED_AENS	 1 | 
|  | 975 | *	RELOGIN_DDB_CHANGED_AENS 2 | 
|  | 976 | **/ | 
|  | 977 | void qla4xxx_process_aen(struct scsi_qla_host * ha, uint8_t process_aen) | 
|  | 978 | { | 
|  | 979 | uint32_t mbox_sts[MBOX_AEN_REG_COUNT]; | 
|  | 980 | struct aen *aen; | 
|  | 981 | int i; | 
|  | 982 | unsigned long flags; | 
|  | 983 |  | 
|  | 984 | spin_lock_irqsave(&ha->hardware_lock, flags); | 
|  | 985 | while (ha->aen_out != ha->aen_in) { | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 986 | aen = &ha->aen_q[ha->aen_out]; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 987 | /* copy aen information to local structure */ | 
|  | 988 | for (i = 0; i < MBOX_AEN_REG_COUNT; i++) | 
|  | 989 | mbox_sts[i] = aen->mbox_sts[i]; | 
|  | 990 |  | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 991 | ha->aen_q_count++; | 
|  | 992 | ha->aen_out++; | 
|  | 993 |  | 
|  | 994 | if (ha->aen_out == MAX_AEN_ENTRIES) | 
|  | 995 | ha->aen_out = 0; | 
|  | 996 |  | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 997 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | 
|  | 998 |  | 
| David C Somayajulu | 401425b | 2007-05-23 18:03:20 -0700 | [diff] [blame] | 999 | DEBUG2(printk("qla4xxx(%ld): AEN[%d]=0x%08x, mbx1=0x%08x mbx2=0x%08x" | 
|  | 1000 | " mbx3=0x%08x mbx4=0x%08x\n", ha->host_no, | 
|  | 1001 | (ha->aen_out ? (ha->aen_out-1): (MAX_AEN_ENTRIES-1)), | 
|  | 1002 | mbox_sts[0], mbox_sts[1], mbox_sts[2], | 
|  | 1003 | mbox_sts[3], mbox_sts[4])); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 1004 |  | 
|  | 1005 | switch (mbox_sts[0]) { | 
|  | 1006 | case MBOX_ASTS_DATABASE_CHANGED: | 
|  | 1007 | if (process_aen == FLUSH_DDB_CHANGED_AENS) { | 
|  | 1008 | DEBUG2(printk("scsi%ld: AEN[%d] %04x, index " | 
|  | 1009 | "[%d] state=%04x FLUSHED!\n", | 
|  | 1010 | ha->host_no, ha->aen_out, | 
|  | 1011 | mbox_sts[0], mbox_sts[2], | 
|  | 1012 | mbox_sts[3])); | 
|  | 1013 | break; | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 1014 | } | 
| Karen Higgins | 7edd9a7 | 2011-03-21 03:34:27 -0700 | [diff] [blame] | 1015 | case PROCESS_ALL_AENS: | 
|  | 1016 | default: | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 1017 | if (mbox_sts[1] == 0) {	/* Global DB change. */ | 
|  | 1018 | qla4xxx_reinitialize_ddb_list(ha); | 
|  | 1019 | } else if (mbox_sts[1] == 1) {	/* Specific device. */ | 
|  | 1020 | qla4xxx_process_ddb_changed(ha, mbox_sts[2], | 
| Vikas Chaudhary | 821d6e5 | 2010-04-28 11:41:21 +0530 | [diff] [blame] | 1021 | mbox_sts[3], mbox_sts[4]); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 1022 | } | 
|  | 1023 | break; | 
|  | 1024 | } | 
|  | 1025 | spin_lock_irqsave(&ha->hardware_lock, flags); | 
|  | 1026 | } | 
|  | 1027 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | 
| David Somayajulu | afaf5a2 | 2006-09-19 10:28:00 -0700 | [diff] [blame] | 1028 | } | 
|  | 1029 |  | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 1030 | int qla4xxx_request_irqs(struct scsi_qla_host *ha) | 
|  | 1031 | { | 
|  | 1032 | int ret; | 
|  | 1033 |  | 
|  | 1034 | if (!is_qla8022(ha)) | 
|  | 1035 | goto try_intx; | 
|  | 1036 |  | 
|  | 1037 | if (ql4xenablemsix == 2) | 
|  | 1038 | goto try_msi; | 
|  | 1039 |  | 
|  | 1040 | if (ql4xenablemsix == 0 || ql4xenablemsix != 1) | 
|  | 1041 | goto try_intx; | 
|  | 1042 |  | 
|  | 1043 | /* Trying MSI-X */ | 
|  | 1044 | ret = qla4_8xxx_enable_msix(ha); | 
|  | 1045 | if (!ret) { | 
|  | 1046 | DEBUG2(ql4_printk(KERN_INFO, ha, | 
|  | 1047 | "MSI-X: Enabled (0x%X).\n", ha->revision_id)); | 
|  | 1048 | goto irq_attached; | 
|  | 1049 | } | 
|  | 1050 |  | 
|  | 1051 | ql4_printk(KERN_WARNING, ha, | 
|  | 1052 | "MSI-X: Falling back-to MSI mode -- %d.\n", ret); | 
|  | 1053 |  | 
|  | 1054 | try_msi: | 
|  | 1055 | /* Trying MSI */ | 
|  | 1056 | ret = pci_enable_msi(ha->pdev); | 
|  | 1057 | if (!ret) { | 
|  | 1058 | ret = request_irq(ha->pdev->irq, qla4_8xxx_msi_handler, | 
| Shyam Sundar | 61391d3 | 2010-12-02 22:12:08 -0800 | [diff] [blame] | 1059 | 0, DRIVER_NAME, ha); | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 1060 | if (!ret) { | 
|  | 1061 | DEBUG2(ql4_printk(KERN_INFO, ha, "MSI: Enabled.\n")); | 
|  | 1062 | set_bit(AF_MSI_ENABLED, &ha->flags); | 
|  | 1063 | goto irq_attached; | 
|  | 1064 | } else { | 
|  | 1065 | ql4_printk(KERN_WARNING, ha, | 
|  | 1066 | "MSI: Failed to reserve interrupt %d " | 
|  | 1067 | "already in use.\n", ha->pdev->irq); | 
|  | 1068 | pci_disable_msi(ha->pdev); | 
|  | 1069 | } | 
|  | 1070 | } | 
|  | 1071 | ql4_printk(KERN_WARNING, ha, | 
|  | 1072 | "MSI: Falling back-to INTx mode -- %d.\n", ret); | 
|  | 1073 |  | 
|  | 1074 | try_intx: | 
|  | 1075 | /* Trying INTx */ | 
|  | 1076 | ret = request_irq(ha->pdev->irq, ha->isp_ops->intr_handler, | 
| Vikas Chaudhary | 3e1350c | 2010-12-02 22:12:03 -0800 | [diff] [blame] | 1077 | IRQF_SHARED, DRIVER_NAME, ha); | 
| Vikas Chaudhary | f4f5df23 | 2010-07-28 15:53:44 +0530 | [diff] [blame] | 1078 | if (!ret) { | 
|  | 1079 | DEBUG2(ql4_printk(KERN_INFO, ha, "INTx: Enabled.\n")); | 
|  | 1080 | set_bit(AF_INTx_ENABLED, &ha->flags); | 
|  | 1081 | goto irq_attached; | 
|  | 1082 |  | 
|  | 1083 | } else { | 
|  | 1084 | ql4_printk(KERN_WARNING, ha, | 
|  | 1085 | "INTx: Failed to reserve interrupt %d already in" | 
|  | 1086 | " use.\n", ha->pdev->irq); | 
|  | 1087 | return ret; | 
|  | 1088 | } | 
|  | 1089 |  | 
|  | 1090 | irq_attached: | 
|  | 1091 | set_bit(AF_IRQ_ATTACHED, &ha->flags); | 
|  | 1092 | ha->host->irq = ha->pdev->irq; | 
|  | 1093 | ql4_printk(KERN_INFO, ha, "%s: irq %d attached\n", | 
|  | 1094 | __func__, ha->pdev->irq); | 
|  | 1095 | return ret; | 
|  | 1096 | } | 
|  | 1097 |  | 
|  | 1098 | void qla4xxx_free_irqs(struct scsi_qla_host *ha) | 
|  | 1099 | { | 
|  | 1100 | if (test_bit(AF_MSIX_ENABLED, &ha->flags)) | 
|  | 1101 | qla4_8xxx_disable_msix(ha); | 
|  | 1102 | else if (test_and_clear_bit(AF_MSI_ENABLED, &ha->flags)) { | 
|  | 1103 | free_irq(ha->pdev->irq, ha); | 
|  | 1104 | pci_disable_msi(ha->pdev); | 
|  | 1105 | } else if (test_and_clear_bit(AF_INTx_ENABLED, &ha->flags)) | 
|  | 1106 | free_irq(ha->pdev->irq, ha); | 
|  | 1107 | } |