qla_isr.c 83.9 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
A
Andrew Vasquez 已提交
2
 * QLogic Fibre Channel HBA Driver
3
 * Copyright (c)  2003-2014 QLogic Corporation
L
Linus Torvalds 已提交
4
 *
A
Andrew Vasquez 已提交
5
 * See LICENSE.qla2xxx for copyright and licensing details.
L
Linus Torvalds 已提交
6 7
 */
#include "qla_def.h"
8
#include "qla_target.h"
L
Linus Torvalds 已提交
9

10
#include <linux/delay.h>
11
#include <linux/slab.h>
12
#include <scsi/scsi_tcq.h>
13
#include <scsi/scsi_bsg_fc.h>
14
#include <scsi/scsi_eh.h>
15

L
Linus Torvalds 已提交
16
static void qla2x00_mbx_completion(scsi_qla_host_t *, uint16_t);
17
static void qla2x00_status_entry(scsi_qla_host_t *, struct rsp_que *, void *);
18
static void qla2x00_status_cont_entry(struct rsp_que *, sts_cont_entry_t *);
19 20
static void qla2x00_error_entry(scsi_qla_host_t *, struct rsp_que *,
	sts_entry_t *);
21

L
Linus Torvalds 已提交
22 23 24 25 26 27 28 29 30 31
/**
 * qla2100_intr_handler() - Process interrupts for the ISP2100 and ISP2200.
 * @irq:
 * @dev_id: SCSI driver HA context
 *
 * Called by system whenever the host adapter generates an interrupt.
 *
 * Returns handled flag.
 */
irqreturn_t
32
qla2100_intr_handler(int irq, void *dev_id)
L
Linus Torvalds 已提交
33
{
34 35
	scsi_qla_host_t	*vha;
	struct qla_hw_data *ha;
36
	struct device_reg_2xxx __iomem *reg;
L
Linus Torvalds 已提交
37 38
	int		status;
	unsigned long	iter;
39
	uint16_t	hccr;
40
	uint16_t	mb[4];
41
	struct rsp_que *rsp;
42
	unsigned long	flags;
L
Linus Torvalds 已提交
43

44 45
	rsp = (struct rsp_que *) dev_id;
	if (!rsp) {
46 47
		ql_log(ql_log_info, NULL, 0x505d,
		    "%s: NULL response queue pointer.\n", __func__);
L
Linus Torvalds 已提交
48 49 50
		return (IRQ_NONE);
	}

51
	ha = rsp->hw;
52
	reg = &ha->iobase->isp;
L
Linus Torvalds 已提交
53 54
	status = 0;

55
	spin_lock_irqsave(&ha->hardware_lock, flags);
56
	vha = pci_get_drvdata(ha->pdev);
L
Linus Torvalds 已提交
57
	for (iter = 50; iter--; ) {
58
		hccr = RD_REG_WORD(&reg->hccr);
59
		if (qla2x00_check_reg16_for_disconnect(vha, hccr))
60
			break;
61 62 63 64 65 66
		if (hccr & HCCR_RISC_PAUSE) {
			if (pci_channel_offline(ha->pdev))
				break;

			/*
			 * Issue a "HARD" reset in order for the RISC interrupt
67
			 * bit to be cleared.  Schedule a big hammer to get
68 69 70 71 72
			 * out of the RISC PAUSED state.
			 */
			WRT_REG_WORD(&reg->hccr, HCCR_RESET_RISC);
			RD_REG_WORD(&reg->hccr);

73 74
			ha->isp_ops->fw_dump(vha, 1);
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
75 76
			break;
		} else if ((RD_REG_WORD(&reg->istatus) & ISR_RISC_INT) == 0)
L
Linus Torvalds 已提交
77 78 79 80 81 82 83
			break;

		if (RD_REG_WORD(&reg->semaphore) & BIT_0) {
			WRT_REG_WORD(&reg->hccr, HCCR_CLR_RISC_INT);
			RD_REG_WORD(&reg->hccr);

			/* Get mailbox data. */
84 85
			mb[0] = RD_MAILBOX_REG(ha, reg, 0);
			if (mb[0] > 0x3fff && mb[0] < 0x8000) {
86
				qla2x00_mbx_completion(vha, mb[0]);
L
Linus Torvalds 已提交
87
				status |= MBX_INTERRUPT;
88 89 90 91
			} else if (mb[0] > 0x7fff && mb[0] < 0xc000) {
				mb[1] = RD_MAILBOX_REG(ha, reg, 1);
				mb[2] = RD_MAILBOX_REG(ha, reg, 2);
				mb[3] = RD_MAILBOX_REG(ha, reg, 3);
92
				qla2x00_async_event(vha, rsp, mb);
L
Linus Torvalds 已提交
93 94
			} else {
				/*EMPTY*/
95 96 97
				ql_dbg(ql_dbg_async, vha, 0x5025,
				    "Unrecognized interrupt type (%d).\n",
				    mb[0]);
L
Linus Torvalds 已提交
98 99 100 101 102
			}
			/* Release mailbox registers. */
			WRT_REG_WORD(&reg->semaphore, 0);
			RD_REG_WORD(&reg->semaphore);
		} else {
103
			qla2x00_process_response_queue(rsp);
L
Linus Torvalds 已提交
104 105 106 107 108

			WRT_REG_WORD(&reg->hccr, HCCR_CLR_RISC_INT);
			RD_REG_WORD(&reg->hccr);
		}
	}
109
	qla2x00_handle_mbx_completion(ha, status);
110
	spin_unlock_irqrestore(&ha->hardware_lock, flags);
L
Linus Torvalds 已提交
111 112 113 114

	return (IRQ_HANDLED);
}

115
bool
116
qla2x00_check_reg32_for_disconnect(scsi_qla_host_t *vha, uint32_t reg)
117 118 119
{
	/* Check for PCI disconnection */
	if (reg == 0xffffffff) {
120 121
		if (!test_and_set_bit(PFLG_DISCONNECTED, &vha->pci_flags) &&
		    !test_bit(PFLG_DRIVER_REMOVING, &vha->pci_flags)) {
122 123 124 125 126 127 128
			/*
			 * Schedule this (only once) on the default system
			 * workqueue so that all the adapter workqueues and the
			 * DPC thread can be shutdown cleanly.
			 */
			schedule_work(&vha->hw->board_disable);
		}
129 130 131 132 133
		return true;
	} else
		return false;
}

134 135 136 137 138 139
bool
qla2x00_check_reg16_for_disconnect(scsi_qla_host_t *vha, uint16_t reg)
{
	return qla2x00_check_reg32_for_disconnect(vha, 0xffff0000 | reg);
}

L
Linus Torvalds 已提交
140 141 142 143 144 145 146 147 148 149
/**
 * qla2300_intr_handler() - Process interrupts for the ISP23xx and ISP63xx.
 * @irq:
 * @dev_id: SCSI driver HA context
 *
 * Called by system whenever the host adapter generates an interrupt.
 *
 * Returns handled flag.
 */
irqreturn_t
150
qla2300_intr_handler(int irq, void *dev_id)
L
Linus Torvalds 已提交
151
{
152
	scsi_qla_host_t	*vha;
153
	struct device_reg_2xxx __iomem *reg;
L
Linus Torvalds 已提交
154 155 156 157
	int		status;
	unsigned long	iter;
	uint32_t	stat;
	uint16_t	hccr;
158
	uint16_t	mb[4];
159 160
	struct rsp_que *rsp;
	struct qla_hw_data *ha;
161
	unsigned long	flags;
L
Linus Torvalds 已提交
162

163 164
	rsp = (struct rsp_que *) dev_id;
	if (!rsp) {
165 166
		ql_log(ql_log_info, NULL, 0x5058,
		    "%s: NULL response queue pointer.\n", __func__);
L
Linus Torvalds 已提交
167 168 169
		return (IRQ_NONE);
	}

170
	ha = rsp->hw;
171
	reg = &ha->iobase->isp;
L
Linus Torvalds 已提交
172 173
	status = 0;

174
	spin_lock_irqsave(&ha->hardware_lock, flags);
175
	vha = pci_get_drvdata(ha->pdev);
L
Linus Torvalds 已提交
176 177
	for (iter = 50; iter--; ) {
		stat = RD_REG_DWORD(&reg->u.isp2300.host_status);
178
		if (qla2x00_check_reg32_for_disconnect(vha, stat))
179
			break;
L
Linus Torvalds 已提交
180
		if (stat & HSR_RISC_PAUSED) {
181
			if (unlikely(pci_channel_offline(ha->pdev)))
182 183
				break;

L
Linus Torvalds 已提交
184
			hccr = RD_REG_WORD(&reg->hccr);
185

L
Linus Torvalds 已提交
186
			if (hccr & (BIT_15 | BIT_13 | BIT_11 | BIT_8))
187 188 189
				ql_log(ql_log_warn, vha, 0x5026,
				    "Parity error -- HCCR=%x, Dumping "
				    "firmware.\n", hccr);
L
Linus Torvalds 已提交
190
			else
191 192 193
				ql_log(ql_log_warn, vha, 0x5027,
				    "RISC paused -- HCCR=%x, Dumping "
				    "firmware.\n", hccr);
L
Linus Torvalds 已提交
194 195 196 197

			/*
			 * Issue a "HARD" reset in order for the RISC
			 * interrupt bit to be cleared.  Schedule a big
198
			 * hammer to get out of the RISC PAUSED state.
L
Linus Torvalds 已提交
199 200 201
			 */
			WRT_REG_WORD(&reg->hccr, HCCR_RESET_RISC);
			RD_REG_WORD(&reg->hccr);
202

203 204
			ha->isp_ops->fw_dump(vha, 1);
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
205 206 207 208 209 210 211 212 213
			break;
		} else if ((stat & HSR_RISC_INT) == 0)
			break;

		switch (stat & 0xff) {
		case 0x1:
		case 0x2:
		case 0x10:
		case 0x11:
214
			qla2x00_mbx_completion(vha, MSW(stat));
L
Linus Torvalds 已提交
215 216 217 218 219 220
			status |= MBX_INTERRUPT;

			/* Release mailbox registers. */
			WRT_REG_WORD(&reg->semaphore, 0);
			break;
		case 0x12:
221 222 223 224
			mb[0] = MSW(stat);
			mb[1] = RD_MAILBOX_REG(ha, reg, 1);
			mb[2] = RD_MAILBOX_REG(ha, reg, 2);
			mb[3] = RD_MAILBOX_REG(ha, reg, 3);
225
			qla2x00_async_event(vha, rsp, mb);
226 227
			break;
		case 0x13:
228
			qla2x00_process_response_queue(rsp);
L
Linus Torvalds 已提交
229 230
			break;
		case 0x15:
231 232
			mb[0] = MBA_CMPLT_1_16BIT;
			mb[1] = MSW(stat);
233
			qla2x00_async_event(vha, rsp, mb);
L
Linus Torvalds 已提交
234 235
			break;
		case 0x16:
236 237 238
			mb[0] = MBA_SCSI_COMPLETION;
			mb[1] = MSW(stat);
			mb[2] = RD_MAILBOX_REG(ha, reg, 2);
239
			qla2x00_async_event(vha, rsp, mb);
L
Linus Torvalds 已提交
240 241
			break;
		default:
242 243
			ql_dbg(ql_dbg_async, vha, 0x5028,
			    "Unrecognized interrupt type (%d).\n", stat & 0xff);
L
Linus Torvalds 已提交
244 245 246 247 248
			break;
		}
		WRT_REG_WORD(&reg->hccr, HCCR_CLR_RISC_INT);
		RD_REG_WORD_RELAXED(&reg->hccr);
	}
249
	qla2x00_handle_mbx_completion(ha, status);
250
	spin_unlock_irqrestore(&ha->hardware_lock, flags);
L
Linus Torvalds 已提交
251 252 253 254 255 256 257 258 259 260

	return (IRQ_HANDLED);
}

/**
 * qla2x00_mbx_completion() - Process mailbox command completions.
 * @ha: SCSI driver HA context
 * @mb0: Mailbox0 register
 */
static void
261
qla2x00_mbx_completion(scsi_qla_host_t *vha, uint16_t mb0)
L
Linus Torvalds 已提交
262 263
{
	uint16_t	cnt;
264
	uint32_t	mboxes;
L
Linus Torvalds 已提交
265
	uint16_t __iomem *wptr;
266
	struct qla_hw_data *ha = vha->hw;
267
	struct device_reg_2xxx __iomem *reg = &ha->iobase->isp;
L
Linus Torvalds 已提交
268

269 270 271
	/* Read all mbox registers? */
	mboxes = (1 << ha->mbx_count) - 1;
	if (!ha->mcp)
272
		ql_dbg(ql_dbg_async, vha, 0x5001, "MBX pointer ERROR.\n");
273 274 275
	else
		mboxes = ha->mcp->in_mb;

L
Linus Torvalds 已提交
276 277 278
	/* Load return mailbox registers. */
	ha->flags.mbox_int = 1;
	ha->mailbox_out[0] = mb0;
279
	mboxes >>= 1;
L
Linus Torvalds 已提交
280 281 282
	wptr = (uint16_t __iomem *)MAILBOX_REG(ha, reg, 1);

	for (cnt = 1; cnt < ha->mbx_count; cnt++) {
A
Andrew Vasquez 已提交
283
		if (IS_QLA2200(ha) && cnt == 8)
L
Linus Torvalds 已提交
284
			wptr = (uint16_t __iomem *)MAILBOX_REG(ha, reg, 8);
285
		if ((cnt == 4 || cnt == 5) && (mboxes & BIT_0))
L
Linus Torvalds 已提交
286
			ha->mailbox_out[cnt] = qla2x00_debounce_register(wptr);
287
		else if (mboxes & BIT_0)
L
Linus Torvalds 已提交
288
			ha->mailbox_out[cnt] = RD_REG_WORD(wptr);
A
Andrew Vasquez 已提交
289

L
Linus Torvalds 已提交
290
		wptr++;
291
		mboxes >>= 1;
L
Linus Torvalds 已提交
292 293 294
	}
}

295 296 297 298 299 300 301
static void
qla81xx_idc_event(scsi_qla_host_t *vha, uint16_t aen, uint16_t descr)
{
	static char *event[] =
		{ "Complete", "Request Notification", "Time Extension" };
	int rval;
	struct device_reg_24xx __iomem *reg24 = &vha->hw->iobase->isp24;
302
	struct device_reg_82xx __iomem *reg82 = &vha->hw->iobase->isp82;
303 304 305 306
	uint16_t __iomem *wptr;
	uint16_t cnt, timeout, mb[QLA_IDC_ACK_REGS];

	/* Seed data -- mailbox1 -> mailbox7. */
307 308 309 310 311 312 313
	if (IS_QLA81XX(vha->hw) || IS_QLA83XX(vha->hw))
		wptr = (uint16_t __iomem *)&reg24->mailbox1;
	else if (IS_QLA8044(vha->hw))
		wptr = (uint16_t __iomem *)&reg82->mailbox_out[1];
	else
		return;

314 315 316
	for (cnt = 0; cnt < QLA_IDC_ACK_REGS; cnt++, wptr++)
		mb[cnt] = RD_REG_WORD(wptr);

317
	ql_dbg(ql_dbg_async, vha, 0x5021,
318
	    "Inter-Driver Communication %s -- "
319 320 321
	    "%04x %04x %04x %04x %04x %04x %04x.\n",
	    event[aen & 0xff], mb[0], mb[1], mb[2], mb[3],
	    mb[4], mb[5], mb[6]);
322 323 324 325 326
	switch (aen) {
	/* Handle IDC Error completion case. */
	case MBA_IDC_COMPLETE:
		if (mb[1] >> 15) {
			vha->hw->flags.idc_compl_status = 1;
327
			if (vha->hw->notify_dcbx_comp && !vha->vp_idx)
328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352
				complete(&vha->hw->dcbx_comp);
		}
		break;

	case MBA_IDC_NOTIFY:
		/* Acknowledgement needed? [Notify && non-zero timeout]. */
		timeout = (descr >> 8) & 0xf;
		ql_dbg(ql_dbg_async, vha, 0x5022,
		    "%lu Inter-Driver Communication %s -- ACK timeout=%d.\n",
		    vha->host_no, event[aen & 0xff], timeout);

		if (!timeout)
			return;
		rval = qla2x00_post_idc_ack_work(vha, mb);
		if (rval != QLA_SUCCESS)
			ql_log(ql_log_warn, vha, 0x5023,
			    "IDC failed to post ACK.\n");
		break;
	case MBA_IDC_TIME_EXT:
		vha->hw->idc_extend_tmo = descr;
		ql_dbg(ql_dbg_async, vha, 0x5087,
		    "%lu Inter-Driver Communication %s -- "
		    "Extend timeout by=%d.\n",
		    vha->host_no, event[aen & 0xff], vha->hw->idc_extend_tmo);
		break;
353
	}
354 355
}

356
#define LS_UNKNOWN	2
357 358
const char *
qla2x00_get_link_speed_str(struct qla_hw_data *ha, uint16_t speed)
359
{
360 361
	static const char *const link_speeds[] = {
		"1", "2", "?", "4", "8", "16", "32", "10"
362
	};
363
#define	QLA_LAST_SPEED	7
364 365

	if (IS_QLA2100(ha) || IS_QLA2200(ha))
366 367
		return link_speeds[0];
	else if (speed == 0x13)
368 369
		return link_speeds[QLA_LAST_SPEED];
	else if (speed < QLA_LAST_SPEED)
370 371 372
		return link_speeds[speed];
	else
		return link_speeds[LS_UNKNOWN];
373 374
}

375
static void
376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530
qla83xx_handle_8200_aen(scsi_qla_host_t *vha, uint16_t *mb)
{
	struct qla_hw_data *ha = vha->hw;

	/*
	 * 8200 AEN Interpretation:
	 * mb[0] = AEN code
	 * mb[1] = AEN Reason code
	 * mb[2] = LSW of Peg-Halt Status-1 Register
	 * mb[6] = MSW of Peg-Halt Status-1 Register
	 * mb[3] = LSW of Peg-Halt Status-2 register
	 * mb[7] = MSW of Peg-Halt Status-2 register
	 * mb[4] = IDC Device-State Register value
	 * mb[5] = IDC Driver-Presence Register value
	 */
	ql_dbg(ql_dbg_async, vha, 0x506b, "AEN Code: mb[0] = 0x%x AEN reason: "
	    "mb[1] = 0x%x PH-status1: mb[2] = 0x%x PH-status1: mb[6] = 0x%x.\n",
	    mb[0], mb[1], mb[2], mb[6]);
	ql_dbg(ql_dbg_async, vha, 0x506c, "PH-status2: mb[3] = 0x%x "
	    "PH-status2: mb[7] = 0x%x Device-State: mb[4] = 0x%x "
	    "Drv-Presence: mb[5] = 0x%x.\n", mb[3], mb[7], mb[4], mb[5]);

	if (mb[1] & (IDC_PEG_HALT_STATUS_CHANGE | IDC_NIC_FW_REPORTED_FAILURE |
				IDC_HEARTBEAT_FAILURE)) {
		ha->flags.nic_core_hung = 1;
		ql_log(ql_log_warn, vha, 0x5060,
		    "83XX: F/W Error Reported: Check if reset required.\n");

		if (mb[1] & IDC_PEG_HALT_STATUS_CHANGE) {
			uint32_t protocol_engine_id, fw_err_code, err_level;

			/*
			 * IDC_PEG_HALT_STATUS_CHANGE interpretation:
			 *  - PEG-Halt Status-1 Register:
			 *	(LSW = mb[2], MSW = mb[6])
			 *	Bits 0-7   = protocol-engine ID
			 *	Bits 8-28  = f/w error code
			 *	Bits 29-31 = Error-level
			 *	    Error-level 0x1 = Non-Fatal error
			 *	    Error-level 0x2 = Recoverable Fatal error
			 *	    Error-level 0x4 = UnRecoverable Fatal error
			 *  - PEG-Halt Status-2 Register:
			 *	(LSW = mb[3], MSW = mb[7])
			 */
			protocol_engine_id = (mb[2] & 0xff);
			fw_err_code = (((mb[2] & 0xff00) >> 8) |
			    ((mb[6] & 0x1fff) << 8));
			err_level = ((mb[6] & 0xe000) >> 13);
			ql_log(ql_log_warn, vha, 0x5061, "PegHalt Status-1 "
			    "Register: protocol_engine_id=0x%x "
			    "fw_err_code=0x%x err_level=0x%x.\n",
			    protocol_engine_id, fw_err_code, err_level);
			ql_log(ql_log_warn, vha, 0x5062, "PegHalt Status-2 "
			    "Register: 0x%x%x.\n", mb[7], mb[3]);
			if (err_level == ERR_LEVEL_NON_FATAL) {
				ql_log(ql_log_warn, vha, 0x5063,
				    "Not a fatal error, f/w has recovered "
				    "iteself.\n");
			} else if (err_level == ERR_LEVEL_RECOVERABLE_FATAL) {
				ql_log(ql_log_fatal, vha, 0x5064,
				    "Recoverable Fatal error: Chip reset "
				    "required.\n");
				qla83xx_schedule_work(vha,
				    QLA83XX_NIC_CORE_RESET);
			} else if (err_level == ERR_LEVEL_UNRECOVERABLE_FATAL) {
				ql_log(ql_log_fatal, vha, 0x5065,
				    "Unrecoverable Fatal error: Set FAILED "
				    "state, reboot required.\n");
				qla83xx_schedule_work(vha,
				    QLA83XX_NIC_CORE_UNRECOVERABLE);
			}
		}

		if (mb[1] & IDC_NIC_FW_REPORTED_FAILURE) {
			uint16_t peg_fw_state, nw_interface_link_up;
			uint16_t nw_interface_signal_detect, sfp_status;
			uint16_t htbt_counter, htbt_monitor_enable;
			uint16_t sfp_additonal_info, sfp_multirate;
			uint16_t sfp_tx_fault, link_speed, dcbx_status;

			/*
			 * IDC_NIC_FW_REPORTED_FAILURE interpretation:
			 *  - PEG-to-FC Status Register:
			 *	(LSW = mb[2], MSW = mb[6])
			 *	Bits 0-7   = Peg-Firmware state
			 *	Bit 8      = N/W Interface Link-up
			 *	Bit 9      = N/W Interface signal detected
			 *	Bits 10-11 = SFP Status
			 *	  SFP Status 0x0 = SFP+ transceiver not expected
			 *	  SFP Status 0x1 = SFP+ transceiver not present
			 *	  SFP Status 0x2 = SFP+ transceiver invalid
			 *	  SFP Status 0x3 = SFP+ transceiver present and
			 *	  valid
			 *	Bits 12-14 = Heartbeat Counter
			 *	Bit 15     = Heartbeat Monitor Enable
			 *	Bits 16-17 = SFP Additional Info
			 *	  SFP info 0x0 = Unregocnized transceiver for
			 *	  Ethernet
			 *	  SFP info 0x1 = SFP+ brand validation failed
			 *	  SFP info 0x2 = SFP+ speed validation failed
			 *	  SFP info 0x3 = SFP+ access error
			 *	Bit 18     = SFP Multirate
			 *	Bit 19     = SFP Tx Fault
			 *	Bits 20-22 = Link Speed
			 *	Bits 23-27 = Reserved
			 *	Bits 28-30 = DCBX Status
			 *	  DCBX Status 0x0 = DCBX Disabled
			 *	  DCBX Status 0x1 = DCBX Enabled
			 *	  DCBX Status 0x2 = DCBX Exchange error
			 *	Bit 31     = Reserved
			 */
			peg_fw_state = (mb[2] & 0x00ff);
			nw_interface_link_up = ((mb[2] & 0x0100) >> 8);
			nw_interface_signal_detect = ((mb[2] & 0x0200) >> 9);
			sfp_status = ((mb[2] & 0x0c00) >> 10);
			htbt_counter = ((mb[2] & 0x7000) >> 12);
			htbt_monitor_enable = ((mb[2] & 0x8000) >> 15);
			sfp_additonal_info = (mb[6] & 0x0003);
			sfp_multirate = ((mb[6] & 0x0004) >> 2);
			sfp_tx_fault = ((mb[6] & 0x0008) >> 3);
			link_speed = ((mb[6] & 0x0070) >> 4);
			dcbx_status = ((mb[6] & 0x7000) >> 12);

			ql_log(ql_log_warn, vha, 0x5066,
			    "Peg-to-Fc Status Register:\n"
			    "peg_fw_state=0x%x, nw_interface_link_up=0x%x, "
			    "nw_interface_signal_detect=0x%x"
			    "\nsfp_statis=0x%x.\n ", peg_fw_state,
			    nw_interface_link_up, nw_interface_signal_detect,
			    sfp_status);
			ql_log(ql_log_warn, vha, 0x5067,
			    "htbt_counter=0x%x, htbt_monitor_enable=0x%x, "
			    "sfp_additonal_info=0x%x, sfp_multirate=0x%x.\n ",
			    htbt_counter, htbt_monitor_enable,
			    sfp_additonal_info, sfp_multirate);
			ql_log(ql_log_warn, vha, 0x5068,
			    "sfp_tx_fault=0x%x, link_state=0x%x, "
			    "dcbx_status=0x%x.\n", sfp_tx_fault, link_speed,
			    dcbx_status);

			qla83xx_schedule_work(vha, QLA83XX_NIC_CORE_RESET);
		}

		if (mb[1] & IDC_HEARTBEAT_FAILURE) {
			ql_log(ql_log_warn, vha, 0x5069,
			    "Heartbeat Failure encountered, chip reset "
			    "required.\n");

			qla83xx_schedule_work(vha, QLA83XX_NIC_CORE_RESET);
		}
	}

	if (mb[1] & IDC_DEVICE_STATE_CHANGE) {
		ql_log(ql_log_info, vha, 0x506a,
		    "IDC Device-State changed = 0x%x.\n", mb[4]);
531 532
		if (ha->flags.nic_core_reset_owner)
			return;
533 534 535 536
		qla83xx_schedule_work(vha, MBA_IDC_AEN);
	}
}

537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561
int
qla2x00_is_a_vp_did(scsi_qla_host_t *vha, uint32_t rscn_entry)
{
	struct qla_hw_data *ha = vha->hw;
	scsi_qla_host_t *vp;
	uint32_t vp_did;
	unsigned long flags;
	int ret = 0;

	if (!ha->num_vhosts)
		return ret;

	spin_lock_irqsave(&ha->vport_slock, flags);
	list_for_each_entry(vp, &ha->vp_list, list) {
		vp_did = vp->d_id.b24;
		if (vp_did == rscn_entry) {
			ret = 1;
			break;
		}
	}
	spin_unlock_irqrestore(&ha->vport_slock, flags);

	return ret;
}

L
Linus Torvalds 已提交
562 563 564
/**
 * qla2x00_async_event() - Process aynchronous events.
 * @ha: SCSI driver HA context
565
 * @mb: Mailbox registers (0 - 3)
L
Linus Torvalds 已提交
566
 */
567
void
568
qla2x00_async_event(scsi_qla_host_t *vha, struct rsp_que *rsp, uint16_t *mb)
L
Linus Torvalds 已提交
569 570
{
	uint16_t	handle_cnt;
571
	uint16_t	cnt, mbx;
L
Linus Torvalds 已提交
572
	uint32_t	handles[5];
573
	struct qla_hw_data *ha = vha->hw;
574
	struct device_reg_2xxx __iomem *reg = &ha->iobase->isp;
575
	struct device_reg_24xx __iomem *reg24 = &ha->iobase->isp24;
576
	struct device_reg_82xx __iomem *reg82 = &ha->iobase->isp82;
L
Linus Torvalds 已提交
577
	uint32_t	rscn_entry, host_pid;
578
	unsigned long	flags;
L
Linus Torvalds 已提交
579 580 581

	/* Setup to process RIO completion. */
	handle_cnt = 0;
582
	if (IS_CNA_CAPABLE(ha))
583
		goto skip_rio;
L
Linus Torvalds 已提交
584 585
	switch (mb[0]) {
	case MBA_SCSI_COMPLETION:
586
		handles[0] = le32_to_cpu((uint32_t)((mb[2] << 16) | mb[1]));
L
Linus Torvalds 已提交
587 588 589
		handle_cnt = 1;
		break;
	case MBA_CMPLT_1_16BIT:
590
		handles[0] = mb[1];
L
Linus Torvalds 已提交
591 592 593 594
		handle_cnt = 1;
		mb[0] = MBA_SCSI_COMPLETION;
		break;
	case MBA_CMPLT_2_16BIT:
595 596
		handles[0] = mb[1];
		handles[1] = mb[2];
L
Linus Torvalds 已提交
597 598 599 600
		handle_cnt = 2;
		mb[0] = MBA_SCSI_COMPLETION;
		break;
	case MBA_CMPLT_3_16BIT:
601 602 603
		handles[0] = mb[1];
		handles[1] = mb[2];
		handles[2] = mb[3];
L
Linus Torvalds 已提交
604 605 606 607
		handle_cnt = 3;
		mb[0] = MBA_SCSI_COMPLETION;
		break;
	case MBA_CMPLT_4_16BIT:
608 609 610
		handles[0] = mb[1];
		handles[1] = mb[2];
		handles[2] = mb[3];
L
Linus Torvalds 已提交
611 612 613 614 615
		handles[3] = (uint32_t)RD_MAILBOX_REG(ha, reg, 6);
		handle_cnt = 4;
		mb[0] = MBA_SCSI_COMPLETION;
		break;
	case MBA_CMPLT_5_16BIT:
616 617 618
		handles[0] = mb[1];
		handles[1] = mb[2];
		handles[2] = mb[3];
L
Linus Torvalds 已提交
619 620 621 622 623 624
		handles[3] = (uint32_t)RD_MAILBOX_REG(ha, reg, 6);
		handles[4] = (uint32_t)RD_MAILBOX_REG(ha, reg, 7);
		handle_cnt = 5;
		mb[0] = MBA_SCSI_COMPLETION;
		break;
	case MBA_CMPLT_2_32BIT:
625
		handles[0] = le32_to_cpu((uint32_t)((mb[2] << 16) | mb[1]));
L
Linus Torvalds 已提交
626 627 628 629 630 631 632 633 634
		handles[1] = le32_to_cpu(
		    ((uint32_t)(RD_MAILBOX_REG(ha, reg, 7) << 16)) |
		    RD_MAILBOX_REG(ha, reg, 6));
		handle_cnt = 2;
		mb[0] = MBA_SCSI_COMPLETION;
		break;
	default:
		break;
	}
635
skip_rio:
L
Linus Torvalds 已提交
636 637
	switch (mb[0]) {
	case MBA_SCSI_COMPLETION:	/* Fast Post */
638
		if (!vha->flags.online)
L
Linus Torvalds 已提交
639 640 641
			break;

		for (cnt = 0; cnt < handle_cnt; cnt++)
642 643
			qla2x00_process_completed_request(vha, rsp->req,
				handles[cnt]);
L
Linus Torvalds 已提交
644 645 646
		break;

	case MBA_RESET:			/* Reset */
647 648
		ql_dbg(ql_dbg_async, vha, 0x5002,
		    "Asynchronous RESET.\n");
L
Linus Torvalds 已提交
649

650
		set_bit(RESET_MARKER_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
651 652 653
		break;

	case MBA_SYSTEM_ERR:		/* System Error */
654
		mbx = (IS_QLA81XX(ha) || IS_QLA83XX(ha) || IS_QLA27XX(ha)) ?
655
			RD_REG_WORD(&reg24->mailbox7) : 0;
656
		ql_log(ql_log_warn, vha, 0x5003,
657 658
		    "ISP System Error - mbx1=%xh mbx2=%xh mbx3=%xh "
		    "mbx7=%xh.\n", mb[1], mb[2], mb[3], mbx);
L
Linus Torvalds 已提交
659

660
		ha->isp_ops->fw_dump(vha, 1);
L
Linus Torvalds 已提交
661

662
		if (IS_FWI2_CAPABLE(ha)) {
663
			if (mb[1] == 0 && mb[2] == 0) {
664
				ql_log(ql_log_fatal, vha, 0x5004,
665 666
				    "Unrecoverable Hardware Error: adapter "
				    "marked OFFLINE!\n");
667
				vha->flags.online = 0;
668
				vha->device_flags |= DFLG_DEV_FAILED;
669
			} else {
L
Lucas De Marchi 已提交
670
				/* Check to see if MPI timeout occurred */
671
				if ((mbx & MBX_3) && (ha->port_no == 0))
672 673 674
					set_bit(MPI_RESET_NEEDED,
					    &vha->dpc_flags);

675
				set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
676
			}
677
		} else if (mb[1] == 0) {
678
			ql_log(ql_log_fatal, vha, 0x5005,
L
Linus Torvalds 已提交
679 680
			    "Unrecoverable Hardware Error: adapter marked "
			    "OFFLINE!\n");
681
			vha->flags.online = 0;
682
			vha->device_flags |= DFLG_DEV_FAILED;
L
Linus Torvalds 已提交
683
		} else
684
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
685 686 687
		break;

	case MBA_REQ_TRANSFER_ERR:	/* Request Transfer Error */
688 689
		ql_log(ql_log_warn, vha, 0x5006,
		    "ISP Request Transfer Error (%x).\n",  mb[1]);
L
Linus Torvalds 已提交
690

691
		set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
692 693 694
		break;

	case MBA_RSP_TRANSFER_ERR:	/* Response Transfer Error */
695 696
		ql_log(ql_log_warn, vha, 0x5007,
		    "ISP Response Transfer Error.\n");
L
Linus Torvalds 已提交
697

698
		set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
699 700 701
		break;

	case MBA_WAKEUP_THRES:		/* Request Queue Wake-up */
702 703
		ql_dbg(ql_dbg_async, vha, 0x5008,
		    "Asynchronous WAKEUP_THRES.\n");
L
Linus Torvalds 已提交
704

705
		break;
L
Linus Torvalds 已提交
706
	case MBA_LIP_OCCURRED:		/* Loop Initialization Procedure */
707
		ql_dbg(ql_dbg_async, vha, 0x5009,
708
		    "LIP occurred (%x).\n", mb[1]);
L
Linus Torvalds 已提交
709

710 711 712 713
		if (atomic_read(&vha->loop_state) != LOOP_DOWN) {
			atomic_set(&vha->loop_state, LOOP_DOWN);
			atomic_set(&vha->loop_down_timer, LOOP_DOWN_TIME);
			qla2x00_mark_all_devices_lost(vha, 1);
L
Linus Torvalds 已提交
714 715
		}

716 717 718
		if (vha->vp_idx) {
			atomic_set(&vha->vp_state, VP_FAILED);
			fc_vport_set_state(vha->fc_vport, FC_VPORT_FAILED);
719 720
		}

721 722
		set_bit(REGISTER_FC4_NEEDED, &vha->dpc_flags);
		set_bit(REGISTER_FDMI_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
723

724 725
		vha->flags.management_server_logged_in = 0;
		qla2x00_post_aen_work(vha, FCH_EVT_LIP, mb[1]);
L
Linus Torvalds 已提交
726 727 728
		break;

	case MBA_LOOP_UP:		/* Loop Up Event */
729
		if (IS_QLA2100(ha) || IS_QLA2200(ha))
730
			ha->link_data_rate = PORT_SPEED_1GB;
731
		else
L
Linus Torvalds 已提交
732 733
			ha->link_data_rate = mb[1];

734
		ql_log(ql_log_info, vha, 0x500a,
735
		    "LOOP UP detected (%s Gbps).\n",
736
		    qla2x00_get_link_speed_str(ha, ha->link_data_rate));
L
Linus Torvalds 已提交
737

738 739
		vha->flags.management_server_logged_in = 0;
		qla2x00_post_aen_work(vha, FCH_EVT_LINKUP, ha->link_data_rate);
L
Linus Torvalds 已提交
740 741 742
		break;

	case MBA_LOOP_DOWN:		/* Loop Down Event */
743 744
		mbx = (IS_QLA81XX(ha) || IS_QLA8031(ha))
			? RD_REG_WORD(&reg24->mailbox4) : 0;
745 746
		mbx = (IS_P3P_TYPE(ha)) ? RD_REG_WORD(&reg82->mailbox_out[4])
			: mbx;
747
		ql_log(ql_log_info, vha, 0x500b,
748 749
		    "LOOP DOWN detected (%x %x %x %x).\n",
		    mb[1], mb[2], mb[3], mbx);
L
Linus Torvalds 已提交
750

751 752 753 754 755
		if (atomic_read(&vha->loop_state) != LOOP_DOWN) {
			atomic_set(&vha->loop_state, LOOP_DOWN);
			atomic_set(&vha->loop_down_timer, LOOP_DOWN_TIME);
			vha->device_flags |= DFLG_NO_CABLE;
			qla2x00_mark_all_devices_lost(vha, 1);
L
Linus Torvalds 已提交
756 757
		}

758 759 760
		if (vha->vp_idx) {
			atomic_set(&vha->vp_state, VP_FAILED);
			fc_vport_set_state(vha->fc_vport, FC_VPORT_FAILED);
761 762
		}

763
		vha->flags.management_server_logged_in = 0;
764
		ha->link_data_rate = PORT_SPEED_UNKNOWN;
765
		qla2x00_post_aen_work(vha, FCH_EVT_LINKDOWN, 0);
L
Linus Torvalds 已提交
766 767 768
		break;

	case MBA_LIP_RESET:		/* LIP reset occurred */
769
		ql_dbg(ql_dbg_async, vha, 0x500c,
770
		    "LIP reset occurred (%x).\n", mb[1]);
L
Linus Torvalds 已提交
771

772 773 774 775
		if (atomic_read(&vha->loop_state) != LOOP_DOWN) {
			atomic_set(&vha->loop_state, LOOP_DOWN);
			atomic_set(&vha->loop_down_timer, LOOP_DOWN_TIME);
			qla2x00_mark_all_devices_lost(vha, 1);
L
Linus Torvalds 已提交
776 777
		}

778 779 780
		if (vha->vp_idx) {
			atomic_set(&vha->vp_state, VP_FAILED);
			fc_vport_set_state(vha->fc_vport, FC_VPORT_FAILED);
781 782
		}

783
		set_bit(RESET_MARKER_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
784 785

		ha->operating_mode = LOOP;
786 787
		vha->flags.management_server_logged_in = 0;
		qla2x00_post_aen_work(vha, FCH_EVT_LIPRESET, mb[1]);
L
Linus Torvalds 已提交
788 789
		break;

790
	/* case MBA_DCBX_COMPLETE: */
L
Linus Torvalds 已提交
791 792 793 794
	case MBA_POINT_TO_POINT:	/* Point-to-Point */
		if (IS_QLA2100(ha))
			break;

795
		if (IS_CNA_CAPABLE(ha)) {
796 797 798
			ql_dbg(ql_dbg_async, vha, 0x500d,
			    "DCBX Completed -- %04x %04x %04x.\n",
			    mb[1], mb[2], mb[3]);
799
			if (ha->notify_dcbx_comp && !vha->vp_idx)
800 801 802
				complete(&ha->dcbx_comp);

		} else
803 804
			ql_dbg(ql_dbg_async, vha, 0x500e,
			    "Asynchronous P2P MODE received.\n");
L
Linus Torvalds 已提交
805 806 807 808 809

		/*
		 * Until there's a transition from loop down to loop up, treat
		 * this as loop down only.
		 */
810 811 812 813
		if (atomic_read(&vha->loop_state) != LOOP_DOWN) {
			atomic_set(&vha->loop_state, LOOP_DOWN);
			if (!atomic_read(&vha->loop_down_timer))
				atomic_set(&vha->loop_down_timer,
L
Linus Torvalds 已提交
814
				    LOOP_DOWN_TIME);
815
			qla2x00_mark_all_devices_lost(vha, 1);
L
Linus Torvalds 已提交
816 817
		}

818 819 820
		if (vha->vp_idx) {
			atomic_set(&vha->vp_state, VP_FAILED);
			fc_vport_set_state(vha->fc_vport, FC_VPORT_FAILED);
821 822
		}

823 824 825 826 827
		if (!(test_bit(ABORT_ISP_ACTIVE, &vha->dpc_flags)))
			set_bit(RESET_MARKER_NEEDED, &vha->dpc_flags);

		set_bit(REGISTER_FC4_NEEDED, &vha->dpc_flags);
		set_bit(REGISTER_FDMI_NEEDED, &vha->dpc_flags);
828 829

		ha->flags.gpsc_supported = 1;
830
		vha->flags.management_server_logged_in = 0;
L
Linus Torvalds 已提交
831 832 833 834 835 836
		break;

	case MBA_CHG_IN_CONNECTION:	/* Change in connection mode */
		if (IS_QLA2100(ha))
			break;

837
		ql_dbg(ql_dbg_async, vha, 0x500f,
L
Linus Torvalds 已提交
838 839
		    "Configuration change detected: value=%x.\n", mb[1]);

840 841 842 843
		if (atomic_read(&vha->loop_state) != LOOP_DOWN) {
			atomic_set(&vha->loop_state, LOOP_DOWN);
			if (!atomic_read(&vha->loop_down_timer))
				atomic_set(&vha->loop_down_timer,
L
Linus Torvalds 已提交
844
				    LOOP_DOWN_TIME);
845
			qla2x00_mark_all_devices_lost(vha, 1);
L
Linus Torvalds 已提交
846 847
		}

848 849 850
		if (vha->vp_idx) {
			atomic_set(&vha->vp_state, VP_FAILED);
			fc_vport_set_state(vha->fc_vport, FC_VPORT_FAILED);
851 852
		}

853 854
		set_bit(LOOP_RESYNC_NEEDED, &vha->dpc_flags);
		set_bit(LOCAL_LOOP_UPDATE, &vha->dpc_flags);
L
Linus Torvalds 已提交
855 856 857
		break;

	case MBA_PORT_UPDATE:		/* Port database update */
858 859 860 861 862 863 864 865 866 867 868 869 870 871 872
		/*
		 * Handle only global and vn-port update events
		 *
		 * Relevant inputs:
		 * mb[1] = N_Port handle of changed port
		 * OR 0xffff for global event
		 * mb[2] = New login state
		 * 7 = Port logged out
		 * mb[3] = LSB is vp_idx, 0xff = all vps
		 *
		 * Skip processing if:
		 *       Event is global, vp_idx is NOT all vps,
		 *           vp_idx does not match
		 *       Event is not global, vp_idx does not match
		 */
873 874 875 876
		if (IS_QLA2XXX_MIDTYPE(ha) &&
		    ((mb[1] == 0xffff && (mb[3] & 0xff) != 0xff) ||
			(mb[1] != 0xffff)) && vha->vp_idx != (mb[3] & 0xff))
			break;
877

878 879
		/* Global event -- port logout or port unavailable. */
		if (mb[1] == 0xffff && mb[2] == 0x7) {
880 881 882
			ql_dbg(ql_dbg_async, vha, 0x5010,
			    "Port unavailable %04x %04x %04x.\n",
			    mb[1], mb[2], mb[3]);
883 884
			ql_log(ql_log_warn, vha, 0x505e,
			    "Link is offline.\n");
885 886 887 888 889 890 891 892 893 894 895 896 897

			if (atomic_read(&vha->loop_state) != LOOP_DOWN) {
				atomic_set(&vha->loop_state, LOOP_DOWN);
				atomic_set(&vha->loop_down_timer,
				    LOOP_DOWN_TIME);
				vha->device_flags |= DFLG_NO_CABLE;
				qla2x00_mark_all_devices_lost(vha, 1);
			}

			if (vha->vp_idx) {
				atomic_set(&vha->vp_state, VP_FAILED);
				fc_vport_set_state(vha->fc_vport,
				    FC_VPORT_FAILED);
898
				qla2x00_mark_all_devices_lost(vha, 1);
899 900 901 902 903 904 905
			}

			vha->flags.management_server_logged_in = 0;
			ha->link_data_rate = PORT_SPEED_UNKNOWN;
			break;
		}

L
Linus Torvalds 已提交
906
		/*
907
		 * If PORT UPDATE is global (received LIP_OCCURRED/LIP_RESET
L
Linus Torvalds 已提交
908 909 910
		 * event etc. earlier indicating loop is down) then process
		 * it.  Otherwise ignore it and Wait for RSCN to come in.
		 */
911
		atomic_set(&vha->loop_down_timer, 0);
912 913
		if (atomic_read(&vha->loop_state) != LOOP_DOWN &&
		    atomic_read(&vha->loop_state) != LOOP_DEAD) {
914 915 916
			ql_dbg(ql_dbg_async, vha, 0x5011,
			    "Asynchronous PORT UPDATE ignored %04x/%04x/%04x.\n",
			    mb[1], mb[2], mb[3]);
917 918

			qlt_async_event(mb[0], vha, mb);
L
Linus Torvalds 已提交
919 920 921
			break;
		}

922 923 924
		ql_dbg(ql_dbg_async, vha, 0x5012,
		    "Port database changed %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
L
Linus Torvalds 已提交
925 926 927 928

		/*
		 * Mark all devices as missing so we will login again.
		 */
929
		atomic_set(&vha->loop_state, LOOP_UP);
L
Linus Torvalds 已提交
930

931
		qla2x00_mark_all_devices_lost(vha, 1);
L
Linus Torvalds 已提交
932

933 934 935
		if (vha->vp_idx == 0 && !qla_ini_mode_enabled(vha))
			set_bit(SCR_PENDING, &vha->dpc_flags);

936 937
		set_bit(LOOP_RESYNC_NEEDED, &vha->dpc_flags);
		set_bit(LOCAL_LOOP_UPDATE, &vha->dpc_flags);
938 939

		qlt_async_event(mb[0], vha, mb);
L
Linus Torvalds 已提交
940 941 942
		break;

	case MBA_RSCN_UPDATE:		/* State Change Registration */
943
		/* Check if the Vport has issued a SCR */
944
		if (vha->vp_idx && test_bit(VP_SCR_NEEDED, &vha->vp_flags))
945 946
			break;
		/* Only handle SCNs for our Vport index. */
947
		if (ha->flags.npiv_supported && vha->vp_idx != (mb[3] & 0xff))
948
			break;
949

950 951 952
		ql_dbg(ql_dbg_async, vha, 0x5013,
		    "RSCN database changed -- %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
L
Linus Torvalds 已提交
953

954
		rscn_entry = ((mb[1] & 0xff) << 16) | mb[2];
955 956
		host_pid = (vha->d_id.b.domain << 16) | (vha->d_id.b.area << 8)
				| vha->d_id.b.al_pa;
L
Linus Torvalds 已提交
957
		if (rscn_entry == host_pid) {
958 959 960
			ql_dbg(ql_dbg_async, vha, 0x5014,
			    "Ignoring RSCN update to local host "
			    "port ID (%06x).\n", host_pid);
L
Linus Torvalds 已提交
961 962 963
			break;
		}

964 965
		/* Ignore reserved bits from RSCN-payload. */
		rscn_entry = ((mb[1] & 0x3ff) << 16) | mb[2];
L
Linus Torvalds 已提交
966

967 968 969 970
		/* Skip RSCNs for virtual ports on the same physical port */
		if (qla2x00_is_a_vp_did(vha, rscn_entry))
			break;

971 972
		atomic_set(&vha->loop_down_timer, 0);
		vha->flags.management_server_logged_in = 0;
L
Linus Torvalds 已提交
973

974 975 976
		set_bit(LOOP_RESYNC_NEEDED, &vha->dpc_flags);
		set_bit(RSCN_UPDATE, &vha->dpc_flags);
		qla2x00_post_aen_work(vha, FCH_EVT_RSCN, rscn_entry);
L
Linus Torvalds 已提交
977 978 979 980
		break;

	/* case MBA_RIO_RESPONSE: */
	case MBA_ZIO_RESPONSE:
981 982
		ql_dbg(ql_dbg_async, vha, 0x5015,
		    "[R|Z]IO update completion.\n");
L
Linus Torvalds 已提交
983

984
		if (IS_FWI2_CAPABLE(ha))
985
			qla24xx_process_response_queue(vha, rsp);
986
		else
987
			qla2x00_process_response_queue(rsp);
L
Linus Torvalds 已提交
988
		break;
989 990

	case MBA_DISCARD_RND_FRAME:
991 992 993
		ql_dbg(ql_dbg_async, vha, 0x5016,
		    "Discard RND Frame -- %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
994
		break;
995 996

	case MBA_TRACE_NOTIFICATION:
997 998
		ql_dbg(ql_dbg_async, vha, 0x5017,
		    "Trace Notification -- %04x %04x.\n", mb[1], mb[2]);
999
		break;
1000 1001

	case MBA_ISP84XX_ALERT:
1002 1003 1004
		ql_dbg(ql_dbg_async, vha, 0x5018,
		    "ISP84XX Alert Notification -- %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
1005 1006 1007 1008

		spin_lock_irqsave(&ha->cs84xx->access_lock, flags);
		switch (mb[1]) {
		case A84_PANIC_RECOVERY:
1009 1010 1011
			ql_log(ql_log_info, vha, 0x5019,
			    "Alert 84XX: panic recovery %04x %04x.\n",
			    mb[2], mb[3]);
1012 1013 1014
			break;
		case A84_OP_LOGIN_COMPLETE:
			ha->cs84xx->op_fw_version = mb[3] << 16 | mb[2];
1015 1016 1017
			ql_log(ql_log_info, vha, 0x501a,
			    "Alert 84XX: firmware version %x.\n",
			    ha->cs84xx->op_fw_version);
1018 1019 1020
			break;
		case A84_DIAG_LOGIN_COMPLETE:
			ha->cs84xx->diag_fw_version = mb[3] << 16 | mb[2];
1021 1022 1023
			ql_log(ql_log_info, vha, 0x501b,
			    "Alert 84XX: diagnostic firmware version %x.\n",
			    ha->cs84xx->diag_fw_version);
1024 1025 1026 1027
			break;
		case A84_GOLD_LOGIN_COMPLETE:
			ha->cs84xx->diag_fw_version = mb[3] << 16 | mb[2];
			ha->cs84xx->fw_update = 1;
1028 1029 1030
			ql_log(ql_log_info, vha, 0x501c,
			    "Alert 84XX: gold firmware version %x.\n",
			    ha->cs84xx->gold_fw_version);
1031 1032
			break;
		default:
1033 1034
			ql_log(ql_log_warn, vha, 0x501d,
			    "Alert 84xx: Invalid Alert %04x %04x %04x.\n",
1035 1036 1037 1038
			    mb[1], mb[2], mb[3]);
		}
		spin_unlock_irqrestore(&ha->cs84xx->access_lock, flags);
		break;
1039
	case MBA_DCBX_START:
1040 1041 1042
		ql_dbg(ql_dbg_async, vha, 0x501e,
		    "DCBX Started -- %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
1043 1044
		break;
	case MBA_DCBX_PARAM_UPDATE:
1045 1046 1047
		ql_dbg(ql_dbg_async, vha, 0x501f,
		    "DCBX Parameters Updated -- %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
1048 1049
		break;
	case MBA_FCF_CONF_ERR:
1050 1051 1052
		ql_dbg(ql_dbg_async, vha, 0x5020,
		    "FCF Configuration Error -- %04x %04x %04x.\n",
		    mb[1], mb[2], mb[3]);
1053 1054
		break;
	case MBA_IDC_NOTIFY:
1055
		if (IS_QLA8031(vha->hw) || IS_QLA8044(ha)) {
1056 1057 1058 1059
			mb[4] = RD_REG_WORD(&reg24->mailbox4);
			if (((mb[2] & 0x7fff) == MBC_PORT_RESET ||
			    (mb[2] & 0x7fff) == MBC_SET_PORT_CONFIG) &&
			    (mb[4] & INTERNAL_LOOPBACK_MASK) != 0) {
1060
				set_bit(ISP_QUIESCE_NEEDED, &vha->dpc_flags);
1061 1062 1063 1064 1065 1066
				/*
				 * Extend loop down timer since port is active.
				 */
				if (atomic_read(&vha->loop_state) == LOOP_DOWN)
					atomic_set(&vha->loop_down_timer,
					    LOOP_DOWN_TIME);
1067 1068
				qla2xxx_wake_dpc(vha);
			}
1069
		}
1070
	case MBA_IDC_COMPLETE:
1071
		if (ha->notify_lb_portup_comp && !vha->vp_idx)
1072 1073
			complete(&ha->lb_portup_comp);
		/* Fallthru */
1074
	case MBA_IDC_TIME_EXT:
1075 1076
		if (IS_QLA81XX(vha->hw) || IS_QLA8031(vha->hw) ||
		    IS_QLA8044(ha))
1077 1078 1079 1080 1081 1082 1083 1084 1085
			qla81xx_idc_event(vha, mb[0], mb[1]);
		break;

	case MBA_IDC_AEN:
		mb[4] = RD_REG_WORD(&reg24->mailbox4);
		mb[5] = RD_REG_WORD(&reg24->mailbox5);
		mb[6] = RD_REG_WORD(&reg24->mailbox6);
		mb[7] = RD_REG_WORD(&reg24->mailbox7);
		qla83xx_handle_8200_aen(vha, mb);
1086
		break;
1087

1088 1089 1090 1091
	default:
		ql_dbg(ql_dbg_async, vha, 0x5057,
		    "Unknown AEN:%04x %04x %04x %04x\n",
		    mb[0], mb[1], mb[2], mb[3]);
L
Linus Torvalds 已提交
1092
	}
1093

1094 1095
	qlt_async_event(mb[0], vha, mb);

1096
	if (!vha->vp_idx && ha->num_vhosts)
1097
		qla2x00_alert_all_vps(rsp, mb);
L
Linus Torvalds 已提交
1098 1099 1100 1101 1102 1103 1104
}

/**
 * qla2x00_process_completed_request() - Process a Fast Post response.
 * @ha: SCSI driver HA context
 * @index: SRB index
 */
1105
void
1106
qla2x00_process_completed_request(struct scsi_qla_host *vha,
1107
				  struct req_que *req, uint32_t index)
L
Linus Torvalds 已提交
1108 1109
{
	srb_t *sp;
1110
	struct qla_hw_data *ha = vha->hw;
L
Linus Torvalds 已提交
1111 1112

	/* Validate handle. */
1113
	if (index >= req->num_outstanding_cmds) {
1114 1115
		ql_log(ql_log_warn, vha, 0x3014,
		    "Invalid SCSI command index (%x).\n", index);
L
Linus Torvalds 已提交
1116

1117
		if (IS_P3P_TYPE(ha))
1118 1119 1120
			set_bit(FCOE_CTX_RESET_NEEDED, &vha->dpc_flags);
		else
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
1121 1122 1123
		return;
	}

1124
	sp = req->outstanding_cmds[index];
L
Linus Torvalds 已提交
1125 1126
	if (sp) {
		/* Free outstanding command slot. */
1127
		req->outstanding_cmds[index] = NULL;
L
Linus Torvalds 已提交
1128 1129

		/* Save ISP completion status */
1130
		sp->done(ha, sp, DID_OK << 16);
L
Linus Torvalds 已提交
1131
	} else {
1132
		ql_log(ql_log_warn, vha, 0x3016, "Invalid SCSI SRB.\n");
L
Linus Torvalds 已提交
1133

1134
		if (IS_P3P_TYPE(ha))
1135 1136 1137
			set_bit(FCOE_CTX_RESET_NEEDED, &vha->dpc_flags);
		else
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
L
Linus Torvalds 已提交
1138 1139 1140
	}
}

1141
srb_t *
1142 1143 1144 1145 1146 1147 1148 1149 1150
qla2x00_get_sp_from_handle(scsi_qla_host_t *vha, const char *func,
    struct req_que *req, void *iocb)
{
	struct qla_hw_data *ha = vha->hw;
	sts_entry_t *pkt = iocb;
	srb_t *sp = NULL;
	uint16_t index;

	index = LSW(pkt->handle);
1151
	if (index >= req->num_outstanding_cmds) {
1152 1153
		ql_log(ql_log_warn, vha, 0x5031,
		    "Invalid command index (%x).\n", index);
1154
		if (IS_P3P_TYPE(ha))
1155 1156 1157
			set_bit(FCOE_CTX_RESET_NEEDED, &vha->dpc_flags);
		else
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
1158 1159 1160 1161
		goto done;
	}
	sp = req->outstanding_cmds[index];
	if (!sp) {
1162 1163
		ql_log(ql_log_warn, vha, 0x5032,
		    "Invalid completion handle (%x) -- timed-out.\n", index);
1164 1165 1166
		return sp;
	}
	if (sp->handle != index) {
1167 1168
		ql_log(ql_log_warn, vha, 0x5033,
		    "SRB handle (%x) mismatch %x.\n", sp->handle, index);
1169 1170
		return NULL;
	}
1171

1172
	req->outstanding_cmds[index] = NULL;
1173

1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185
done:
	return sp;
}

static void
qla2x00_mbx_iocb_entry(scsi_qla_host_t *vha, struct req_que *req,
    struct mbx_entry *mbx)
{
	const char func[] = "MBX-IOCB";
	const char *type;
	fc_port_t *fcport;
	srb_t *sp;
1186
	struct srb_iocb *lio;
1187
	uint16_t *data;
1188
	uint16_t status;
1189 1190 1191 1192 1193

	sp = qla2x00_get_sp_from_handle(vha, func, req, mbx);
	if (!sp)
		return;

1194 1195
	lio = &sp->u.iocb_cmd;
	type = sp->name;
1196
	fcport = sp->fcport;
1197
	data = lio->u.logio.data;
1198

1199
	data[0] = MBS_COMMAND_ERROR;
1200
	data[1] = lio->u.logio.flags & SRB_LOGIN_RETRIED ?
1201
	    QLA_LOGIO_LOGIN_RETRIED : 0;
1202
	if (mbx->entry_status) {
1203
		ql_dbg(ql_dbg_async, vha, 0x5043,
1204
		    "Async-%s error entry - hdl=%x portid=%02x%02x%02x "
1205
		    "entry-status=%x status=%x state-flag=%x "
1206 1207
		    "status-flags=%x.\n", type, sp->handle,
		    fcport->d_id.b.domain, fcport->d_id.b.area,
1208 1209
		    fcport->d_id.b.al_pa, mbx->entry_status,
		    le16_to_cpu(mbx->status), le16_to_cpu(mbx->state_flags),
1210
		    le16_to_cpu(mbx->status_flags));
1211

1212
		ql_dump_buffer(ql_dbg_async + ql_dbg_buffer, vha, 0x5029,
1213
		    (uint8_t *)mbx, sizeof(*mbx));
1214

1215
		goto logio_done;
1216 1217
	}

1218
	status = le16_to_cpu(mbx->status);
1219
	if (status == 0x30 && sp->type == SRB_LOGIN_CMD &&
1220 1221 1222
	    le16_to_cpu(mbx->mb0) == MBS_COMMAND_COMPLETE)
		status = 0;
	if (!status && le16_to_cpu(mbx->mb0) == MBS_COMMAND_COMPLETE) {
1223
		ql_dbg(ql_dbg_async, vha, 0x5045,
1224 1225 1226 1227
		    "Async-%s complete - hdl=%x portid=%02x%02x%02x mbx1=%x.\n",
		    type, sp->handle, fcport->d_id.b.domain,
		    fcport->d_id.b.area, fcport->d_id.b.al_pa,
		    le16_to_cpu(mbx->mb1));
1228 1229

		data[0] = MBS_COMMAND_COMPLETE;
1230
		if (sp->type == SRB_LOGIN_CMD) {
1231 1232 1233
			fcport->port_type = FCT_TARGET;
			if (le16_to_cpu(mbx->mb1) & BIT_0)
				fcport->port_type = FCT_INITIATOR;
1234
			else if (le16_to_cpu(mbx->mb1) & BIT_1)
1235
				fcport->flags |= FCF_FCP2_DEVICE;
1236
		}
1237
		goto logio_done;
1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251
	}

	data[0] = le16_to_cpu(mbx->mb0);
	switch (data[0]) {
	case MBS_PORT_ID_USED:
		data[1] = le16_to_cpu(mbx->mb1);
		break;
	case MBS_LOOP_ID_USED:
		break;
	default:
		data[0] = MBS_COMMAND_ERROR;
		break;
	}

1252
	ql_log(ql_log_warn, vha, 0x5046,
1253 1254 1255 1256
	    "Async-%s failed - hdl=%x portid=%02x%02x%02x status=%x "
	    "mb0=%x mb1=%x mb2=%x mb6=%x mb7=%x.\n", type, sp->handle,
	    fcport->d_id.b.domain, fcport->d_id.b.area, fcport->d_id.b.al_pa,
	    status, le16_to_cpu(mbx->mb0), le16_to_cpu(mbx->mb1),
1257
	    le16_to_cpu(mbx->mb2), le16_to_cpu(mbx->mb6),
1258
	    le16_to_cpu(mbx->mb7));
1259

1260
logio_done:
1261
	sp->done(vha, sp, 0);
1262 1263
}

1264 1265 1266 1267 1268 1269 1270 1271 1272
static void
qla2x00_ct_entry(scsi_qla_host_t *vha, struct req_que *req,
    sts_entry_t *pkt, int iocb_type)
{
	const char func[] = "CT_IOCB";
	const char *type;
	srb_t *sp;
	struct fc_bsg_job *bsg_job;
	uint16_t comp_status;
1273
	int res;
1274 1275 1276 1277 1278

	sp = qla2x00_get_sp_from_handle(vha, func, req, pkt);
	if (!sp)
		return;

1279
	bsg_job = sp->u.bsg_job;
1280

1281
	type = "ct pass-through";
1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292

	comp_status = le16_to_cpu(pkt->comp_status);

	/* return FC_CTELS_STATUS_OK and leave the decoding of the ELS/CT
	 * fc payload  to the caller
	 */
	bsg_job->reply->reply_data.ctels_reply.status = FC_CTELS_STATUS_OK;
	bsg_job->reply_len = sizeof(struct fc_bsg_reply);

	if (comp_status != CS_COMPLETE) {
		if (comp_status == CS_DATA_UNDERRUN) {
1293
			res = DID_OK << 16;
1294 1295 1296
			bsg_job->reply->reply_payload_rcv_len =
			    le16_to_cpu(((sts_entry_t *)pkt)->rsp_info_len);

1297 1298
			ql_log(ql_log_warn, vha, 0x5048,
			    "CT pass-through-%s error "
1299
			    "comp_status-status=0x%x total_byte = 0x%x.\n",
1300 1301
			    type, comp_status,
			    bsg_job->reply->reply_payload_rcv_len);
1302
		} else {
1303 1304 1305
			ql_log(ql_log_warn, vha, 0x5049,
			    "CT pass-through-%s error "
			    "comp_status-status=0x%x.\n", type, comp_status);
1306
			res = DID_ERROR << 16;
1307 1308
			bsg_job->reply->reply_payload_rcv_len = 0;
		}
1309
		ql_dump_buffer(ql_dbg_async + ql_dbg_buffer, vha, 0x5035,
1310
		    (uint8_t *)pkt, sizeof(*pkt));
1311
	} else {
1312
		res = DID_OK << 16;
1313 1314 1315 1316 1317
		bsg_job->reply->reply_payload_rcv_len =
		    bsg_job->reply_payload.payload_len;
		bsg_job->reply_len = 0;
	}

1318
	sp->done(vha, sp, res);
1319 1320
}

1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331
static void
qla24xx_els_ct_entry(scsi_qla_host_t *vha, struct req_que *req,
    struct sts_entry_24xx *pkt, int iocb_type)
{
	const char func[] = "ELS_CT_IOCB";
	const char *type;
	srb_t *sp;
	struct fc_bsg_job *bsg_job;
	uint16_t comp_status;
	uint32_t fw_status[3];
	uint8_t* fw_sts_ptr;
1332
	int res;
1333 1334 1335 1336

	sp = qla2x00_get_sp_from_handle(vha, func, req, pkt);
	if (!sp)
		return;
1337
	bsg_job = sp->u.bsg_job;
1338 1339

	type = NULL;
1340
	switch (sp->type) {
1341 1342 1343 1344 1345 1346 1347 1348
	case SRB_ELS_CMD_RPT:
	case SRB_ELS_CMD_HST:
		type = "els";
		break;
	case SRB_CT_CMD:
		type = "ct pass-through";
		break;
	default:
1349
		ql_dbg(ql_dbg_user, vha, 0x503e,
1350
		    "Unrecognized SRB: (%p) type=%d.\n", sp, sp->type);
1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365
		return;
	}

	comp_status = fw_status[0] = le16_to_cpu(pkt->comp_status);
	fw_status[1] = le16_to_cpu(((struct els_sts_entry_24xx*)pkt)->error_subcode_1);
	fw_status[2] = le16_to_cpu(((struct els_sts_entry_24xx*)pkt)->error_subcode_2);

	/* return FC_CTELS_STATUS_OK and leave the decoding of the ELS/CT
	 * fc payload  to the caller
	 */
	bsg_job->reply->reply_data.ctels_reply.status = FC_CTELS_STATUS_OK;
	bsg_job->reply_len = sizeof(struct fc_bsg_reply) + sizeof(fw_status);

	if (comp_status != CS_COMPLETE) {
		if (comp_status == CS_DATA_UNDERRUN) {
1366
			res = DID_OK << 16;
1367
			bsg_job->reply->reply_payload_rcv_len =
1368
			    le16_to_cpu(((struct els_sts_entry_24xx *)pkt)->total_byte_count);
1369

1370
			ql_dbg(ql_dbg_user, vha, 0x503f,
1371
			    "ELS-CT pass-through-%s error hdl=%x comp_status-status=0x%x "
1372
			    "error subcode 1=0x%x error subcode 2=0x%x total_byte = 0x%x.\n",
1373
			    type, sp->handle, comp_status, fw_status[1], fw_status[2],
1374 1375
			    le16_to_cpu(((struct els_sts_entry_24xx *)
				pkt)->total_byte_count));
1376 1377 1378 1379
			fw_sts_ptr = ((uint8_t*)bsg_job->req->sense) + sizeof(struct fc_bsg_reply);
			memcpy( fw_sts_ptr, fw_status, sizeof(fw_status));
		}
		else {
1380
			ql_dbg(ql_dbg_user, vha, 0x5040,
1381
			    "ELS-CT pass-through-%s error hdl=%x comp_status-status=0x%x "
1382
			    "error subcode 1=0x%x error subcode 2=0x%x.\n",
1383
			    type, sp->handle, comp_status,
1384 1385 1386 1387
			    le16_to_cpu(((struct els_sts_entry_24xx *)
				pkt)->error_subcode_1),
			    le16_to_cpu(((struct els_sts_entry_24xx *)
				    pkt)->error_subcode_2));
1388
			res = DID_ERROR << 16;
1389 1390 1391 1392
			bsg_job->reply->reply_payload_rcv_len = 0;
			fw_sts_ptr = ((uint8_t*)bsg_job->req->sense) + sizeof(struct fc_bsg_reply);
			memcpy( fw_sts_ptr, fw_status, sizeof(fw_status));
		}
1393
		ql_dump_buffer(ql_dbg_user + ql_dbg_buffer, vha, 0x5056,
1394
				(uint8_t *)pkt, sizeof(*pkt));
1395 1396
	}
	else {
1397
		res =  DID_OK << 16;
1398 1399 1400 1401
		bsg_job->reply->reply_payload_rcv_len = bsg_job->reply_payload.payload_len;
		bsg_job->reply_len = 0;
	}

1402
	sp->done(vha, sp, res);
1403 1404
}

1405 1406 1407 1408 1409 1410 1411 1412
static void
qla24xx_logio_entry(scsi_qla_host_t *vha, struct req_que *req,
    struct logio_entry_24xx *logio)
{
	const char func[] = "LOGIO-IOCB";
	const char *type;
	fc_port_t *fcport;
	srb_t *sp;
1413
	struct srb_iocb *lio;
1414
	uint16_t *data;
1415 1416 1417 1418 1419 1420
	uint32_t iop[2];

	sp = qla2x00_get_sp_from_handle(vha, func, req, logio);
	if (!sp)
		return;

1421 1422
	lio = &sp->u.iocb_cmd;
	type = sp->name;
1423
	fcport = sp->fcport;
1424
	data = lio->u.logio.data;
1425

1426
	data[0] = MBS_COMMAND_ERROR;
1427
	data[1] = lio->u.logio.flags & SRB_LOGIN_RETRIED ?
1428
		QLA_LOGIO_LOGIN_RETRIED : 0;
1429
	if (logio->entry_status) {
1430
		ql_log(ql_log_warn, fcport->vha, 0x5034,
1431
		    "Async-%s error entry - hdl=%x"
1432
		    "portid=%02x%02x%02x entry-status=%x.\n",
1433 1434 1435 1436
		    type, sp->handle, fcport->d_id.b.domain,
		    fcport->d_id.b.area, fcport->d_id.b.al_pa,
		    logio->entry_status);
		ql_dump_buffer(ql_dbg_async + ql_dbg_buffer, vha, 0x504d,
1437
		    (uint8_t *)logio, sizeof(*logio));
1438

1439
		goto logio_done;
1440 1441 1442
	}

	if (le16_to_cpu(logio->comp_status) == CS_COMPLETE) {
1443
		ql_dbg(ql_dbg_async, fcport->vha, 0x5036,
1444 1445 1446
		    "Async-%s complete - hdl=%x portid=%02x%02x%02x "
		    "iop0=%x.\n", type, sp->handle, fcport->d_id.b.domain,
		    fcport->d_id.b.area, fcport->d_id.b.al_pa,
1447
		    le32_to_cpu(logio->io_parameter[0]));
1448 1449

		data[0] = MBS_COMMAND_COMPLETE;
1450
		if (sp->type != SRB_LOGIN_CMD)
1451
			goto logio_done;
1452 1453 1454 1455 1456

		iop[0] = le32_to_cpu(logio->io_parameter[0]);
		if (iop[0] & BIT_4) {
			fcport->port_type = FCT_TARGET;
			if (iop[0] & BIT_8)
1457
				fcport->flags |= FCF_FCP2_DEVICE;
1458
		} else if (iop[0] & BIT_5)
1459
			fcport->port_type = FCT_INITIATOR;
1460

1461 1462 1463
		if (iop[0] & BIT_7)
			fcport->flags |= FCF_CONF_COMP_SUPPORTED;

1464 1465 1466 1467 1468
		if (logio->io_parameter[7] || logio->io_parameter[8])
			fcport->supported_classes |= FC_COS_CLASS2;
		if (logio->io_parameter[9] || logio->io_parameter[10])
			fcport->supported_classes |= FC_COS_CLASS3;

1469
		goto logio_done;
1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486
	}

	iop[0] = le32_to_cpu(logio->io_parameter[0]);
	iop[1] = le32_to_cpu(logio->io_parameter[1]);
	switch (iop[0]) {
	case LSC_SCODE_PORTID_USED:
		data[0] = MBS_PORT_ID_USED;
		data[1] = LSW(iop[1]);
		break;
	case LSC_SCODE_NPORT_USED:
		data[0] = MBS_LOOP_ID_USED;
		break;
	default:
		data[0] = MBS_COMMAND_ERROR;
		break;
	}

1487
	ql_dbg(ql_dbg_async, fcport->vha, 0x5037,
1488 1489
	    "Async-%s failed - hdl=%x portid=%02x%02x%02x comp=%x "
	    "iop0=%x iop1=%x.\n", type, sp->handle, fcport->d_id.b.domain,
1490
	    fcport->d_id.b.area, fcport->d_id.b.al_pa,
1491 1492
	    le16_to_cpu(logio->comp_status),
	    le32_to_cpu(logio->io_parameter[0]),
1493
	    le32_to_cpu(logio->io_parameter[1]));
1494

1495
logio_done:
1496
	sp->done(vha, sp, 0);
1497 1498
}

1499
static void
1500
qla24xx_tm_iocb_entry(scsi_qla_host_t *vha, struct req_que *req, void *tsk)
1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512
{
	const char func[] = "TMF-IOCB";
	const char *type;
	fc_port_t *fcport;
	srb_t *sp;
	struct srb_iocb *iocb;
	struct sts_entry_24xx *sts = (struct sts_entry_24xx *)tsk;

	sp = qla2x00_get_sp_from_handle(vha, func, req, tsk);
	if (!sp)
		return;

1513 1514
	iocb = &sp->u.iocb_cmd;
	type = sp->name;
1515
	fcport = sp->fcport;
1516
	iocb->u.tmf.data = QLA_SUCCESS;
1517 1518

	if (sts->entry_status) {
1519
		ql_log(ql_log_warn, fcport->vha, 0x5038,
1520 1521
		    "Async-%s error - hdl=%x entry-status(%x).\n",
		    type, sp->handle, sts->entry_status);
1522
		iocb->u.tmf.data = QLA_FUNCTION_FAILED;
1523
	} else if (sts->comp_status != __constant_cpu_to_le16(CS_COMPLETE)) {
1524
		ql_log(ql_log_warn, fcport->vha, 0x5039,
1525 1526
		    "Async-%s error - hdl=%x completion status(%x).\n",
		    type, sp->handle, sts->comp_status);
1527 1528
		iocb->u.tmf.data = QLA_FUNCTION_FAILED;
	} else if ((le16_to_cpu(sts->scsi_status) &
1529
	    SS_RESPONSE_INFO_LEN_VALID)) {
1530 1531 1532 1533 1534 1535 1536 1537 1538 1539
		if (le32_to_cpu(sts->rsp_data_len) < 4) {
			ql_log(ql_log_warn, fcport->vha, 0x503b,
			    "Async-%s error - hdl=%x not enough response(%d).\n",
			    type, sp->handle, sts->rsp_data_len);
		} else if (sts->data[3]) {
			ql_log(ql_log_warn, fcport->vha, 0x503c,
			    "Async-%s error - hdl=%x response(%x).\n",
			    type, sp->handle, sts->data[3]);
		iocb->u.tmf.data = QLA_FUNCTION_FAILED;
		}
1540 1541
	}

1542
	if (iocb->u.tmf.data != QLA_SUCCESS)
1543 1544
		ql_dump_buffer(ql_dbg_async + ql_dbg_buffer, vha, 0x5055,
		    (uint8_t *)sts, sizeof(*sts));
1545

1546
	sp->done(vha, sp, 0);
1547 1548
}

L
Linus Torvalds 已提交
1549 1550 1551 1552 1553
/**
 * qla2x00_process_response_queue() - Process response queue entries.
 * @ha: SCSI driver HA context
 */
void
1554
qla2x00_process_response_queue(struct rsp_que *rsp)
L
Linus Torvalds 已提交
1555
{
1556 1557
	struct scsi_qla_host *vha;
	struct qla_hw_data *ha = rsp->hw;
1558
	struct device_reg_2xxx __iomem *reg = &ha->iobase->isp;
L
Linus Torvalds 已提交
1559 1560 1561
	sts_entry_t	*pkt;
	uint16_t        handle_cnt;
	uint16_t        cnt;
1562

1563
	vha = pci_get_drvdata(ha->pdev);
L
Linus Torvalds 已提交
1564

1565
	if (!vha->flags.online)
L
Linus Torvalds 已提交
1566 1567
		return;

1568 1569
	while (rsp->ring_ptr->signature != RESPONSE_PROCESSED) {
		pkt = (sts_entry_t *)rsp->ring_ptr;
L
Linus Torvalds 已提交
1570

1571 1572 1573 1574
		rsp->ring_index++;
		if (rsp->ring_index == rsp->length) {
			rsp->ring_index = 0;
			rsp->ring_ptr = rsp->ring;
L
Linus Torvalds 已提交
1575
		} else {
1576
			rsp->ring_ptr++;
L
Linus Torvalds 已提交
1577 1578 1579
		}

		if (pkt->entry_status != 0) {
1580
			qla2x00_error_entry(vha, rsp, pkt);
L
Linus Torvalds 已提交
1581 1582 1583 1584 1585 1586 1587
			((response_t *)pkt)->signature = RESPONSE_PROCESSED;
			wmb();
			continue;
		}

		switch (pkt->entry_type) {
		case STATUS_TYPE:
1588
			qla2x00_status_entry(vha, rsp, pkt);
L
Linus Torvalds 已提交
1589 1590 1591 1592
			break;
		case STATUS_TYPE_21:
			handle_cnt = ((sts21_entry_t *)pkt)->handle_count;
			for (cnt = 0; cnt < handle_cnt; cnt++) {
1593
				qla2x00_process_completed_request(vha, rsp->req,
L
Linus Torvalds 已提交
1594 1595 1596 1597 1598 1599
				    ((sts21_entry_t *)pkt)->handle[cnt]);
			}
			break;
		case STATUS_TYPE_22:
			handle_cnt = ((sts22_entry_t *)pkt)->handle_count;
			for (cnt = 0; cnt < handle_cnt; cnt++) {
1600
				qla2x00_process_completed_request(vha, rsp->req,
L
Linus Torvalds 已提交
1601 1602 1603 1604
				    ((sts22_entry_t *)pkt)->handle[cnt]);
			}
			break;
		case STATUS_CONT_TYPE:
1605
			qla2x00_status_cont_entry(rsp, (sts_cont_entry_t *)pkt);
L
Linus Torvalds 已提交
1606
			break;
1607 1608 1609
		case MBX_IOCB_TYPE:
			qla2x00_mbx_iocb_entry(vha, rsp->req,
			    (struct mbx_entry *)pkt);
1610
			break;
1611 1612 1613
		case CT_IOCB_TYPE:
			qla2x00_ct_entry(vha, rsp->req, pkt, CT_IOCB_TYPE);
			break;
L
Linus Torvalds 已提交
1614 1615
		default:
			/* Type Not Supported. */
1616 1617
			ql_log(ql_log_warn, vha, 0x504a,
			    "Received unknown response pkt type %x "
L
Linus Torvalds 已提交
1618
			    "entry status=%x.\n",
1619
			    pkt->entry_type, pkt->entry_status);
L
Linus Torvalds 已提交
1620 1621 1622 1623 1624 1625 1626
			break;
		}
		((response_t *)pkt)->signature = RESPONSE_PROCESSED;
		wmb();
	}

	/* Adjust ring index */
1627
	WRT_REG_WORD(ISP_RSP_Q_OUT(ha, reg), rsp->ring_index);
L
Linus Torvalds 已提交
1628 1629
}

1630
static inline void
1631
qla2x00_handle_sense(srb_t *sp, uint8_t *sense_data, uint32_t par_sense_len,
1632
		     uint32_t sense_len, struct rsp_que *rsp, int res)
1633
{
1634
	struct scsi_qla_host *vha = sp->fcport->vha;
1635 1636
	struct scsi_cmnd *cp = GET_CMD_SP(sp);
	uint32_t track_sense_len;
1637 1638 1639 1640

	if (sense_len >= SCSI_SENSE_BUFFERSIZE)
		sense_len = SCSI_SENSE_BUFFERSIZE;

1641 1642 1643 1644 1645
	SET_CMD_SENSE_LEN(sp, sense_len);
	SET_CMD_SENSE_PTR(sp, cp->sense_buffer);
	track_sense_len = sense_len;

	if (sense_len > par_sense_len)
1646
		sense_len = par_sense_len;
1647 1648 1649

	memcpy(cp->sense_buffer, sense_data, sense_len);

1650 1651 1652 1653 1654
	SET_CMD_SENSE_PTR(sp, cp->sense_buffer + sense_len);
	track_sense_len -= sense_len;
	SET_CMD_SENSE_LEN(sp, track_sense_len);

	if (track_sense_len != 0) {
1655
		rsp->status_srb = sp;
1656 1657
		cp->result = res;
	}
1658

1659 1660
	if (sense_len) {
		ql_dbg(ql_dbg_io + ql_dbg_buffer, vha, 0x301c,
H
Hannes Reinecke 已提交
1661
		    "Check condition Sense data, nexus%ld:%d:%llu cmd=%p.\n",
1662 1663
		    sp->fcport->vha->host_no, cp->device->id, cp->device->lun,
		    cp);
1664 1665
		ql_dump_buffer(ql_dbg_io + ql_dbg_buffer, vha, 0x302b,
		    cp->sense_buffer, sense_len);
1666
	}
1667 1668
}

1669 1670
struct scsi_dif_tuple {
	__be16 guard;       /* Checksum */
1671
	__be16 app_tag;         /* APPL identifier */
1672 1673 1674 1675 1676 1677 1678 1679 1680
	__be32 ref_tag;         /* Target LBA or indirect LBA */
};

/*
 * Checks the guard or meta-data for the type of error
 * detected by the HBA. In case of errors, we set the
 * ASC/ASCQ fields in the sense buffer with ILLEGAL_REQUEST
 * to indicate to the kernel that the HBA detected error.
 */
1681
static inline int
1682 1683
qla2x00_handle_dif_error(srb_t *sp, struct sts_entry_24xx *sts24)
{
1684
	struct scsi_qla_host *vha = sp->fcport->vha;
1685
	struct scsi_cmnd *cmd = GET_CMD_SP(sp);
1686 1687
	uint8_t		*ap = &sts24->data[12];
	uint8_t		*ep = &sts24->data[20];
1688 1689 1690 1691
	uint32_t	e_ref_tag, a_ref_tag;
	uint16_t	e_app_tag, a_app_tag;
	uint16_t	e_guard, a_guard;

1692 1693 1694 1695 1696 1697 1698 1699 1700 1701
	/*
	 * swab32 of the "data" field in the beginning of qla2x00_status_entry()
	 * would make guard field appear at offset 2
	 */
	a_guard   = le16_to_cpu(*(uint16_t *)(ap + 2));
	a_app_tag = le16_to_cpu(*(uint16_t *)(ap + 0));
	a_ref_tag = le32_to_cpu(*(uint32_t *)(ap + 4));
	e_guard   = le16_to_cpu(*(uint16_t *)(ep + 2));
	e_app_tag = le16_to_cpu(*(uint16_t *)(ep + 0));
	e_ref_tag = le32_to_cpu(*(uint32_t *)(ep + 4));
1702

1703 1704
	ql_dbg(ql_dbg_io, vha, 0x3023,
	    "iocb(s) %p Returned STATUS.\n", sts24);
1705

1706 1707
	ql_dbg(ql_dbg_io, vha, 0x3024,
	    "DIF ERROR in cmd 0x%x lba 0x%llx act ref"
1708
	    " tag=0x%x, exp ref_tag=0x%x, act app tag=0x%x, exp app"
1709
	    " tag=0x%x, act guard=0x%x, exp guard=0x%x.\n",
1710
	    cmd->cmnd[0], (u64)scsi_get_lba(cmd), a_ref_tag, e_ref_tag,
1711
	    a_app_tag, e_app_tag, a_guard, e_guard);
1712

1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752
	/*
	 * Ignore sector if:
	 * For type     3: ref & app tag is all 'f's
	 * For type 0,1,2: app tag is all 'f's
	 */
	if ((a_app_tag == 0xffff) &&
	    ((scsi_get_prot_type(cmd) != SCSI_PROT_DIF_TYPE3) ||
	     (a_ref_tag == 0xffffffff))) {
		uint32_t blocks_done, resid;
		sector_t lba_s = scsi_get_lba(cmd);

		/* 2TB boundary case covered automatically with this */
		blocks_done = e_ref_tag - (uint32_t)lba_s + 1;

		resid = scsi_bufflen(cmd) - (blocks_done *
		    cmd->device->sector_size);

		scsi_set_resid(cmd, resid);
		cmd->result = DID_OK << 16;

		/* Update protection tag */
		if (scsi_prot_sg_count(cmd)) {
			uint32_t i, j = 0, k = 0, num_ent;
			struct scatterlist *sg;
			struct sd_dif_tuple *spt;

			/* Patch the corresponding protection tags */
			scsi_for_each_prot_sg(cmd, sg,
			    scsi_prot_sg_count(cmd), i) {
				num_ent = sg_dma_len(sg) / 8;
				if (k + num_ent < blocks_done) {
					k += num_ent;
					continue;
				}
				j = blocks_done - k - 1;
				k = blocks_done;
				break;
			}

			if (k != blocks_done) {
1753
				ql_log(ql_log_warn, vha, 0x302f,
1754 1755
				    "unexpected tag values tag:lba=%x:%llx)\n",
				    e_ref_tag, (unsigned long long)lba_s);
1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769
				return 1;
			}

			spt = page_address(sg_page(sg)) + sg->offset;
			spt += j;

			spt->app_tag = 0xffff;
			if (scsi_get_prot_type(cmd) == SCSI_PROT_DIF_TYPE3)
				spt->ref_tag = 0xffffffff;
		}

		return 0;
	}

1770 1771 1772 1773 1774 1775 1776
	/* check guard */
	if (e_guard != a_guard) {
		scsi_build_sense_buffer(1, cmd->sense_buffer, ILLEGAL_REQUEST,
		    0x10, 0x1);
		set_driver_byte(cmd, DRIVER_SENSE);
		set_host_byte(cmd, DID_ABORT);
		cmd->result |= SAM_STAT_CHECK_CONDITION << 1;
1777
		return 1;
1778 1779
	}

1780 1781
	/* check ref tag */
	if (e_ref_tag != a_ref_tag) {
1782
		scsi_build_sense_buffer(1, cmd->sense_buffer, ILLEGAL_REQUEST,
1783
		    0x10, 0x3);
1784 1785 1786
		set_driver_byte(cmd, DRIVER_SENSE);
		set_host_byte(cmd, DID_ABORT);
		cmd->result |= SAM_STAT_CHECK_CONDITION << 1;
1787
		return 1;
1788 1789
	}

1790 1791
	/* check appl tag */
	if (e_app_tag != a_app_tag) {
1792
		scsi_build_sense_buffer(1, cmd->sense_buffer, ILLEGAL_REQUEST,
1793
		    0x10, 0x2);
1794 1795 1796
		set_driver_byte(cmd, DRIVER_SENSE);
		set_host_byte(cmd, DID_ABORT);
		cmd->result |= SAM_STAT_CHECK_CONDITION << 1;
1797
		return 1;
1798
	}
1799

1800
	return 1;
1801 1802
}

1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819
static void
qla25xx_process_bidir_status_iocb(scsi_qla_host_t *vha, void *pkt,
				  struct req_que *req, uint32_t index)
{
	struct qla_hw_data *ha = vha->hw;
	srb_t *sp;
	uint16_t	comp_status;
	uint16_t	scsi_status;
	uint16_t thread_id;
	uint32_t rval = EXT_STATUS_OK;
	struct fc_bsg_job *bsg_job = NULL;
	sts_entry_t *sts;
	struct sts_entry_24xx *sts24;
	sts = (sts_entry_t *) pkt;
	sts24 = (struct sts_entry_24xx *) pkt;

	/* Validate handle. */
1820
	if (index >= req->num_outstanding_cmds) {
1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854
		ql_log(ql_log_warn, vha, 0x70af,
		    "Invalid SCSI completion handle 0x%x.\n", index);
		set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
		return;
	}

	sp = req->outstanding_cmds[index];
	if (sp) {
		/* Free outstanding command slot. */
		req->outstanding_cmds[index] = NULL;
		bsg_job = sp->u.bsg_job;
	} else {
		ql_log(ql_log_warn, vha, 0x70b0,
		    "Req:%d: Invalid ISP SCSI completion handle(0x%x)\n",
		    req->id, index);

		set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
		return;
	}

	if (IS_FWI2_CAPABLE(ha)) {
		comp_status = le16_to_cpu(sts24->comp_status);
		scsi_status = le16_to_cpu(sts24->scsi_status) & SS_MASK;
	} else {
		comp_status = le16_to_cpu(sts->comp_status);
		scsi_status = le16_to_cpu(sts->scsi_status) & SS_MASK;
	}

	thread_id = bsg_job->request->rqst_data.h_vendor.vendor_cmd[1];
	switch (comp_status) {
	case CS_COMPLETE:
		if (scsi_status == 0) {
			bsg_job->reply->reply_payload_rcv_len =
					bsg_job->reply_payload.payload_len;
1855 1856 1857
			vha->qla_stats.input_bytes +=
				bsg_job->reply->reply_payload_rcv_len;
			vha->qla_stats.input_requests++;
1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913 1914 1915 1916 1917 1918 1919 1920 1921 1922 1923 1924 1925 1926 1927 1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948
			rval = EXT_STATUS_OK;
		}
		goto done;

	case CS_DATA_OVERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b1,
		    "Command completed with date overrun thread_id=%d\n",
		    thread_id);
		rval = EXT_STATUS_DATA_OVERRUN;
		break;

	case CS_DATA_UNDERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b2,
		    "Command completed with date underrun thread_id=%d\n",
		    thread_id);
		rval = EXT_STATUS_DATA_UNDERRUN;
		break;
	case CS_BIDIR_RD_OVERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b3,
		    "Command completed with read data overrun thread_id=%d\n",
		    thread_id);
		rval = EXT_STATUS_DATA_OVERRUN;
		break;

	case CS_BIDIR_RD_WR_OVERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b4,
		    "Command completed with read and write data overrun "
		    "thread_id=%d\n", thread_id);
		rval = EXT_STATUS_DATA_OVERRUN;
		break;

	case CS_BIDIR_RD_OVERRUN_WR_UNDERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b5,
		    "Command completed with read data over and write data "
		    "underrun thread_id=%d\n", thread_id);
		rval = EXT_STATUS_DATA_OVERRUN;
		break;

	case CS_BIDIR_RD_UNDERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b6,
		    "Command completed with read data data underrun "
		    "thread_id=%d\n", thread_id);
		rval = EXT_STATUS_DATA_UNDERRUN;
		break;

	case CS_BIDIR_RD_UNDERRUN_WR_OVERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b7,
		    "Command completed with read data under and write data "
		    "overrun thread_id=%d\n", thread_id);
		rval = EXT_STATUS_DATA_UNDERRUN;
		break;

	case CS_BIDIR_RD_WR_UNDERRUN:
		ql_dbg(ql_dbg_user, vha, 0x70b8,
		    "Command completed with read and write data underrun "
		    "thread_id=%d\n", thread_id);
		rval = EXT_STATUS_DATA_UNDERRUN;
		break;

	case CS_BIDIR_DMA:
		ql_dbg(ql_dbg_user, vha, 0x70b9,
		    "Command completed with data DMA error thread_id=%d\n",
		    thread_id);
		rval = EXT_STATUS_DMA_ERR;
		break;

	case CS_TIMEOUT:
		ql_dbg(ql_dbg_user, vha, 0x70ba,
		    "Command completed with timeout thread_id=%d\n",
		    thread_id);
		rval = EXT_STATUS_TIMEOUT;
		break;
	default:
		ql_dbg(ql_dbg_user, vha, 0x70bb,
		    "Command completed with completion status=0x%x "
		    "thread_id=%d\n", comp_status, thread_id);
		rval = EXT_STATUS_ERR;
		break;
	}
		bsg_job->reply->reply_payload_rcv_len = 0;

done:
	/* Return the vendor specific reply to API */
	bsg_job->reply->reply_data.vendor_reply.vendor_rsp[0] = rval;
	bsg_job->reply_len = sizeof(struct fc_bsg_reply);
	/* Always return DID_OK, bsg will send the vendor specific response
	 * in this case only */
	sp->done(vha, sp, (DID_OK << 6));

}

L
Linus Torvalds 已提交
1949 1950 1951 1952 1953 1954
/**
 * qla2x00_status_entry() - Process a Status IOCB entry.
 * @ha: SCSI driver HA context
 * @pkt: Entry pointer
 */
static void
1955
qla2x00_status_entry(scsi_qla_host_t *vha, struct rsp_que *rsp, void *pkt)
L
Linus Torvalds 已提交
1956 1957 1958 1959
{
	srb_t		*sp;
	fc_port_t	*fcport;
	struct scsi_cmnd *cp;
1960 1961
	sts_entry_t *sts;
	struct sts_entry_24xx *sts24;
L
Linus Torvalds 已提交
1962 1963
	uint16_t	comp_status;
	uint16_t	scsi_status;
1964
	uint16_t	ox_id;
L
Linus Torvalds 已提交
1965 1966
	uint8_t		lscsi_status;
	int32_t		resid;
1967 1968
	uint32_t sense_len, par_sense_len, rsp_info_len, resid_len,
	    fw_resid_len;
1969
	uint8_t		*rsp_info, *sense_data;
1970
	struct qla_hw_data *ha = vha->hw;
1971 1972 1973
	uint32_t handle;
	uint16_t que;
	struct req_que *req;
1974
	int logit = 1;
1975
	int res = 0;
1976
	uint16_t state_flags = 0;
1977 1978 1979

	sts = (sts_entry_t *) pkt;
	sts24 = (struct sts_entry_24xx *) pkt;
1980
	if (IS_FWI2_CAPABLE(ha)) {
1981 1982
		comp_status = le16_to_cpu(sts24->comp_status);
		scsi_status = le16_to_cpu(sts24->scsi_status) & SS_MASK;
1983
		state_flags = le16_to_cpu(sts24->state_flags);
1984 1985 1986 1987
	} else {
		comp_status = le16_to_cpu(sts->comp_status);
		scsi_status = le16_to_cpu(sts->scsi_status) & SS_MASK;
	}
1988 1989 1990
	handle = (uint32_t) LSW(sts->handle);
	que = MSW(sts->handle);
	req = ha->req_q_map[que];
1991

1992 1993 1994 1995 1996 1997 1998 1999 2000
	/* Check for invalid queue pointer */
	if (req == NULL ||
	    que >= find_first_zero_bit(ha->req_qid_map, ha->max_req_queues)) {
		ql_dbg(ql_dbg_io, vha, 0x3059,
		    "Invalid status handle (0x%x): Bad req pointer. req=%p, "
		    "que=%u.\n", sts->handle, req, que);
		return;
	}

L
Linus Torvalds 已提交
2001
	/* Validate handle. */
2002
	if (handle < req->num_outstanding_cmds)
2003
		sp = req->outstanding_cmds[handle];
2004
	else
L
Linus Torvalds 已提交
2005 2006 2007
		sp = NULL;

	if (sp == NULL) {
2008
		ql_dbg(ql_dbg_io, vha, 0x3017,
2009
		    "Invalid status handle (0x%x).\n", sts->handle);
L
Linus Torvalds 已提交
2010

2011 2012 2013 2014 2015 2016 2017
		if (!test_bit(ABORT_ISP_ACTIVE, &vha->dpc_flags)) {
			if (IS_P3P_TYPE(ha))
				set_bit(FCOE_CTX_RESET_NEEDED, &vha->dpc_flags);
			else
				set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
			qla2xxx_wake_dpc(vha);
		}
L
Linus Torvalds 已提交
2018 2019
		return;
	}
2020 2021 2022 2023 2024 2025

	if (unlikely((state_flags & BIT_1) && (sp->type == SRB_BIDI_CMD))) {
		qla25xx_process_bidir_status_iocb(vha, pkt, req, handle);
		return;
	}

2026 2027 2028 2029 2030 2031
	/* Task Management completion. */
	if (sp->type == SRB_TM_CMD) {
		qla24xx_tm_iocb_entry(vha, req, pkt);
		return;
	}

2032 2033 2034 2035 2036 2037 2038 2039
	/* Fast path completion. */
	if (comp_status == CS_COMPLETE && scsi_status == 0) {
		qla2x00_process_completed_request(vha, req, handle);

		return;
	}

	req->outstanding_cmds[handle] = NULL;
2040
	cp = GET_CMD_SP(sp);
L
Linus Torvalds 已提交
2041
	if (cp == NULL) {
2042
		ql_dbg(ql_dbg_io, vha, 0x3018,
2043 2044
		    "Command already returned (0x%x/%p).\n",
		    sts->handle, sp);
L
Linus Torvalds 已提交
2045 2046 2047 2048

		return;
	}

2049
	lscsi_status = scsi_status & STATUS_MASK;
L
Linus Torvalds 已提交
2050

2051
	fcport = sp->fcport;
L
Linus Torvalds 已提交
2052

2053
	ox_id = 0;
2054 2055
	sense_len = par_sense_len = rsp_info_len = resid_len =
	    fw_resid_len = 0;
2056
	if (IS_FWI2_CAPABLE(ha)) {
2057 2058 2059 2060 2061 2062 2063 2064
		if (scsi_status & SS_SENSE_LEN_VALID)
			sense_len = le32_to_cpu(sts24->sense_len);
		if (scsi_status & SS_RESPONSE_INFO_LEN_VALID)
			rsp_info_len = le32_to_cpu(sts24->rsp_data_len);
		if (scsi_status & (SS_RESIDUAL_UNDER | SS_RESIDUAL_OVER))
			resid_len = le32_to_cpu(sts24->rsp_residual_count);
		if (comp_status == CS_DATA_UNDERRUN)
			fw_resid_len = le32_to_cpu(sts24->residual_len);
2065 2066 2067
		rsp_info = sts24->data;
		sense_data = sts24->data;
		host_to_fcp_swap(sts24->data, sizeof(sts24->data));
2068
		ox_id = le16_to_cpu(sts24->ox_id);
2069
		par_sense_len = sizeof(sts24->data);
2070
	} else {
2071 2072 2073 2074
		if (scsi_status & SS_SENSE_LEN_VALID)
			sense_len = le16_to_cpu(sts->req_sense_length);
		if (scsi_status & SS_RESPONSE_INFO_LEN_VALID)
			rsp_info_len = le16_to_cpu(sts->rsp_info_len);
2075 2076 2077
		resid_len = le32_to_cpu(sts->residual_length);
		rsp_info = sts->rsp_info;
		sense_data = sts->req_sense_data;
2078
		par_sense_len = sizeof(sts->req_sense_data);
2079 2080
	}

L
Linus Torvalds 已提交
2081 2082
	/* Check for any FCP transport errors. */
	if (scsi_status & SS_RESPONSE_INFO_LEN_VALID) {
2083
		/* Sense data lies beyond any FCP RESPONSE data. */
2084
		if (IS_FWI2_CAPABLE(ha)) {
2085
			sense_data += rsp_info_len;
2086 2087
			par_sense_len -= rsp_info_len;
		}
2088
		if (rsp_info_len > 3 && rsp_info[3]) {
2089
			ql_dbg(ql_dbg_io, fcport->vha, 0x3019,
2090 2091
			    "FCP I/O protocol failure (0x%x/0x%x).\n",
			    rsp_info_len, rsp_info[3]);
L
Linus Torvalds 已提交
2092

2093
			res = DID_BUS_BUSY << 16;
2094
			goto out;
L
Linus Torvalds 已提交
2095 2096 2097
		}
	}

2098 2099 2100 2101 2102
	/* Check for overrun. */
	if (IS_FWI2_CAPABLE(ha) && comp_status == CS_COMPLETE &&
	    scsi_status & SS_RESIDUAL_OVER)
		comp_status = CS_DATA_OVERRUN;

L
Linus Torvalds 已提交
2103 2104 2105 2106 2107
	/*
	 * Based on Host and scsi status generate status code for Linux
	 */
	switch (comp_status) {
	case CS_COMPLETE:
2108
	case CS_QUEUE_FULL:
L
Linus Torvalds 已提交
2109
		if (scsi_status == 0) {
2110
			res = DID_OK << 16;
L
Linus Torvalds 已提交
2111 2112 2113
			break;
		}
		if (scsi_status & (SS_RESIDUAL_UNDER | SS_RESIDUAL_OVER)) {
2114
			resid = resid_len;
2115
			scsi_set_resid(cp, resid);
2116 2117

			if (!lscsi_status &&
2118
			    ((unsigned)(scsi_bufflen(cp) - resid) <
2119
			     cp->underflow)) {
2120
				ql_dbg(ql_dbg_io, fcport->vha, 0x301a,
2121
				    "Mid-layer underflow "
2122
				    "detected (0x%x of 0x%x bytes).\n",
2123
				    resid, scsi_bufflen(cp));
2124

2125
				res = DID_ERROR << 16;
2126 2127
				break;
			}
L
Linus Torvalds 已提交
2128
		}
2129
		res = DID_OK << 16 | lscsi_status;
L
Linus Torvalds 已提交
2130

2131
		if (lscsi_status == SAM_STAT_TASK_SET_FULL) {
2132
			ql_dbg(ql_dbg_io, fcport->vha, 0x301b,
2133
			    "QUEUE FULL detected.\n");
2134 2135
			break;
		}
2136
		logit = 0;
L
Linus Torvalds 已提交
2137 2138 2139
		if (lscsi_status != SS_CHECK_CONDITION)
			break;

2140
		memset(cp->sense_buffer, 0, SCSI_SENSE_BUFFERSIZE);
L
Linus Torvalds 已提交
2141 2142 2143
		if (!(scsi_status & SS_SENSE_LEN_VALID))
			break;

2144
		qla2x00_handle_sense(sp, sense_data, par_sense_len, sense_len,
2145
		    rsp, res);
L
Linus Torvalds 已提交
2146 2147 2148
		break;

	case CS_DATA_UNDERRUN:
2149
		/* Use F/W calculated residual length. */
2150 2151 2152 2153
		resid = IS_FWI2_CAPABLE(ha) ? fw_resid_len : resid_len;
		scsi_set_resid(cp, resid);
		if (scsi_status & SS_RESIDUAL_UNDER) {
			if (IS_FWI2_CAPABLE(ha) && fw_resid_len != resid_len) {
2154
				ql_dbg(ql_dbg_io, fcport->vha, 0x301d,
2155 2156 2157
				    "Dropped frame(s) detected "
				    "(0x%x of 0x%x bytes).\n",
				    resid, scsi_bufflen(cp));
2158

2159
				res = DID_ERROR << 16 | lscsi_status;
2160
				goto check_scsi_status;
2161
			}
2162

2163 2164 2165
			if (!lscsi_status &&
			    ((unsigned)(scsi_bufflen(cp) - resid) <
			    cp->underflow)) {
2166
				ql_dbg(ql_dbg_io, fcport->vha, 0x301e,
2167
				    "Mid-layer underflow "
2168
				    "detected (0x%x of 0x%x bytes).\n",
2169
				    resid, scsi_bufflen(cp));
2170

2171
				res = DID_ERROR << 16;
2172 2173
				break;
			}
2174 2175 2176 2177 2178 2179 2180
		} else if (lscsi_status != SAM_STAT_TASK_SET_FULL &&
			    lscsi_status != SAM_STAT_BUSY) {
			/*
			 * scsi status of task set and busy are considered to be
			 * task not completed.
			 */

2181
			ql_dbg(ql_dbg_io, fcport->vha, 0x301f,
2182
			    "Dropped frame(s) detected (0x%x "
2183 2184
			    "of 0x%x bytes).\n", resid,
			    scsi_bufflen(cp));
2185

2186
			res = DID_ERROR << 16 | lscsi_status;
2187
			goto check_scsi_status;
2188 2189 2190 2191
		} else {
			ql_dbg(ql_dbg_io, fcport->vha, 0x3030,
			    "scsi_status: 0x%x, lscsi_status: 0x%x\n",
			    scsi_status, lscsi_status);
L
Linus Torvalds 已提交
2192 2193
		}

2194
		res = DID_OK << 16 | lscsi_status;
2195
		logit = 0;
2196

2197
check_scsi_status:
L
Linus Torvalds 已提交
2198
		/*
A
Andrew Vasquez 已提交
2199
		 * Check to see if SCSI Status is non zero. If so report SCSI
L
Linus Torvalds 已提交
2200 2201 2202
		 * Status.
		 */
		if (lscsi_status != 0) {
2203
			if (lscsi_status == SAM_STAT_TASK_SET_FULL) {
2204
				ql_dbg(ql_dbg_io, fcport->vha, 0x3020,
2205
				    "QUEUE FULL detected.\n");
2206
				logit = 1;
2207 2208
				break;
			}
L
Linus Torvalds 已提交
2209 2210 2211
			if (lscsi_status != SS_CHECK_CONDITION)
				break;

2212
			memset(cp->sense_buffer, 0, SCSI_SENSE_BUFFERSIZE);
L
Linus Torvalds 已提交
2213 2214 2215
			if (!(scsi_status & SS_SENSE_LEN_VALID))
				break;

2216
			qla2x00_handle_sense(sp, sense_data, par_sense_len,
2217
			    sense_len, rsp, res);
L
Linus Torvalds 已提交
2218 2219 2220 2221 2222 2223 2224 2225
		}
		break;

	case CS_PORT_LOGGED_OUT:
	case CS_PORT_CONFIG_CHG:
	case CS_PORT_BUSY:
	case CS_INCOMPLETE:
	case CS_PORT_UNAVAILABLE:
2226
	case CS_TIMEOUT:
2227 2228
	case CS_RESET:

2229 2230 2231 2232 2233
		/*
		 * We are going to have the fc class block the rport
		 * while we try to recover so instruct the mid layer
		 * to requeue until the class decides how to handle this.
		 */
2234
		res = DID_TRANSPORT_DISRUPTED << 16;
2235 2236 2237 2238 2239 2240 2241 2242 2243

		if (comp_status == CS_TIMEOUT) {
			if (IS_FWI2_CAPABLE(ha))
				break;
			else if ((le16_to_cpu(sts->status_flags) &
			    SF_LOGOUT_SENT) == 0)
				break;
		}

2244
		ql_dbg(ql_dbg_io, fcport->vha, 0x3021,
2245 2246 2247 2248
		    "Port to be marked lost on fcport=%02x%02x%02x, current "
		    "port state= %s.\n", fcport->d_id.b.domain,
		    fcport->d_id.b.area, fcport->d_id.b.al_pa,
		    port_state_str[atomic_read(&fcport->state)]);
2249

2250
		if (atomic_read(&fcport->state) == FCS_ONLINE)
2251
			qla2x00_mark_device_lost(fcport->vha, fcport, 1, 1);
L
Linus Torvalds 已提交
2252 2253 2254
		break;

	case CS_ABORTED:
2255
		res = DID_RESET << 16;
L
Linus Torvalds 已提交
2256
		break;
2257 2258

	case CS_DIF_ERROR:
2259
		logit = qla2x00_handle_dif_error(sp, sts24);
2260
		res = cp->result;
2261
		break;
2262 2263 2264 2265 2266 2267 2268 2269 2270 2271 2272 2273 2274

	case CS_TRANSPORT:
		res = DID_ERROR << 16;

		if (!IS_PI_SPLIT_DET_CAPABLE(ha))
			break;

		if (state_flags & BIT_4)
			scmd_printk(KERN_WARNING, cp,
			    "Unsupported device '%s' found.\n",
			    cp->device->vendor);
		break;

L
Linus Torvalds 已提交
2275
	default:
2276
		res = DID_ERROR << 16;
L
Linus Torvalds 已提交
2277 2278 2279
		break;
	}

2280 2281
out:
	if (logit)
2282
		ql_dbg(ql_dbg_io, fcport->vha, 0x3022,
H
Hannes Reinecke 已提交
2283
		    "FCP command status: 0x%x-0x%x (0x%x) nexus=%ld:%d:%llu "
2284
		    "portid=%02x%02x%02x oxid=0x%x cdb=%10phN len=0x%x "
2285
		    "rsp_info=0x%x resid=0x%x fw_resid=0x%x.\n",
2286
		    comp_status, scsi_status, res, vha->host_no,
2287 2288
		    cp->device->id, cp->device->lun, fcport->d_id.b.domain,
		    fcport->d_id.b.area, fcport->d_id.b.al_pa, ox_id,
2289
		    cp->cmnd, scsi_bufflen(cp), rsp_info_len,
2290
		    resid_len, fw_resid_len);
2291

2292
	if (rsp->status_srb == NULL)
2293
		sp->done(ha, sp, res);
L
Linus Torvalds 已提交
2294 2295 2296 2297 2298 2299 2300 2301 2302 2303
}

/**
 * qla2x00_status_cont_entry() - Process a Status Continuations entry.
 * @ha: SCSI driver HA context
 * @pkt: Entry pointer
 *
 * Extended sense data.
 */
static void
2304
qla2x00_status_cont_entry(struct rsp_que *rsp, sts_cont_entry_t *pkt)
L
Linus Torvalds 已提交
2305
{
2306
	uint8_t	sense_sz = 0;
2307
	struct qla_hw_data *ha = rsp->hw;
2308
	struct scsi_qla_host *vha = pci_get_drvdata(ha->pdev);
2309
	srb_t *sp = rsp->status_srb;
L
Linus Torvalds 已提交
2310
	struct scsi_cmnd *cp;
2311 2312
	uint32_t sense_len;
	uint8_t *sense_ptr;
L
Linus Torvalds 已提交
2313

2314 2315
	if (!sp || !GET_CMD_SENSE_LEN(sp))
		return;
L
Linus Torvalds 已提交
2316

2317 2318
	sense_len = GET_CMD_SENSE_LEN(sp);
	sense_ptr = GET_CMD_SENSE_PTR(sp);
L
Linus Torvalds 已提交
2319

2320 2321 2322 2323
	cp = GET_CMD_SP(sp);
	if (cp == NULL) {
		ql_log(ql_log_warn, vha, 0x3025,
		    "cmd is NULL: already returned to OS (sp=%p).\n", sp);
L
Linus Torvalds 已提交
2324

2325 2326
		rsp->status_srb = NULL;
		return;
L
Linus Torvalds 已提交
2327 2328
	}

2329 2330 2331 2332
	if (sense_len > sizeof(pkt->data))
		sense_sz = sizeof(pkt->data);
	else
		sense_sz = sense_len;
2333

2334 2335 2336 2337 2338 2339
	/* Move sense data. */
	if (IS_FWI2_CAPABLE(ha))
		host_to_fcp_swap(pkt->data, sizeof(pkt->data));
	memcpy(sense_ptr, pkt->data, sense_sz);
	ql_dump_buffer(ql_dbg_io + ql_dbg_buffer, vha, 0x302c,
		sense_ptr, sense_sz);
2340

2341 2342
	sense_len -= sense_sz;
	sense_ptr += sense_sz;
2343

2344 2345 2346 2347 2348 2349 2350
	SET_CMD_SENSE_PTR(sp, sense_ptr);
	SET_CMD_SENSE_LEN(sp, sense_len);

	/* Place command on done queue. */
	if (sense_len == 0) {
		rsp->status_srb = NULL;
		sp->done(ha, sp, cp->result);
2351 2352 2353
	}
}

L
Linus Torvalds 已提交
2354 2355 2356 2357 2358 2359
/**
 * qla2x00_error_entry() - Process an error entry.
 * @ha: SCSI driver HA context
 * @pkt: Entry pointer
 */
static void
2360
qla2x00_error_entry(scsi_qla_host_t *vha, struct rsp_que *rsp, sts_entry_t *pkt)
L
Linus Torvalds 已提交
2361 2362
{
	srb_t *sp;
2363
	struct qla_hw_data *ha = vha->hw;
2364
	const char func[] = "ERROR-IOCB";
2365
	uint16_t que = MSW(pkt->handle);
2366
	struct req_que *req = NULL;
2367
	int res = DID_ERROR << 16;
2368

2369 2370 2371
	ql_dbg(ql_dbg_async, vha, 0x502a,
	    "type of error status in response: 0x%x\n", pkt->entry_status);

2372 2373 2374 2375 2376
	if (que >= ha->max_req_queues || !ha->req_q_map[que])
		goto fatal;

	req = ha->req_q_map[que];

2377 2378
	if (pkt->entry_status & RF_BUSY)
		res = DID_BUS_BUSY << 16;
L
Linus Torvalds 已提交
2379

2380
	sp = qla2x00_get_sp_from_handle(vha, func, req, pkt);
2381
	if (sp) {
2382
		sp->done(ha, sp, res);
2383
		return;
L
Linus Torvalds 已提交
2384
	}
2385 2386 2387 2388
fatal:
	ql_log(ql_log_warn, vha, 0x5030,
	    "Error entry - invalid handle/queue.\n");

2389
	if (IS_P3P_TYPE(ha))
2390 2391 2392 2393
		set_bit(FCOE_CTX_RESET_NEEDED, &vha->dpc_flags);
	else
		set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
	qla2xxx_wake_dpc(vha);
L
Linus Torvalds 已提交
2394 2395
}

2396 2397 2398 2399 2400 2401
/**
 * qla24xx_mbx_completion() - Process mailbox command completions.
 * @ha: SCSI driver HA context
 * @mb0: Mailbox0 register
 */
static void
2402
qla24xx_mbx_completion(scsi_qla_host_t *vha, uint16_t mb0)
2403 2404
{
	uint16_t	cnt;
2405
	uint32_t	mboxes;
2406
	uint16_t __iomem *wptr;
2407
	struct qla_hw_data *ha = vha->hw;
2408 2409
	struct device_reg_24xx __iomem *reg = &ha->iobase->isp24;

2410 2411 2412
	/* Read all mbox registers? */
	mboxes = (1 << ha->mbx_count) - 1;
	if (!ha->mcp)
2413
		ql_dbg(ql_dbg_async, vha, 0x504e, "MBX pointer ERROR.\n");
2414 2415 2416
	else
		mboxes = ha->mcp->in_mb;

2417 2418 2419
	/* Load return mailbox registers. */
	ha->flags.mbox_int = 1;
	ha->mailbox_out[0] = mb0;
2420
	mboxes >>= 1;
2421 2422 2423
	wptr = (uint16_t __iomem *)&reg->mailbox1;

	for (cnt = 1; cnt < ha->mbx_count; cnt++) {
2424 2425 2426 2427
		if (mboxes & BIT_0)
			ha->mailbox_out[cnt] = RD_REG_WORD(wptr);

		mboxes >>= 1;
2428 2429 2430 2431
		wptr++;
	}
}

2432 2433 2434 2435 2436 2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448
static void
qla24xx_abort_iocb_entry(scsi_qla_host_t *vha, struct req_que *req,
	struct abort_entry_24xx *pkt)
{
	const char func[] = "ABT_IOCB";
	srb_t *sp;
	struct srb_iocb *abt;

	sp = qla2x00_get_sp_from_handle(vha, func, req, pkt);
	if (!sp)
		return;

	abt = &sp->u.iocb_cmd;
	abt->u.abt.comp_status = le32_to_cpu(pkt->nport_handle);
	sp->done(vha, sp, 0);
}

2449 2450 2451 2452
/**
 * qla24xx_process_response_queue() - Process response queue entries.
 * @ha: SCSI driver HA context
 */
2453 2454
void qla24xx_process_response_queue(struct scsi_qla_host *vha,
	struct rsp_que *rsp)
2455 2456
{
	struct sts_entry_24xx *pkt;
2457
	struct qla_hw_data *ha = vha->hw;
2458

2459
	if (!vha->flags.online)
2460 2461
		return;

2462 2463
	while (rsp->ring_ptr->signature != RESPONSE_PROCESSED) {
		pkt = (struct sts_entry_24xx *)rsp->ring_ptr;
2464

2465 2466 2467 2468
		rsp->ring_index++;
		if (rsp->ring_index == rsp->length) {
			rsp->ring_index = 0;
			rsp->ring_ptr = rsp->ring;
2469
		} else {
2470
			rsp->ring_ptr++;
2471 2472 2473
		}

		if (pkt->entry_status != 0) {
2474
			qla2x00_error_entry(vha, rsp, (sts_entry_t *) pkt);
2475

2476 2477
			if (qlt_24xx_process_response_error(vha, pkt))
				goto process_err;
2478

2479 2480 2481 2482
			((response_t *)pkt)->signature = RESPONSE_PROCESSED;
			wmb();
			continue;
		}
2483
process_err:
2484 2485 2486

		switch (pkt->entry_type) {
		case STATUS_TYPE:
2487
			qla2x00_status_entry(vha, rsp, pkt);
2488 2489
			break;
		case STATUS_CONT_TYPE:
2490
			qla2x00_status_cont_entry(rsp, (sts_cont_entry_t *)pkt);
2491
			break;
2492
		case VP_RPT_ID_IOCB_TYPE:
2493
			qla24xx_report_id_acquisition(vha,
2494 2495
			    (struct vp_rpt_id_entry_24xx *)pkt);
			break;
2496 2497 2498 2499
		case LOGINOUT_PORT_IOCB_TYPE:
			qla24xx_logio_entry(vha, rsp->req,
			    (struct logio_entry_24xx *)pkt);
			break;
2500
		case CT_IOCB_TYPE:
2501 2502
			qla24xx_els_ct_entry(vha, rsp->req, pkt, CT_IOCB_TYPE);
			break;
2503
		case ELS_IOCB_TYPE:
2504 2505
			qla24xx_els_ct_entry(vha, rsp->req, pkt, ELS_IOCB_TYPE);
			break;
2506 2507 2508 2509 2510 2511
		case ABTS_RECV_24XX:
			/* ensure that the ATIO queue is empty */
			qlt_24xx_process_atio_queue(vha);
		case ABTS_RESP_24XX:
		case CTIO_TYPE7:
		case NOTIFY_ACK_TYPE:
2512
		case CTIO_CRC2:
2513 2514
			qlt_response_pkt_all_vps(vha, (response_t *)pkt);
			break;
2515 2516 2517 2518 2519
		case MARKER_TYPE:
			/* Do nothing in this case, this check is to prevent it
			 * from falling into default case
			 */
			break;
2520 2521 2522 2523
		case ABORT_IOCB_TYPE:
			qla24xx_abort_iocb_entry(vha, rsp->req,
			    (struct abort_entry_24xx *)pkt);
			break;
2524 2525
		default:
			/* Type Not Supported. */
2526 2527
			ql_dbg(ql_dbg_async, vha, 0x5042,
			    "Received unknown response pkt type %x "
2528
			    "entry status=%x.\n",
2529
			    pkt->entry_type, pkt->entry_status);
2530 2531 2532 2533 2534 2535 2536
			break;
		}
		((response_t *)pkt)->signature = RESPONSE_PROCESSED;
		wmb();
	}

	/* Adjust ring index */
2537
	if (IS_P3P_TYPE(ha)) {
2538 2539 2540 2541
		struct device_reg_82xx __iomem *reg = &ha->iobase->isp82;
		WRT_REG_DWORD(&reg->rsp_q_out[0], rsp->ring_index);
	} else
		WRT_REG_DWORD(rsp->rsp_q_out, rsp->ring_index);
2542 2543
}

2544
static void
2545
qla2xxx_check_risc_status(scsi_qla_host_t *vha)
2546 2547 2548
{
	int rval;
	uint32_t cnt;
2549
	struct qla_hw_data *ha = vha->hw;
2550 2551
	struct device_reg_24xx __iomem *reg = &ha->iobase->isp24;

2552 2553
	if (!IS_QLA25XX(ha) && !IS_QLA81XX(ha) && !IS_QLA83XX(ha) &&
	    !IS_QLA27XX(ha))
2554 2555 2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570
		return;

	rval = QLA_SUCCESS;
	WRT_REG_DWORD(&reg->iobase_addr, 0x7C00);
	RD_REG_DWORD(&reg->iobase_addr);
	WRT_REG_DWORD(&reg->iobase_window, 0x0001);
	for (cnt = 10000; (RD_REG_DWORD(&reg->iobase_window) & BIT_0) == 0 &&
	    rval == QLA_SUCCESS; cnt--) {
		if (cnt) {
			WRT_REG_DWORD(&reg->iobase_window, 0x0001);
			udelay(10);
		} else
			rval = QLA_FUNCTION_TIMEOUT;
	}
	if (rval == QLA_SUCCESS)
		goto next_test;

2571
	rval = QLA_SUCCESS;
2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585
	WRT_REG_DWORD(&reg->iobase_window, 0x0003);
	for (cnt = 100; (RD_REG_DWORD(&reg->iobase_window) & BIT_0) == 0 &&
	    rval == QLA_SUCCESS; cnt--) {
		if (cnt) {
			WRT_REG_DWORD(&reg->iobase_window, 0x0003);
			udelay(10);
		} else
			rval = QLA_FUNCTION_TIMEOUT;
	}
	if (rval != QLA_SUCCESS)
		goto done;

next_test:
	if (RD_REG_DWORD(&reg->iobase_c8) & BIT_3)
2586 2587
		ql_log(ql_log_info, vha, 0x504c,
		    "Additional code -- 0x55AA.\n");
2588 2589 2590 2591 2592 2593

done:
	WRT_REG_DWORD(&reg->iobase_window, 0x0000);
	RD_REG_DWORD(&reg->iobase_window);
}

2594
/**
2595
 * qla24xx_intr_handler() - Process interrupts for the ISP23xx and ISP24xx.
2596 2597 2598 2599 2600 2601 2602 2603
 * @irq:
 * @dev_id: SCSI driver HA context
 *
 * Called by system whenever the host adapter generates an interrupt.
 *
 * Returns handled flag.
 */
irqreturn_t
2604
qla24xx_intr_handler(int irq, void *dev_id)
2605
{
2606 2607
	scsi_qla_host_t	*vha;
	struct qla_hw_data *ha;
2608 2609 2610 2611 2612
	struct device_reg_24xx __iomem *reg;
	int		status;
	unsigned long	iter;
	uint32_t	stat;
	uint32_t	hccr;
2613
	uint16_t	mb[8];
2614
	struct rsp_que *rsp;
2615
	unsigned long	flags;
2616

2617 2618
	rsp = (struct rsp_que *) dev_id;
	if (!rsp) {
2619 2620
		ql_log(ql_log_info, NULL, 0x5059,
		    "%s: NULL response queue pointer.\n", __func__);
2621 2622 2623
		return IRQ_NONE;
	}

2624
	ha = rsp->hw;
2625 2626 2627
	reg = &ha->iobase->isp24;
	status = 0;

2628 2629 2630
	if (unlikely(pci_channel_offline(ha->pdev)))
		return IRQ_HANDLED;

2631
	spin_lock_irqsave(&ha->hardware_lock, flags);
2632
	vha = pci_get_drvdata(ha->pdev);
2633 2634
	for (iter = 50; iter--; ) {
		stat = RD_REG_DWORD(&reg->host_status);
2635
		if (qla2x00_check_reg32_for_disconnect(vha, stat))
2636
			break;
2637
		if (stat & HSRX_RISC_PAUSED) {
2638
			if (unlikely(pci_channel_offline(ha->pdev)))
2639 2640
				break;

2641 2642
			hccr = RD_REG_DWORD(&reg->hccr);

2643 2644 2645
			ql_log(ql_log_warn, vha, 0x504b,
			    "RISC paused -- HCCR=%x, Dumping firmware.\n",
			    hccr);
2646

2647
			qla2xxx_check_risc_status(vha);
2648

2649 2650
			ha->isp_ops->fw_dump(vha, 1);
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
2651 2652 2653 2654 2655
			break;
		} else if ((stat & HSRX_RISC_INT) == 0)
			break;

		switch (stat & 0xff) {
2656 2657 2658 2659
		case INTR_ROM_MB_SUCCESS:
		case INTR_ROM_MB_FAILED:
		case INTR_MB_SUCCESS:
		case INTR_MB_FAILED:
2660
			qla24xx_mbx_completion(vha, MSW(stat));
2661 2662 2663
			status |= MBX_INTERRUPT;

			break;
2664
		case INTR_ASYNC_EVENT:
2665 2666 2667 2668
			mb[0] = MSW(stat);
			mb[1] = RD_REG_WORD(&reg->mailbox1);
			mb[2] = RD_REG_WORD(&reg->mailbox2);
			mb[3] = RD_REG_WORD(&reg->mailbox3);
2669
			qla2x00_async_event(vha, rsp, mb);
2670
			break;
2671 2672
		case INTR_RSP_QUE_UPDATE:
		case INTR_RSP_QUE_UPDATE_83XX:
2673
			qla24xx_process_response_queue(vha, rsp);
2674
			break;
2675
		case INTR_ATIO_QUE_UPDATE:
2676 2677
			qlt_24xx_process_atio_queue(vha);
			break;
2678
		case INTR_ATIO_RSP_QUE_UPDATE:
2679 2680 2681
			qlt_24xx_process_atio_queue(vha);
			qla24xx_process_response_queue(vha, rsp);
			break;
2682
		default:
2683 2684
			ql_dbg(ql_dbg_async, vha, 0x504f,
			    "Unrecognized interrupt type (%d).\n", stat * 0xff);
2685 2686 2687 2688
			break;
		}
		WRT_REG_DWORD(&reg->hccr, HCCRX_CLR_RISC_INT);
		RD_REG_DWORD_RELAXED(&reg->hccr);
2689 2690
		if (unlikely(IS_QLA83XX(ha) && (ha->pdev->revision == 1)))
			ndelay(3500);
2691
	}
2692
	qla2x00_handle_mbx_completion(ha, status);
2693
	spin_unlock_irqrestore(&ha->hardware_lock, flags);
2694 2695 2696 2697

	return IRQ_HANDLED;
}

2698 2699 2700
static irqreturn_t
qla24xx_msix_rsp_q(int irq, void *dev_id)
{
2701 2702
	struct qla_hw_data *ha;
	struct rsp_que *rsp;
2703
	struct device_reg_24xx __iomem *reg;
2704
	struct scsi_qla_host *vha;
2705
	unsigned long flags;
2706
	uint32_t stat = 0;
2707

2708 2709
	rsp = (struct rsp_que *) dev_id;
	if (!rsp) {
2710 2711
		ql_log(ql_log_info, NULL, 0x505a,
		    "%s: NULL response queue pointer.\n", __func__);
2712 2713 2714
		return IRQ_NONE;
	}
	ha = rsp->hw;
2715 2716
	reg = &ha->iobase->isp24;

2717
	spin_lock_irqsave(&ha->hardware_lock, flags);
2718

2719
	vha = pci_get_drvdata(ha->pdev);
2720 2721 2722 2723 2724
	/*
	 * Use host_status register to check to PCI disconnection before we
	 * we process the response queue.
	 */
	stat = RD_REG_DWORD(&reg->host_status);
2725
	if (qla2x00_check_reg32_for_disconnect(vha, stat))
2726
		goto out;
2727
	qla24xx_process_response_queue(vha, rsp);
2728
	if (!ha->flags.disable_msix_handshake) {
2729 2730 2731
		WRT_REG_DWORD(&reg->hccr, HCCRX_CLR_RISC_INT);
		RD_REG_DWORD_RELAXED(&reg->hccr);
	}
2732
out:
2733
	spin_unlock_irqrestore(&ha->hardware_lock, flags);
2734 2735 2736 2737

	return IRQ_HANDLED;
}

2738 2739 2740 2741
static irqreturn_t
qla25xx_msix_rsp_q(int irq, void *dev_id)
{
	struct qla_hw_data *ha;
2742
	scsi_qla_host_t *vha;
2743
	struct rsp_que *rsp;
2744
	struct device_reg_24xx __iomem *reg;
2745
	unsigned long flags;
2746
	uint32_t hccr = 0;
2747 2748 2749

	rsp = (struct rsp_que *) dev_id;
	if (!rsp) {
2750 2751
		ql_log(ql_log_info, NULL, 0x505b,
		    "%s: NULL response queue pointer.\n", __func__);
2752 2753 2754
		return IRQ_NONE;
	}
	ha = rsp->hw;
2755
	vha = pci_get_drvdata(ha->pdev);
2756

2757
	/* Clear the interrupt, if enabled, for this response queue */
2758
	if (!ha->flags.disable_msix_handshake) {
2759
		reg = &ha->iobase->isp24;
2760
		spin_lock_irqsave(&ha->hardware_lock, flags);
2761
		WRT_REG_DWORD(&reg->hccr, HCCRX_CLR_RISC_INT);
2762
		hccr = RD_REG_DWORD_RELAXED(&reg->hccr);
2763
		spin_unlock_irqrestore(&ha->hardware_lock, flags);
2764
	}
2765
	if (qla2x00_check_reg32_for_disconnect(vha, hccr))
2766
		goto out;
2767 2768
	queue_work_on((int) (rsp->id - 1), ha->wq, &rsp->q_work);

2769
out:
2770 2771 2772
	return IRQ_HANDLED;
}

2773 2774 2775
static irqreturn_t
qla24xx_msix_default(int irq, void *dev_id)
{
2776 2777 2778
	scsi_qla_host_t	*vha;
	struct qla_hw_data *ha;
	struct rsp_que *rsp;
2779 2780 2781 2782
	struct device_reg_24xx __iomem *reg;
	int		status;
	uint32_t	stat;
	uint32_t	hccr;
2783
	uint16_t	mb[8];
2784
	unsigned long flags;
2785

2786 2787
	rsp = (struct rsp_que *) dev_id;
	if (!rsp) {
2788 2789
		ql_log(ql_log_info, NULL, 0x505c,
		    "%s: NULL response queue pointer.\n", __func__);
2790 2791 2792
		return IRQ_NONE;
	}
	ha = rsp->hw;
2793 2794 2795
	reg = &ha->iobase->isp24;
	status = 0;

2796
	spin_lock_irqsave(&ha->hardware_lock, flags);
2797
	vha = pci_get_drvdata(ha->pdev);
2798
	do {
2799
		stat = RD_REG_DWORD(&reg->host_status);
2800
		if (qla2x00_check_reg32_for_disconnect(vha, stat))
2801
			break;
2802
		if (stat & HSRX_RISC_PAUSED) {
2803
			if (unlikely(pci_channel_offline(ha->pdev)))
2804 2805
				break;

2806 2807
			hccr = RD_REG_DWORD(&reg->hccr);

2808 2809 2810
			ql_log(ql_log_info, vha, 0x5050,
			    "RISC paused -- HCCR=%x, Dumping firmware.\n",
			    hccr);
2811

2812
			qla2xxx_check_risc_status(vha);
2813

2814 2815
			ha->isp_ops->fw_dump(vha, 1);
			set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
2816 2817 2818 2819 2820
			break;
		} else if ((stat & HSRX_RISC_INT) == 0)
			break;

		switch (stat & 0xff) {
2821 2822 2823 2824
		case INTR_ROM_MB_SUCCESS:
		case INTR_ROM_MB_FAILED:
		case INTR_MB_SUCCESS:
		case INTR_MB_FAILED:
2825
			qla24xx_mbx_completion(vha, MSW(stat));
2826 2827 2828
			status |= MBX_INTERRUPT;

			break;
2829
		case INTR_ASYNC_EVENT:
2830 2831 2832 2833
			mb[0] = MSW(stat);
			mb[1] = RD_REG_WORD(&reg->mailbox1);
			mb[2] = RD_REG_WORD(&reg->mailbox2);
			mb[3] = RD_REG_WORD(&reg->mailbox3);
2834
			qla2x00_async_event(vha, rsp, mb);
2835
			break;
2836 2837
		case INTR_RSP_QUE_UPDATE:
		case INTR_RSP_QUE_UPDATE_83XX:
2838
			qla24xx_process_response_queue(vha, rsp);
2839
			break;
2840
		case INTR_ATIO_QUE_UPDATE:
2841 2842
			qlt_24xx_process_atio_queue(vha);
			break;
2843
		case INTR_ATIO_RSP_QUE_UPDATE:
2844 2845 2846
			qlt_24xx_process_atio_queue(vha);
			qla24xx_process_response_queue(vha, rsp);
			break;
2847
		default:
2848 2849
			ql_dbg(ql_dbg_async, vha, 0x5051,
			    "Unrecognized interrupt type (%d).\n", stat & 0xff);
2850 2851 2852
			break;
		}
		WRT_REG_DWORD(&reg->hccr, HCCRX_CLR_RISC_INT);
2853
	} while (0);
2854
	qla2x00_handle_mbx_completion(ha, status);
2855
	spin_unlock_irqrestore(&ha->hardware_lock, flags);
2856 2857 2858 2859 2860 2861 2862 2863

	return IRQ_HANDLED;
}

/* Interrupt handling helpers. */

struct qla_init_msix_entry {
	const char *name;
2864
	irq_handler_t handler;
2865 2866
};

2867
static struct qla_init_msix_entry msix_entries[3] = {
2868 2869
	{ "qla2xxx (default)", qla24xx_msix_default },
	{ "qla2xxx (rsp_q)", qla24xx_msix_rsp_q },
2870
	{ "qla2xxx (multiq)", qla25xx_msix_rsp_q },
2871 2872
};

2873 2874 2875 2876 2877
static struct qla_init_msix_entry qla82xx_msix_entries[2] = {
	{ "qla2xxx (default)", qla82xx_msix_default },
	{ "qla2xxx (rsp_q)", qla82xx_msix_rsp_q },
};

2878 2879 2880 2881 2882 2883
static struct qla_init_msix_entry qla83xx_msix_entries[3] = {
	{ "qla2xxx (default)", qla24xx_msix_default },
	{ "qla2xxx (rsp_q)", qla24xx_msix_rsp_q },
	{ "qla2xxx (atio_q)", qla83xx_msix_atio_q },
};

2884
static void
2885
qla24xx_disable_msix(struct qla_hw_data *ha)
2886 2887 2888
{
	int i;
	struct qla_msix_entry *qentry;
2889
	scsi_qla_host_t *vha = pci_get_drvdata(ha->pdev);
2890

2891 2892
	for (i = 0; i < ha->msix_count; i++) {
		qentry = &ha->msix_entries[i];
2893
		if (qentry->have_irq)
2894
			free_irq(qentry->vector, qentry->rsp);
2895 2896
	}
	pci_disable_msix(ha->pdev);
2897 2898 2899
	kfree(ha->msix_entries);
	ha->msix_entries = NULL;
	ha->flags.msix_enabled = 0;
2900 2901
	ql_dbg(ql_dbg_init, vha, 0x0042,
	    "Disabled the MSI.\n");
2902 2903 2904
}

static int
2905
qla24xx_enable_msix(struct qla_hw_data *ha, struct rsp_que *rsp)
2906
{
2907
#define MIN_MSIX_COUNT	2
2908
#define ATIO_VECTOR	2
2909
	int i, ret;
2910
	struct msix_entry *entries;
2911
	struct qla_msix_entry *qentry;
2912
	scsi_qla_host_t *vha = pci_get_drvdata(ha->pdev);
2913 2914

	entries = kzalloc(sizeof(struct msix_entry) * ha->msix_count,
2915
			GFP_KERNEL);
2916 2917 2918
	if (!entries) {
		ql_log(ql_log_warn, vha, 0x00bc,
		    "Failed to allocate memory for msix_entry.\n");
2919
		return -ENOMEM;
2920
	}
2921

2922 2923
	for (i = 0; i < ha->msix_count; i++)
		entries[i].entry = i;
2924

2925 2926 2927 2928 2929 2930 2931 2932 2933
	ret = pci_enable_msix_range(ha->pdev,
				    entries, MIN_MSIX_COUNT, ha->msix_count);
	if (ret < 0) {
		ql_log(ql_log_fatal, vha, 0x00c7,
		    "MSI-X: Failed to enable support, "
		    "giving   up -- %d/%d.\n",
		    ha->msix_count, ret);
		goto msix_out;
	} else if (ret < ha->msix_count) {
2934 2935 2936 2937
		ql_log(ql_log_warn, vha, 0x00c6,
		    "MSI-X: Failed to enable support "
		    "-- %d/%d\n Retry with %d vectors.\n",
		    ha->msix_count, ret, ret);
2938
	}
2939 2940
	ha->msix_count = ret;
	ha->max_rsp_queues = ha->msix_count - 1;
2941 2942 2943
	ha->msix_entries = kzalloc(sizeof(struct qla_msix_entry) *
				ha->msix_count, GFP_KERNEL);
	if (!ha->msix_entries) {
2944 2945
		ql_log(ql_log_fatal, vha, 0x00c8,
		    "Failed to allocate memory for ha->msix_entries.\n");
2946
		ret = -ENOMEM;
2947 2948 2949 2950
		goto msix_out;
	}
	ha->flags.msix_enabled = 1;

2951 2952 2953 2954
	for (i = 0; i < ha->msix_count; i++) {
		qentry = &ha->msix_entries[i];
		qentry->vector = entries[i].vector;
		qentry->entry = entries[i].entry;
2955
		qentry->have_irq = 0;
2956
		qentry->rsp = NULL;
2957 2958
	}

2959
	/* Enable MSI-X vectors for the base queue */
2960
	for (i = 0; i < 2; i++) {
2961
		qentry = &ha->msix_entries[i];
2962
		if (IS_P3P_TYPE(ha))
2963 2964 2965
			ret = request_irq(qentry->vector,
				qla82xx_msix_entries[i].handler,
				0, qla82xx_msix_entries[i].name, rsp);
2966
		else
2967 2968 2969
			ret = request_irq(qentry->vector,
				msix_entries[i].handler,
				0, msix_entries[i].name, rsp);
2970 2971 2972 2973 2974 2975 2976 2977 2978 2979 2980 2981 2982 2983 2984 2985
		if (ret)
			goto msix_register_fail;
		qentry->have_irq = 1;
		qentry->rsp = rsp;
		rsp->msix = qentry;
	}

	/*
	 * If target mode is enable, also request the vector for the ATIO
	 * queue.
	 */
	if (QLA_TGT_MODE_ENABLED() && IS_ATIO_MSIX_CAPABLE(ha)) {
		qentry = &ha->msix_entries[ATIO_VECTOR];
		ret = request_irq(qentry->vector,
			qla83xx_msix_entries[ATIO_VECTOR].handler,
			0, qla83xx_msix_entries[ATIO_VECTOR].name, rsp);
2986 2987 2988
		qentry->have_irq = 1;
		qentry->rsp = rsp;
		rsp->msix = qentry;
2989 2990
	}

2991 2992 2993 2994 2995 2996 2997 2998 2999 3000
msix_register_fail:
	if (ret) {
		ql_log(ql_log_fatal, vha, 0x00cb,
		    "MSI-X: unable to register handler -- %x/%d.\n",
		    qentry->vector, ret);
		qla24xx_disable_msix(ha);
		ha->mqenable = 0;
		goto msix_out;
	}

3001
	/* Enable MSI-X vector for response queue update for queue 0 */
3002
	if (IS_QLA83XX(ha) || IS_QLA27XX(ha)) {
3003 3004 3005 3006 3007 3008 3009
		if (ha->msixbase && ha->mqiobase &&
		    (ha->max_rsp_queues > 1 || ha->max_req_queues > 1))
			ha->mqenable = 1;
	} else
		if (ha->mqiobase
		    && (ha->max_rsp_queues > 1 || ha->max_req_queues > 1))
			ha->mqenable = 1;
3010 3011 3012 3013 3014 3015
	ql_dbg(ql_dbg_multiq, vha, 0xc005,
	    "mqiobase=%p, max_rsp_queues=%d, max_req_queues=%d.\n",
	    ha->mqiobase, ha->max_rsp_queues, ha->max_req_queues);
	ql_dbg(ql_dbg_init, vha, 0x0055,
	    "mqiobase=%p, max_rsp_queues=%d, max_req_queues=%d.\n",
	    ha->mqiobase, ha->max_rsp_queues, ha->max_req_queues);
3016

3017
msix_out:
3018
	kfree(entries);
3019 3020 3021 3022
	return ret;
}

int
3023
qla2x00_request_irqs(struct qla_hw_data *ha, struct rsp_que *rsp)
3024
{
3025
	int ret = QLA_FUNCTION_FAILED;
3026
	device_reg_t *reg = ha->iobase;
3027
	scsi_qla_host_t *vha = pci_get_drvdata(ha->pdev);
3028 3029

	/* If possible, enable MSI-X. */
3030
	if (!IS_QLA2432(ha) && !IS_QLA2532(ha) && !IS_QLA8432(ha) &&
3031 3032
	    !IS_CNA_CAPABLE(ha) && !IS_QLA2031(ha) && !IS_QLAFX00(ha) &&
	    !IS_QLA27XX(ha))
3033 3034 3035 3036 3037 3038
		goto skip_msi;

	if (ha->pdev->subsystem_vendor == PCI_VENDOR_ID_HP &&
		(ha->pdev->subsystem_device == 0x7040 ||
		ha->pdev->subsystem_device == 0x7041 ||
		ha->pdev->subsystem_device == 0x1705)) {
3039 3040
		ql_log(ql_log_warn, vha, 0x0034,
		    "MSI-X: Unsupported ISP 2432 SSVID/SSDID (0x%X,0x%X).\n",
3041
			ha->pdev->subsystem_vendor,
3042
			ha->pdev->subsystem_device);
3043 3044
		goto skip_msi;
	}
3045

3046
	if (IS_QLA2432(ha) && (ha->pdev->revision < QLA_MSIX_CHIP_REV_24XX)) {
3047 3048
		ql_log(ql_log_warn, vha, 0x0035,
		    "MSI-X; Unsupported ISP2432 (0x%X, 0x%X).\n",
3049
		    ha->pdev->revision, QLA_MSIX_CHIP_REV_24XX);
3050 3051 3052
		goto skip_msix;
	}

3053
	ret = qla24xx_enable_msix(ha, rsp);
3054
	if (!ret) {
3055 3056 3057
		ql_dbg(ql_dbg_init, vha, 0x0036,
		    "MSI-X: Enabled (0x%X, 0x%X).\n",
		    ha->chip_revision, ha->fw_attributes);
3058
		goto clear_risc_ints;
3059
	}
3060

3061
skip_msix:
3062

3063 3064 3065
	ql_log(ql_log_info, vha, 0x0037,
	    "Falling back-to MSI mode -%d.\n", ret);

3066
	if (!IS_QLA24XX(ha) && !IS_QLA2532(ha) && !IS_QLA8432(ha) &&
3067 3068
	    !IS_QLA8001(ha) && !IS_P3P_TYPE(ha) && !IS_QLAFX00(ha) &&
	    !IS_QLA27XX(ha))
3069 3070 3071 3072
		goto skip_msi;

	ret = pci_enable_msi(ha->pdev);
	if (!ret) {
3073 3074
		ql_dbg(ql_dbg_init, vha, 0x0038,
		    "MSI: Enabled.\n");
3075
		ha->flags.msi_enabled = 1;
3076
	} else
3077
		ql_log(ql_log_warn, vha, 0x0039,
3078 3079
		    "Falling back-to INTa mode -- %d.\n", ret);
skip_msi:
3080 3081 3082 3083 3084

	/* Skip INTx on ISP82xx. */
	if (!ha->flags.msi_enabled && IS_QLA82XX(ha))
		return QLA_FUNCTION_FAILED;

3085
	ret = request_irq(ha->pdev->irq, ha->isp_ops->intr_handler,
3086 3087
	    ha->flags.msi_enabled ? 0 : IRQF_SHARED,
	    QLA2XXX_DRIVER_NAME, rsp);
3088
	if (ret) {
3089
		ql_log(ql_log_warn, vha, 0x003a,
3090 3091
		    "Failed to reserve interrupt %d already in use.\n",
		    ha->pdev->irq);
3092
		goto fail;
3093
	} else if (!ha->flags.msi_enabled) {
3094 3095
		ql_dbg(ql_dbg_init, vha, 0x0125,
		    "INTa mode: Enabled.\n");
3096 3097
		ha->flags.mr_intr_valid = 1;
	}
3098

3099 3100
clear_risc_ints:

3101
	spin_lock_irq(&ha->hardware_lock);
3102
	if (!IS_FWI2_CAPABLE(ha))
3103
		WRT_REG_WORD(&reg->isp.semaphore, 0);
3104
	spin_unlock_irq(&ha->hardware_lock);
3105

3106
fail:
3107 3108 3109 3110
	return ret;
}

void
3111
qla2x00_free_irqs(scsi_qla_host_t *vha)
3112
{
3113
	struct qla_hw_data *ha = vha->hw;
3114 3115 3116 3117 3118 3119 3120 3121 3122
	struct rsp_que *rsp;

	/*
	 * We need to check that ha->rsp_q_map is valid in case we are called
	 * from a probe failure context.
	 */
	if (!ha->rsp_q_map || !ha->rsp_q_map[0])
		return;
	rsp = ha->rsp_q_map[0];
3123 3124 3125

	if (ha->flags.msix_enabled)
		qla24xx_disable_msix(ha);
3126
	else if (ha->flags.msi_enabled) {
3127
		free_irq(ha->pdev->irq, rsp);
3128
		pci_disable_msi(ha->pdev);
3129 3130
	} else
		free_irq(ha->pdev->irq, rsp);
3131
}
3132

3133 3134 3135 3136

int qla25xx_request_irq(struct rsp_que *rsp)
{
	struct qla_hw_data *ha = rsp->hw;
3137
	struct qla_init_msix_entry *intr = &msix_entries[2];
3138
	struct qla_msix_entry *msix = rsp->msix;
3139
	scsi_qla_host_t *vha = pci_get_drvdata(ha->pdev);
3140 3141 3142 3143
	int ret;

	ret = request_irq(msix->vector, intr->handler, 0, intr->name, rsp);
	if (ret) {
3144 3145 3146
		ql_log(ql_log_fatal, vha, 0x00e6,
		    "MSI-X: Unable to register handler -- %x/%d.\n",
		    msix->vector, ret);
3147 3148 3149 3150 3151 3152
		return ret;
	}
	msix->have_irq = 1;
	msix->rsp = rsp;
	return ret;
}