scsi: qla2xxx: Flush mailbox commands on chip reset
[linux-block.git] / drivers / scsi / qla2xxx / qla_mbx.c
CommitLineData
77adf3f0 1// SPDX-License-Identifier: GPL-2.0-only
1da177e4 2/*
fa90c54f 3 * QLogic Fibre Channel HBA Driver
bd21eaf9 4 * Copyright (c) 2003-2014 QLogic Corporation
1da177e4
LT
5 */
6#include "qla_def.h"
2d70c103 7#include "qla_target.h"
1da177e4
LT
8
9#include <linux/delay.h>
5a0e3ad6 10#include <linux/gfp.h>
1da177e4 11
0d6a536c
JC
12#ifdef CONFIG_PPC
13#define IS_PPCARCH true
14#else
15#define IS_PPCARCH false
16#endif
17
15f30a57
QT
18static struct mb_cmd_name {
19 uint16_t cmd;
20 const char *str;
21} mb_str[] = {
22 {MBC_GET_PORT_DATABASE, "GPDB"},
23 {MBC_GET_ID_LIST, "GIDList"},
24 {MBC_GET_LINK_PRIV_STATS, "Stats"},
94d83e36 25 {MBC_GET_RESOURCE_COUNTS, "ResCnt"},
15f30a57
QT
26};
27
28static const char *mb_to_str(uint16_t cmd)
29{
30 int i;
31 struct mb_cmd_name *e;
32
33 for (i = 0; i < ARRAY_SIZE(mb_str); i++) {
34 e = mb_str + i;
35 if (cmd == e->cmd)
36 return e->str;
37 }
38 return "unknown";
39}
40
ca825828 41static struct rom_cmd {
77ddb94a 42 uint16_t cmd;
43} rom_cmds[] = {
44 { MBC_LOAD_RAM },
45 { MBC_EXECUTE_FIRMWARE },
46 { MBC_READ_RAM_WORD },
47 { MBC_MAILBOX_REGISTER_TEST },
48 { MBC_VERIFY_CHECKSUM },
49 { MBC_GET_FIRMWARE_VERSION },
50 { MBC_LOAD_RISC_RAM },
51 { MBC_DUMP_RISC_RAM },
52 { MBC_LOAD_RISC_RAM_EXTENDED },
53 { MBC_DUMP_RISC_RAM_EXTENDED },
54 { MBC_WRITE_RAM_WORD_EXTENDED },
55 { MBC_READ_RAM_EXTENDED },
56 { MBC_GET_RESOURCE_COUNTS },
57 { MBC_SET_FIRMWARE_OPTION },
58 { MBC_MID_INITIALIZE_FIRMWARE },
59 { MBC_GET_FIRMWARE_STATE },
60 { MBC_GET_MEM_OFFLOAD_CNTRL_STAT },
61 { MBC_GET_RETRY_COUNT },
62 { MBC_TRACE_CONTROL },
b7edfa23 63 { MBC_INITIALIZE_MULTIQ },
1608cc4a
QT
64 { MBC_IOCB_COMMAND_A64 },
65 { MBC_GET_ADAPTER_LOOP_ID },
e4e3a2ce 66 { MBC_READ_SFP },
62e9dd17 67 { MBC_SET_RNID_PARAMS },
8777e431 68 { MBC_GET_RNID_PARAMS },
8b4673ba 69 { MBC_GET_SET_ZIO_THRESHOLD },
77ddb94a 70};
71
72static int is_rom_cmd(uint16_t cmd)
73{
74 int i;
75 struct rom_cmd *wc;
76
77 for (i = 0; i < ARRAY_SIZE(rom_cmds); i++) {
78 wc = rom_cmds + i;
79 if (wc->cmd == cmd)
80 return 1;
81 }
82
83 return 0;
84}
1da177e4
LT
85
86/*
87 * qla2x00_mailbox_command
88 * Issue mailbox command and waits for completion.
89 *
90 * Input:
91 * ha = adapter block pointer.
92 * mcp = driver internal mbx struct pointer.
93 *
94 * Output:
95 * mb[MAX_MAILBOX_REGISTER_COUNT] = returned mailbox data.
96 *
97 * Returns:
98 * 0 : QLA_SUCCESS = cmd performed success
99 * 1 : QLA_FUNCTION_FAILED (error encountered)
100 * 6 : QLA_FUNCTION_TIMEOUT (timeout condition encountered)
101 *
102 * Context:
103 * Kernel context.
104 */
105static int
7b867cf7 106qla2x00_mailbox_command(scsi_qla_host_t *vha, mbx_cmd_t *mcp)
1da177e4 107{
d14e72fb 108 int rval, i;
1da177e4 109 unsigned long flags = 0;
f73cb695 110 device_reg_t *reg;
daafc8d3 111 uint8_t abort_active, eeh_delay;
2c3dfe3f 112 uint8_t io_lock_on;
cdbb0a4f 113 uint16_t command = 0;
1da177e4 114 uint16_t *iptr;
37139da1 115 __le16 __iomem *optr;
1da177e4
LT
116 uint32_t cnt;
117 uint32_t mboxes;
1da177e4 118 unsigned long wait_time;
7b867cf7
AC
119 struct qla_hw_data *ha = vha->hw;
120 scsi_qla_host_t *base_vha = pci_get_drvdata(ha->pdev);
b2000805 121 u32 chip_reset;
2c3dfe3f 122
d14e72fb 123
5e19ed90 124 ql_dbg(ql_dbg_mbx, vha, 0x1000, "Entered %s.\n", __func__);
7c3df132 125
471298ca 126 if (ha->pdev->error_state == pci_channel_io_perm_failure) {
5e19ed90 127 ql_log(ql_log_warn, vha, 0x1001,
471298ca 128 "PCI channel failed permanently, exiting.\n");
b9b12f73 129 return QLA_FUNCTION_TIMEOUT;
7c3df132 130 }
b9b12f73 131
a9083016 132 if (vha->device_flags & DFLG_DEV_FAILED) {
5e19ed90 133 ql_log(ql_log_warn, vha, 0x1002,
7c3df132 134 "Device in failed state, exiting.\n");
a9083016
GM
135 return QLA_FUNCTION_TIMEOUT;
136 }
137
c2a5d94f 138 /* if PCI error, then avoid mbx processing.*/
ba175891
SC
139 if (test_bit(PFLG_DISCONNECTED, &base_vha->dpc_flags) &&
140 test_bit(UNLOADING, &base_vha->dpc_flags)) {
83548fe2 141 ql_log(ql_log_warn, vha, 0xd04e,
783e0dc4
SC
142 "PCI error, exiting.\n");
143 return QLA_FUNCTION_TIMEOUT;
c2a5d94f 144 }
daafc8d3 145 eeh_delay = 0;
2c3dfe3f 146 reg = ha->iobase;
7b867cf7 147 io_lock_on = base_vha->flags.init_done;
1da177e4
LT
148
149 rval = QLA_SUCCESS;
7b867cf7 150 abort_active = test_bit(ABORT_ISP_ACTIVE, &base_vha->dpc_flags);
b2000805 151 chip_reset = ha->chip_reset;
1da177e4 152
85880801 153 if (ha->flags.pci_channel_io_perm_failure) {
5e19ed90 154 ql_log(ql_log_warn, vha, 0x1003,
7c3df132 155 "Perm failure on EEH timeout MBX, exiting.\n");
85880801
AV
156 return QLA_FUNCTION_TIMEOUT;
157 }
158
7ec0effd 159 if (IS_P3P_TYPE(ha) && ha->flags.isp82xx_fw_hung) {
862cd01e
GM
160 /* Setting Link-Down error */
161 mcp->mb[0] = MBS_LINK_DOWN_ERROR;
5e19ed90 162 ql_log(ql_log_warn, vha, 0x1004,
7c3df132 163 "FW hung = %d.\n", ha->flags.isp82xx_fw_hung);
1806fcd5 164 return QLA_FUNCTION_TIMEOUT;
862cd01e
GM
165 }
166
77ddb94a 167 /* check if ISP abort is active and return cmd with timeout */
daafc8d3
QT
168 if (((test_bit(ABORT_ISP_ACTIVE, &base_vha->dpc_flags) ||
169 test_bit(ISP_ABORT_RETRY, &base_vha->dpc_flags) ||
170 test_bit(ISP_ABORT_NEEDED, &base_vha->dpc_flags)) &&
171 !is_rom_cmd(mcp->mb[0])) || ha->flags.eeh_busy) {
77ddb94a 172 ql_log(ql_log_info, vha, 0x1005,
173 "Cmd 0x%x aborted with timeout since ISP Abort is pending\n",
174 mcp->mb[0]);
175 return QLA_FUNCTION_TIMEOUT;
176 }
177
b2000805 178 atomic_inc(&ha->num_pend_mbx_stage1);
1da177e4 179 /*
1c7c6357
AV
180 * Wait for active mailbox commands to finish by waiting at most tov
181 * seconds. This is to serialize actual issuing of mailbox cmds during
182 * non ISP abort time.
1da177e4 183 */
8eca3f39
AV
184 if (!wait_for_completion_timeout(&ha->mbx_cmd_comp, mcp->tov * HZ)) {
185 /* Timeout occurred. Return error. */
83548fe2 186 ql_log(ql_log_warn, vha, 0xd035,
d8c0d546
CD
187 "Cmd access timeout, cmd=0x%x, Exiting.\n",
188 mcp->mb[0]);
dbf1f53c 189 vha->hw_err_cnt++;
b2000805 190 atomic_dec(&ha->num_pend_mbx_stage1);
8eca3f39 191 return QLA_FUNCTION_TIMEOUT;
1da177e4 192 }
b2000805 193 atomic_dec(&ha->num_pend_mbx_stage1);
daafc8d3
QT
194 if (ha->flags.purge_mbox || chip_reset != ha->chip_reset ||
195 ha->flags.eeh_busy) {
196 ql_log(ql_log_warn, vha, 0xd035,
197 "Error detected: purge[%d] eeh[%d] cmd=0x%x, Exiting.\n",
198 ha->flags.purge_mbox, ha->flags.eeh_busy, mcp->mb[0]);
b2000805
QT
199 rval = QLA_ABORTED;
200 goto premature_exit;
201 }
1da177e4 202
b6faaaf7 203
1da177e4
LT
204 /* Save mailbox command for debug */
205 ha->mcp = mcp;
206
5e19ed90 207 ql_dbg(ql_dbg_mbx, vha, 0x1006,
7c3df132 208 "Prepare to issue mbox cmd=0x%x.\n", mcp->mb[0]);
1da177e4
LT
209
210 spin_lock_irqsave(&ha->hardware_lock, flags);
211
b6faaaf7
QT
212 if (ha->flags.purge_mbox || chip_reset != ha->chip_reset ||
213 ha->flags.mbox_busy) {
b2000805 214 rval = QLA_ABORTED;
b2000805
QT
215 spin_unlock_irqrestore(&ha->hardware_lock, flags);
216 goto premature_exit;
217 }
b6faaaf7 218 ha->flags.mbox_busy = 1;
b2000805 219
1da177e4 220 /* Load mailbox registers. */
7ec0effd 221 if (IS_P3P_TYPE(ha))
7ffa5b93 222 optr = &reg->isp82.mailbox_in[0];
7ec0effd 223 else if (IS_FWI2_CAPABLE(ha) && !(IS_P3P_TYPE(ha)))
7ffa5b93 224 optr = &reg->isp24.mailbox0;
1c7c6357 225 else
7ffa5b93 226 optr = MAILBOX_REG(ha, &reg->isp, 0);
1da177e4
LT
227
228 iptr = mcp->mb;
229 command = mcp->mb[0];
230 mboxes = mcp->out_mb;
231
7b711623 232 ql_dbg(ql_dbg_mbx, vha, 0x1111,
0e31a2c8 233 "Mailbox registers (OUT):\n");
1da177e4
LT
234 for (cnt = 0; cnt < ha->mbx_count; cnt++) {
235 if (IS_QLA2200(ha) && cnt == 8)
7ffa5b93 236 optr = MAILBOX_REG(ha, &reg->isp, 8);
0e31a2c8
JC
237 if (mboxes & BIT_0) {
238 ql_dbg(ql_dbg_mbx, vha, 0x1112,
239 "mbox[%d]<-0x%04x\n", cnt, *iptr);
04474d3a 240 wrt_reg_word(optr, *iptr);
6c96a3c7
BH
241 } else {
242 wrt_reg_word(optr, 0);
0e31a2c8 243 }
1da177e4
LT
244
245 mboxes >>= 1;
246 optr++;
247 iptr++;
248 }
249
5e19ed90 250 ql_dbg(ql_dbg_mbx + ql_dbg_buffer, vha, 0x1117,
7c3df132 251 "I/O Address = %p.\n", optr);
1da177e4
LT
252
253 /* Issue set host interrupt command to send cmd out. */
254 ha->flags.mbox_int = 0;
255 clear_bit(MBX_INTERRUPT, &ha->mbx_cmd_flags);
256
257 /* Unlock mbx registers and wait for interrupt */
5e19ed90 258 ql_dbg(ql_dbg_mbx, vha, 0x100f,
7c3df132
SK
259 "Going to unlock irq & waiting for interrupts. "
260 "jiffies=%lx.\n", jiffies);
1da177e4
LT
261
262 /* Wait for mbx cmd completion until timeout */
b2000805 263 atomic_inc(&ha->num_pend_mbx_stage2);
124f85e6 264 if ((!abort_active && io_lock_on) || IS_NOPOLLING_TYPE(ha)) {
1da177e4
LT
265 set_bit(MBX_INTR_WAIT, &ha->mbx_cmd_flags);
266
32a13df2 267 if (IS_P3P_TYPE(ha))
04474d3a 268 wrt_reg_dword(&reg->isp82.hint, HINT_MBX_INT_PENDING);
32a13df2 269 else if (IS_FWI2_CAPABLE(ha))
04474d3a 270 wrt_reg_dword(&reg->isp24.hccr, HCCRX_SET_HOST_INT);
1c7c6357 271 else
04474d3a 272 wrt_reg_word(&reg->isp.hccr, HCCR_SET_HOST_INT);
1da177e4
LT
273 spin_unlock_irqrestore(&ha->hardware_lock, flags);
274
77ddb94a 275 wait_time = jiffies;
754d1243
GM
276 if (!wait_for_completion_timeout(&ha->mbx_intr_comp,
277 mcp->tov * HZ)) {
f260694e
QT
278 ql_dbg(ql_dbg_mbx, vha, 0x117a,
279 "cmd=%x Timeout.\n", command);
280 spin_lock_irqsave(&ha->hardware_lock, flags);
281 clear_bit(MBX_INTR_WAIT, &ha->mbx_cmd_flags);
282 spin_unlock_irqrestore(&ha->hardware_lock, flags);
283
b6faaaf7 284 if (chip_reset != ha->chip_reset) {
daafc8d3
QT
285 eeh_delay = ha->flags.eeh_busy ? 1 : 0;
286
b6faaaf7
QT
287 spin_lock_irqsave(&ha->hardware_lock, flags);
288 ha->flags.mbox_busy = 0;
289 spin_unlock_irqrestore(&ha->hardware_lock,
290 flags);
291 atomic_dec(&ha->num_pend_mbx_stage2);
b6faaaf7
QT
292 rval = QLA_ABORTED;
293 goto premature_exit;
294 }
b2000805
QT
295 } else if (ha->flags.purge_mbox ||
296 chip_reset != ha->chip_reset) {
daafc8d3
QT
297 eeh_delay = ha->flags.eeh_busy ? 1 : 0;
298
b6faaaf7 299 spin_lock_irqsave(&ha->hardware_lock, flags);
b2000805 300 ha->flags.mbox_busy = 0;
b6faaaf7 301 spin_unlock_irqrestore(&ha->hardware_lock, flags);
b2000805 302 atomic_dec(&ha->num_pend_mbx_stage2);
b2000805
QT
303 rval = QLA_ABORTED;
304 goto premature_exit;
754d1243 305 }
b2000805 306
77ddb94a 307 if (time_after(jiffies, wait_time + 5 * HZ))
308 ql_log(ql_log_warn, vha, 0x1015, "cmd=0x%x, waited %d msecs\n",
309 command, jiffies_to_msecs(jiffies - wait_time));
1da177e4 310 } else {
5e19ed90 311 ql_dbg(ql_dbg_mbx, vha, 0x1011,
7c3df132 312 "Cmd=%x Polling Mode.\n", command);
1da177e4 313
7ec0effd 314 if (IS_P3P_TYPE(ha)) {
04474d3a 315 if (rd_reg_dword(&reg->isp82.hint) &
a9083016 316 HINT_MBX_INT_PENDING) {
b6faaaf7 317 ha->flags.mbox_busy = 0;
a9083016
GM
318 spin_unlock_irqrestore(&ha->hardware_lock,
319 flags);
b2000805 320 atomic_dec(&ha->num_pend_mbx_stage2);
5e19ed90 321 ql_dbg(ql_dbg_mbx, vha, 0x1012,
7c3df132 322 "Pending mailbox timeout, exiting.\n");
dbf1f53c 323 vha->hw_err_cnt++;
cdbb0a4f
SV
324 rval = QLA_FUNCTION_TIMEOUT;
325 goto premature_exit;
a9083016 326 }
04474d3a 327 wrt_reg_dword(&reg->isp82.hint, HINT_MBX_INT_PENDING);
a9083016 328 } else if (IS_FWI2_CAPABLE(ha))
04474d3a 329 wrt_reg_dword(&reg->isp24.hccr, HCCRX_SET_HOST_INT);
1c7c6357 330 else
04474d3a 331 wrt_reg_word(&reg->isp.hccr, HCCR_SET_HOST_INT);
1da177e4 332 spin_unlock_irqrestore(&ha->hardware_lock, flags);
1da177e4
LT
333
334 wait_time = jiffies + mcp->tov * HZ; /* wait at most tov secs */
335 while (!ha->flags.mbox_int) {
b2000805
QT
336 if (ha->flags.purge_mbox ||
337 chip_reset != ha->chip_reset) {
daafc8d3
QT
338 eeh_delay = ha->flags.eeh_busy ? 1 : 0;
339
b6faaaf7 340 spin_lock_irqsave(&ha->hardware_lock, flags);
b2000805 341 ha->flags.mbox_busy = 0;
b6faaaf7
QT
342 spin_unlock_irqrestore(&ha->hardware_lock,
343 flags);
b2000805
QT
344 atomic_dec(&ha->num_pend_mbx_stage2);
345 rval = QLA_ABORTED;
346 goto premature_exit;
347 }
348
1da177e4
LT
349 if (time_after(jiffies, wait_time))
350 break;
351
352 /* Check for pending interrupts. */
73208dfd 353 qla2x00_poll(ha->rsp_q_map[0]);
1da177e4 354
85880801
AV
355 if (!ha->flags.mbox_int &&
356 !(IS_QLA2200(ha) &&
357 command == MBC_LOAD_RISC_RAM_EXTENDED))
59989831 358 msleep(10);
1da177e4 359 } /* while */
5e19ed90 360 ql_dbg(ql_dbg_mbx, vha, 0x1013,
7c3df132
SK
361 "Waited %d sec.\n",
362 (uint)((jiffies - (wait_time - (mcp->tov * HZ)))/HZ));
1da177e4 363 }
b2000805 364 atomic_dec(&ha->num_pend_mbx_stage2);
1da177e4 365
1da177e4
LT
366 /* Check whether we timed out */
367 if (ha->flags.mbox_int) {
368 uint16_t *iptr2;
369
5e19ed90 370 ql_dbg(ql_dbg_mbx, vha, 0x1014,
7c3df132 371 "Cmd=%x completed.\n", command);
1da177e4
LT
372
373 /* Got interrupt. Clear the flag. */
374 ha->flags.mbox_int = 0;
375 clear_bit(MBX_INTERRUPT, &ha->mbx_cmd_flags);
376
7ec0effd 377 if (IS_P3P_TYPE(ha) && ha->flags.isp82xx_fw_hung) {
b6faaaf7 378 spin_lock_irqsave(&ha->hardware_lock, flags);
cdbb0a4f 379 ha->flags.mbox_busy = 0;
b6faaaf7
QT
380 spin_unlock_irqrestore(&ha->hardware_lock, flags);
381
cdbb0a4f
SV
382 /* Setting Link-Down error */
383 mcp->mb[0] = MBS_LINK_DOWN_ERROR;
384 ha->mcp = NULL;
385 rval = QLA_FUNCTION_FAILED;
83548fe2 386 ql_log(ql_log_warn, vha, 0xd048,
7c3df132 387 "FW hung = %d.\n", ha->flags.isp82xx_fw_hung);
cdbb0a4f
SV
388 goto premature_exit;
389 }
390
b3e9772d
BVA
391 if (ha->mailbox_out[0] != MBS_COMMAND_COMPLETE) {
392 ql_dbg(ql_dbg_mbx, vha, 0x11ff,
393 "mb_out[0] = %#x <> %#x\n", ha->mailbox_out[0],
394 MBS_COMMAND_COMPLETE);
1da177e4 395 rval = QLA_FUNCTION_FAILED;
b3e9772d 396 }
1da177e4
LT
397
398 /* Load return mailbox registers. */
399 iptr2 = mcp->mb;
400 iptr = (uint16_t *)&ha->mailbox_out[0];
401 mboxes = mcp->in_mb;
0e31a2c8
JC
402
403 ql_dbg(ql_dbg_mbx, vha, 0x1113,
404 "Mailbox registers (IN):\n");
1da177e4 405 for (cnt = 0; cnt < ha->mbx_count; cnt++) {
0e31a2c8 406 if (mboxes & BIT_0) {
1da177e4 407 *iptr2 = *iptr;
0e31a2c8
JC
408 ql_dbg(ql_dbg_mbx, vha, 0x1114,
409 "mbox[%d]->0x%04x\n", cnt, *iptr2);
410 }
1da177e4
LT
411
412 mboxes >>= 1;
413 iptr2++;
414 iptr++;
415 }
416 } else {
417
8d3c9c23
QT
418 uint16_t mb[8];
419 uint32_t ictrl, host_status, hccr;
783e0dc4 420 uint16_t w;
1c7c6357 421
e428924c 422 if (IS_FWI2_CAPABLE(ha)) {
04474d3a
BVA
423 mb[0] = rd_reg_word(&reg->isp24.mailbox0);
424 mb[1] = rd_reg_word(&reg->isp24.mailbox1);
425 mb[2] = rd_reg_word(&reg->isp24.mailbox2);
426 mb[3] = rd_reg_word(&reg->isp24.mailbox3);
427 mb[7] = rd_reg_word(&reg->isp24.mailbox7);
428 ictrl = rd_reg_dword(&reg->isp24.ictrl);
429 host_status = rd_reg_dword(&reg->isp24.host_status);
430 hccr = rd_reg_dword(&reg->isp24.hccr);
8d3c9c23 431
83548fe2 432 ql_log(ql_log_warn, vha, 0xd04c,
8d3c9c23
QT
433 "MBX Command timeout for cmd %x, iocontrol=%x jiffies=%lx "
434 "mb[0-3]=[0x%x 0x%x 0x%x 0x%x] mb7 0x%x host_status 0x%x hccr 0x%x\n",
435 command, ictrl, jiffies, mb[0], mb[1], mb[2], mb[3],
436 mb[7], host_status, hccr);
dbf1f53c 437 vha->hw_err_cnt++;
8d3c9c23 438
1c7c6357 439 } else {
8d3c9c23 440 mb[0] = RD_MAILBOX_REG(ha, &reg->isp, 0);
04474d3a 441 ictrl = rd_reg_word(&reg->isp.ictrl);
8d3c9c23
QT
442 ql_dbg(ql_dbg_mbx + ql_dbg_buffer, vha, 0x1119,
443 "MBX Command timeout for cmd %x, iocontrol=%x jiffies=%lx "
444 "mb[0]=0x%x\n", command, ictrl, jiffies, mb[0]);
dbf1f53c 445 vha->hw_err_cnt++;
1c7c6357 446 }
5e19ed90 447 ql_dump_regs(ql_dbg_mbx + ql_dbg_buffer, vha, 0x1019);
1da177e4 448
783e0dc4
SC
449 /* Capture FW dump only, if PCI device active */
450 if (!pci_channel_offline(vha->hw->pdev)) {
451 pci_read_config_word(ha->pdev, PCI_VENDOR_ID, &w);
b2000805
QT
452 if (w == 0xffff || ictrl == 0xffffffff ||
453 (chip_reset != ha->chip_reset)) {
783e0dc4
SC
454 /* This is special case if there is unload
455 * of driver happening and if PCI device go
456 * into bad state due to PCI error condition
457 * then only PCI ERR flag would be set.
458 * we will do premature exit for above case.
459 */
b6faaaf7 460 spin_lock_irqsave(&ha->hardware_lock, flags);
783e0dc4 461 ha->flags.mbox_busy = 0;
b6faaaf7
QT
462 spin_unlock_irqrestore(&ha->hardware_lock,
463 flags);
783e0dc4
SC
464 rval = QLA_FUNCTION_TIMEOUT;
465 goto premature_exit;
466 }
f55bfc88 467
783e0dc4
SC
468 /* Attempt to capture firmware dump for further
469 * anallysis of the current formware state. we do not
470 * need to do this if we are intentionally generating
471 * a dump
472 */
473 if (mcp->mb[0] != MBC_GEN_SYSTEM_ERROR)
8ae17876 474 qla2xxx_dump_fw(vha);
783e0dc4
SC
475 rval = QLA_FUNCTION_TIMEOUT;
476 }
1da177e4 477 }
b6faaaf7 478 spin_lock_irqsave(&ha->hardware_lock, flags);
1da177e4 479 ha->flags.mbox_busy = 0;
b6faaaf7 480 spin_unlock_irqrestore(&ha->hardware_lock, flags);
1da177e4
LT
481
482 /* Clean up */
483 ha->mcp = NULL;
484
124f85e6 485 if ((abort_active || !io_lock_on) && !IS_NOPOLLING_TYPE(ha)) {
5e19ed90 486 ql_dbg(ql_dbg_mbx, vha, 0x101a,
7c3df132 487 "Checking for additional resp interrupt.\n");
1da177e4
LT
488
489 /* polling mode for non isp_abort commands. */
73208dfd 490 qla2x00_poll(ha->rsp_q_map[0]);
1da177e4
LT
491 }
492
1c7c6357
AV
493 if (rval == QLA_FUNCTION_TIMEOUT &&
494 mcp->mb[0] != MBC_GEN_SYSTEM_ERROR) {
85880801
AV
495 if (!io_lock_on || (mcp->flags & IOCTL_CMD) ||
496 ha->flags.eeh_busy) {
1da177e4 497 /* not in dpc. schedule it for dpc to take over. */
5e19ed90 498 ql_dbg(ql_dbg_mbx, vha, 0x101b,
7c3df132 499 "Timeout, schedule isp_abort_needed.\n");
cdbb0a4f
SV
500
501 if (!test_bit(ISP_ABORT_NEEDED, &vha->dpc_flags) &&
502 !test_bit(ABORT_ISP_ACTIVE, &vha->dpc_flags) &&
503 !test_bit(ISP_ABORT_RETRY, &vha->dpc_flags)) {
63154916
GM
504 if (IS_QLA82XX(ha)) {
505 ql_dbg(ql_dbg_mbx, vha, 0x112a,
506 "disabling pause transmit on port "
507 "0 & 1.\n");
508 qla82xx_wr_32(ha,
509 QLA82XX_CRB_NIU + 0x98,
510 CRB_NIU_XG_PAUSE_CTL_P0|
511 CRB_NIU_XG_PAUSE_CTL_P1);
512 }
7c3df132 513 ql_log(ql_log_info, base_vha, 0x101c,
24d9ee85 514 "Mailbox cmd timeout occurred, cmd=0x%x, "
d8c0d546
CD
515 "mb[0]=0x%x, eeh_busy=0x%x. Scheduling ISP "
516 "abort.\n", command, mcp->mb[0],
517 ha->flags.eeh_busy);
dbf1f53c 518 vha->hw_err_cnt++;
cdbb0a4f
SV
519 set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
520 qla2xxx_wake_dpc(vha);
521 }
710bc78f 522 } else if (current == ha->dpc_thread) {
1da177e4 523 /* call abort directly since we are in the DPC thread */
5e19ed90 524 ql_dbg(ql_dbg_mbx, vha, 0x101d,
7c3df132 525 "Timeout, calling abort_isp.\n");
cdbb0a4f
SV
526
527 if (!test_bit(ISP_ABORT_NEEDED, &vha->dpc_flags) &&
528 !test_bit(ABORT_ISP_ACTIVE, &vha->dpc_flags) &&
529 !test_bit(ISP_ABORT_RETRY, &vha->dpc_flags)) {
63154916
GM
530 if (IS_QLA82XX(ha)) {
531 ql_dbg(ql_dbg_mbx, vha, 0x112b,
532 "disabling pause transmit on port "
533 "0 & 1.\n");
534 qla82xx_wr_32(ha,
535 QLA82XX_CRB_NIU + 0x98,
536 CRB_NIU_XG_PAUSE_CTL_P0|
537 CRB_NIU_XG_PAUSE_CTL_P1);
538 }
7c3df132 539 ql_log(ql_log_info, base_vha, 0x101e,
24d9ee85 540 "Mailbox cmd timeout occurred, cmd=0x%x, "
d8c0d546
CD
541 "mb[0]=0x%x. Scheduling ISP abort ",
542 command, mcp->mb[0]);
dbf1f53c 543 vha->hw_err_cnt++;
cdbb0a4f
SV
544 set_bit(ABORT_ISP_ACTIVE, &vha->dpc_flags);
545 clear_bit(ISP_ABORT_NEEDED, &vha->dpc_flags);
d3360960
GM
546 /* Allow next mbx cmd to come in. */
547 complete(&ha->mbx_cmd_comp);
daafc8d3
QT
548 if (ha->isp_ops->abort_isp(vha) &&
549 !ha->flags.eeh_busy) {
cdbb0a4f
SV
550 /* Failed. retry later. */
551 set_bit(ISP_ABORT_NEEDED,
552 &vha->dpc_flags);
553 }
554 clear_bit(ABORT_ISP_ACTIVE, &vha->dpc_flags);
5e19ed90 555 ql_dbg(ql_dbg_mbx, vha, 0x101f,
7c3df132 556 "Finished abort_isp.\n");
d3360960 557 goto mbx_done;
1da177e4 558 }
1da177e4
LT
559 }
560 }
561
cdbb0a4f 562premature_exit:
1da177e4 563 /* Allow next mbx cmd to come in. */
8eca3f39 564 complete(&ha->mbx_cmd_comp);
1da177e4 565
d3360960 566mbx_done:
b2000805
QT
567 if (rval == QLA_ABORTED) {
568 ql_log(ql_log_info, vha, 0xd035,
569 "Chip Reset in progress. Purging Mbox cmd=0x%x.\n",
570 mcp->mb[0]);
571 } else if (rval) {
050dc76a 572 if (ql2xextended_error_logging & (ql_dbg_disc|ql_dbg_mbx)) {
3f918ffa 573 pr_warn("%s [%s]-%04x:%ld: **** Failed=%x", QL_MSGHDR,
050dc76a 574 dev_name(&ha->pdev->dev), 0x1020+0x800,
3f918ffa 575 vha->host_no, rval);
050dc76a
JC
576 mboxes = mcp->in_mb;
577 cnt = 4;
578 for (i = 0; i < ha->mbx_count && cnt; i++, mboxes >>= 1)
579 if (mboxes & BIT_0) {
580 printk(" mb[%u]=%x", i, mcp->mb[i]);
581 cnt--;
582 }
583 pr_warn(" cmd=%x ****\n", command);
584 }
f7e59e99
MR
585 if (IS_FWI2_CAPABLE(ha) && !(IS_P3P_TYPE(ha))) {
586 ql_dbg(ql_dbg_mbx, vha, 0x1198,
587 "host_status=%#x intr_ctrl=%#x intr_status=%#x\n",
04474d3a
BVA
588 rd_reg_dword(&reg->isp24.host_status),
589 rd_reg_dword(&reg->isp24.ictrl),
590 rd_reg_dword(&reg->isp24.istatus));
f7e59e99
MR
591 } else {
592 ql_dbg(ql_dbg_mbx, vha, 0x1206,
593 "ctrl_status=%#x ictrl=%#x istatus=%#x\n",
04474d3a
BVA
594 rd_reg_word(&reg->isp.ctrl_status),
595 rd_reg_word(&reg->isp.ictrl),
596 rd_reg_word(&reg->isp.istatus));
f7e59e99 597 }
1da177e4 598 } else {
7c3df132 599 ql_dbg(ql_dbg_mbx, base_vha, 0x1021, "Done %s.\n", __func__);
1da177e4
LT
600 }
601
daafc8d3
QT
602 i = 500;
603 while (i && eeh_delay && (ha->pci_error_state < QLA_PCI_SLOT_RESET)) {
604 /*
605 * The caller of this mailbox encounter pci error.
606 * Hold the thread until PCIE link reset complete to make
607 * sure caller does not unmap dma while recovery is
608 * in progress.
609 */
610 msleep(1);
611 i--;
612 }
1da177e4
LT
613 return rval;
614}
615
1da177e4 616int
7b867cf7 617qla2x00_load_ram(scsi_qla_host_t *vha, dma_addr_t req_dma, uint32_t risc_addr,
590f98e5 618 uint32_t risc_code_size)
1da177e4
LT
619{
620 int rval;
7b867cf7 621 struct qla_hw_data *ha = vha->hw;
1da177e4
LT
622 mbx_cmd_t mc;
623 mbx_cmd_t *mcp = &mc;
1da177e4 624
5f28d2d7
SK
625 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1022,
626 "Entered %s.\n", __func__);
1da177e4 627
e428924c 628 if (MSW(risc_addr) || IS_FWI2_CAPABLE(ha)) {
590f98e5 629 mcp->mb[0] = MBC_LOAD_RISC_RAM_EXTENDED;
630 mcp->mb[8] = MSW(risc_addr);
631 mcp->out_mb = MBX_8|MBX_0;
1da177e4 632 } else {
590f98e5 633 mcp->mb[0] = MBC_LOAD_RISC_RAM;
634 mcp->out_mb = MBX_0;
1da177e4 635 }
1da177e4
LT
636 mcp->mb[1] = LSW(risc_addr);
637 mcp->mb[2] = MSW(req_dma);
638 mcp->mb[3] = LSW(req_dma);
1da177e4
LT
639 mcp->mb[6] = MSW(MSD(req_dma));
640 mcp->mb[7] = LSW(MSD(req_dma));
590f98e5 641 mcp->out_mb |= MBX_7|MBX_6|MBX_3|MBX_2|MBX_1;
e428924c 642 if (IS_FWI2_CAPABLE(ha)) {
1c7c6357
AV
643 mcp->mb[4] = MSW(risc_code_size);
644 mcp->mb[5] = LSW(risc_code_size);
645 mcp->out_mb |= MBX_5|MBX_4;
646 } else {
647 mcp->mb[4] = LSW(risc_code_size);
648 mcp->out_mb |= MBX_4;
649 }
650
2a3192a3 651 mcp->in_mb = MBX_1|MBX_0;
b93480e3 652 mcp->tov = MBX_TOV_SECONDS;
1da177e4 653 mcp->flags = 0;
7b867cf7 654 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4 655
1da177e4 656 if (rval != QLA_SUCCESS) {
7c3df132 657 ql_dbg(ql_dbg_mbx, vha, 0x1023,
2a3192a3
JC
658 "Failed=%x mb[0]=%x mb[1]=%x.\n",
659 rval, mcp->mb[0], mcp->mb[1]);
dbf1f53c 660 vha->hw_err_cnt++;
1da177e4 661 } else {
5f28d2d7
SK
662 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1024,
663 "Done %s.\n", __func__);
1da177e4
LT
664 }
665
666 return rval;
667}
668
e84067d7 669#define NVME_ENABLE_FLAG BIT_3
d07b75ba 670#define EDIF_HW_SUPPORT BIT_10
e84067d7 671
1da177e4
LT
672/*
673 * qla2x00_execute_fw
1c7c6357 674 * Start adapter firmware.
1da177e4
LT
675 *
676 * Input:
1c7c6357
AV
677 * ha = adapter block pointer.
678 * TARGET_QUEUE_LOCK must be released.
679 * ADAPTER_STATE_LOCK must be released.
1da177e4
LT
680 *
681 * Returns:
1c7c6357 682 * qla2x00 local function return status code.
1da177e4
LT
683 *
684 * Context:
1c7c6357 685 * Kernel context.
1da177e4
LT
686 */
687int
7b867cf7 688qla2x00_execute_fw(scsi_qla_host_t *vha, uint32_t risc_addr)
1da177e4
LT
689{
690 int rval;
7b867cf7 691 struct qla_hw_data *ha = vha->hw;
1da177e4
LT
692 mbx_cmd_t mc;
693 mbx_cmd_t *mcp = &mc;
cad9c2d2
QT
694 u8 semaphore = 0;
695#define EXE_FW_FORCE_SEMAPHORE BIT_7
355f5ffe 696 u8 retry = 5;
1da177e4 697
5f28d2d7
SK
698 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1025,
699 "Entered %s.\n", __func__);
1da177e4 700
cad9c2d2 701again:
1da177e4 702 mcp->mb[0] = MBC_EXECUTE_FIRMWARE;
1c7c6357
AV
703 mcp->out_mb = MBX_0;
704 mcp->in_mb = MBX_0;
e428924c 705 if (IS_FWI2_CAPABLE(ha)) {
1c7c6357
AV
706 mcp->mb[1] = MSW(risc_addr);
707 mcp->mb[2] = LSW(risc_addr);
708 mcp->mb[3] = 0;
e4e3a2ce 709 mcp->mb[4] = 0;
c2ff2a36 710 mcp->mb[11] = 0;
b0f18eee
AV
711
712 /* Enable BPM? */
713 if (ha->flags.lr_detected) {
714 mcp->mb[4] = BIT_0;
715 if (IS_BPM_RANGE_CAPABLE(ha))
716 mcp->mb[4] |=
717 ha->lr_distance << LR_DIST_FW_POS;
e4e3a2ce 718 }
b0d6cabd 719
ecc89f25 720 if (ql2xnvmeenable && (IS_QLA27XX(ha) || IS_QLA28XX(ha)))
e84067d7
DG
721 mcp->mb[4] |= NVME_ENABLE_FLAG;
722
ecc89f25 723 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
92d4408e
SC
724 struct nvram_81xx *nv = ha->nvram;
725 /* set minimum speed if specified in nvram */
72a92df2
JC
726 if (nv->min_supported_speed >= 2 &&
727 nv->min_supported_speed <= 5) {
92d4408e 728 mcp->mb[4] |= BIT_4;
72a92df2 729 mcp->mb[11] |= nv->min_supported_speed & 0xF;
92d4408e
SC
730 mcp->out_mb |= MBX_11;
731 mcp->in_mb |= BIT_5;
72a92df2
JC
732 vha->min_supported_speed =
733 nv->min_supported_speed;
92d4408e 734 }
0d6a536c
JC
735
736 if (IS_PPCARCH)
737 mcp->mb[11] |= BIT_4;
92d4408e
SC
738 }
739
b0d6cabd
HM
740 if (ha->flags.exlogins_enabled)
741 mcp->mb[4] |= ENABLE_EXTENDED_LOGIN;
742
2f56a7f1
HM
743 if (ha->flags.exchoffld_enabled)
744 mcp->mb[4] |= ENABLE_EXCHANGE_OFFLD;
745
cad9c2d2
QT
746 if (semaphore)
747 mcp->mb[11] |= EXE_FW_FORCE_SEMAPHORE;
748
c2ff2a36 749 mcp->out_mb |= MBX_4 | MBX_3 | MBX_2 | MBX_1 | MBX_11;
44d01857 750 mcp->in_mb |= MBX_5 | MBX_3 | MBX_2 | MBX_1;
1c7c6357
AV
751 } else {
752 mcp->mb[1] = LSW(risc_addr);
753 mcp->out_mb |= MBX_1;
754 if (IS_QLA2322(ha) || IS_QLA6322(ha)) {
755 mcp->mb[2] = 0;
756 mcp->out_mb |= MBX_2;
757 }
1da177e4
LT
758 }
759
b93480e3 760 mcp->tov = MBX_TOV_SECONDS;
1da177e4 761 mcp->flags = 0;
7b867cf7 762 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4 763
1c7c6357 764 if (rval != QLA_SUCCESS) {
cad9c2d2
QT
765 if (IS_QLA28XX(ha) && rval == QLA_COMMAND_ERROR &&
766 mcp->mb[1] == 0x27 && retry) {
767 semaphore = 1;
768 retry--;
769 ql_dbg(ql_dbg_async, vha, 0x1026,
770 "Exe FW: force semaphore.\n");
771 goto again;
772 }
773
355f5ffe
QT
774 if (retry) {
775 retry--;
776 ql_dbg(ql_dbg_async, vha, 0x509d,
777 "Exe FW retry: mb[0]=%x retry[%d]\n", mcp->mb[0], retry);
778 goto again;
779 }
7c3df132
SK
780 ql_dbg(ql_dbg_mbx, vha, 0x1026,
781 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
dbf1f53c 782 vha->hw_err_cnt++;
72a92df2
JC
783 return rval;
784 }
785
786 if (!IS_FWI2_CAPABLE(ha))
787 goto done;
788
789 ha->fw_ability_mask = mcp->mb[3] << 16 | mcp->mb[2];
790 ql_dbg(ql_dbg_mbx, vha, 0x119a,
791 "fw_ability_mask=%x.\n", ha->fw_ability_mask);
792 ql_dbg(ql_dbg_mbx, vha, 0x1027, "exchanges=%x.\n", mcp->mb[1]);
793 if (IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
794 ha->max_supported_speed = mcp->mb[2] & (BIT_0|BIT_1);
795 ql_dbg(ql_dbg_mbx, vha, 0x119b, "max_supported_speed=%s.\n",
796 ha->max_supported_speed == 0 ? "16Gps" :
797 ha->max_supported_speed == 1 ? "32Gps" :
798 ha->max_supported_speed == 2 ? "64Gps" : "unknown");
799 if (vha->min_supported_speed) {
800 ha->min_supported_speed = mcp->mb[5] &
801 (BIT_0 | BIT_1 | BIT_2);
802 ql_dbg(ql_dbg_mbx, vha, 0x119c,
803 "min_supported_speed=%s.\n",
804 ha->min_supported_speed == 6 ? "64Gps" :
805 ha->min_supported_speed == 5 ? "32Gps" :
806 ha->min_supported_speed == 4 ? "16Gps" :
807 ha->min_supported_speed == 3 ? "8Gps" :
808 ha->min_supported_speed == 2 ? "4Gps" : "unknown");
1c7c6357
AV
809 }
810 }
1da177e4 811
d07b75ba
QT
812 if (IS_QLA28XX(ha) && (mcp->mb[5] & EDIF_HW_SUPPORT)) {
813 ha->flags.edif_hw = 1;
44d01857 814 ql_log(ql_log_info, vha, 0xffff,
d07b75ba 815 "%s: edif HW\n", __func__);
44d01857
QT
816 }
817
72a92df2
JC
818done:
819 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1028,
820 "Done %s.\n", __func__);
821
1da177e4
LT
822 return rval;
823}
824
b0d6cabd
HM
825/*
826 * qla_get_exlogin_status
827 * Get extended login status
828 * uses the memory offload control/status Mailbox
829 *
830 * Input:
831 * ha: adapter state pointer.
832 * fwopt: firmware options
833 *
834 * Returns:
835 * qla2x00 local function status
836 *
837 * Context:
838 * Kernel context.
839 */
840#define FETCH_XLOGINS_STAT 0x8
841int
842qla_get_exlogin_status(scsi_qla_host_t *vha, uint16_t *buf_sz,
843 uint16_t *ex_logins_cnt)
844{
845 int rval;
846 mbx_cmd_t mc;
847 mbx_cmd_t *mcp = &mc;
848
849 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x118f,
850 "Entered %s\n", __func__);
851
852 memset(mcp->mb, 0 , sizeof(mcp->mb));
853 mcp->mb[0] = MBC_GET_MEM_OFFLOAD_CNTRL_STAT;
854 mcp->mb[1] = FETCH_XLOGINS_STAT;
855 mcp->out_mb = MBX_1|MBX_0;
856 mcp->in_mb = MBX_10|MBX_4|MBX_0;
857 mcp->tov = MBX_TOV_SECONDS;
858 mcp->flags = 0;
859
860 rval = qla2x00_mailbox_command(vha, mcp);
861 if (rval != QLA_SUCCESS) {
862 ql_dbg(ql_dbg_mbx, vha, 0x1115, "Failed=%x.\n", rval);
863 } else {
864 *buf_sz = mcp->mb[4];
865 *ex_logins_cnt = mcp->mb[10];
866
867 ql_log(ql_log_info, vha, 0x1190,
868 "buffer size 0x%x, exchange login count=%d\n",
869 mcp->mb[4], mcp->mb[10]);
870
871 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1116,
872 "Done %s.\n", __func__);
873 }
874
875 return rval;
876}
877
878/*
879 * qla_set_exlogin_mem_cfg
880 * set extended login memory configuration
881 * Mbx needs to be issues before init_cb is set
882 *
883 * Input:
884 * ha: adapter state pointer.
885 * buffer: buffer pointer
886 * phys_addr: physical address of buffer
887 * size: size of buffer
888 * TARGET_QUEUE_LOCK must be released
889 * ADAPTER_STATE_LOCK must be release
890 *
891 * Returns:
892 * qla2x00 local funxtion status code.
893 *
894 * Context:
895 * Kernel context.
896 */
d38cb849 897#define CONFIG_XLOGINS_MEM 0x9
b0d6cabd
HM
898int
899qla_set_exlogin_mem_cfg(scsi_qla_host_t *vha, dma_addr_t phys_addr)
900{
901 int rval;
902 mbx_cmd_t mc;
903 mbx_cmd_t *mcp = &mc;
904 struct qla_hw_data *ha = vha->hw;
b0d6cabd
HM
905
906 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x111a,
907 "Entered %s.\n", __func__);
908
909 memset(mcp->mb, 0 , sizeof(mcp->mb));
910 mcp->mb[0] = MBC_GET_MEM_OFFLOAD_CNTRL_STAT;
911 mcp->mb[1] = CONFIG_XLOGINS_MEM;
912 mcp->mb[2] = MSW(phys_addr);
913 mcp->mb[3] = LSW(phys_addr);
914 mcp->mb[6] = MSW(MSD(phys_addr));
915 mcp->mb[7] = LSW(MSD(phys_addr));
916 mcp->mb[8] = MSW(ha->exlogin_size);
917 mcp->mb[9] = LSW(ha->exlogin_size);
918 mcp->out_mb = MBX_9|MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
919 mcp->in_mb = MBX_11|MBX_0;
920 mcp->tov = MBX_TOV_SECONDS;
921 mcp->flags = 0;
922 rval = qla2x00_mailbox_command(vha, mcp);
923 if (rval != QLA_SUCCESS) {
d38cb849
QT
924 ql_dbg(ql_dbg_mbx, vha, 0x111b,
925 "EXlogin Failed=%x. MB0=%x MB11=%x\n",
926 rval, mcp->mb[0], mcp->mb[11]);
b0d6cabd 927 } else {
b0d6cabd
HM
928 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x118c,
929 "Done %s.\n", __func__);
930 }
931
932 return rval;
933}
934
2f56a7f1
HM
935/*
936 * qla_get_exchoffld_status
937 * Get exchange offload status
938 * uses the memory offload control/status Mailbox
939 *
940 * Input:
941 * ha: adapter state pointer.
942 * fwopt: firmware options
943 *
944 * Returns:
945 * qla2x00 local function status
946 *
947 * Context:
948 * Kernel context.
949 */
950#define FETCH_XCHOFFLD_STAT 0x2
951int
952qla_get_exchoffld_status(scsi_qla_host_t *vha, uint16_t *buf_sz,
953 uint16_t *ex_logins_cnt)
954{
955 int rval;
956 mbx_cmd_t mc;
957 mbx_cmd_t *mcp = &mc;
958
959 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1019,
960 "Entered %s\n", __func__);
961
962 memset(mcp->mb, 0 , sizeof(mcp->mb));
963 mcp->mb[0] = MBC_GET_MEM_OFFLOAD_CNTRL_STAT;
964 mcp->mb[1] = FETCH_XCHOFFLD_STAT;
965 mcp->out_mb = MBX_1|MBX_0;
966 mcp->in_mb = MBX_10|MBX_4|MBX_0;
967 mcp->tov = MBX_TOV_SECONDS;
968 mcp->flags = 0;
969
970 rval = qla2x00_mailbox_command(vha, mcp);
971 if (rval != QLA_SUCCESS) {
972 ql_dbg(ql_dbg_mbx, vha, 0x1155, "Failed=%x.\n", rval);
973 } else {
974 *buf_sz = mcp->mb[4];
975 *ex_logins_cnt = mcp->mb[10];
976
977 ql_log(ql_log_info, vha, 0x118e,
978 "buffer size 0x%x, exchange offload count=%d\n",
979 mcp->mb[4], mcp->mb[10]);
980
981 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1156,
982 "Done %s.\n", __func__);
983 }
984
985 return rval;
986}
987
988/*
989 * qla_set_exchoffld_mem_cfg
990 * Set exchange offload memory configuration
991 * Mbx needs to be issues before init_cb is set
992 *
993 * Input:
994 * ha: adapter state pointer.
995 * buffer: buffer pointer
996 * phys_addr: physical address of buffer
997 * size: size of buffer
998 * TARGET_QUEUE_LOCK must be released
999 * ADAPTER_STATE_LOCK must be release
1000 *
1001 * Returns:
1002 * qla2x00 local funxtion status code.
1003 *
1004 * Context:
1005 * Kernel context.
1006 */
1007#define CONFIG_XCHOFFLD_MEM 0x3
1008int
99e1b683 1009qla_set_exchoffld_mem_cfg(scsi_qla_host_t *vha)
2f56a7f1
HM
1010{
1011 int rval;
1012 mbx_cmd_t mc;
1013 mbx_cmd_t *mcp = &mc;
1014 struct qla_hw_data *ha = vha->hw;
1015
1016 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1157,
1017 "Entered %s.\n", __func__);
1018
1019 memset(mcp->mb, 0 , sizeof(mcp->mb));
1020 mcp->mb[0] = MBC_GET_MEM_OFFLOAD_CNTRL_STAT;
1021 mcp->mb[1] = CONFIG_XCHOFFLD_MEM;
99e1b683
QT
1022 mcp->mb[2] = MSW(ha->exchoffld_buf_dma);
1023 mcp->mb[3] = LSW(ha->exchoffld_buf_dma);
1024 mcp->mb[6] = MSW(MSD(ha->exchoffld_buf_dma));
1025 mcp->mb[7] = LSW(MSD(ha->exchoffld_buf_dma));
1026 mcp->mb[8] = MSW(ha->exchoffld_size);
1027 mcp->mb[9] = LSW(ha->exchoffld_size);
2f56a7f1
HM
1028 mcp->out_mb = MBX_9|MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
1029 mcp->in_mb = MBX_11|MBX_0;
1030 mcp->tov = MBX_TOV_SECONDS;
1031 mcp->flags = 0;
1032 rval = qla2x00_mailbox_command(vha, mcp);
1033 if (rval != QLA_SUCCESS) {
1034 /*EMPTY*/
1035 ql_dbg(ql_dbg_mbx, vha, 0x1158, "Failed=%x.\n", rval);
1036 } else {
1037 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1192,
1038 "Done %s.\n", __func__);
1039 }
1040
1041 return rval;
1042}
1043
1da177e4
LT
1044/*
1045 * qla2x00_get_fw_version
1046 * Get firmware version.
1047 *
1048 * Input:
1049 * ha: adapter state pointer.
1050 * major: pointer for major number.
1051 * minor: pointer for minor number.
1052 * subminor: pointer for subminor number.
1053 *
1054 * Returns:
1055 * qla2x00 local function return status code.
1056 *
1057 * Context:
1058 * Kernel context.
1059 */
ca9e9c3e 1060int
6246b8a1 1061qla2x00_get_fw_version(scsi_qla_host_t *vha)
1da177e4
LT
1062{
1063 int rval;
1064 mbx_cmd_t mc;
1065 mbx_cmd_t *mcp = &mc;
6246b8a1 1066 struct qla_hw_data *ha = vha->hw;
1da177e4 1067
5f28d2d7
SK
1068 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1029,
1069 "Entered %s.\n", __func__);
1da177e4
LT
1070
1071 mcp->mb[0] = MBC_GET_FIRMWARE_VERSION;
1072 mcp->out_mb = MBX_0;
1073 mcp->in_mb = MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
7ec0effd 1074 if (IS_QLA81XX(vha->hw) || IS_QLA8031(ha) || IS_QLA8044(ha))
55a96158 1075 mcp->in_mb |= MBX_13|MBX_12|MBX_11|MBX_10|MBX_9|MBX_8;
fb0effee 1076 if (IS_FWI2_CAPABLE(ha))
6246b8a1 1077 mcp->in_mb |= MBX_17|MBX_16|MBX_15;
ecc89f25 1078 if (IS_QLA27XX(ha) || IS_QLA28XX(ha))
ad1ef177
JC
1079 mcp->in_mb |=
1080 MBX_25|MBX_24|MBX_23|MBX_22|MBX_21|MBX_20|MBX_19|MBX_18|
2a3192a3 1081 MBX_14|MBX_13|MBX_11|MBX_10|MBX_9|MBX_8|MBX_7;
03aa868c 1082
1da177e4 1083 mcp->flags = 0;
b93480e3 1084 mcp->tov = MBX_TOV_SECONDS;
7b867cf7 1085 rval = qla2x00_mailbox_command(vha, mcp);
ca9e9c3e
AV
1086 if (rval != QLA_SUCCESS)
1087 goto failed;
1da177e4
LT
1088
1089 /* Return mailbox data. */
6246b8a1
GM
1090 ha->fw_major_version = mcp->mb[1];
1091 ha->fw_minor_version = mcp->mb[2];
1092 ha->fw_subminor_version = mcp->mb[3];
1093 ha->fw_attributes = mcp->mb[6];
7b867cf7 1094 if (IS_QLA2100(vha->hw) || IS_QLA2200(vha->hw))
6246b8a1 1095 ha->fw_memory_size = 0x1FFFF; /* Defaults to 128KB. */
1da177e4 1096 else
6246b8a1 1097 ha->fw_memory_size = (mcp->mb[5] << 16) | mcp->mb[4];
03aa868c 1098
7ec0effd 1099 if (IS_QLA81XX(vha->hw) || IS_QLA8031(vha->hw) || IS_QLA8044(ha)) {
6246b8a1
GM
1100 ha->mpi_version[0] = mcp->mb[10] & 0xff;
1101 ha->mpi_version[1] = mcp->mb[11] >> 8;
1102 ha->mpi_version[2] = mcp->mb[11] & 0xff;
1103 ha->mpi_capabilities = (mcp->mb[12] << 16) | mcp->mb[13];
1104 ha->phy_version[0] = mcp->mb[8] & 0xff;
1105 ha->phy_version[1] = mcp->mb[9] >> 8;
1106 ha->phy_version[2] = mcp->mb[9] & 0xff;
1107 }
03aa868c 1108
81178772
SK
1109 if (IS_FWI2_CAPABLE(ha)) {
1110 ha->fw_attributes_h = mcp->mb[15];
1111 ha->fw_attributes_ext[0] = mcp->mb[16];
1112 ha->fw_attributes_ext[1] = mcp->mb[17];
1113 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1139,
1114 "%s: FW_attributes Upper: 0x%x, Lower: 0x%x.\n",
1115 __func__, mcp->mb[15], mcp->mb[6]);
1116 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x112f,
1117 "%s: Ext_FwAttributes Upper: 0x%x, Lower: 0x%x.\n",
1118 __func__, mcp->mb[17], mcp->mb[16]);
2f56a7f1 1119
b0d6cabd
HM
1120 if (ha->fw_attributes_h & 0x4)
1121 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x118d,
1122 "%s: Firmware supports Extended Login 0x%x\n",
1123 __func__, ha->fw_attributes_h);
2f56a7f1
HM
1124
1125 if (ha->fw_attributes_h & 0x8)
1126 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1191,
1127 "%s: Firmware supports Exchange Offload 0x%x\n",
1128 __func__, ha->fw_attributes_h);
e84067d7 1129
deeae7a6
DG
1130 /*
1131 * FW supports nvme and driver load parameter requested nvme.
1132 * BIT 26 of fw_attributes indicates NVMe support.
1133 */
171e4909
GM
1134 if ((ha->fw_attributes_h &
1135 (FW_ATTR_H_NVME | FW_ATTR_H_NVME_UPDATED)) &&
1136 ql2xnvmeenable) {
03aaa89f
DT
1137 if (ha->fw_attributes_h & FW_ATTR_H_NVME_FBURST)
1138 vha->flags.nvme_first_burst = 1;
1139
e84067d7 1140 vha->flags.nvme_enabled = 1;
1cbc0efc
DT
1141 ql_log(ql_log_info, vha, 0xd302,
1142 "%s: FC-NVMe is Enabled (0x%x)\n",
1143 __func__, ha->fw_attributes_h);
1144 }
cf3c54fb
SK
1145
1146 /* BIT_13 of Extended FW Attributes informs about NVMe2 support */
1147 if (ha->fw_attributes_ext[0] & FW_ATTR_EXT0_NVME2) {
1148 ql_log(ql_log_info, vha, 0xd302,
1149 "Firmware supports NVMe2 0x%x\n",
1150 ha->fw_attributes_ext[0]);
1151 vha->flags.nvme2_enabled = 1;
1152 }
d07b75ba
QT
1153
1154 if (IS_QLA28XX(ha) && ha->flags.edif_hw && ql2xsecenable &&
1155 (ha->fw_attributes_ext[0] & FW_ATTR_EXT0_EDIF)) {
1156 ha->flags.edif_enabled = 1;
4de067e5 1157 ql_log(ql_log_info, vha, 0xffff,
d07b75ba
QT
1158 "%s: edif is enabled\n", __func__);
1159 }
3a03eb79 1160 }
03aa868c 1161
ecc89f25 1162 if (IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
2a3192a3
JC
1163 ha->serdes_version[0] = mcp->mb[7] & 0xff;
1164 ha->serdes_version[1] = mcp->mb[8] >> 8;
1165 ha->serdes_version[2] = mcp->mb[8] & 0xff;
03aa868c
SC
1166 ha->mpi_version[0] = mcp->mb[10] & 0xff;
1167 ha->mpi_version[1] = mcp->mb[11] >> 8;
1168 ha->mpi_version[2] = mcp->mb[11] & 0xff;
1169 ha->pep_version[0] = mcp->mb[13] & 0xff;
1170 ha->pep_version[1] = mcp->mb[14] >> 8;
1171 ha->pep_version[2] = mcp->mb[14] & 0xff;
f73cb695
CD
1172 ha->fw_shared_ram_start = (mcp->mb[19] << 16) | mcp->mb[18];
1173 ha->fw_shared_ram_end = (mcp->mb[21] << 16) | mcp->mb[20];
ad1ef177
JC
1174 ha->fw_ddr_ram_start = (mcp->mb[23] << 16) | mcp->mb[22];
1175 ha->fw_ddr_ram_end = (mcp->mb[25] << 16) | mcp->mb[24];
3f006ac3 1176 if (IS_QLA28XX(ha)) {
4ba836f6 1177 if (mcp->mb[16] & BIT_10)
3f006ac3 1178 ha->flags.secure_fw = 1;
4ba836f6
MH
1179
1180 ql_log(ql_log_info, vha, 0xffff,
1181 "Secure Flash Update in FW: %s\n",
1182 (ha->flags.secure_fw) ? "Supported" :
1183 "Not Supported");
3f006ac3 1184 }
9f2475fe
SS
1185
1186 if (ha->flags.scm_supported_a &&
1187 (ha->fw_attributes_ext[0] & FW_ATTR_EXT0_SCM_SUPPORTED)) {
1188 ha->flags.scm_supported_f = 1;
8a78dd6e 1189 ha->sf_init_cb->flags |= cpu_to_le16(BIT_13);
9f2475fe
SS
1190 }
1191 ql_log(ql_log_info, vha, 0x11a3, "SCM in FW: %s\n",
1192 (ha->flags.scm_supported_f) ? "Supported" :
1193 "Not Supported");
cf3c54fb
SK
1194
1195 if (vha->flags.nvme2_enabled) {
1196 /* set BIT_15 of special feature control block for SLER */
8a78dd6e 1197 ha->sf_init_cb->flags |= cpu_to_le16(BIT_15);
cf3c54fb 1198 /* set BIT_14 of special feature control block for PI CTRL*/
8a78dd6e 1199 ha->sf_init_cb->flags |= cpu_to_le16(BIT_14);
cf3c54fb 1200 }
f73cb695 1201 }
6246b8a1 1202
ca9e9c3e 1203failed:
1da177e4
LT
1204 if (rval != QLA_SUCCESS) {
1205 /*EMPTY*/
7c3df132 1206 ql_dbg(ql_dbg_mbx, vha, 0x102a, "Failed=%x.\n", rval);
1da177e4
LT
1207 } else {
1208 /*EMPTY*/
5f28d2d7
SK
1209 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x102b,
1210 "Done %s.\n", __func__);
1da177e4 1211 }
ca9e9c3e 1212 return rval;
1da177e4
LT
1213}
1214
1215/*
1216 * qla2x00_get_fw_options
1217 * Set firmware options.
1218 *
1219 * Input:
1220 * ha = adapter block pointer.
1221 * fwopt = pointer for firmware options.
1222 *
1223 * Returns:
1224 * qla2x00 local function return status code.
1225 *
1226 * Context:
1227 * Kernel context.
1228 */
1229int
7b867cf7 1230qla2x00_get_fw_options(scsi_qla_host_t *vha, uint16_t *fwopts)
1da177e4
LT
1231{
1232 int rval;
1233 mbx_cmd_t mc;
1234 mbx_cmd_t *mcp = &mc;
1235
5f28d2d7
SK
1236 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x102c,
1237 "Entered %s.\n", __func__);
1da177e4
LT
1238
1239 mcp->mb[0] = MBC_GET_FIRMWARE_OPTION;
1240 mcp->out_mb = MBX_0;
1241 mcp->in_mb = MBX_3|MBX_2|MBX_1|MBX_0;
b93480e3 1242 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1243 mcp->flags = 0;
7b867cf7 1244 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1245
1246 if (rval != QLA_SUCCESS) {
1247 /*EMPTY*/
7c3df132 1248 ql_dbg(ql_dbg_mbx, vha, 0x102d, "Failed=%x.\n", rval);
1da177e4 1249 } else {
1c7c6357 1250 fwopts[0] = mcp->mb[0];
1da177e4
LT
1251 fwopts[1] = mcp->mb[1];
1252 fwopts[2] = mcp->mb[2];
1253 fwopts[3] = mcp->mb[3];
1254
5f28d2d7
SK
1255 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x102e,
1256 "Done %s.\n", __func__);
1da177e4
LT
1257 }
1258
1259 return rval;
1260}
1261
1262
1263/*
1264 * qla2x00_set_fw_options
1265 * Set firmware options.
1266 *
1267 * Input:
1268 * ha = adapter block pointer.
1269 * fwopt = pointer for firmware options.
1270 *
1271 * Returns:
1272 * qla2x00 local function return status code.
1273 *
1274 * Context:
1275 * Kernel context.
1276 */
1277int
7b867cf7 1278qla2x00_set_fw_options(scsi_qla_host_t *vha, uint16_t *fwopts)
1da177e4
LT
1279{
1280 int rval;
1281 mbx_cmd_t mc;
1282 mbx_cmd_t *mcp = &mc;
1283
5f28d2d7
SK
1284 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x102f,
1285 "Entered %s.\n", __func__);
1da177e4
LT
1286
1287 mcp->mb[0] = MBC_SET_FIRMWARE_OPTION;
1288 mcp->mb[1] = fwopts[1];
1289 mcp->mb[2] = fwopts[2];
1290 mcp->mb[3] = fwopts[3];
1c7c6357 1291 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
1da177e4 1292 mcp->in_mb = MBX_0;
7b867cf7 1293 if (IS_FWI2_CAPABLE(vha->hw)) {
1c7c6357 1294 mcp->in_mb |= MBX_1;
2da52737
QT
1295 mcp->mb[10] = fwopts[10];
1296 mcp->out_mb |= MBX_10;
1c7c6357
AV
1297 } else {
1298 mcp->mb[10] = fwopts[10];
1299 mcp->mb[11] = fwopts[11];
1300 mcp->mb[12] = 0; /* Undocumented, but used */
1301 mcp->out_mb |= MBX_12|MBX_11|MBX_10;
1302 }
b93480e3 1303 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1304 mcp->flags = 0;
7b867cf7 1305 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4 1306
1c7c6357
AV
1307 fwopts[0] = mcp->mb[0];
1308
1da177e4
LT
1309 if (rval != QLA_SUCCESS) {
1310 /*EMPTY*/
7c3df132
SK
1311 ql_dbg(ql_dbg_mbx, vha, 0x1030,
1312 "Failed=%x (%x/%x).\n", rval, mcp->mb[0], mcp->mb[1]);
1da177e4
LT
1313 } else {
1314 /*EMPTY*/
5f28d2d7
SK
1315 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1031,
1316 "Done %s.\n", __func__);
1da177e4
LT
1317 }
1318
1319 return rval;
1320}
1321
1322/*
1323 * qla2x00_mbx_reg_test
1324 * Mailbox register wrap test.
1325 *
1326 * Input:
1327 * ha = adapter block pointer.
1328 * TARGET_QUEUE_LOCK must be released.
1329 * ADAPTER_STATE_LOCK must be released.
1330 *
1331 * Returns:
1332 * qla2x00 local function return status code.
1333 *
1334 * Context:
1335 * Kernel context.
1336 */
1337int
7b867cf7 1338qla2x00_mbx_reg_test(scsi_qla_host_t *vha)
1da177e4
LT
1339{
1340 int rval;
1341 mbx_cmd_t mc;
1342 mbx_cmd_t *mcp = &mc;
1343
5f28d2d7
SK
1344 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1032,
1345 "Entered %s.\n", __func__);
1da177e4
LT
1346
1347 mcp->mb[0] = MBC_MAILBOX_REGISTER_TEST;
1348 mcp->mb[1] = 0xAAAA;
1349 mcp->mb[2] = 0x5555;
1350 mcp->mb[3] = 0xAA55;
1351 mcp->mb[4] = 0x55AA;
1352 mcp->mb[5] = 0xA5A5;
1353 mcp->mb[6] = 0x5A5A;
1354 mcp->mb[7] = 0x2525;
1355 mcp->out_mb = MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
1356 mcp->in_mb = MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
b93480e3 1357 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1358 mcp->flags = 0;
7b867cf7 1359 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1360
1361 if (rval == QLA_SUCCESS) {
1362 if (mcp->mb[1] != 0xAAAA || mcp->mb[2] != 0x5555 ||
1363 mcp->mb[3] != 0xAA55 || mcp->mb[4] != 0x55AA)
1364 rval = QLA_FUNCTION_FAILED;
1365 if (mcp->mb[5] != 0xA5A5 || mcp->mb[6] != 0x5A5A ||
1366 mcp->mb[7] != 0x2525)
1367 rval = QLA_FUNCTION_FAILED;
1368 }
1369
1370 if (rval != QLA_SUCCESS) {
1371 /*EMPTY*/
7c3df132 1372 ql_dbg(ql_dbg_mbx, vha, 0x1033, "Failed=%x.\n", rval);
dbf1f53c 1373 vha->hw_err_cnt++;
1da177e4
LT
1374 } else {
1375 /*EMPTY*/
5f28d2d7
SK
1376 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1034,
1377 "Done %s.\n", __func__);
1da177e4
LT
1378 }
1379
1380 return rval;
1381}
1382
1383/*
1384 * qla2x00_verify_checksum
1385 * Verify firmware checksum.
1386 *
1387 * Input:
1388 * ha = adapter block pointer.
1389 * TARGET_QUEUE_LOCK must be released.
1390 * ADAPTER_STATE_LOCK must be released.
1391 *
1392 * Returns:
1393 * qla2x00 local function return status code.
1394 *
1395 * Context:
1396 * Kernel context.
1397 */
1398int
7b867cf7 1399qla2x00_verify_checksum(scsi_qla_host_t *vha, uint32_t risc_addr)
1da177e4
LT
1400{
1401 int rval;
1402 mbx_cmd_t mc;
1403 mbx_cmd_t *mcp = &mc;
1404
5f28d2d7
SK
1405 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1035,
1406 "Entered %s.\n", __func__);
1da177e4
LT
1407
1408 mcp->mb[0] = MBC_VERIFY_CHECKSUM;
1c7c6357
AV
1409 mcp->out_mb = MBX_0;
1410 mcp->in_mb = MBX_0;
7b867cf7 1411 if (IS_FWI2_CAPABLE(vha->hw)) {
1c7c6357
AV
1412 mcp->mb[1] = MSW(risc_addr);
1413 mcp->mb[2] = LSW(risc_addr);
1414 mcp->out_mb |= MBX_2|MBX_1;
1415 mcp->in_mb |= MBX_2|MBX_1;
1416 } else {
1417 mcp->mb[1] = LSW(risc_addr);
1418 mcp->out_mb |= MBX_1;
1419 mcp->in_mb |= MBX_1;
1420 }
1421
b93480e3 1422 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1423 mcp->flags = 0;
7b867cf7 1424 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1425
1426 if (rval != QLA_SUCCESS) {
7c3df132
SK
1427 ql_dbg(ql_dbg_mbx, vha, 0x1036,
1428 "Failed=%x chm sum=%x.\n", rval, IS_FWI2_CAPABLE(vha->hw) ?
1429 (mcp->mb[2] << 16) | mcp->mb[1] : mcp->mb[1]);
1da177e4 1430 } else {
5f28d2d7
SK
1431 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1037,
1432 "Done %s.\n", __func__);
1da177e4
LT
1433 }
1434
1435 return rval;
1436}
1437
1438/*
1439 * qla2x00_issue_iocb
1440 * Issue IOCB using mailbox command
1441 *
1442 * Input:
1443 * ha = adapter state pointer.
1444 * buffer = buffer pointer.
1445 * phys_addr = physical address of buffer.
1446 * size = size of buffer.
1447 * TARGET_QUEUE_LOCK must be released.
1448 * ADAPTER_STATE_LOCK must be released.
1449 *
1450 * Returns:
1451 * qla2x00 local function return status code.
1452 *
1453 * Context:
1454 * Kernel context.
1455 */
6e98016c 1456int
7b867cf7 1457qla2x00_issue_iocb_timeout(scsi_qla_host_t *vha, void *buffer,
4d4df193 1458 dma_addr_t phys_addr, size_t size, uint32_t tov)
1da177e4
LT
1459{
1460 int rval;
1461 mbx_cmd_t mc;
1462 mbx_cmd_t *mcp = &mc;
1463
ab391abd 1464 if (!vha->hw->flags.fw_started)
345f574d
HM
1465 return QLA_INVALID_COMMAND;
1466
5f28d2d7
SK
1467 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1038,
1468 "Entered %s.\n", __func__);
7c3df132 1469
1da177e4
LT
1470 mcp->mb[0] = MBC_IOCB_COMMAND_A64;
1471 mcp->mb[1] = 0;
818c7f87
JC
1472 mcp->mb[2] = MSW(LSD(phys_addr));
1473 mcp->mb[3] = LSW(LSD(phys_addr));
1da177e4
LT
1474 mcp->mb[6] = MSW(MSD(phys_addr));
1475 mcp->mb[7] = LSW(MSD(phys_addr));
1476 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
818c7f87 1477 mcp->in_mb = MBX_1|MBX_0;
4d4df193 1478 mcp->tov = tov;
1da177e4 1479 mcp->flags = 0;
7b867cf7 1480 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1481
1482 if (rval != QLA_SUCCESS) {
1483 /*EMPTY*/
7c3df132 1484 ql_dbg(ql_dbg_mbx, vha, 0x1039, "Failed=%x.\n", rval);
1da177e4 1485 } else {
818c7f87 1486 sts_entry_t *sts_entry = buffer;
8c958a99
AV
1487
1488 /* Mask reserved bits. */
1489 sts_entry->entry_status &=
7b867cf7 1490 IS_FWI2_CAPABLE(vha->hw) ? RF_MASK_24XX : RF_MASK;
5f28d2d7 1491 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x103a,
818c7f87
JC
1492 "Done %s (status=%x).\n", __func__,
1493 sts_entry->entry_status);
1da177e4
LT
1494 }
1495
1496 return rval;
1497}
1498
4d4df193 1499int
7b867cf7 1500qla2x00_issue_iocb(scsi_qla_host_t *vha, void *buffer, dma_addr_t phys_addr,
4d4df193
HK
1501 size_t size)
1502{
7b867cf7 1503 return qla2x00_issue_iocb_timeout(vha, buffer, phys_addr, size,
4d4df193
HK
1504 MBX_TOV_SECONDS);
1505}
1506
1da177e4
LT
1507/*
1508 * qla2x00_abort_command
1509 * Abort command aborts a specified IOCB.
1510 *
1511 * Input:
1512 * ha = adapter block pointer.
1513 * sp = SB structure pointer.
1514 *
1515 * Returns:
1516 * qla2x00 local function return status code.
1517 *
1518 * Context:
1519 * Kernel context.
1520 */
1521int
2afa19a9 1522qla2x00_abort_command(srb_t *sp)
1da177e4
LT
1523{
1524 unsigned long flags = 0;
1da177e4 1525 int rval;
73208dfd 1526 uint32_t handle = 0;
1da177e4
LT
1527 mbx_cmd_t mc;
1528 mbx_cmd_t *mcp = &mc;
2afa19a9
AC
1529 fc_port_t *fcport = sp->fcport;
1530 scsi_qla_host_t *vha = fcport->vha;
7b867cf7 1531 struct qla_hw_data *ha = vha->hw;
d7459527 1532 struct req_que *req;
9ba56b95 1533 struct scsi_cmnd *cmd = GET_CMD_SP(sp);
1da177e4 1534
5f28d2d7
SK
1535 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x103b,
1536 "Entered %s.\n", __func__);
1da177e4 1537
345f574d 1538 if (sp->qpair)
d7459527
MH
1539 req = sp->qpair->req;
1540 else
1541 req = vha->req;
1542
c9c5ced9 1543 spin_lock_irqsave(&ha->hardware_lock, flags);
8d93f550 1544 for (handle = 1; handle < req->num_outstanding_cmds; handle++) {
7b867cf7 1545 if (req->outstanding_cmds[handle] == sp)
1da177e4
LT
1546 break;
1547 }
c9c5ced9 1548 spin_unlock_irqrestore(&ha->hardware_lock, flags);
1da177e4 1549
8d93f550 1550 if (handle == req->num_outstanding_cmds) {
1da177e4
LT
1551 /* command not found */
1552 return QLA_FUNCTION_FAILED;
1553 }
1554
1555 mcp->mb[0] = MBC_ABORT_COMMAND;
1556 if (HAS_EXTENDED_IDS(ha))
1557 mcp->mb[1] = fcport->loop_id;
1558 else
1559 mcp->mb[1] = fcport->loop_id << 8;
1560 mcp->mb[2] = (uint16_t)handle;
1561 mcp->mb[3] = (uint16_t)(handle >> 16);
9ba56b95 1562 mcp->mb[6] = (uint16_t)cmd->device->lun;
1da177e4
LT
1563 mcp->out_mb = MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
1564 mcp->in_mb = MBX_0;
b93480e3 1565 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1566 mcp->flags = 0;
7b867cf7 1567 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1568
1569 if (rval != QLA_SUCCESS) {
7c3df132 1570 ql_dbg(ql_dbg_mbx, vha, 0x103c, "Failed=%x.\n", rval);
1da177e4 1571 } else {
5f28d2d7
SK
1572 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x103d,
1573 "Done %s.\n", __func__);
1da177e4
LT
1574 }
1575
1576 return rval;
1577}
1578
1da177e4 1579int
9cb78c16 1580qla2x00_abort_target(struct fc_port *fcport, uint64_t l, int tag)
1da177e4 1581{
523ec773 1582 int rval, rval2;
1da177e4
LT
1583 mbx_cmd_t mc;
1584 mbx_cmd_t *mcp = &mc;
7b867cf7 1585 scsi_qla_host_t *vha;
1da177e4 1586
7b867cf7 1587 vha = fcport->vha;
7c3df132 1588
5f28d2d7
SK
1589 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x103e,
1590 "Entered %s.\n", __func__);
7c3df132 1591
1da177e4 1592 mcp->mb[0] = MBC_ABORT_TARGET;
523ec773 1593 mcp->out_mb = MBX_9|MBX_2|MBX_1|MBX_0;
7b867cf7 1594 if (HAS_EXTENDED_IDS(vha->hw)) {
1da177e4
LT
1595 mcp->mb[1] = fcport->loop_id;
1596 mcp->mb[10] = 0;
1597 mcp->out_mb |= MBX_10;
1598 } else {
1599 mcp->mb[1] = fcport->loop_id << 8;
1600 }
7b867cf7
AC
1601 mcp->mb[2] = vha->hw->loop_reset_delay;
1602 mcp->mb[9] = vha->vp_idx;
1da177e4
LT
1603
1604 mcp->in_mb = MBX_0;
b93480e3 1605 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1606 mcp->flags = 0;
7b867cf7 1607 rval = qla2x00_mailbox_command(vha, mcp);
523ec773 1608 if (rval != QLA_SUCCESS) {
5f28d2d7
SK
1609 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x103f,
1610 "Failed=%x.\n", rval);
523ec773
AV
1611 }
1612
1613 /* Issue marker IOCB. */
9eb9c6dc 1614 rval2 = qla2x00_marker(vha, vha->hw->base_qpair, fcport->loop_id, 0,
73208dfd 1615 MK_SYNC_ID);
523ec773 1616 if (rval2 != QLA_SUCCESS) {
7c3df132
SK
1617 ql_dbg(ql_dbg_mbx, vha, 0x1040,
1618 "Failed to issue marker IOCB (%x).\n", rval2);
523ec773 1619 } else {
5f28d2d7
SK
1620 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1041,
1621 "Done %s.\n", __func__);
523ec773
AV
1622 }
1623
1624 return rval;
1625}
1626
1627int
9cb78c16 1628qla2x00_lun_reset(struct fc_port *fcport, uint64_t l, int tag)
523ec773
AV
1629{
1630 int rval, rval2;
1631 mbx_cmd_t mc;
1632 mbx_cmd_t *mcp = &mc;
7b867cf7 1633 scsi_qla_host_t *vha;
523ec773 1634
7b867cf7 1635 vha = fcport->vha;
7c3df132 1636
5f28d2d7
SK
1637 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1042,
1638 "Entered %s.\n", __func__);
7c3df132 1639
523ec773
AV
1640 mcp->mb[0] = MBC_LUN_RESET;
1641 mcp->out_mb = MBX_9|MBX_3|MBX_2|MBX_1|MBX_0;
7b867cf7 1642 if (HAS_EXTENDED_IDS(vha->hw))
523ec773
AV
1643 mcp->mb[1] = fcport->loop_id;
1644 else
1645 mcp->mb[1] = fcport->loop_id << 8;
9cb78c16 1646 mcp->mb[2] = (u32)l;
523ec773 1647 mcp->mb[3] = 0;
7b867cf7 1648 mcp->mb[9] = vha->vp_idx;
1da177e4 1649
523ec773 1650 mcp->in_mb = MBX_0;
b93480e3 1651 mcp->tov = MBX_TOV_SECONDS;
523ec773 1652 mcp->flags = 0;
7b867cf7 1653 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4 1654 if (rval != QLA_SUCCESS) {
7c3df132 1655 ql_dbg(ql_dbg_mbx, vha, 0x1043, "Failed=%x.\n", rval);
523ec773
AV
1656 }
1657
1658 /* Issue marker IOCB. */
9eb9c6dc 1659 rval2 = qla2x00_marker(vha, vha->hw->base_qpair, fcport->loop_id, l,
73208dfd 1660 MK_SYNC_ID_LUN);
523ec773 1661 if (rval2 != QLA_SUCCESS) {
7c3df132
SK
1662 ql_dbg(ql_dbg_mbx, vha, 0x1044,
1663 "Failed to issue marker IOCB (%x).\n", rval2);
1da177e4 1664 } else {
5f28d2d7
SK
1665 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1045,
1666 "Done %s.\n", __func__);
1da177e4
LT
1667 }
1668
1669 return rval;
1670}
1da177e4 1671
1da177e4
LT
1672/*
1673 * qla2x00_get_adapter_id
1674 * Get adapter ID and topology.
1675 *
1676 * Input:
1677 * ha = adapter block pointer.
1678 * id = pointer for loop ID.
1679 * al_pa = pointer for AL_PA.
1680 * area = pointer for area.
1681 * domain = pointer for domain.
1682 * top = pointer for topology.
1683 * TARGET_QUEUE_LOCK must be released.
1684 * ADAPTER_STATE_LOCK must be released.
1685 *
1686 * Returns:
1687 * qla2x00 local function return status code.
1688 *
1689 * Context:
1690 * Kernel context.
1691 */
1692int
7b867cf7 1693qla2x00_get_adapter_id(scsi_qla_host_t *vha, uint16_t *id, uint8_t *al_pa,
2c3dfe3f 1694 uint8_t *area, uint8_t *domain, uint16_t *top, uint16_t *sw_cap)
1da177e4
LT
1695{
1696 int rval;
1697 mbx_cmd_t mc;
1698 mbx_cmd_t *mcp = &mc;
1699
5f28d2d7
SK
1700 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1046,
1701 "Entered %s.\n", __func__);
1da177e4
LT
1702
1703 mcp->mb[0] = MBC_GET_ADAPTER_LOOP_ID;
7b867cf7 1704 mcp->mb[9] = vha->vp_idx;
eb66dc60 1705 mcp->out_mb = MBX_9|MBX_0;
2c3dfe3f 1706 mcp->in_mb = MBX_9|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
6246b8a1 1707 if (IS_CNA_CAPABLE(vha->hw))
bad7001c 1708 mcp->in_mb |= MBX_13|MBX_12|MBX_11|MBX_10;
7c9c4766
JC
1709 if (IS_FWI2_CAPABLE(vha->hw))
1710 mcp->in_mb |= MBX_19|MBX_18|MBX_17|MBX_16;
39200687
EM
1711 if (IS_QLA27XX(vha->hw) || IS_QLA28XX(vha->hw))
1712 mcp->in_mb |= MBX_15|MBX_21|MBX_22|MBX_23;
9f2475fe 1713
b93480e3 1714 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1715 mcp->flags = 0;
7b867cf7 1716 rval = qla2x00_mailbox_command(vha, mcp);
33135aa2
RA
1717 if (mcp->mb[0] == MBS_COMMAND_ERROR)
1718 rval = QLA_COMMAND_ERROR;
42e421b1
AV
1719 else if (mcp->mb[0] == MBS_INVALID_COMMAND)
1720 rval = QLA_INVALID_COMMAND;
1da177e4
LT
1721
1722 /* Return data. */
1723 *id = mcp->mb[1];
1724 *al_pa = LSB(mcp->mb[2]);
1725 *area = MSB(mcp->mb[2]);
1726 *domain = LSB(mcp->mb[3]);
1727 *top = mcp->mb[6];
2c3dfe3f 1728 *sw_cap = mcp->mb[7];
1da177e4
LT
1729
1730 if (rval != QLA_SUCCESS) {
1731 /*EMPTY*/
7c3df132 1732 ql_dbg(ql_dbg_mbx, vha, 0x1047, "Failed=%x.\n", rval);
1da177e4 1733 } else {
5f28d2d7
SK
1734 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1048,
1735 "Done %s.\n", __func__);
bad7001c 1736
6246b8a1 1737 if (IS_CNA_CAPABLE(vha->hw)) {
bad7001c
AV
1738 vha->fcoe_vlan_id = mcp->mb[9] & 0xfff;
1739 vha->fcoe_fcf_idx = mcp->mb[10];
1740 vha->fcoe_vn_port_mac[5] = mcp->mb[11] >> 8;
1741 vha->fcoe_vn_port_mac[4] = mcp->mb[11] & 0xff;
1742 vha->fcoe_vn_port_mac[3] = mcp->mb[12] >> 8;
1743 vha->fcoe_vn_port_mac[2] = mcp->mb[12] & 0xff;
1744 vha->fcoe_vn_port_mac[1] = mcp->mb[13] >> 8;
1745 vha->fcoe_vn_port_mac[0] = mcp->mb[13] & 0xff;
1746 }
7c9c4766 1747 /* If FA-WWN supported */
d6b9b42b
SK
1748 if (IS_FAWWN_CAPABLE(vha->hw)) {
1749 if (mcp->mb[7] & BIT_14) {
1750 vha->port_name[0] = MSB(mcp->mb[16]);
1751 vha->port_name[1] = LSB(mcp->mb[16]);
1752 vha->port_name[2] = MSB(mcp->mb[17]);
1753 vha->port_name[3] = LSB(mcp->mb[17]);
1754 vha->port_name[4] = MSB(mcp->mb[18]);
1755 vha->port_name[5] = LSB(mcp->mb[18]);
1756 vha->port_name[6] = MSB(mcp->mb[19]);
1757 vha->port_name[7] = LSB(mcp->mb[19]);
1758 fc_host_port_name(vha->host) =
1759 wwn_to_u64(vha->port_name);
1760 ql_dbg(ql_dbg_mbx, vha, 0x10ca,
1761 "FA-WWN acquired %016llx\n",
1762 wwn_to_u64(vha->port_name));
1763 }
7c9c4766 1764 }
969a6199 1765
9f2475fe 1766 if (IS_QLA27XX(vha->hw) || IS_QLA28XX(vha->hw)) {
969a6199 1767 vha->bbcr = mcp->mb[15];
9f2475fe
SS
1768 if (mcp->mb[7] & SCM_EDC_ACC_RECEIVED) {
1769 ql_log(ql_log_info, vha, 0x11a4,
1770 "SCM: EDC ELS completed, flags 0x%x\n",
1771 mcp->mb[21]);
1772 }
1773 if (mcp->mb[7] & SCM_RDF_ACC_RECEIVED) {
1774 vha->hw->flags.scm_enabled = 1;
1775 vha->scm_fabric_connection_flags |=
1776 SCM_FLAG_RDF_COMPLETED;
1777 ql_log(ql_log_info, vha, 0x11a5,
1778 "SCM: RDF ELS completed, flags 0x%x\n",
1779 mcp->mb[23]);
1780 }
1781 }
1da177e4
LT
1782 }
1783
1784 return rval;
1785}
1786
1787/*
1788 * qla2x00_get_retry_cnt
1789 * Get current firmware login retry count and delay.
1790 *
1791 * Input:
1792 * ha = adapter block pointer.
1793 * retry_cnt = pointer to login retry count.
1794 * tov = pointer to login timeout value.
1795 *
1796 * Returns:
1797 * qla2x00 local function return status code.
1798 *
1799 * Context:
1800 * Kernel context.
1801 */
1802int
7b867cf7 1803qla2x00_get_retry_cnt(scsi_qla_host_t *vha, uint8_t *retry_cnt, uint8_t *tov,
1da177e4
LT
1804 uint16_t *r_a_tov)
1805{
1806 int rval;
1807 uint16_t ratov;
1808 mbx_cmd_t mc;
1809 mbx_cmd_t *mcp = &mc;
1810
5f28d2d7
SK
1811 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1049,
1812 "Entered %s.\n", __func__);
1da177e4
LT
1813
1814 mcp->mb[0] = MBC_GET_RETRY_COUNT;
1815 mcp->out_mb = MBX_0;
1816 mcp->in_mb = MBX_3|MBX_2|MBX_1|MBX_0;
b93480e3 1817 mcp->tov = MBX_TOV_SECONDS;
1da177e4 1818 mcp->flags = 0;
7b867cf7 1819 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1820
1821 if (rval != QLA_SUCCESS) {
1822 /*EMPTY*/
7c3df132
SK
1823 ql_dbg(ql_dbg_mbx, vha, 0x104a,
1824 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
1da177e4
LT
1825 } else {
1826 /* Convert returned data and check our values. */
1827 *r_a_tov = mcp->mb[3] / 2;
1828 ratov = (mcp->mb[3]/2) / 10; /* mb[3] value is in 100ms */
1829 if (mcp->mb[1] * ratov > (*retry_cnt) * (*tov)) {
1830 /* Update to the larger values */
1831 *retry_cnt = (uint8_t)mcp->mb[1];
1832 *tov = ratov;
1833 }
1834
5f28d2d7 1835 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x104b,
7c3df132 1836 "Done %s mb3=%d ratov=%d.\n", __func__, mcp->mb[3], ratov);
1da177e4
LT
1837 }
1838
1839 return rval;
1840}
1841
1842/*
1843 * qla2x00_init_firmware
1844 * Initialize adapter firmware.
1845 *
1846 * Input:
1847 * ha = adapter block pointer.
1848 * dptr = Initialization control block pointer.
1849 * size = size of initialization control block.
1850 * TARGET_QUEUE_LOCK must be released.
1851 * ADAPTER_STATE_LOCK must be released.
1852 *
1853 * Returns:
1854 * qla2x00 local function return status code.
1855 *
1856 * Context:
1857 * Kernel context.
1858 */
1859int
7b867cf7 1860qla2x00_init_firmware(scsi_qla_host_t *vha, uint16_t size)
1da177e4
LT
1861{
1862 int rval;
1863 mbx_cmd_t mc;
1864 mbx_cmd_t *mcp = &mc;
7b867cf7 1865 struct qla_hw_data *ha = vha->hw;
1da177e4 1866
5f28d2d7
SK
1867 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x104c,
1868 "Entered %s.\n", __func__);
1da177e4 1869
7ec0effd 1870 if (IS_P3P_TYPE(ha) && ql2xdbwr)
8dfa4b5a 1871 qla82xx_wr_32(ha, (uintptr_t __force)ha->nxdb_wr_ptr,
a9083016
GM
1872 (0x04 | (ha->portnum << 5) | (0 << 8) | (0 << 16)));
1873
e6e074f1 1874 if (ha->flags.npiv_supported)
2c3dfe3f
SJ
1875 mcp->mb[0] = MBC_MID_INITIALIZE_FIRMWARE;
1876 else
1877 mcp->mb[0] = MBC_INITIALIZE_FIRMWARE;
1878
b64b0e8f 1879 mcp->mb[1] = 0;
1da177e4
LT
1880 mcp->mb[2] = MSW(ha->init_cb_dma);
1881 mcp->mb[3] = LSW(ha->init_cb_dma);
1da177e4
LT
1882 mcp->mb[6] = MSW(MSD(ha->init_cb_dma));
1883 mcp->mb[7] = LSW(MSD(ha->init_cb_dma));
b64b0e8f 1884 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
4ef21bd4 1885 if (ha->ex_init_cb && ha->ex_init_cb->ex_version) {
b64b0e8f
AV
1886 mcp->mb[1] = BIT_0;
1887 mcp->mb[10] = MSW(ha->ex_init_cb_dma);
1888 mcp->mb[11] = LSW(ha->ex_init_cb_dma);
1889 mcp->mb[12] = MSW(MSD(ha->ex_init_cb_dma));
1890 mcp->mb[13] = LSW(MSD(ha->ex_init_cb_dma));
1891 mcp->mb[14] = sizeof(*ha->ex_init_cb);
1892 mcp->out_mb |= MBX_14|MBX_13|MBX_12|MBX_11|MBX_10;
1893 }
9f2475fe 1894
cf3c54fb 1895 if (ha->flags.scm_supported_f || vha->flags.nvme2_enabled) {
9f2475fe
SS
1896 mcp->mb[1] |= BIT_1;
1897 mcp->mb[16] = MSW(ha->sf_init_cb_dma);
1898 mcp->mb[17] = LSW(ha->sf_init_cb_dma);
1899 mcp->mb[18] = MSW(MSD(ha->sf_init_cb_dma));
1900 mcp->mb[19] = LSW(MSD(ha->sf_init_cb_dma));
1901 mcp->mb[15] = sizeof(*ha->sf_init_cb);
1902 mcp->out_mb |= MBX_19|MBX_18|MBX_17|MBX_16|MBX_15;
1903 }
1904
6246b8a1
GM
1905 /* 1 and 2 should normally be captured. */
1906 mcp->in_mb = MBX_2|MBX_1|MBX_0;
ecc89f25 1907 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha))
6246b8a1
GM
1908 /* mb3 is additional info about the installed SFP. */
1909 mcp->in_mb |= MBX_3;
1da177e4
LT
1910 mcp->buf_size = size;
1911 mcp->flags = MBX_DMA_OUT;
b93480e3 1912 mcp->tov = MBX_TOV_SECONDS;
7b867cf7 1913 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
1914
1915 if (rval != QLA_SUCCESS) {
1916 /*EMPTY*/
7c3df132 1917 ql_dbg(ql_dbg_mbx, vha, 0x104d,
f8f97b0c 1918 "Failed=%x mb[0]=%x, mb[1]=%x, mb[2]=%x, mb[3]=%x.\n",
6246b8a1 1919 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2], mcp->mb[3]);
f8f97b0c
JC
1920 if (ha->init_cb) {
1921 ql_dbg(ql_dbg_mbx, vha, 0x104d, "init_cb:\n");
1922 ql_dump_buffer(ql_dbg_init + ql_dbg_verbose, vha,
1923 0x0104d, ha->init_cb, sizeof(*ha->init_cb));
1924 }
1925 if (ha->ex_init_cb && ha->ex_init_cb->ex_version) {
1926 ql_dbg(ql_dbg_mbx, vha, 0x104d, "ex_init_cb:\n");
1927 ql_dump_buffer(ql_dbg_init + ql_dbg_verbose, vha,
1928 0x0104d, ha->ex_init_cb, sizeof(*ha->ex_init_cb));
1929 }
1da177e4 1930 } else {
ecc89f25 1931 if (IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
92d4408e
SC
1932 if (mcp->mb[2] == 6 || mcp->mb[3] == 2)
1933 ql_dbg(ql_dbg_mbx, vha, 0x119d,
1934 "Invalid SFP/Validation Failed\n");
1935 }
5f28d2d7
SK
1936 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x104e,
1937 "Done %s.\n", __func__);
1da177e4
LT
1938 }
1939
1940 return rval;
1941}
1942
2d70c103 1943
1da177e4
LT
1944/*
1945 * qla2x00_get_port_database
1946 * Issue normal/enhanced get port database mailbox command
1947 * and copy device name as necessary.
1948 *
1949 * Input:
1950 * ha = adapter state pointer.
1951 * dev = structure pointer.
1952 * opt = enhanced cmd option byte.
1953 *
1954 * Returns:
1955 * qla2x00 local function return status code.
1956 *
1957 * Context:
1958 * Kernel context.
1959 */
1960int
7b867cf7 1961qla2x00_get_port_database(scsi_qla_host_t *vha, fc_port_t *fcport, uint8_t opt)
1da177e4
LT
1962{
1963 int rval;
1964 mbx_cmd_t mc;
1965 mbx_cmd_t *mcp = &mc;
1966 port_database_t *pd;
1c7c6357 1967 struct port_database_24xx *pd24;
1da177e4 1968 dma_addr_t pd_dma;
7b867cf7 1969 struct qla_hw_data *ha = vha->hw;
1da177e4 1970
5f28d2d7
SK
1971 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x104f,
1972 "Entered %s.\n", __func__);
1da177e4 1973
1c7c6357 1974 pd24 = NULL;
08eb7f45 1975 pd = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &pd_dma);
1da177e4 1976 if (pd == NULL) {
7c3df132
SK
1977 ql_log(ql_log_warn, vha, 0x1050,
1978 "Failed to allocate port database structure.\n");
edd05de1 1979 fcport->query = 0;
1da177e4
LT
1980 return QLA_MEMORY_ALLOC_FAILED;
1981 }
1da177e4 1982
1c7c6357 1983 mcp->mb[0] = MBC_GET_PORT_DATABASE;
e428924c 1984 if (opt != 0 && !IS_FWI2_CAPABLE(ha))
1da177e4 1985 mcp->mb[0] = MBC_ENHANCED_GET_PORT_DATABASE;
1da177e4
LT
1986 mcp->mb[2] = MSW(pd_dma);
1987 mcp->mb[3] = LSW(pd_dma);
1988 mcp->mb[6] = MSW(MSD(pd_dma));
1989 mcp->mb[7] = LSW(MSD(pd_dma));
7b867cf7 1990 mcp->mb[9] = vha->vp_idx;
2c3dfe3f 1991 mcp->out_mb = MBX_9|MBX_7|MBX_6|MBX_3|MBX_2|MBX_0;
1da177e4 1992 mcp->in_mb = MBX_0;
e428924c 1993 if (IS_FWI2_CAPABLE(ha)) {
1c7c6357
AV
1994 mcp->mb[1] = fcport->loop_id;
1995 mcp->mb[10] = opt;
1996 mcp->out_mb |= MBX_10|MBX_1;
1997 mcp->in_mb |= MBX_1;
1998 } else if (HAS_EXTENDED_IDS(ha)) {
1999 mcp->mb[1] = fcport->loop_id;
2000 mcp->mb[10] = opt;
2001 mcp->out_mb |= MBX_10|MBX_1;
2002 } else {
2003 mcp->mb[1] = fcport->loop_id << 8 | opt;
2004 mcp->out_mb |= MBX_1;
2005 }
e428924c
AV
2006 mcp->buf_size = IS_FWI2_CAPABLE(ha) ?
2007 PORT_DATABASE_24XX_SIZE : PORT_DATABASE_SIZE;
1da177e4
LT
2008 mcp->flags = MBX_DMA_IN;
2009 mcp->tov = (ha->login_timeout * 2) + (ha->login_timeout / 2);
7b867cf7 2010 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2011 if (rval != QLA_SUCCESS)
2012 goto gpd_error_out;
2013
e428924c 2014 if (IS_FWI2_CAPABLE(ha)) {
0eba25df 2015 uint64_t zero = 0;
c0c462c8
DG
2016 u8 current_login_state, last_login_state;
2017
1c7c6357
AV
2018 pd24 = (struct port_database_24xx *) pd;
2019
2020 /* Check for logged in state. */
84ed362a 2021 if (NVME_TARGET(ha, fcport)) {
c0c462c8
DG
2022 current_login_state = pd24->current_login_state >> 4;
2023 last_login_state = pd24->last_login_state >> 4;
2024 } else {
2025 current_login_state = pd24->current_login_state & 0xf;
2026 last_login_state = pd24->last_login_state & 0xf;
2027 }
2028 fcport->current_login_state = pd24->current_login_state;
2029 fcport->last_login_state = pd24->last_login_state;
2030
2031 /* Check for logged in state. */
2032 if (current_login_state != PDS_PRLI_COMPLETE &&
2033 last_login_state != PDS_PRLI_COMPLETE) {
2034 ql_dbg(ql_dbg_mbx, vha, 0x119a,
2035 "Unable to verify login-state (%x/%x) for loop_id %x.\n",
2036 current_login_state, last_login_state,
2037 fcport->loop_id);
1c7c6357 2038 rval = QLA_FUNCTION_FAILED;
c0c462c8
DG
2039
2040 if (!fcport->query)
2041 goto gpd_error_out;
1c7c6357 2042 }
1da177e4 2043
0eba25df
AE
2044 if (fcport->loop_id == FC_NO_LOOP_ID ||
2045 (memcmp(fcport->port_name, (uint8_t *)&zero, 8) &&
2046 memcmp(fcport->port_name, pd24->port_name, 8))) {
2047 /* We lost the device mid way. */
2048 rval = QLA_NOT_LOGGED_IN;
2049 goto gpd_error_out;
2050 }
2051
1c7c6357
AV
2052 /* Names are little-endian. */
2053 memcpy(fcport->node_name, pd24->node_name, WWN_SIZE);
2054 memcpy(fcport->port_name, pd24->port_name, WWN_SIZE);
2055
2056 /* Get port_id of device. */
2057 fcport->d_id.b.domain = pd24->port_id[0];
2058 fcport->d_id.b.area = pd24->port_id[1];
2059 fcport->d_id.b.al_pa = pd24->port_id[2];
2060 fcport->d_id.b.rsvd_1 = 0;
2061
2062 /* If not target must be initiator or unknown type. */
2063 if ((pd24->prli_svc_param_word_3[0] & BIT_4) == 0)
2064 fcport->port_type = FCT_INITIATOR;
2065 else
2066 fcport->port_type = FCT_TARGET;
2d70c103
NB
2067
2068 /* Passback COS information. */
2069 fcport->supported_classes = (pd24->flags & PDF_CLASS_2) ?
2070 FC_COS_CLASS2 : FC_COS_CLASS3;
2071
2072 if (pd24->prli_svc_param_word_3[0] & BIT_7)
2073 fcport->flags |= FCF_CONF_COMP_SUPPORTED;
1c7c6357 2074 } else {
0eba25df
AE
2075 uint64_t zero = 0;
2076
1c7c6357
AV
2077 /* Check for logged in state. */
2078 if (pd->master_state != PD_STATE_PORT_LOGGED_IN &&
2079 pd->slave_state != PD_STATE_PORT_LOGGED_IN) {
7c3df132
SK
2080 ql_dbg(ql_dbg_mbx, vha, 0x100a,
2081 "Unable to verify login-state (%x/%x) - "
2082 "portid=%02x%02x%02x.\n", pd->master_state,
2083 pd->slave_state, fcport->d_id.b.domain,
2084 fcport->d_id.b.area, fcport->d_id.b.al_pa);
1c7c6357
AV
2085 rval = QLA_FUNCTION_FAILED;
2086 goto gpd_error_out;
2087 }
1da177e4 2088
0eba25df
AE
2089 if (fcport->loop_id == FC_NO_LOOP_ID ||
2090 (memcmp(fcport->port_name, (uint8_t *)&zero, 8) &&
2091 memcmp(fcport->port_name, pd->port_name, 8))) {
2092 /* We lost the device mid way. */
2093 rval = QLA_NOT_LOGGED_IN;
2094 goto gpd_error_out;
2095 }
2096
1c7c6357
AV
2097 /* Names are little-endian. */
2098 memcpy(fcport->node_name, pd->node_name, WWN_SIZE);
2099 memcpy(fcport->port_name, pd->port_name, WWN_SIZE);
2100
2101 /* Get port_id of device. */
2102 fcport->d_id.b.domain = pd->port_id[0];
2103 fcport->d_id.b.area = pd->port_id[3];
2104 fcport->d_id.b.al_pa = pd->port_id[2];
2105 fcport->d_id.b.rsvd_1 = 0;
2106
1c7c6357
AV
2107 /* If not target must be initiator or unknown type. */
2108 if ((pd->prli_svc_param_word_3[0] & BIT_4) == 0)
2109 fcport->port_type = FCT_INITIATOR;
2110 else
2111 fcport->port_type = FCT_TARGET;
ad3e0eda
AV
2112
2113 /* Passback COS information. */
2114 fcport->supported_classes = (pd->options & BIT_4) ?
58e2753c 2115 FC_COS_CLASS2 : FC_COS_CLASS3;
1c7c6357 2116 }
1da177e4
LT
2117
2118gpd_error_out:
2119 dma_pool_free(ha->s_dma_pool, pd, pd_dma);
edd05de1 2120 fcport->query = 0;
1da177e4
LT
2121
2122 if (rval != QLA_SUCCESS) {
7c3df132
SK
2123 ql_dbg(ql_dbg_mbx, vha, 0x1052,
2124 "Failed=%x mb[0]=%x mb[1]=%x.\n", rval,
2125 mcp->mb[0], mcp->mb[1]);
1da177e4 2126 } else {
5f28d2d7
SK
2127 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1053,
2128 "Done %s.\n", __func__);
1da177e4
LT
2129 }
2130
2131 return rval;
2132}
2133
818c7f87
JC
2134int
2135qla24xx_get_port_database(scsi_qla_host_t *vha, u16 nport_handle,
2136 struct port_database_24xx *pdb)
2137{
2138 mbx_cmd_t mc;
2139 mbx_cmd_t *mcp = &mc;
2140 dma_addr_t pdb_dma;
2141 int rval;
2142
2143 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1115,
2144 "Entered %s.\n", __func__);
2145
2146 memset(pdb, 0, sizeof(*pdb));
2147
2148 pdb_dma = dma_map_single(&vha->hw->pdev->dev, pdb,
2149 sizeof(*pdb), DMA_FROM_DEVICE);
2150 if (!pdb_dma) {
2151 ql_log(ql_log_warn, vha, 0x1116, "Failed to map dma buffer.\n");
2152 return QLA_MEMORY_ALLOC_FAILED;
2153 }
2154
2155 mcp->mb[0] = MBC_GET_PORT_DATABASE;
2156 mcp->mb[1] = nport_handle;
2157 mcp->mb[2] = MSW(LSD(pdb_dma));
2158 mcp->mb[3] = LSW(LSD(pdb_dma));
2159 mcp->mb[6] = MSW(MSD(pdb_dma));
2160 mcp->mb[7] = LSW(MSD(pdb_dma));
2161 mcp->mb[9] = 0;
2162 mcp->mb[10] = 0;
2163 mcp->out_mb = MBX_10|MBX_9|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
2164 mcp->in_mb = MBX_1|MBX_0;
2165 mcp->buf_size = sizeof(*pdb);
2166 mcp->flags = MBX_DMA_IN;
2167 mcp->tov = vha->hw->login_timeout * 2;
2168 rval = qla2x00_mailbox_command(vha, mcp);
2169
2170 if (rval != QLA_SUCCESS) {
2171 ql_dbg(ql_dbg_mbx, vha, 0x111a,
2172 "Failed=%x mb[0]=%x mb[1]=%x.\n",
2173 rval, mcp->mb[0], mcp->mb[1]);
2174 } else {
2175 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x111b,
2176 "Done %s.\n", __func__);
2177 }
2178
2179 dma_unmap_single(&vha->hw->pdev->dev, pdb_dma,
2180 sizeof(*pdb), DMA_FROM_DEVICE);
2181
2182 return rval;
2183}
2184
1da177e4
LT
2185/*
2186 * qla2x00_get_firmware_state
2187 * Get adapter firmware state.
2188 *
2189 * Input:
2190 * ha = adapter block pointer.
2191 * dptr = pointer for firmware state.
2192 * TARGET_QUEUE_LOCK must be released.
2193 * ADAPTER_STATE_LOCK must be released.
2194 *
2195 * Returns:
2196 * qla2x00 local function return status code.
2197 *
2198 * Context:
2199 * Kernel context.
2200 */
2201int
7b867cf7 2202qla2x00_get_firmware_state(scsi_qla_host_t *vha, uint16_t *states)
1da177e4
LT
2203{
2204 int rval;
2205 mbx_cmd_t mc;
2206 mbx_cmd_t *mcp = &mc;
92d4408e 2207 struct qla_hw_data *ha = vha->hw;
1da177e4 2208
5f28d2d7
SK
2209 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1054,
2210 "Entered %s.\n", __func__);
1da177e4 2211
efa74a62
QT
2212 if (!ha->flags.fw_started)
2213 return QLA_FUNCTION_FAILED;
2214
1da177e4
LT
2215 mcp->mb[0] = MBC_GET_FIRMWARE_STATE;
2216 mcp->out_mb = MBX_0;
9d2683c0 2217 if (IS_FWI2_CAPABLE(vha->hw))
b5a340dd 2218 mcp->in_mb = MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
9d2683c0
AV
2219 else
2220 mcp->in_mb = MBX_1|MBX_0;
b93480e3 2221 mcp->tov = MBX_TOV_SECONDS;
1da177e4 2222 mcp->flags = 0;
7b867cf7 2223 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4 2224
4d4df193
HK
2225 /* Return firmware states. */
2226 states[0] = mcp->mb[1];
9d2683c0
AV
2227 if (IS_FWI2_CAPABLE(vha->hw)) {
2228 states[1] = mcp->mb[2];
ec891462 2229 states[2] = mcp->mb[3]; /* SFP info */
9d2683c0
AV
2230 states[3] = mcp->mb[4];
2231 states[4] = mcp->mb[5];
b5a340dd 2232 states[5] = mcp->mb[6]; /* DPORT status */
9d2683c0 2233 }
1da177e4
LT
2234
2235 if (rval != QLA_SUCCESS) {
2236 /*EMPTY*/
7c3df132 2237 ql_dbg(ql_dbg_mbx, vha, 0x1055, "Failed=%x.\n", rval);
1da177e4 2238 } else {
ecc89f25 2239 if (IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
92d4408e
SC
2240 if (mcp->mb[2] == 6 || mcp->mb[3] == 2)
2241 ql_dbg(ql_dbg_mbx, vha, 0x119e,
2242 "Invalid SFP/Validation Failed\n");
2243 }
5f28d2d7
SK
2244 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1056,
2245 "Done %s.\n", __func__);
1da177e4
LT
2246 }
2247
2248 return rval;
2249}
2250
2251/*
2252 * qla2x00_get_port_name
2253 * Issue get port name mailbox command.
2254 * Returned name is in big endian format.
2255 *
2256 * Input:
2257 * ha = adapter block pointer.
2258 * loop_id = loop ID of device.
2259 * name = pointer for name.
2260 * TARGET_QUEUE_LOCK must be released.
2261 * ADAPTER_STATE_LOCK must be released.
2262 *
2263 * Returns:
2264 * qla2x00 local function return status code.
2265 *
2266 * Context:
2267 * Kernel context.
2268 */
2269int
7b867cf7 2270qla2x00_get_port_name(scsi_qla_host_t *vha, uint16_t loop_id, uint8_t *name,
1da177e4
LT
2271 uint8_t opt)
2272{
2273 int rval;
2274 mbx_cmd_t mc;
2275 mbx_cmd_t *mcp = &mc;
2276
5f28d2d7
SK
2277 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1057,
2278 "Entered %s.\n", __func__);
1da177e4
LT
2279
2280 mcp->mb[0] = MBC_GET_PORT_NAME;
7b867cf7 2281 mcp->mb[9] = vha->vp_idx;
2c3dfe3f 2282 mcp->out_mb = MBX_9|MBX_1|MBX_0;
7b867cf7 2283 if (HAS_EXTENDED_IDS(vha->hw)) {
1da177e4
LT
2284 mcp->mb[1] = loop_id;
2285 mcp->mb[10] = opt;
2286 mcp->out_mb |= MBX_10;
2287 } else {
2288 mcp->mb[1] = loop_id << 8 | opt;
2289 }
2290
2291 mcp->in_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
b93480e3 2292 mcp->tov = MBX_TOV_SECONDS;
1da177e4 2293 mcp->flags = 0;
7b867cf7 2294 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2295
2296 if (rval != QLA_SUCCESS) {
2297 /*EMPTY*/
7c3df132 2298 ql_dbg(ql_dbg_mbx, vha, 0x1058, "Failed=%x.\n", rval);
1da177e4
LT
2299 } else {
2300 if (name != NULL) {
2301 /* This function returns name in big endian. */
1196ae02
RL
2302 name[0] = MSB(mcp->mb[2]);
2303 name[1] = LSB(mcp->mb[2]);
2304 name[2] = MSB(mcp->mb[3]);
2305 name[3] = LSB(mcp->mb[3]);
2306 name[4] = MSB(mcp->mb[6]);
2307 name[5] = LSB(mcp->mb[6]);
2308 name[6] = MSB(mcp->mb[7]);
2309 name[7] = LSB(mcp->mb[7]);
1da177e4
LT
2310 }
2311
5f28d2d7
SK
2312 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1059,
2313 "Done %s.\n", __func__);
1da177e4
LT
2314 }
2315
2316 return rval;
2317}
2318
61e1b269
JC
2319/*
2320 * qla24xx_link_initialization
2321 * Issue link initialization mailbox command.
2322 *
2323 * Input:
2324 * ha = adapter block pointer.
2325 * TARGET_QUEUE_LOCK must be released.
2326 * ADAPTER_STATE_LOCK must be released.
2327 *
2328 * Returns:
2329 * qla2x00 local function return status code.
2330 *
2331 * Context:
2332 * Kernel context.
2333 */
2334int
2335qla24xx_link_initialize(scsi_qla_host_t *vha)
2336{
2337 int rval;
2338 mbx_cmd_t mc;
2339 mbx_cmd_t *mcp = &mc;
2340
2341 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1152,
2342 "Entered %s.\n", __func__);
2343
2344 if (!IS_FWI2_CAPABLE(vha->hw) || IS_CNA_CAPABLE(vha->hw))
2345 return QLA_FUNCTION_FAILED;
2346
2347 mcp->mb[0] = MBC_LINK_INITIALIZATION;
5a5c27b6
JC
2348 mcp->mb[1] = BIT_4;
2349 if (vha->hw->operating_mode == LOOP)
2350 mcp->mb[1] |= BIT_6;
2351 else
2352 mcp->mb[1] |= BIT_5;
61e1b269
JC
2353 mcp->mb[2] = 0;
2354 mcp->mb[3] = 0;
2355 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
2356 mcp->in_mb = MBX_0;
2357 mcp->tov = MBX_TOV_SECONDS;
2358 mcp->flags = 0;
2359 rval = qla2x00_mailbox_command(vha, mcp);
2360
2361 if (rval != QLA_SUCCESS) {
2362 ql_dbg(ql_dbg_mbx, vha, 0x1153, "Failed=%x.\n", rval);
2363 } else {
2364 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1154,
2365 "Done %s.\n", __func__);
2366 }
2367
2368 return rval;
2369}
2370
1da177e4
LT
2371/*
2372 * qla2x00_lip_reset
2373 * Issue LIP reset mailbox command.
2374 *
2375 * Input:
2376 * ha = adapter block pointer.
2377 * TARGET_QUEUE_LOCK must be released.
2378 * ADAPTER_STATE_LOCK must be released.
2379 *
2380 * Returns:
2381 * qla2x00 local function return status code.
2382 *
2383 * Context:
2384 * Kernel context.
2385 */
2386int
7b867cf7 2387qla2x00_lip_reset(scsi_qla_host_t *vha)
1da177e4
LT
2388{
2389 int rval;
2390 mbx_cmd_t mc;
2391 mbx_cmd_t *mcp = &mc;
2392
7f2a398d 2393 ql_dbg(ql_dbg_disc, vha, 0x105a,
5f28d2d7 2394 "Entered %s.\n", __func__);
1da177e4 2395
6246b8a1 2396 if (IS_CNA_CAPABLE(vha->hw)) {
3a03eb79
AV
2397 /* Logout across all FCFs. */
2398 mcp->mb[0] = MBC_LIP_FULL_LOGIN;
2399 mcp->mb[1] = BIT_1;
2400 mcp->mb[2] = 0;
2401 mcp->out_mb = MBX_2|MBX_1|MBX_0;
2402 } else if (IS_FWI2_CAPABLE(vha->hw)) {
1c7c6357 2403 mcp->mb[0] = MBC_LIP_FULL_LOGIN;
87d6814a 2404 mcp->mb[1] = BIT_4;
0c8c39af 2405 mcp->mb[2] = 0;
7b867cf7 2406 mcp->mb[3] = vha->hw->loop_reset_delay;
1c7c6357 2407 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
1da177e4 2408 } else {
1c7c6357
AV
2409 mcp->mb[0] = MBC_LIP_RESET;
2410 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
7b867cf7 2411 if (HAS_EXTENDED_IDS(vha->hw)) {
1c7c6357
AV
2412 mcp->mb[1] = 0x00ff;
2413 mcp->mb[10] = 0;
2414 mcp->out_mb |= MBX_10;
2415 } else {
2416 mcp->mb[1] = 0xff00;
2417 }
7b867cf7 2418 mcp->mb[2] = vha->hw->loop_reset_delay;
1c7c6357 2419 mcp->mb[3] = 0;
1da177e4 2420 }
1da177e4 2421 mcp->in_mb = MBX_0;
b93480e3 2422 mcp->tov = MBX_TOV_SECONDS;
1da177e4 2423 mcp->flags = 0;
7b867cf7 2424 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2425
2426 if (rval != QLA_SUCCESS) {
2427 /*EMPTY*/
7c3df132 2428 ql_dbg(ql_dbg_mbx, vha, 0x105b, "Failed=%x.\n", rval);
1da177e4
LT
2429 } else {
2430 /*EMPTY*/
5f28d2d7
SK
2431 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x105c,
2432 "Done %s.\n", __func__);
1da177e4
LT
2433 }
2434
2435 return rval;
2436}
2437
2438/*
2439 * qla2x00_send_sns
2440 * Send SNS command.
2441 *
2442 * Input:
2443 * ha = adapter block pointer.
2444 * sns = pointer for command.
2445 * cmd_size = command size.
2446 * buf_size = response/command size.
2447 * TARGET_QUEUE_LOCK must be released.
2448 * ADAPTER_STATE_LOCK must be released.
2449 *
2450 * Returns:
2451 * qla2x00 local function return status code.
2452 *
2453 * Context:
2454 * Kernel context.
2455 */
2456int
7b867cf7 2457qla2x00_send_sns(scsi_qla_host_t *vha, dma_addr_t sns_phys_address,
1da177e4
LT
2458 uint16_t cmd_size, size_t buf_size)
2459{
2460 int rval;
2461 mbx_cmd_t mc;
2462 mbx_cmd_t *mcp = &mc;
2463
5f28d2d7
SK
2464 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x105d,
2465 "Entered %s.\n", __func__);
1da177e4 2466
5f28d2d7 2467 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x105e,
7c3df132
SK
2468 "Retry cnt=%d ratov=%d total tov=%d.\n",
2469 vha->hw->retry_count, vha->hw->login_timeout, mcp->tov);
1da177e4
LT
2470
2471 mcp->mb[0] = MBC_SEND_SNS_COMMAND;
2472 mcp->mb[1] = cmd_size;
2473 mcp->mb[2] = MSW(sns_phys_address);
2474 mcp->mb[3] = LSW(sns_phys_address);
2475 mcp->mb[6] = MSW(MSD(sns_phys_address));
2476 mcp->mb[7] = LSW(MSD(sns_phys_address));
2477 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
2478 mcp->in_mb = MBX_0|MBX_1;
2479 mcp->buf_size = buf_size;
2480 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN;
7b867cf7
AC
2481 mcp->tov = (vha->hw->login_timeout * 2) + (vha->hw->login_timeout / 2);
2482 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2483
2484 if (rval != QLA_SUCCESS) {
2485 /*EMPTY*/
7c3df132
SK
2486 ql_dbg(ql_dbg_mbx, vha, 0x105f,
2487 "Failed=%x mb[0]=%x mb[1]=%x.\n",
2488 rval, mcp->mb[0], mcp->mb[1]);
1da177e4
LT
2489 } else {
2490 /*EMPTY*/
5f28d2d7
SK
2491 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1060,
2492 "Done %s.\n", __func__);
1da177e4
LT
2493 }
2494
2495 return rval;
2496}
2497
1c7c6357 2498int
7b867cf7 2499qla24xx_login_fabric(scsi_qla_host_t *vha, uint16_t loop_id, uint8_t domain,
1c7c6357
AV
2500 uint8_t area, uint8_t al_pa, uint16_t *mb, uint8_t opt)
2501{
2502 int rval;
2503
2504 struct logio_entry_24xx *lg;
2505 dma_addr_t lg_dma;
2506 uint32_t iop[2];
7b867cf7 2507 struct qla_hw_data *ha = vha->hw;
2afa19a9 2508 struct req_que *req;
1c7c6357 2509
5f28d2d7
SK
2510 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1061,
2511 "Entered %s.\n", __func__);
1c7c6357 2512
d7459527
MH
2513 if (vha->vp_idx && vha->qpair)
2514 req = vha->qpair->req;
68ca949c 2515 else
d7459527 2516 req = ha->req_q_map[0];
2afa19a9 2517
08eb7f45 2518 lg = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &lg_dma);
1c7c6357 2519 if (lg == NULL) {
7c3df132
SK
2520 ql_log(ql_log_warn, vha, 0x1062,
2521 "Failed to allocate login IOCB.\n");
1c7c6357
AV
2522 return QLA_MEMORY_ALLOC_FAILED;
2523 }
1c7c6357
AV
2524
2525 lg->entry_type = LOGINOUT_PORT_IOCB_TYPE;
2526 lg->entry_count = 1;
c25eb70a 2527 lg->handle = make_handle(req->id, lg->handle);
1c7c6357 2528 lg->nport_handle = cpu_to_le16(loop_id);
ad950360 2529 lg->control_flags = cpu_to_le16(LCF_COMMAND_PLOGI);
1c7c6357 2530 if (opt & BIT_0)
ad950360 2531 lg->control_flags |= cpu_to_le16(LCF_COND_PLOGI);
8baa51a6 2532 if (opt & BIT_1)
ad950360 2533 lg->control_flags |= cpu_to_le16(LCF_SKIP_PRLI);
1c7c6357
AV
2534 lg->port_id[0] = al_pa;
2535 lg->port_id[1] = area;
2536 lg->port_id[2] = domain;
7b867cf7 2537 lg->vp_index = vha->vp_idx;
7f45dd0b
AV
2538 rval = qla2x00_issue_iocb_timeout(vha, lg, lg_dma, 0,
2539 (ha->r_a_tov / 10 * 2) + 2);
1c7c6357 2540 if (rval != QLA_SUCCESS) {
7c3df132
SK
2541 ql_dbg(ql_dbg_mbx, vha, 0x1063,
2542 "Failed to issue login IOCB (%x).\n", rval);
1c7c6357 2543 } else if (lg->entry_status != 0) {
7c3df132
SK
2544 ql_dbg(ql_dbg_mbx, vha, 0x1064,
2545 "Failed to complete IOCB -- error status (%x).\n",
2546 lg->entry_status);
1c7c6357 2547 rval = QLA_FUNCTION_FAILED;
ad950360 2548 } else if (lg->comp_status != cpu_to_le16(CS_COMPLETE)) {
1c7c6357
AV
2549 iop[0] = le32_to_cpu(lg->io_parameter[0]);
2550 iop[1] = le32_to_cpu(lg->io_parameter[1]);
2551
7c3df132
SK
2552 ql_dbg(ql_dbg_mbx, vha, 0x1065,
2553 "Failed to complete IOCB -- completion status (%x) "
2554 "ioparam=%x/%x.\n", le16_to_cpu(lg->comp_status),
2555 iop[0], iop[1]);
1c7c6357
AV
2556
2557 switch (iop[0]) {
2558 case LSC_SCODE_PORTID_USED:
2559 mb[0] = MBS_PORT_ID_USED;
2560 mb[1] = LSW(iop[1]);
2561 break;
2562 case LSC_SCODE_NPORT_USED:
2563 mb[0] = MBS_LOOP_ID_USED;
2564 break;
2565 case LSC_SCODE_NOLINK:
2566 case LSC_SCODE_NOIOCB:
2567 case LSC_SCODE_NOXCB:
2568 case LSC_SCODE_CMD_FAILED:
2569 case LSC_SCODE_NOFABRIC:
2570 case LSC_SCODE_FW_NOT_READY:
2571 case LSC_SCODE_NOT_LOGGED_IN:
2572 case LSC_SCODE_NOPCB:
2573 case LSC_SCODE_ELS_REJECT:
2574 case LSC_SCODE_CMD_PARAM_ERR:
2575 case LSC_SCODE_NONPORT:
2576 case LSC_SCODE_LOGGED_IN:
2577 case LSC_SCODE_NOFLOGI_ACC:
2578 default:
2579 mb[0] = MBS_COMMAND_ERROR;
2580 break;
2581 }
2582 } else {
5f28d2d7
SK
2583 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1066,
2584 "Done %s.\n", __func__);
1c7c6357
AV
2585
2586 iop[0] = le32_to_cpu(lg->io_parameter[0]);
2587
2588 mb[0] = MBS_COMMAND_COMPLETE;
2589 mb[1] = 0;
2590 if (iop[0] & BIT_4) {
2591 if (iop[0] & BIT_8)
2592 mb[1] |= BIT_1;
2593 } else
2594 mb[1] = BIT_0;
ad3e0eda
AV
2595
2596 /* Passback COS information. */
2597 mb[10] = 0;
2598 if (lg->io_parameter[7] || lg->io_parameter[8])
2599 mb[10] |= BIT_0; /* Class 2. */
2600 if (lg->io_parameter[9] || lg->io_parameter[10])
2601 mb[10] |= BIT_1; /* Class 3. */
ad950360 2602 if (lg->io_parameter[0] & cpu_to_le32(BIT_7))
2d70c103
NB
2603 mb[10] |= BIT_7; /* Confirmed Completion
2604 * Allowed
2605 */
1c7c6357
AV
2606 }
2607
2608 dma_pool_free(ha->s_dma_pool, lg, lg_dma);
2609
2610 return rval;
2611}
2612
1da177e4
LT
2613/*
2614 * qla2x00_login_fabric
2615 * Issue login fabric port mailbox command.
2616 *
2617 * Input:
2618 * ha = adapter block pointer.
2619 * loop_id = device loop ID.
2620 * domain = device domain.
2621 * area = device area.
2622 * al_pa = device AL_PA.
2623 * status = pointer for return status.
2624 * opt = command options.
2625 * TARGET_QUEUE_LOCK must be released.
2626 * ADAPTER_STATE_LOCK must be released.
2627 *
2628 * Returns:
2629 * qla2x00 local function return status code.
2630 *
2631 * Context:
2632 * Kernel context.
2633 */
2634int
7b867cf7 2635qla2x00_login_fabric(scsi_qla_host_t *vha, uint16_t loop_id, uint8_t domain,
1da177e4
LT
2636 uint8_t area, uint8_t al_pa, uint16_t *mb, uint8_t opt)
2637{
2638 int rval;
2639 mbx_cmd_t mc;
2640 mbx_cmd_t *mcp = &mc;
7b867cf7 2641 struct qla_hw_data *ha = vha->hw;
1da177e4 2642
5f28d2d7
SK
2643 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1067,
2644 "Entered %s.\n", __func__);
1da177e4
LT
2645
2646 mcp->mb[0] = MBC_LOGIN_FABRIC_PORT;
2647 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
2648 if (HAS_EXTENDED_IDS(ha)) {
2649 mcp->mb[1] = loop_id;
2650 mcp->mb[10] = opt;
2651 mcp->out_mb |= MBX_10;
2652 } else {
2653 mcp->mb[1] = (loop_id << 8) | opt;
2654 }
2655 mcp->mb[2] = domain;
2656 mcp->mb[3] = area << 8 | al_pa;
2657
2658 mcp->in_mb = MBX_7|MBX_6|MBX_2|MBX_1|MBX_0;
2659 mcp->tov = (ha->login_timeout * 2) + (ha->login_timeout / 2);
2660 mcp->flags = 0;
7b867cf7 2661 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2662
2663 /* Return mailbox statuses. */
2664 if (mb != NULL) {
2665 mb[0] = mcp->mb[0];
2666 mb[1] = mcp->mb[1];
2667 mb[2] = mcp->mb[2];
2668 mb[6] = mcp->mb[6];
2669 mb[7] = mcp->mb[7];
ad3e0eda
AV
2670 /* COS retrieved from Get-Port-Database mailbox command. */
2671 mb[10] = 0;
1da177e4
LT
2672 }
2673
2674 if (rval != QLA_SUCCESS) {
2675 /* RLU tmp code: need to change main mailbox_command function to
2676 * return ok even when the mailbox completion value is not
2677 * SUCCESS. The caller needs to be responsible to interpret
2678 * the return values of this mailbox command if we're not
2679 * to change too much of the existing code.
2680 */
2681 if (mcp->mb[0] == 0x4001 || mcp->mb[0] == 0x4002 ||
2682 mcp->mb[0] == 0x4003 || mcp->mb[0] == 0x4005 ||
2683 mcp->mb[0] == 0x4006)
2684 rval = QLA_SUCCESS;
2685
2686 /*EMPTY*/
7c3df132
SK
2687 ql_dbg(ql_dbg_mbx, vha, 0x1068,
2688 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x.\n",
2689 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2]);
1da177e4
LT
2690 } else {
2691 /*EMPTY*/
5f28d2d7
SK
2692 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1069,
2693 "Done %s.\n", __func__);
1da177e4
LT
2694 }
2695
2696 return rval;
2697}
2698
2699/*
2700 * qla2x00_login_local_device
2701 * Issue login loop port mailbox command.
fa2a1ce5 2702 *
1da177e4
LT
2703 * Input:
2704 * ha = adapter block pointer.
2705 * loop_id = device loop ID.
2706 * opt = command options.
fa2a1ce5 2707 *
1da177e4
LT
2708 * Returns:
2709 * Return status code.
fa2a1ce5 2710 *
1da177e4
LT
2711 * Context:
2712 * Kernel context.
fa2a1ce5 2713 *
1da177e4
LT
2714 */
2715int
7b867cf7 2716qla2x00_login_local_device(scsi_qla_host_t *vha, fc_port_t *fcport,
1da177e4
LT
2717 uint16_t *mb_ret, uint8_t opt)
2718{
2719 int rval;
2720 mbx_cmd_t mc;
2721 mbx_cmd_t *mcp = &mc;
7b867cf7 2722 struct qla_hw_data *ha = vha->hw;
1da177e4 2723
5f28d2d7
SK
2724 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x106a,
2725 "Entered %s.\n", __func__);
7c3df132 2726
e428924c 2727 if (IS_FWI2_CAPABLE(ha))
7b867cf7 2728 return qla24xx_login_fabric(vha, fcport->loop_id,
9a52a57c 2729 fcport->d_id.b.domain, fcport->d_id.b.area,
2730 fcport->d_id.b.al_pa, mb_ret, opt);
2731
1da177e4
LT
2732 mcp->mb[0] = MBC_LOGIN_LOOP_PORT;
2733 if (HAS_EXTENDED_IDS(ha))
9a52a57c 2734 mcp->mb[1] = fcport->loop_id;
1da177e4 2735 else
9a52a57c 2736 mcp->mb[1] = fcport->loop_id << 8;
1da177e4
LT
2737 mcp->mb[2] = opt;
2738 mcp->out_mb = MBX_2|MBX_1|MBX_0;
2739 mcp->in_mb = MBX_7|MBX_6|MBX_1|MBX_0;
2740 mcp->tov = (ha->login_timeout * 2) + (ha->login_timeout / 2);
2741 mcp->flags = 0;
7b867cf7 2742 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2743
2744 /* Return mailbox statuses. */
2745 if (mb_ret != NULL) {
2746 mb_ret[0] = mcp->mb[0];
2747 mb_ret[1] = mcp->mb[1];
2748 mb_ret[6] = mcp->mb[6];
2749 mb_ret[7] = mcp->mb[7];
2750 }
2751
2752 if (rval != QLA_SUCCESS) {
2753 /* AV tmp code: need to change main mailbox_command function to
2754 * return ok even when the mailbox completion value is not
2755 * SUCCESS. The caller needs to be responsible to interpret
2756 * the return values of this mailbox command if we're not
2757 * to change too much of the existing code.
2758 */
2759 if (mcp->mb[0] == 0x4005 || mcp->mb[0] == 0x4006)
2760 rval = QLA_SUCCESS;
2761
7c3df132
SK
2762 ql_dbg(ql_dbg_mbx, vha, 0x106b,
2763 "Failed=%x mb[0]=%x mb[1]=%x mb[6]=%x mb[7]=%x.\n",
2764 rval, mcp->mb[0], mcp->mb[1], mcp->mb[6], mcp->mb[7]);
1da177e4
LT
2765 } else {
2766 /*EMPTY*/
5f28d2d7
SK
2767 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x106c,
2768 "Done %s.\n", __func__);
1da177e4
LT
2769 }
2770
2771 return (rval);
2772}
2773
1c7c6357 2774int
7b867cf7 2775qla24xx_fabric_logout(scsi_qla_host_t *vha, uint16_t loop_id, uint8_t domain,
1c7c6357
AV
2776 uint8_t area, uint8_t al_pa)
2777{
2778 int rval;
2779 struct logio_entry_24xx *lg;
2780 dma_addr_t lg_dma;
7b867cf7 2781 struct qla_hw_data *ha = vha->hw;
2afa19a9 2782 struct req_que *req;
1c7c6357 2783
5f28d2d7
SK
2784 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x106d,
2785 "Entered %s.\n", __func__);
1c7c6357 2786
08eb7f45 2787 lg = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &lg_dma);
1c7c6357 2788 if (lg == NULL) {
7c3df132
SK
2789 ql_log(ql_log_warn, vha, 0x106e,
2790 "Failed to allocate logout IOCB.\n");
1c7c6357
AV
2791 return QLA_MEMORY_ALLOC_FAILED;
2792 }
1c7c6357 2793
d7459527 2794 req = vha->req;
1c7c6357
AV
2795 lg->entry_type = LOGINOUT_PORT_IOCB_TYPE;
2796 lg->entry_count = 1;
c25eb70a 2797 lg->handle = make_handle(req->id, lg->handle);
1c7c6357
AV
2798 lg->nport_handle = cpu_to_le16(loop_id);
2799 lg->control_flags =
ad950360 2800 cpu_to_le16(LCF_COMMAND_LOGO|LCF_IMPL_LOGO|
c8d6691b 2801 LCF_FREE_NPORT);
1c7c6357
AV
2802 lg->port_id[0] = al_pa;
2803 lg->port_id[1] = area;
2804 lg->port_id[2] = domain;
7b867cf7 2805 lg->vp_index = vha->vp_idx;
7f45dd0b
AV
2806 rval = qla2x00_issue_iocb_timeout(vha, lg, lg_dma, 0,
2807 (ha->r_a_tov / 10 * 2) + 2);
1c7c6357 2808 if (rval != QLA_SUCCESS) {
7c3df132
SK
2809 ql_dbg(ql_dbg_mbx, vha, 0x106f,
2810 "Failed to issue logout IOCB (%x).\n", rval);
1c7c6357 2811 } else if (lg->entry_status != 0) {
7c3df132
SK
2812 ql_dbg(ql_dbg_mbx, vha, 0x1070,
2813 "Failed to complete IOCB -- error status (%x).\n",
2814 lg->entry_status);
1c7c6357 2815 rval = QLA_FUNCTION_FAILED;
ad950360 2816 } else if (lg->comp_status != cpu_to_le16(CS_COMPLETE)) {
7c3df132
SK
2817 ql_dbg(ql_dbg_mbx, vha, 0x1071,
2818 "Failed to complete IOCB -- completion status (%x) "
2819 "ioparam=%x/%x.\n", le16_to_cpu(lg->comp_status),
1c7c6357 2820 le32_to_cpu(lg->io_parameter[0]),
7c3df132 2821 le32_to_cpu(lg->io_parameter[1]));
1c7c6357
AV
2822 } else {
2823 /*EMPTY*/
5f28d2d7
SK
2824 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1072,
2825 "Done %s.\n", __func__);
1c7c6357
AV
2826 }
2827
2828 dma_pool_free(ha->s_dma_pool, lg, lg_dma);
2829
2830 return rval;
2831}
2832
1da177e4
LT
2833/*
2834 * qla2x00_fabric_logout
2835 * Issue logout fabric port mailbox command.
2836 *
2837 * Input:
2838 * ha = adapter block pointer.
2839 * loop_id = device loop ID.
2840 * TARGET_QUEUE_LOCK must be released.
2841 * ADAPTER_STATE_LOCK must be released.
2842 *
2843 * Returns:
2844 * qla2x00 local function return status code.
2845 *
2846 * Context:
2847 * Kernel context.
2848 */
2849int
7b867cf7 2850qla2x00_fabric_logout(scsi_qla_host_t *vha, uint16_t loop_id, uint8_t domain,
1c7c6357 2851 uint8_t area, uint8_t al_pa)
1da177e4
LT
2852{
2853 int rval;
2854 mbx_cmd_t mc;
2855 mbx_cmd_t *mcp = &mc;
2856
5f28d2d7
SK
2857 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1073,
2858 "Entered %s.\n", __func__);
1da177e4
LT
2859
2860 mcp->mb[0] = MBC_LOGOUT_FABRIC_PORT;
2861 mcp->out_mb = MBX_1|MBX_0;
7b867cf7 2862 if (HAS_EXTENDED_IDS(vha->hw)) {
1da177e4
LT
2863 mcp->mb[1] = loop_id;
2864 mcp->mb[10] = 0;
2865 mcp->out_mb |= MBX_10;
2866 } else {
2867 mcp->mb[1] = loop_id << 8;
2868 }
2869
2870 mcp->in_mb = MBX_1|MBX_0;
b93480e3 2871 mcp->tov = MBX_TOV_SECONDS;
1da177e4 2872 mcp->flags = 0;
7b867cf7 2873 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2874
2875 if (rval != QLA_SUCCESS) {
2876 /*EMPTY*/
7c3df132
SK
2877 ql_dbg(ql_dbg_mbx, vha, 0x1074,
2878 "Failed=%x mb[1]=%x.\n", rval, mcp->mb[1]);
1da177e4
LT
2879 } else {
2880 /*EMPTY*/
5f28d2d7
SK
2881 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1075,
2882 "Done %s.\n", __func__);
1da177e4
LT
2883 }
2884
2885 return rval;
2886}
2887
2888/*
2889 * qla2x00_full_login_lip
2890 * Issue full login LIP mailbox command.
2891 *
2892 * Input:
2893 * ha = adapter block pointer.
2894 * TARGET_QUEUE_LOCK must be released.
2895 * ADAPTER_STATE_LOCK must be released.
2896 *
2897 * Returns:
2898 * qla2x00 local function return status code.
2899 *
2900 * Context:
2901 * Kernel context.
2902 */
2903int
7b867cf7 2904qla2x00_full_login_lip(scsi_qla_host_t *vha)
1da177e4
LT
2905{
2906 int rval;
2907 mbx_cmd_t mc;
2908 mbx_cmd_t *mcp = &mc;
2909
5f28d2d7
SK
2910 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1076,
2911 "Entered %s.\n", __func__);
1da177e4
LT
2912
2913 mcp->mb[0] = MBC_LIP_FULL_LOGIN;
87d6814a 2914 mcp->mb[1] = IS_FWI2_CAPABLE(vha->hw) ? BIT_4 : 0;
0c8c39af 2915 mcp->mb[2] = 0;
1da177e4
LT
2916 mcp->mb[3] = 0;
2917 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
2918 mcp->in_mb = MBX_0;
b93480e3 2919 mcp->tov = MBX_TOV_SECONDS;
1da177e4 2920 mcp->flags = 0;
7b867cf7 2921 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2922
2923 if (rval != QLA_SUCCESS) {
2924 /*EMPTY*/
7c3df132 2925 ql_dbg(ql_dbg_mbx, vha, 0x1077, "Failed=%x.\n", rval);
1da177e4
LT
2926 } else {
2927 /*EMPTY*/
5f28d2d7
SK
2928 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1078,
2929 "Done %s.\n", __func__);
1da177e4
LT
2930 }
2931
2932 return rval;
2933}
2934
2935/*
2936 * qla2x00_get_id_list
2937 *
2938 * Input:
2939 * ha = adapter block pointer.
2940 *
2941 * Returns:
2942 * qla2x00 local function return status code.
2943 *
2944 * Context:
2945 * Kernel context.
2946 */
2947int
7b867cf7 2948qla2x00_get_id_list(scsi_qla_host_t *vha, void *id_list, dma_addr_t id_list_dma,
1da177e4
LT
2949 uint16_t *entries)
2950{
2951 int rval;
2952 mbx_cmd_t mc;
2953 mbx_cmd_t *mcp = &mc;
2954
5f28d2d7
SK
2955 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1079,
2956 "Entered %s.\n", __func__);
1da177e4
LT
2957
2958 if (id_list == NULL)
2959 return QLA_FUNCTION_FAILED;
2960
2961 mcp->mb[0] = MBC_GET_ID_LIST;
1c7c6357 2962 mcp->out_mb = MBX_0;
7b867cf7 2963 if (IS_FWI2_CAPABLE(vha->hw)) {
1c7c6357
AV
2964 mcp->mb[2] = MSW(id_list_dma);
2965 mcp->mb[3] = LSW(id_list_dma);
2966 mcp->mb[6] = MSW(MSD(id_list_dma));
2967 mcp->mb[7] = LSW(MSD(id_list_dma));
247ec457 2968 mcp->mb[8] = 0;
7b867cf7 2969 mcp->mb[9] = vha->vp_idx;
2c3dfe3f 2970 mcp->out_mb |= MBX_9|MBX_8|MBX_7|MBX_6|MBX_3|MBX_2;
1c7c6357
AV
2971 } else {
2972 mcp->mb[1] = MSW(id_list_dma);
2973 mcp->mb[2] = LSW(id_list_dma);
2974 mcp->mb[3] = MSW(MSD(id_list_dma));
2975 mcp->mb[6] = LSW(MSD(id_list_dma));
2976 mcp->out_mb |= MBX_6|MBX_3|MBX_2|MBX_1;
2977 }
1da177e4 2978 mcp->in_mb = MBX_1|MBX_0;
b93480e3 2979 mcp->tov = MBX_TOV_SECONDS;
1da177e4 2980 mcp->flags = 0;
7b867cf7 2981 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
2982
2983 if (rval != QLA_SUCCESS) {
2984 /*EMPTY*/
7c3df132 2985 ql_dbg(ql_dbg_mbx, vha, 0x107a, "Failed=%x.\n", rval);
1da177e4
LT
2986 } else {
2987 *entries = mcp->mb[1];
5f28d2d7
SK
2988 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x107b,
2989 "Done %s.\n", __func__);
1da177e4
LT
2990 }
2991
2992 return rval;
2993}
2994
2995/*
2996 * qla2x00_get_resource_cnts
2997 * Get current firmware resource counts.
2998 *
2999 * Input:
3000 * ha = adapter block pointer.
3001 *
3002 * Returns:
3003 * qla2x00 local function return status code.
3004 *
3005 * Context:
3006 * Kernel context.
3007 */
3008int
03e8c680 3009qla2x00_get_resource_cnts(scsi_qla_host_t *vha)
1da177e4 3010{
03e8c680 3011 struct qla_hw_data *ha = vha->hw;
1da177e4
LT
3012 int rval;
3013 mbx_cmd_t mc;
3014 mbx_cmd_t *mcp = &mc;
3015
5f28d2d7
SK
3016 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x107c,
3017 "Entered %s.\n", __func__);
1da177e4
LT
3018
3019 mcp->mb[0] = MBC_GET_RESOURCE_COUNTS;
3020 mcp->out_mb = MBX_0;
4d0ea247 3021 mcp->in_mb = MBX_11|MBX_10|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
ecc89f25
JC
3022 if (IS_QLA81XX(ha) || IS_QLA83XX(ha) ||
3023 IS_QLA27XX(ha) || IS_QLA28XX(ha))
f3a0a77e 3024 mcp->in_mb |= MBX_12;
b93480e3 3025 mcp->tov = MBX_TOV_SECONDS;
1da177e4 3026 mcp->flags = 0;
7b867cf7 3027 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
3028
3029 if (rval != QLA_SUCCESS) {
3030 /*EMPTY*/
7c3df132
SK
3031 ql_dbg(ql_dbg_mbx, vha, 0x107d,
3032 "Failed mb[0]=%x.\n", mcp->mb[0]);
1da177e4 3033 } else {
5f28d2d7 3034 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x107e,
7c3df132
SK
3035 "Done %s mb1=%x mb2=%x mb3=%x mb6=%x mb7=%x mb10=%x "
3036 "mb11=%x mb12=%x.\n", __func__, mcp->mb[1], mcp->mb[2],
3037 mcp->mb[3], mcp->mb[6], mcp->mb[7], mcp->mb[10],
3038 mcp->mb[11], mcp->mb[12]);
1da177e4 3039
03e8c680
QT
3040 ha->orig_fw_tgt_xcb_count = mcp->mb[1];
3041 ha->cur_fw_tgt_xcb_count = mcp->mb[2];
3042 ha->cur_fw_xcb_count = mcp->mb[3];
3043 ha->orig_fw_xcb_count = mcp->mb[6];
3044 ha->cur_fw_iocb_count = mcp->mb[7];
3045 ha->orig_fw_iocb_count = mcp->mb[10];
3046 if (ha->flags.npiv_supported)
3047 ha->max_npiv_vports = mcp->mb[11];
0d6a536c 3048 if (IS_QLA81XX(ha) || IS_QLA83XX(ha))
03e8c680 3049 ha->fw_max_fcf_count = mcp->mb[12];
1da177e4
LT
3050 }
3051
3052 return (rval);
3053}
3054
1da177e4
LT
3055/*
3056 * qla2x00_get_fcal_position_map
3057 * Get FCAL (LILP) position map using mailbox command
3058 *
3059 * Input:
3060 * ha = adapter state pointer.
3061 * pos_map = buffer pointer (can be NULL).
3062 *
3063 * Returns:
3064 * qla2x00 local function return status code.
3065 *
3066 * Context:
3067 * Kernel context.
3068 */
3069int
47ccb113
AE
3070qla2x00_get_fcal_position_map(scsi_qla_host_t *vha, char *pos_map,
3071 u8 *num_entries)
1da177e4
LT
3072{
3073 int rval;
3074 mbx_cmd_t mc;
3075 mbx_cmd_t *mcp = &mc;
3076 char *pmap;
3077 dma_addr_t pmap_dma;
7b867cf7 3078 struct qla_hw_data *ha = vha->hw;
1da177e4 3079
5f28d2d7
SK
3080 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x107f,
3081 "Entered %s.\n", __func__);
7c3df132 3082
08eb7f45 3083 pmap = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &pmap_dma);
1da177e4 3084 if (pmap == NULL) {
7c3df132
SK
3085 ql_log(ql_log_warn, vha, 0x1080,
3086 "Memory alloc failed.\n");
1da177e4
LT
3087 return QLA_MEMORY_ALLOC_FAILED;
3088 }
1da177e4
LT
3089
3090 mcp->mb[0] = MBC_GET_FC_AL_POSITION_MAP;
3091 mcp->mb[2] = MSW(pmap_dma);
3092 mcp->mb[3] = LSW(pmap_dma);
3093 mcp->mb[6] = MSW(MSD(pmap_dma));
3094 mcp->mb[7] = LSW(MSD(pmap_dma));
3095 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_0;
3096 mcp->in_mb = MBX_1|MBX_0;
3097 mcp->buf_size = FCAL_MAP_SIZE;
3098 mcp->flags = MBX_DMA_IN;
3099 mcp->tov = (ha->login_timeout * 2) + (ha->login_timeout / 2);
7b867cf7 3100 rval = qla2x00_mailbox_command(vha, mcp);
1da177e4
LT
3101
3102 if (rval == QLA_SUCCESS) {
5f28d2d7 3103 ql_dbg(ql_dbg_mbx + ql_dbg_buffer, vha, 0x1081,
7c3df132
SK
3104 "mb0/mb1=%x/%X FC/AL position map size (%x).\n",
3105 mcp->mb[0], mcp->mb[1], (unsigned)pmap[0]);
3106 ql_dump_buffer(ql_dbg_mbx + ql_dbg_buffer, vha, 0x111d,
3107 pmap, pmap[0] + 1);
1da177e4
LT
3108
3109 if (pos_map)
3110 memcpy(pos_map, pmap, FCAL_MAP_SIZE);
47ccb113
AE
3111 if (num_entries)
3112 *num_entries = pmap[0];
1da177e4
LT
3113 }
3114 dma_pool_free(ha->s_dma_pool, pmap, pmap_dma);
3115
3116 if (rval != QLA_SUCCESS) {
7c3df132 3117 ql_dbg(ql_dbg_mbx, vha, 0x1082, "Failed=%x.\n", rval);
1da177e4 3118 } else {
5f28d2d7
SK
3119 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1083,
3120 "Done %s.\n", __func__);
1da177e4
LT
3121 }
3122
3123 return rval;
3124}
392e2f65 3125
3126/*
3127 * qla2x00_get_link_status
3128 *
3129 * Input:
3130 * ha = adapter block pointer.
3131 * loop_id = device loop ID.
3132 * ret_buf = pointer to link status return buffer.
3133 *
3134 * Returns:
3135 * 0 = success.
3136 * BIT_0 = mem alloc error.
3137 * BIT_1 = mailbox error.
3138 */
3139int
7b867cf7 3140qla2x00_get_link_status(scsi_qla_host_t *vha, uint16_t loop_id,
43ef0580 3141 struct link_statistics *stats, dma_addr_t stats_dma)
392e2f65 3142{
3143 int rval;
3144 mbx_cmd_t mc;
3145 mbx_cmd_t *mcp = &mc;
ab053c09 3146 uint32_t *iter = (uint32_t *)stats;
c6dc9905 3147 ushort dwords = offsetof(typeof(*stats), link_up_cnt)/sizeof(*iter);
7b867cf7 3148 struct qla_hw_data *ha = vha->hw;
392e2f65 3149
5f28d2d7
SK
3150 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1084,
3151 "Entered %s.\n", __func__);
392e2f65 3152
392e2f65 3153 mcp->mb[0] = MBC_GET_LINK_STATUS;
c6dc9905
JC
3154 mcp->mb[2] = MSW(LSD(stats_dma));
3155 mcp->mb[3] = LSW(LSD(stats_dma));
43ef0580
AV
3156 mcp->mb[6] = MSW(MSD(stats_dma));
3157 mcp->mb[7] = LSW(MSD(stats_dma));
392e2f65 3158 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_0;
3159 mcp->in_mb = MBX_0;
e428924c 3160 if (IS_FWI2_CAPABLE(ha)) {
392e2f65 3161 mcp->mb[1] = loop_id;
3162 mcp->mb[4] = 0;
3163 mcp->mb[10] = 0;
3164 mcp->out_mb |= MBX_10|MBX_4|MBX_1;
3165 mcp->in_mb |= MBX_1;
3166 } else if (HAS_EXTENDED_IDS(ha)) {
3167 mcp->mb[1] = loop_id;
3168 mcp->mb[10] = 0;
3169 mcp->out_mb |= MBX_10|MBX_1;
3170 } else {
3171 mcp->mb[1] = loop_id << 8;
3172 mcp->out_mb |= MBX_1;
3173 }
b93480e3 3174 mcp->tov = MBX_TOV_SECONDS;
392e2f65 3175 mcp->flags = IOCTL_CMD;
7b867cf7 3176 rval = qla2x00_mailbox_command(vha, mcp);
392e2f65 3177
3178 if (rval == QLA_SUCCESS) {
3179 if (mcp->mb[0] != MBS_COMMAND_COMPLETE) {
7c3df132
SK
3180 ql_dbg(ql_dbg_mbx, vha, 0x1085,
3181 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
43ef0580 3182 rval = QLA_FUNCTION_FAILED;
392e2f65 3183 } else {
c6dc9905 3184 /* Re-endianize - firmware data is le32. */
5f28d2d7
SK
3185 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1086,
3186 "Done %s.\n", __func__);
da08ef5c
JC
3187 for ( ; dwords--; iter++)
3188 le32_to_cpus(iter);
392e2f65 3189 }
3190 } else {
3191 /* Failed. */
7c3df132 3192 ql_dbg(ql_dbg_mbx, vha, 0x1087, "Failed=%x.\n", rval);
392e2f65 3193 }
3194
392e2f65 3195 return rval;
3196}
3197
3198int
7b867cf7 3199qla24xx_get_isp_stats(scsi_qla_host_t *vha, struct link_statistics *stats,
15f30a57 3200 dma_addr_t stats_dma, uint16_t options)
1c7c6357
AV
3201{
3202 int rval;
3203 mbx_cmd_t mc;
3204 mbx_cmd_t *mcp = &mc;
ab053c09 3205 uint32_t *iter = (uint32_t *)stats;
818c7f87 3206 ushort dwords = sizeof(*stats)/sizeof(*iter);
1c7c6357 3207
5f28d2d7
SK
3208 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1088,
3209 "Entered %s.\n", __func__);
1c7c6357 3210
15f30a57
QT
3211 memset(&mc, 0, sizeof(mc));
3212 mc.mb[0] = MBC_GET_LINK_PRIV_STATS;
818c7f87
JC
3213 mc.mb[2] = MSW(LSD(stats_dma));
3214 mc.mb[3] = LSW(LSD(stats_dma));
15f30a57
QT
3215 mc.mb[6] = MSW(MSD(stats_dma));
3216 mc.mb[7] = LSW(MSD(stats_dma));
818c7f87 3217 mc.mb[8] = dwords;
7ffa5b93
BVA
3218 mc.mb[9] = vha->vp_idx;
3219 mc.mb[10] = options;
15f30a57
QT
3220
3221 rval = qla24xx_send_mb_cmd(vha, &mc);
1c7c6357
AV
3222
3223 if (rval == QLA_SUCCESS) {
3224 if (mcp->mb[0] != MBS_COMMAND_COMPLETE) {
7c3df132
SK
3225 ql_dbg(ql_dbg_mbx, vha, 0x1089,
3226 "Failed mb[0]=%x.\n", mcp->mb[0]);
43ef0580 3227 rval = QLA_FUNCTION_FAILED;
1c7c6357 3228 } else {
5f28d2d7
SK
3229 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x108a,
3230 "Done %s.\n", __func__);
c6dc9905 3231 /* Re-endianize - firmware data is le32. */
da08ef5c
JC
3232 for ( ; dwords--; iter++)
3233 le32_to_cpus(iter);
1c7c6357
AV
3234 }
3235 } else {
3236 /* Failed. */
7c3df132 3237 ql_dbg(ql_dbg_mbx, vha, 0x108b, "Failed=%x.\n", rval);
1c7c6357
AV
3238 }
3239
1c7c6357
AV
3240 return rval;
3241}
1c7c6357
AV
3242
3243int
2afa19a9 3244qla24xx_abort_command(srb_t *sp)
1c7c6357
AV
3245{
3246 int rval;
1c7c6357
AV
3247 unsigned long flags = 0;
3248
3249 struct abort_entry_24xx *abt;
3250 dma_addr_t abt_dma;
3251 uint32_t handle;
2afa19a9
AC
3252 fc_port_t *fcport = sp->fcport;
3253 struct scsi_qla_host *vha = fcport->vha;
7b867cf7 3254 struct qla_hw_data *ha = vha->hw;
914418f3 3255 struct req_que *req;
585def9b 3256 struct qla_qpair *qpair = sp->qpair;
1c7c6357 3257
5f28d2d7
SK
3258 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x108c,
3259 "Entered %s.\n", __func__);
1c7c6357 3260
45a76264 3261 if (sp->qpair)
d7459527 3262 req = sp->qpair->req;
585def9b 3263 else
2cabf10d 3264 return QLA_ERR_NO_QPAIR;
d7459527 3265
4440e46d
AB
3266 if (ql2xasynctmfenable)
3267 return qla24xx_async_abort_command(sp);
3268
585def9b 3269 spin_lock_irqsave(qpair->qp_lock_ptr, flags);
8d93f550 3270 for (handle = 1; handle < req->num_outstanding_cmds; handle++) {
7b867cf7 3271 if (req->outstanding_cmds[handle] == sp)
1c7c6357
AV
3272 break;
3273 }
585def9b 3274 spin_unlock_irqrestore(qpair->qp_lock_ptr, flags);
8d93f550 3275 if (handle == req->num_outstanding_cmds) {
1c7c6357 3276 /* Command not found. */
2cabf10d 3277 return QLA_ERR_NOT_FOUND;
1c7c6357
AV
3278 }
3279
08eb7f45 3280 abt = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &abt_dma);
1c7c6357 3281 if (abt == NULL) {
7c3df132
SK
3282 ql_log(ql_log_warn, vha, 0x108d,
3283 "Failed to allocate abort IOCB.\n");
1c7c6357
AV
3284 return QLA_MEMORY_ALLOC_FAILED;
3285 }
1c7c6357
AV
3286
3287 abt->entry_type = ABORT_IOCB_TYPE;
3288 abt->entry_count = 1;
c25eb70a 3289 abt->handle = make_handle(req->id, abt->handle);
1c7c6357 3290 abt->nport_handle = cpu_to_le16(fcport->loop_id);
c25eb70a 3291 abt->handle_to_abort = make_handle(req->id, handle);
1c7c6357
AV
3292 abt->port_id[0] = fcport->d_id.b.al_pa;
3293 abt->port_id[1] = fcport->d_id.b.area;
3294 abt->port_id[2] = fcport->d_id.b.domain;
c6d39e23 3295 abt->vp_index = fcport->vha->vp_idx;
73208dfd
AC
3296
3297 abt->req_que_no = cpu_to_le16(req->id);
a0465859
BH
3298 /* Need to pass original sp */
3299 qla_nvme_abort_set_option(abt, sp);
73208dfd 3300
7b867cf7 3301 rval = qla2x00_issue_iocb(vha, abt, abt_dma, 0);
1c7c6357 3302 if (rval != QLA_SUCCESS) {
7c3df132
SK
3303 ql_dbg(ql_dbg_mbx, vha, 0x108e,
3304 "Failed to issue IOCB (%x).\n", rval);
1c7c6357 3305 } else if (abt->entry_status != 0) {
7c3df132
SK
3306 ql_dbg(ql_dbg_mbx, vha, 0x108f,
3307 "Failed to complete IOCB -- error status (%x).\n",
3308 abt->entry_status);
1c7c6357 3309 rval = QLA_FUNCTION_FAILED;
ad950360 3310 } else if (abt->nport_handle != cpu_to_le16(0)) {
7c3df132
SK
3311 ql_dbg(ql_dbg_mbx, vha, 0x1090,
3312 "Failed to complete IOCB -- completion status (%x).\n",
3313 le16_to_cpu(abt->nport_handle));
7ffa5b93 3314 if (abt->nport_handle == cpu_to_le16(CS_IOCB_ERROR))
f934c9d0
CD
3315 rval = QLA_FUNCTION_PARAMETER_ERROR;
3316 else
3317 rval = QLA_FUNCTION_FAILED;
1c7c6357 3318 } else {
5f28d2d7
SK
3319 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1091,
3320 "Done %s.\n", __func__);
1c7c6357 3321 }
a0465859
BH
3322 if (rval == QLA_SUCCESS)
3323 qla_nvme_abort_process_comp_status(abt, sp);
3324
3325 qla_wait_nvme_release_cmd_kref(sp);
1c7c6357
AV
3326
3327 dma_pool_free(ha->s_dma_pool, abt, abt_dma);
3328
3329 return rval;
3330}
3331
3332struct tsk_mgmt_cmd {
3333 union {
3334 struct tsk_mgmt_entry tsk;
3335 struct sts_entry_24xx sts;
3336 } p;
3337};
3338
523ec773
AV
3339static int
3340__qla24xx_issue_tmf(char *name, uint32_t type, struct fc_port *fcport,
9cb78c16 3341 uint64_t l, int tag)
1c7c6357 3342{
523ec773 3343 int rval, rval2;
1c7c6357 3344 struct tsk_mgmt_cmd *tsk;
9ca1d01f 3345 struct sts_entry_24xx *sts;
1c7c6357 3346 dma_addr_t tsk_dma;
7b867cf7
AC
3347 scsi_qla_host_t *vha;
3348 struct qla_hw_data *ha;
73208dfd 3349 struct req_que *req;
d7459527 3350 struct qla_qpair *qpair;
1c7c6357 3351
7b867cf7
AC
3352 vha = fcport->vha;
3353 ha = vha->hw;
2afa19a9 3354 req = vha->req;
7c3df132 3355
5f28d2d7
SK
3356 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1092,
3357 "Entered %s.\n", __func__);
7c3df132 3358
d7459527
MH
3359 if (vha->vp_idx && vha->qpair) {
3360 /* NPIV port */
3361 qpair = vha->qpair;
d7459527 3362 req = qpair->req;
d7459527
MH
3363 }
3364
08eb7f45 3365 tsk = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &tsk_dma);
1c7c6357 3366 if (tsk == NULL) {
7c3df132
SK
3367 ql_log(ql_log_warn, vha, 0x1093,
3368 "Failed to allocate task management IOCB.\n");
1c7c6357
AV
3369 return QLA_MEMORY_ALLOC_FAILED;
3370 }
1c7c6357
AV
3371
3372 tsk->p.tsk.entry_type = TSK_MGMT_IOCB_TYPE;
3373 tsk->p.tsk.entry_count = 1;
c25eb70a 3374 tsk->p.tsk.handle = make_handle(req->id, tsk->p.tsk.handle);
1c7c6357 3375 tsk->p.tsk.nport_handle = cpu_to_le16(fcport->loop_id);
00a537b8 3376 tsk->p.tsk.timeout = cpu_to_le16(ha->r_a_tov / 10 * 2);
523ec773 3377 tsk->p.tsk.control_flags = cpu_to_le32(type);
1c7c6357
AV
3378 tsk->p.tsk.port_id[0] = fcport->d_id.b.al_pa;
3379 tsk->p.tsk.port_id[1] = fcport->d_id.b.area;
3380 tsk->p.tsk.port_id[2] = fcport->d_id.b.domain;
c6d39e23 3381 tsk->p.tsk.vp_index = fcport->vha->vp_idx;
523ec773
AV
3382 if (type == TCF_LUN_RESET) {
3383 int_to_scsilun(l, &tsk->p.tsk.lun);
3384 host_to_fcp_swap((uint8_t *)&tsk->p.tsk.lun,
3385 sizeof(tsk->p.tsk.lun));
3386 }
2c3dfe3f 3387
9ca1d01f 3388 sts = &tsk->p.sts;
7b867cf7 3389 rval = qla2x00_issue_iocb(vha, tsk, tsk_dma, 0);
1c7c6357 3390 if (rval != QLA_SUCCESS) {
7c3df132
SK
3391 ql_dbg(ql_dbg_mbx, vha, 0x1094,
3392 "Failed to issue %s reset IOCB (%x).\n", name, rval);
9ca1d01f 3393 } else if (sts->entry_status != 0) {
7c3df132
SK
3394 ql_dbg(ql_dbg_mbx, vha, 0x1095,
3395 "Failed to complete IOCB -- error status (%x).\n",
3396 sts->entry_status);
1c7c6357 3397 rval = QLA_FUNCTION_FAILED;
ad950360 3398 } else if (sts->comp_status != cpu_to_le16(CS_COMPLETE)) {
7c3df132
SK
3399 ql_dbg(ql_dbg_mbx, vha, 0x1096,
3400 "Failed to complete IOCB -- completion status (%x).\n",
3401 le16_to_cpu(sts->comp_status));
9ca1d01f 3402 rval = QLA_FUNCTION_FAILED;
97dec564
AV
3403 } else if (le16_to_cpu(sts->scsi_status) &
3404 SS_RESPONSE_INFO_LEN_VALID) {
3405 if (le32_to_cpu(sts->rsp_data_len) < 4) {
5f28d2d7 3406 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1097,
7c3df132
SK
3407 "Ignoring inconsistent data length -- not enough "
3408 "response info (%d).\n",
3409 le32_to_cpu(sts->rsp_data_len));
97dec564 3410 } else if (sts->data[3]) {
7c3df132
SK
3411 ql_dbg(ql_dbg_mbx, vha, 0x1098,
3412 "Failed to complete IOCB -- response (%x).\n",
3413 sts->data[3]);
97dec564
AV
3414 rval = QLA_FUNCTION_FAILED;
3415 }
1c7c6357
AV
3416 }
3417
3418 /* Issue marker IOCB. */
9eb9c6dc 3419 rval2 = qla2x00_marker(vha, ha->base_qpair, fcport->loop_id, l,
58e2753c 3420 type == TCF_LUN_RESET ? MK_SYNC_ID_LUN : MK_SYNC_ID);
523ec773 3421 if (rval2 != QLA_SUCCESS) {
7c3df132
SK
3422 ql_dbg(ql_dbg_mbx, vha, 0x1099,
3423 "Failed to issue marker IOCB (%x).\n", rval2);
1c7c6357 3424 } else {
5f28d2d7
SK
3425 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x109a,
3426 "Done %s.\n", __func__);
1c7c6357
AV
3427 }
3428
7b867cf7 3429 dma_pool_free(ha->s_dma_pool, tsk, tsk_dma);
1c7c6357
AV
3430
3431 return rval;
3432}
3433
523ec773 3434int
9cb78c16 3435qla24xx_abort_target(struct fc_port *fcport, uint64_t l, int tag)
523ec773 3436{
3822263e
MI
3437 struct qla_hw_data *ha = fcport->vha->hw;
3438
3439 if ((ql2xasynctmfenable) && IS_FWI2_CAPABLE(ha))
3440 return qla2x00_async_tm_cmd(fcport, TCF_TARGET_RESET, l, tag);
3441
2afa19a9 3442 return __qla24xx_issue_tmf("Target", TCF_TARGET_RESET, fcport, l, tag);
523ec773
AV
3443}
3444
3445int
9cb78c16 3446qla24xx_lun_reset(struct fc_port *fcport, uint64_t l, int tag)
523ec773 3447{
3822263e
MI
3448 struct qla_hw_data *ha = fcport->vha->hw;
3449
3450 if ((ql2xasynctmfenable) && IS_FWI2_CAPABLE(ha))
3451 return qla2x00_async_tm_cmd(fcport, TCF_LUN_RESET, l, tag);
3452
2afa19a9 3453 return __qla24xx_issue_tmf("Lun", TCF_LUN_RESET, fcport, l, tag);
523ec773
AV
3454}
3455
1c7c6357 3456int
7b867cf7 3457qla2x00_system_error(scsi_qla_host_t *vha)
1c7c6357
AV
3458{
3459 int rval;
3460 mbx_cmd_t mc;
3461 mbx_cmd_t *mcp = &mc;
7b867cf7 3462 struct qla_hw_data *ha = vha->hw;
1c7c6357 3463
68af0811 3464 if (!IS_QLA23XX(ha) && !IS_FWI2_CAPABLE(ha))
1c7c6357
AV
3465 return QLA_FUNCTION_FAILED;
3466
5f28d2d7
SK
3467 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x109b,
3468 "Entered %s.\n", __func__);
1c7c6357
AV
3469
3470 mcp->mb[0] = MBC_GEN_SYSTEM_ERROR;
3471 mcp->out_mb = MBX_0;
3472 mcp->in_mb = MBX_0;
3473 mcp->tov = 5;
3474 mcp->flags = 0;
7b867cf7 3475 rval = qla2x00_mailbox_command(vha, mcp);
1c7c6357
AV
3476
3477 if (rval != QLA_SUCCESS) {
7c3df132 3478 ql_dbg(ql_dbg_mbx, vha, 0x109c, "Failed=%x.\n", rval);
1c7c6357 3479 } else {
5f28d2d7
SK
3480 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x109d,
3481 "Done %s.\n", __func__);
1c7c6357
AV
3482 }
3483
3484 return rval;
3485}
3486
db64e930
JC
3487int
3488qla2x00_write_serdes_word(scsi_qla_host_t *vha, uint16_t addr, uint16_t data)
3489{
3490 int rval;
3491 mbx_cmd_t mc;
3492 mbx_cmd_t *mcp = &mc;
3493
f299c7c2 3494 if (!IS_QLA25XX(vha->hw) && !IS_QLA2031(vha->hw) &&
ecc89f25 3495 !IS_QLA27XX(vha->hw) && !IS_QLA28XX(vha->hw))
db64e930
JC
3496 return QLA_FUNCTION_FAILED;
3497
3498 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1182,
3499 "Entered %s.\n", __func__);
3500
3501 mcp->mb[0] = MBC_WRITE_SERDES;
3502 mcp->mb[1] = addr;
064135e0
AV
3503 if (IS_QLA2031(vha->hw))
3504 mcp->mb[2] = data & 0xff;
3505 else
3506 mcp->mb[2] = data;
3507
db64e930
JC
3508 mcp->mb[3] = 0;
3509 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
3510 mcp->in_mb = MBX_0;
3511 mcp->tov = MBX_TOV_SECONDS;
3512 mcp->flags = 0;
3513 rval = qla2x00_mailbox_command(vha, mcp);
3514
3515 if (rval != QLA_SUCCESS) {
3516 ql_dbg(ql_dbg_mbx, vha, 0x1183,
3517 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
3518 } else {
3519 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1184,
3520 "Done %s.\n", __func__);
3521 }
3522
3523 return rval;
3524}
3525
3526int
3527qla2x00_read_serdes_word(scsi_qla_host_t *vha, uint16_t addr, uint16_t *data)
3528{
3529 int rval;
3530 mbx_cmd_t mc;
3531 mbx_cmd_t *mcp = &mc;
3532
f299c7c2 3533 if (!IS_QLA25XX(vha->hw) && !IS_QLA2031(vha->hw) &&
ecc89f25 3534 !IS_QLA27XX(vha->hw) && !IS_QLA28XX(vha->hw))
db64e930
JC
3535 return QLA_FUNCTION_FAILED;
3536
3537 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1185,
3538 "Entered %s.\n", __func__);
3539
3540 mcp->mb[0] = MBC_READ_SERDES;
3541 mcp->mb[1] = addr;
3542 mcp->mb[3] = 0;
3543 mcp->out_mb = MBX_3|MBX_1|MBX_0;
3544 mcp->in_mb = MBX_1|MBX_0;
3545 mcp->tov = MBX_TOV_SECONDS;
3546 mcp->flags = 0;
3547 rval = qla2x00_mailbox_command(vha, mcp);
3548
064135e0
AV
3549 if (IS_QLA2031(vha->hw))
3550 *data = mcp->mb[1] & 0xff;
3551 else
3552 *data = mcp->mb[1];
db64e930
JC
3553
3554 if (rval != QLA_SUCCESS) {
3555 ql_dbg(ql_dbg_mbx, vha, 0x1186,
3556 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
3557 } else {
3558 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1187,
3559 "Done %s.\n", __func__);
3560 }
3561
3562 return rval;
3563}
3564
e8887c51
JC
3565int
3566qla8044_write_serdes_word(scsi_qla_host_t *vha, uint32_t addr, uint32_t data)
3567{
3568 int rval;
3569 mbx_cmd_t mc;
3570 mbx_cmd_t *mcp = &mc;
3571
3572 if (!IS_QLA8044(vha->hw))
3573 return QLA_FUNCTION_FAILED;
3574
83548fe2 3575 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x11a0,
e8887c51
JC
3576 "Entered %s.\n", __func__);
3577
3578 mcp->mb[0] = MBC_SET_GET_ETH_SERDES_REG;
3579 mcp->mb[1] = HCS_WRITE_SERDES;
3580 mcp->mb[3] = LSW(addr);
3581 mcp->mb[4] = MSW(addr);
3582 mcp->mb[5] = LSW(data);
3583 mcp->mb[6] = MSW(data);
3584 mcp->out_mb = MBX_6|MBX_5|MBX_4|MBX_3|MBX_1|MBX_0;
3585 mcp->in_mb = MBX_0;
3586 mcp->tov = MBX_TOV_SECONDS;
3587 mcp->flags = 0;
3588 rval = qla2x00_mailbox_command(vha, mcp);
3589
3590 if (rval != QLA_SUCCESS) {
83548fe2 3591 ql_dbg(ql_dbg_mbx, vha, 0x11a1,
e8887c51
JC
3592 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
3593 } else {
3594 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1188,
3595 "Done %s.\n", __func__);
3596 }
3597
3598 return rval;
3599}
3600
3601int
3602qla8044_read_serdes_word(scsi_qla_host_t *vha, uint32_t addr, uint32_t *data)
3603{
3604 int rval;
3605 mbx_cmd_t mc;
3606 mbx_cmd_t *mcp = &mc;
3607
3608 if (!IS_QLA8044(vha->hw))
3609 return QLA_FUNCTION_FAILED;
3610
3611 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1189,
3612 "Entered %s.\n", __func__);
3613
3614 mcp->mb[0] = MBC_SET_GET_ETH_SERDES_REG;
3615 mcp->mb[1] = HCS_READ_SERDES;
3616 mcp->mb[3] = LSW(addr);
3617 mcp->mb[4] = MSW(addr);
3618 mcp->out_mb = MBX_4|MBX_3|MBX_1|MBX_0;
3619 mcp->in_mb = MBX_2|MBX_1|MBX_0;
3620 mcp->tov = MBX_TOV_SECONDS;
3621 mcp->flags = 0;
3622 rval = qla2x00_mailbox_command(vha, mcp);
3623
3624 *data = mcp->mb[2] << 16 | mcp->mb[1];
3625
3626 if (rval != QLA_SUCCESS) {
3627 ql_dbg(ql_dbg_mbx, vha, 0x118a,
3628 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
3629 } else {
3630 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x118b,
3631 "Done %s.\n", __func__);
3632 }
3633
3634 return rval;
3635}
3636
1c7c6357
AV
3637/**
3638 * qla2x00_set_serdes_params() -
2db6228d 3639 * @vha: HA context
807eb907
BVA
3640 * @sw_em_1g: serial link options
3641 * @sw_em_2g: serial link options
3642 * @sw_em_4g: serial link options
1c7c6357
AV
3643 *
3644 * Returns
3645 */
3646int
7b867cf7 3647qla2x00_set_serdes_params(scsi_qla_host_t *vha, uint16_t sw_em_1g,
1c7c6357
AV
3648 uint16_t sw_em_2g, uint16_t sw_em_4g)
3649{
3650 int rval;
3651 mbx_cmd_t mc;
3652 mbx_cmd_t *mcp = &mc;
3653
5f28d2d7
SK
3654 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x109e,
3655 "Entered %s.\n", __func__);
1c7c6357
AV
3656
3657 mcp->mb[0] = MBC_SERDES_PARAMS;
3658 mcp->mb[1] = BIT_0;
fdbc6833 3659 mcp->mb[2] = sw_em_1g | BIT_15;
3660 mcp->mb[3] = sw_em_2g | BIT_15;
3661 mcp->mb[4] = sw_em_4g | BIT_15;
1c7c6357
AV
3662 mcp->out_mb = MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
3663 mcp->in_mb = MBX_0;
b93480e3 3664 mcp->tov = MBX_TOV_SECONDS;
1c7c6357 3665 mcp->flags = 0;
7b867cf7 3666 rval = qla2x00_mailbox_command(vha, mcp);
1c7c6357
AV
3667
3668 if (rval != QLA_SUCCESS) {
3669 /*EMPTY*/
7c3df132
SK
3670 ql_dbg(ql_dbg_mbx, vha, 0x109f,
3671 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
1c7c6357
AV
3672 } else {
3673 /*EMPTY*/
5f28d2d7
SK
3674 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a0,
3675 "Done %s.\n", __func__);
1c7c6357
AV
3676 }
3677
3678 return rval;
3679}
f6ef3b18
AV
3680
3681int
7b867cf7 3682qla2x00_stop_firmware(scsi_qla_host_t *vha)
f6ef3b18
AV
3683{
3684 int rval;
3685 mbx_cmd_t mc;
3686 mbx_cmd_t *mcp = &mc;
3687
7b867cf7 3688 if (!IS_FWI2_CAPABLE(vha->hw))
f6ef3b18
AV
3689 return QLA_FUNCTION_FAILED;
3690
5f28d2d7
SK
3691 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a1,
3692 "Entered %s.\n", __func__);
f6ef3b18
AV
3693
3694 mcp->mb[0] = MBC_STOP_FIRMWARE;
4ba988db
AV
3695 mcp->mb[1] = 0;
3696 mcp->out_mb = MBX_1|MBX_0;
f6ef3b18
AV
3697 mcp->in_mb = MBX_0;
3698 mcp->tov = 5;
3699 mcp->flags = 0;
7b867cf7 3700 rval = qla2x00_mailbox_command(vha, mcp);
f6ef3b18
AV
3701
3702 if (rval != QLA_SUCCESS) {
7c3df132 3703 ql_dbg(ql_dbg_mbx, vha, 0x10a2, "Failed=%x.\n", rval);
b469a7cb
AV
3704 if (mcp->mb[0] == MBS_INVALID_COMMAND)
3705 rval = QLA_INVALID_COMMAND;
f6ef3b18 3706 } else {
5f28d2d7
SK
3707 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a3,
3708 "Done %s.\n", __func__);
f6ef3b18
AV
3709 }
3710
3711 return rval;
3712}
a7a167bf
AV
3713
3714int
7b867cf7 3715qla2x00_enable_eft_trace(scsi_qla_host_t *vha, dma_addr_t eft_dma,
a7a167bf
AV
3716 uint16_t buffers)
3717{
3718 int rval;
3719 mbx_cmd_t mc;
3720 mbx_cmd_t *mcp = &mc;
3721
5f28d2d7
SK
3722 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a4,
3723 "Entered %s.\n", __func__);
7c3df132 3724
7b867cf7 3725 if (!IS_FWI2_CAPABLE(vha->hw))
a7a167bf
AV
3726 return QLA_FUNCTION_FAILED;
3727
85880801
AV
3728 if (unlikely(pci_channel_offline(vha->hw->pdev)))
3729 return QLA_FUNCTION_FAILED;
3730
a7a167bf 3731 mcp->mb[0] = MBC_TRACE_CONTROL;
00b6bd25
AV
3732 mcp->mb[1] = TC_EFT_ENABLE;
3733 mcp->mb[2] = LSW(eft_dma);
3734 mcp->mb[3] = MSW(eft_dma);
3735 mcp->mb[4] = LSW(MSD(eft_dma));
3736 mcp->mb[5] = MSW(MSD(eft_dma));
3737 mcp->mb[6] = buffers;
3738 mcp->mb[7] = TC_AEN_DISABLE;
3739 mcp->out_mb = MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
a7a167bf 3740 mcp->in_mb = MBX_1|MBX_0;
b93480e3 3741 mcp->tov = MBX_TOV_SECONDS;
a7a167bf 3742 mcp->flags = 0;
7b867cf7 3743 rval = qla2x00_mailbox_command(vha, mcp);
00b6bd25 3744 if (rval != QLA_SUCCESS) {
7c3df132
SK
3745 ql_dbg(ql_dbg_mbx, vha, 0x10a5,
3746 "Failed=%x mb[0]=%x mb[1]=%x.\n",
3747 rval, mcp->mb[0], mcp->mb[1]);
00b6bd25 3748 } else {
5f28d2d7
SK
3749 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a6,
3750 "Done %s.\n", __func__);
00b6bd25
AV
3751 }
3752
3753 return rval;
3754}
a7a167bf 3755
00b6bd25 3756int
7b867cf7 3757qla2x00_disable_eft_trace(scsi_qla_host_t *vha)
00b6bd25
AV
3758{
3759 int rval;
3760 mbx_cmd_t mc;
3761 mbx_cmd_t *mcp = &mc;
3762
5f28d2d7
SK
3763 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a7,
3764 "Entered %s.\n", __func__);
7c3df132 3765
7b867cf7 3766 if (!IS_FWI2_CAPABLE(vha->hw))
00b6bd25
AV
3767 return QLA_FUNCTION_FAILED;
3768
85880801
AV
3769 if (unlikely(pci_channel_offline(vha->hw->pdev)))
3770 return QLA_FUNCTION_FAILED;
3771
00b6bd25
AV
3772 mcp->mb[0] = MBC_TRACE_CONTROL;
3773 mcp->mb[1] = TC_EFT_DISABLE;
3774 mcp->out_mb = MBX_1|MBX_0;
3775 mcp->in_mb = MBX_1|MBX_0;
b93480e3 3776 mcp->tov = MBX_TOV_SECONDS;
00b6bd25 3777 mcp->flags = 0;
7b867cf7 3778 rval = qla2x00_mailbox_command(vha, mcp);
a7a167bf 3779 if (rval != QLA_SUCCESS) {
7c3df132
SK
3780 ql_dbg(ql_dbg_mbx, vha, 0x10a8,
3781 "Failed=%x mb[0]=%x mb[1]=%x.\n",
3782 rval, mcp->mb[0], mcp->mb[1]);
a7a167bf 3783 } else {
5f28d2d7
SK
3784 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10a9,
3785 "Done %s.\n", __func__);
a7a167bf
AV
3786 }
3787
3788 return rval;
3789}
3790
df613b96 3791int
7b867cf7 3792qla2x00_enable_fce_trace(scsi_qla_host_t *vha, dma_addr_t fce_dma,
df613b96
AV
3793 uint16_t buffers, uint16_t *mb, uint32_t *dwords)
3794{
3795 int rval;
3796 mbx_cmd_t mc;
3797 mbx_cmd_t *mcp = &mc;
3798
5f28d2d7
SK
3799 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10aa,
3800 "Entered %s.\n", __func__);
7c3df132 3801
6246b8a1 3802 if (!IS_QLA25XX(vha->hw) && !IS_QLA81XX(vha->hw) &&
ecc89f25
JC
3803 !IS_QLA83XX(vha->hw) && !IS_QLA27XX(vha->hw) &&
3804 !IS_QLA28XX(vha->hw))
df613b96
AV
3805 return QLA_FUNCTION_FAILED;
3806
85880801
AV
3807 if (unlikely(pci_channel_offline(vha->hw->pdev)))
3808 return QLA_FUNCTION_FAILED;
3809
df613b96
AV
3810 mcp->mb[0] = MBC_TRACE_CONTROL;
3811 mcp->mb[1] = TC_FCE_ENABLE;
3812 mcp->mb[2] = LSW(fce_dma);
3813 mcp->mb[3] = MSW(fce_dma);
3814 mcp->mb[4] = LSW(MSD(fce_dma));
3815 mcp->mb[5] = MSW(MSD(fce_dma));
3816 mcp->mb[6] = buffers;
3817 mcp->mb[7] = TC_AEN_DISABLE;
3818 mcp->mb[8] = 0;
3819 mcp->mb[9] = TC_FCE_DEFAULT_RX_SIZE;
3820 mcp->mb[10] = TC_FCE_DEFAULT_TX_SIZE;
3821 mcp->out_mb = MBX_10|MBX_9|MBX_8|MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|
3822 MBX_1|MBX_0;
3823 mcp->in_mb = MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
b93480e3 3824 mcp->tov = MBX_TOV_SECONDS;
df613b96 3825 mcp->flags = 0;
7b867cf7 3826 rval = qla2x00_mailbox_command(vha, mcp);
df613b96 3827 if (rval != QLA_SUCCESS) {
7c3df132
SK
3828 ql_dbg(ql_dbg_mbx, vha, 0x10ab,
3829 "Failed=%x mb[0]=%x mb[1]=%x.\n",
3830 rval, mcp->mb[0], mcp->mb[1]);
df613b96 3831 } else {
5f28d2d7
SK
3832 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ac,
3833 "Done %s.\n", __func__);
df613b96
AV
3834
3835 if (mb)
3836 memcpy(mb, mcp->mb, 8 * sizeof(*mb));
3837 if (dwords)
fa0926df 3838 *dwords = buffers;
df613b96
AV
3839 }
3840
3841 return rval;
3842}
3843
3844int
7b867cf7 3845qla2x00_disable_fce_trace(scsi_qla_host_t *vha, uint64_t *wr, uint64_t *rd)
df613b96
AV
3846{
3847 int rval;
3848 mbx_cmd_t mc;
3849 mbx_cmd_t *mcp = &mc;
3850
5f28d2d7
SK
3851 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ad,
3852 "Entered %s.\n", __func__);
7c3df132 3853
7b867cf7 3854 if (!IS_FWI2_CAPABLE(vha->hw))
df613b96
AV
3855 return QLA_FUNCTION_FAILED;
3856
85880801
AV
3857 if (unlikely(pci_channel_offline(vha->hw->pdev)))
3858 return QLA_FUNCTION_FAILED;
3859
df613b96
AV
3860 mcp->mb[0] = MBC_TRACE_CONTROL;
3861 mcp->mb[1] = TC_FCE_DISABLE;
3862 mcp->mb[2] = TC_FCE_DISABLE_TRACE;
3863 mcp->out_mb = MBX_2|MBX_1|MBX_0;
3864 mcp->in_mb = MBX_9|MBX_8|MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|
3865 MBX_1|MBX_0;
b93480e3 3866 mcp->tov = MBX_TOV_SECONDS;
df613b96 3867 mcp->flags = 0;
7b867cf7 3868 rval = qla2x00_mailbox_command(vha, mcp);
df613b96 3869 if (rval != QLA_SUCCESS) {
7c3df132
SK
3870 ql_dbg(ql_dbg_mbx, vha, 0x10ae,
3871 "Failed=%x mb[0]=%x mb[1]=%x.\n",
3872 rval, mcp->mb[0], mcp->mb[1]);
df613b96 3873 } else {
5f28d2d7
SK
3874 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10af,
3875 "Done %s.\n", __func__);
df613b96
AV
3876
3877 if (wr)
3878 *wr = (uint64_t) mcp->mb[5] << 48 |
3879 (uint64_t) mcp->mb[4] << 32 |
3880 (uint64_t) mcp->mb[3] << 16 |
3881 (uint64_t) mcp->mb[2];
3882 if (rd)
3883 *rd = (uint64_t) mcp->mb[9] << 48 |
3884 (uint64_t) mcp->mb[8] << 32 |
3885 (uint64_t) mcp->mb[7] << 16 |
3886 (uint64_t) mcp->mb[6];
3887 }
3888
3889 return rval;
3890}
3891
6e98016c
GM
3892int
3893qla2x00_get_idma_speed(scsi_qla_host_t *vha, uint16_t loop_id,
3894 uint16_t *port_speed, uint16_t *mb)
3895{
3896 int rval;
3897 mbx_cmd_t mc;
3898 mbx_cmd_t *mcp = &mc;
3899
5f28d2d7
SK
3900 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10b0,
3901 "Entered %s.\n", __func__);
7c3df132 3902
6e98016c
GM
3903 if (!IS_IIDMA_CAPABLE(vha->hw))
3904 return QLA_FUNCTION_FAILED;
3905
6e98016c
GM
3906 mcp->mb[0] = MBC_PORT_PARAMS;
3907 mcp->mb[1] = loop_id;
3908 mcp->mb[2] = mcp->mb[3] = 0;
3909 mcp->mb[9] = vha->vp_idx;
3910 mcp->out_mb = MBX_9|MBX_3|MBX_2|MBX_1|MBX_0;
3911 mcp->in_mb = MBX_3|MBX_1|MBX_0;
3912 mcp->tov = MBX_TOV_SECONDS;
3913 mcp->flags = 0;
3914 rval = qla2x00_mailbox_command(vha, mcp);
3915
3916 /* Return mailbox statuses. */
2a3192a3 3917 if (mb) {
6e98016c
GM
3918 mb[0] = mcp->mb[0];
3919 mb[1] = mcp->mb[1];
3920 mb[3] = mcp->mb[3];
3921 }
3922
3923 if (rval != QLA_SUCCESS) {
7c3df132 3924 ql_dbg(ql_dbg_mbx, vha, 0x10b1, "Failed=%x.\n", rval);
6e98016c 3925 } else {
5f28d2d7
SK
3926 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10b2,
3927 "Done %s.\n", __func__);
6e98016c
GM
3928 if (port_speed)
3929 *port_speed = mcp->mb[3];
3930 }
3931
3932 return rval;
3933}
3934
d8b45213 3935int
7b867cf7 3936qla2x00_set_idma_speed(scsi_qla_host_t *vha, uint16_t loop_id,
d8b45213
AV
3937 uint16_t port_speed, uint16_t *mb)
3938{
3939 int rval;
3940 mbx_cmd_t mc;
3941 mbx_cmd_t *mcp = &mc;
3942
5f28d2d7
SK
3943 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10b3,
3944 "Entered %s.\n", __func__);
7c3df132 3945
7b867cf7 3946 if (!IS_IIDMA_CAPABLE(vha->hw))
d8b45213
AV
3947 return QLA_FUNCTION_FAILED;
3948
d8b45213
AV
3949 mcp->mb[0] = MBC_PORT_PARAMS;
3950 mcp->mb[1] = loop_id;
3951 mcp->mb[2] = BIT_0;
2a3192a3 3952 mcp->mb[3] = port_speed & 0x3F;
1bb39548
HZ
3953 mcp->mb[9] = vha->vp_idx;
3954 mcp->out_mb = MBX_9|MBX_3|MBX_2|MBX_1|MBX_0;
3955 mcp->in_mb = MBX_3|MBX_1|MBX_0;
b93480e3 3956 mcp->tov = MBX_TOV_SECONDS;
d8b45213 3957 mcp->flags = 0;
7b867cf7 3958 rval = qla2x00_mailbox_command(vha, mcp);
d8b45213
AV
3959
3960 /* Return mailbox statuses. */
2a3192a3 3961 if (mb) {
d8b45213
AV
3962 mb[0] = mcp->mb[0];
3963 mb[1] = mcp->mb[1];
3964 mb[3] = mcp->mb[3];
d8b45213
AV
3965 }
3966
3967 if (rval != QLA_SUCCESS) {
5f28d2d7
SK
3968 ql_dbg(ql_dbg_mbx, vha, 0x10b4,
3969 "Failed=%x.\n", rval);
d8b45213 3970 } else {
5f28d2d7
SK
3971 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10b5,
3972 "Done %s.\n", __func__);
d8b45213
AV
3973 }
3974
3975 return rval;
3976}
2c3dfe3f 3977
2c3dfe3f 3978void
7b867cf7 3979qla24xx_report_id_acquisition(scsi_qla_host_t *vha,
2c3dfe3f
SJ
3980 struct vp_rpt_id_entry_24xx *rptid_entry)
3981{
7b867cf7 3982 struct qla_hw_data *ha = vha->hw;
41dc529a 3983 scsi_qla_host_t *vp = NULL;
feafb7b1 3984 unsigned long flags;
4ac8d4ca 3985 int found;
482c9dc7 3986 port_id_t id;
9cd883f0 3987 struct fc_port *fcport;
2c3dfe3f 3988
5f28d2d7
SK
3989 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10b6,
3990 "Entered %s.\n", __func__);
7c3df132 3991
2c3dfe3f
SJ
3992 if (rptid_entry->entry_status != 0)
3993 return;
2c3dfe3f 3994
482c9dc7
QT
3995 id.b.domain = rptid_entry->port_id[2];
3996 id.b.area = rptid_entry->port_id[1];
3997 id.b.al_pa = rptid_entry->port_id[0];
3998 id.b.rsvd_1 = 0;
1763c1fd 3999 ha->flags.n2n_ae = 0;
482c9dc7 4000
2c3dfe3f 4001 if (rptid_entry->format == 0) {
41dc529a 4002 /* loop */
ec7193e2 4003 ql_dbg(ql_dbg_async, vha, 0x10b7,
7c3df132 4004 "Format 0 : Number of VPs setup %d, number of "
41dc529a
QT
4005 "VPs acquired %d.\n", rptid_entry->vp_setup,
4006 rptid_entry->vp_acquired);
ec7193e2 4007 ql_dbg(ql_dbg_async, vha, 0x10b8,
7c3df132
SK
4008 "Primary port id %02x%02x%02x.\n",
4009 rptid_entry->port_id[2], rptid_entry->port_id[1],
4010 rptid_entry->port_id[0]);
9cd883f0 4011 ha->current_topology = ISP_CFG_NL;
430eef03 4012 qla_update_host_map(vha, id);
41dc529a 4013
2c3dfe3f 4014 } else if (rptid_entry->format == 1) {
41dc529a 4015 /* fabric */
ec7193e2 4016 ql_dbg(ql_dbg_async, vha, 0x10b9,
7c3df132 4017 "Format 1: VP[%d] enabled - status %d - with "
41dc529a
QT
4018 "port id %02x%02x%02x.\n", rptid_entry->vp_idx,
4019 rptid_entry->vp_status,
2c3dfe3f 4020 rptid_entry->port_id[2], rptid_entry->port_id[1],
7c3df132 4021 rptid_entry->port_id[0]);
edd05de1
DG
4022 ql_dbg(ql_dbg_async, vha, 0x5075,
4023 "Format 1: Remote WWPN %8phC.\n",
4024 rptid_entry->u.f1.port_name);
4025
4026 ql_dbg(ql_dbg_async, vha, 0x5075,
4027 "Format 1: WWPN %8phC.\n",
4028 vha->port_name);
4029
8777e431
QT
4030 switch (rptid_entry->u.f1.flags & TOPO_MASK) {
4031 case TOPO_N2N:
4032 ha->current_topology = ISP_CFG_N;
4033 spin_lock_irqsave(&vha->hw->tgt.sess_lock, flags);
7f2a398d
QT
4034 list_for_each_entry(fcport, &vha->vp_fcports, list) {
4035 fcport->scan_state = QLA_FCPORT_SCAN;
4036 fcport->n2n_flag = 0;
4037 }
ad8a260a
QT
4038 id.b24 = 0;
4039 if (wwn_to_u64(vha->port_name) >
4040 wwn_to_u64(rptid_entry->u.f1.port_name)) {
4041 vha->d_id.b24 = 0;
4042 vha->d_id.b.al_pa = 1;
4043 ha->flags.n2n_bigger = 1;
4044
4045 id.b.al_pa = 2;
4046 ql_dbg(ql_dbg_async, vha, 0x5075,
4047 "Format 1: assign local id %x remote id %x\n",
4048 vha->d_id.b24, id.b24);
4049 } else {
4050 ql_dbg(ql_dbg_async, vha, 0x5075,
4051 "Format 1: Remote login - Waiting for WWPN %8phC.\n",
4052 rptid_entry->u.f1.port_name);
4053 ha->flags.n2n_bigger = 0;
4054 }
7f2a398d 4055
8777e431
QT
4056 fcport = qla2x00_find_fcport_by_wwpn(vha,
4057 rptid_entry->u.f1.port_name, 1);
4058 spin_unlock_irqrestore(&vha->hw->tgt.sess_lock, flags);
4059
ad8a260a 4060
8777e431
QT
4061 if (fcport) {
4062 fcport->plogi_nack_done_deadline = jiffies + HZ;
94eda271
AE
4063 fcport->dm_login_expire = jiffies +
4064 QLA_N2N_WAIT_TIME * HZ;
8777e431 4065 fcport->scan_state = QLA_FCPORT_FOUND;
7f2a398d 4066 fcport->n2n_flag = 1;
f3f1938b 4067 fcport->keep_nport_handle = 1;
4de067e5 4068 fcport->login_retry = vha->hw->login_retry_count;
f8844457
QT
4069 fcport->fc4_type = FS_FC4TYPE_FCP;
4070 if (vha->flags.nvme_enabled)
4071 fcport->fc4_type |= FS_FC4TYPE_NVME;
7f2a398d 4072
ad8a260a
QT
4073 if (wwn_to_u64(vha->port_name) >
4074 wwn_to_u64(fcport->port_name)) {
4075 fcport->d_id = id;
4076 }
4077
8777e431
QT
4078 switch (fcport->disc_state) {
4079 case DSC_DELETED:
4080 set_bit(RELOGIN_NEEDED,
4081 &vha->dpc_flags);
4082 break;
4083 case DSC_DELETE_PEND:
4084 break;
4085 default:
4086 qlt_schedule_sess_for_deletion(fcport);
4087 break;
4088 }
edd05de1 4089 } else {
8777e431
QT
4090 qla24xx_post_newsess_work(vha, &id,
4091 rptid_entry->u.f1.port_name,
4092 rptid_entry->u.f1.node_name,
4093 NULL,
7f2a398d 4094 FS_FCP_IS_N2N);
edd05de1
DG
4095 }
4096
8777e431
QT
4097 /* if our portname is higher then initiate N2N login */
4098
edd05de1 4099 set_bit(N2N_LOGIN_NEEDED, &vha->dpc_flags);
edd05de1 4100 return;
8777e431
QT
4101 case TOPO_FL:
4102 ha->current_topology = ISP_CFG_FL;
4103 break;
4104 case TOPO_F:
4105 ha->current_topology = ISP_CFG_F;
4106 break;
4107 default:
4108 break;
edd05de1 4109 }
531a82d1 4110
9cd883f0
QT
4111 ha->flags.gpsc_supported = 1;
4112 ha->current_topology = ISP_CFG_F;
969a6199 4113 /* buffer to buffer credit flag */
41dc529a
QT
4114 vha->flags.bbcr_enable = (rptid_entry->u.f1.bbcr & 0xf) != 0;
4115
4116 if (rptid_entry->vp_idx == 0) {
4117 if (rptid_entry->vp_status == VP_STAT_COMPL) {
4118 /* FA-WWN is only for physical port */
4119 if (qla_ini_mode_enabled(vha) &&
4120 ha->flags.fawwpn_enabled &&
4121 (rptid_entry->u.f1.flags &
fcc5b5cd 4122 BIT_6)) {
41dc529a
QT
4123 memcpy(vha->port_name,
4124 rptid_entry->u.f1.port_name,
4125 WWN_SIZE);
4126 }
7c9c4766 4127
430eef03 4128 qla_update_host_map(vha, id);
7c9c4766 4129 }
41dc529a 4130
41dc529a
QT
4131 set_bit(REGISTER_FC4_NEEDED, &vha->dpc_flags);
4132 set_bit(REGISTER_FDMI_NEEDED, &vha->dpc_flags);
4133 } else {
4134 if (rptid_entry->vp_status != VP_STAT_COMPL &&
4135 rptid_entry->vp_status != VP_STAT_ID_CHG) {
4136 ql_dbg(ql_dbg_mbx, vha, 0x10ba,
4137 "Could not acquire ID for VP[%d].\n",
4138 rptid_entry->vp_idx);
4139 return;
4ac8d4ca 4140 }
feafb7b1 4141
41dc529a
QT
4142 found = 0;
4143 spin_lock_irqsave(&ha->vport_slock, flags);
4144 list_for_each_entry(vp, &ha->vp_list, list) {
4145 if (rptid_entry->vp_idx == vp->vp_idx) {
4146 found = 1;
4147 break;
4148 }
4149 }
4150 spin_unlock_irqrestore(&ha->vport_slock, flags);
2c3dfe3f 4151
41dc529a
QT
4152 if (!found)
4153 return;
2c3dfe3f 4154
430eef03 4155 qla_update_host_map(vp, id);
2c3dfe3f 4156
41dc529a
QT
4157 /*
4158 * Cannot configure here as we are still sitting on the
4159 * response queue. Handle it in dpc context.
4160 */
4161 set_bit(VP_IDX_ACQUIRED, &vp->vp_flags);
4162 set_bit(REGISTER_FC4_NEEDED, &vp->dpc_flags);
4163 set_bit(REGISTER_FDMI_NEEDED, &vp->dpc_flags);
4164 }
531a82d1 4165 set_bit(VP_DPC_NEEDED, &vha->dpc_flags);
7b867cf7 4166 qla2xxx_wake_dpc(vha);
41dc529a 4167 } else if (rptid_entry->format == 2) {
83548fe2 4168 ql_dbg(ql_dbg_async, vha, 0x505f,
41dc529a
QT
4169 "RIDA: format 2/N2N Primary port id %02x%02x%02x.\n",
4170 rptid_entry->port_id[2], rptid_entry->port_id[1],
4171 rptid_entry->port_id[0]);
4172
83548fe2 4173 ql_dbg(ql_dbg_async, vha, 0x5075,
41dc529a
QT
4174 "N2N: Remote WWPN %8phC.\n",
4175 rptid_entry->u.f2.port_name);
4176
4177 /* N2N. direct connect */
9cd883f0
QT
4178 ha->current_topology = ISP_CFG_N;
4179 ha->flags.rida_fmt2 = 1;
41dc529a
QT
4180 vha->d_id.b.domain = rptid_entry->port_id[2];
4181 vha->d_id.b.area = rptid_entry->port_id[1];
4182 vha->d_id.b.al_pa = rptid_entry->port_id[0];
4183
1763c1fd 4184 ha->flags.n2n_ae = 1;
41dc529a 4185 spin_lock_irqsave(&ha->vport_slock, flags);
430eef03 4186 qla_update_vp_map(vha, SET_AL_PA);
41dc529a 4187 spin_unlock_irqrestore(&ha->vport_slock, flags);
9cd883f0
QT
4188
4189 list_for_each_entry(fcport, &vha->vp_fcports, list) {
4190 fcport->scan_state = QLA_FCPORT_SCAN;
7f2a398d 4191 fcport->n2n_flag = 0;
9cd883f0
QT
4192 }
4193
4194 fcport = qla2x00_find_fcport_by_wwpn(vha,
4195 rptid_entry->u.f2.port_name, 1);
4196
4197 if (fcport) {
23dd98a6 4198 fcport->login_retry = vha->hw->login_retry_count;
9cd883f0
QT
4199 fcport->plogi_nack_done_deadline = jiffies + HZ;
4200 fcport->scan_state = QLA_FCPORT_FOUND;
f3f1938b 4201 fcport->keep_nport_handle = 1;
7f2a398d
QT
4202 fcport->n2n_flag = 1;
4203 fcport->d_id.b.domain =
4204 rptid_entry->u.f2.remote_nport_id[2];
4205 fcport->d_id.b.area =
4206 rptid_entry->u.f2.remote_nport_id[1];
4207 fcport->d_id.b.al_pa =
4208 rptid_entry->u.f2.remote_nport_id[0];
7a8ff7d9
QT
4209
4210 /*
4211 * For the case where remote port sending PRLO, FW
4212 * sends up RIDA Format 2 as an indication of session
4213 * loss. In other word, FW state change from PRLI
4214 * complete back to PLOGI complete. Delete the
4215 * session and let relogin drive the reconnect.
4216 */
4217 if (atomic_read(&fcport->state) == FCS_ONLINE)
4218 qlt_schedule_sess_for_deletion(fcport);
9cd883f0 4219 }
2c3dfe3f
SJ
4220 }
4221}
4222
4223/*
4224 * qla24xx_modify_vp_config
4225 * Change VP configuration for vha
4226 *
4227 * Input:
4228 * vha = adapter block pointer.
4229 *
4230 * Returns:
4231 * qla2xxx local function return status code.
4232 *
4233 * Context:
4234 * Kernel context.
4235 */
4236int
4237qla24xx_modify_vp_config(scsi_qla_host_t *vha)
4238{
4239 int rval;
4240 struct vp_config_entry_24xx *vpmod;
4241 dma_addr_t vpmod_dma;
7b867cf7
AC
4242 struct qla_hw_data *ha = vha->hw;
4243 struct scsi_qla_host *base_vha = pci_get_drvdata(ha->pdev);
2c3dfe3f
SJ
4244
4245 /* This can be called by the parent */
2c3dfe3f 4246
5f28d2d7
SK
4247 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10bb,
4248 "Entered %s.\n", __func__);
7c3df132 4249
08eb7f45 4250 vpmod = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &vpmod_dma);
2c3dfe3f 4251 if (!vpmod) {
7c3df132
SK
4252 ql_log(ql_log_warn, vha, 0x10bc,
4253 "Failed to allocate modify VP IOCB.\n");
2c3dfe3f
SJ
4254 return QLA_MEMORY_ALLOC_FAILED;
4255 }
4256
2c3dfe3f
SJ
4257 vpmod->entry_type = VP_CONFIG_IOCB_TYPE;
4258 vpmod->entry_count = 1;
4259 vpmod->command = VCT_COMMAND_MOD_ENABLE_VPS;
4260 vpmod->vp_count = 1;
4261 vpmod->vp_index1 = vha->vp_idx;
4262 vpmod->options_idx1 = BIT_3|BIT_4|BIT_5;
2d70c103
NB
4263
4264 qlt_modify_vp_config(vha, vpmod);
4265
2c3dfe3f
SJ
4266 memcpy(vpmod->node_name_idx1, vha->node_name, WWN_SIZE);
4267 memcpy(vpmod->port_name_idx1, vha->port_name, WWN_SIZE);
4268 vpmod->entry_count = 1;
4269
7b867cf7 4270 rval = qla2x00_issue_iocb(base_vha, vpmod, vpmod_dma, 0);
2c3dfe3f 4271 if (rval != QLA_SUCCESS) {
7c3df132
SK
4272 ql_dbg(ql_dbg_mbx, vha, 0x10bd,
4273 "Failed to issue VP config IOCB (%x).\n", rval);
2c3dfe3f 4274 } else if (vpmod->comp_status != 0) {
7c3df132
SK
4275 ql_dbg(ql_dbg_mbx, vha, 0x10be,
4276 "Failed to complete IOCB -- error status (%x).\n",
4277 vpmod->comp_status);
2c3dfe3f 4278 rval = QLA_FUNCTION_FAILED;
ad950360 4279 } else if (vpmod->comp_status != cpu_to_le16(CS_COMPLETE)) {
7c3df132
SK
4280 ql_dbg(ql_dbg_mbx, vha, 0x10bf,
4281 "Failed to complete IOCB -- completion status (%x).\n",
4282 le16_to_cpu(vpmod->comp_status));
2c3dfe3f
SJ
4283 rval = QLA_FUNCTION_FAILED;
4284 } else {
4285 /* EMPTY */
5f28d2d7
SK
4286 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10c0,
4287 "Done %s.\n", __func__);
2c3dfe3f
SJ
4288 fc_vport_set_state(vha->fc_vport, FC_VPORT_INITIALIZING);
4289 }
7b867cf7 4290 dma_pool_free(ha->s_dma_pool, vpmod, vpmod_dma);
2c3dfe3f
SJ
4291
4292 return rval;
4293}
4294
2c3dfe3f
SJ
4295/*
4296 * qla2x00_send_change_request
4297 * Receive or disable RSCN request from fabric controller
4298 *
4299 * Input:
4300 * ha = adapter block pointer
4301 * format = registration format:
4302 * 0 - Reserved
4303 * 1 - Fabric detected registration
4304 * 2 - N_port detected registration
4305 * 3 - Full registration
4306 * FF - clear registration
4307 * vp_idx = Virtual port index
4308 *
4309 * Returns:
4310 * qla2x00 local function return status code.
4311 *
4312 * Context:
4313 * Kernel Context
4314 */
4315
4316int
7b867cf7 4317qla2x00_send_change_request(scsi_qla_host_t *vha, uint16_t format,
2c3dfe3f
SJ
4318 uint16_t vp_idx)
4319{
4320 int rval;
4321 mbx_cmd_t mc;
4322 mbx_cmd_t *mcp = &mc;
4323
5f28d2d7
SK
4324 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10c7,
4325 "Entered %s.\n", __func__);
7c3df132 4326
2c3dfe3f
SJ
4327 mcp->mb[0] = MBC_SEND_CHANGE_REQUEST;
4328 mcp->mb[1] = format;
4329 mcp->mb[9] = vp_idx;
4330 mcp->out_mb = MBX_9|MBX_1|MBX_0;
4331 mcp->in_mb = MBX_0|MBX_1;
4332 mcp->tov = MBX_TOV_SECONDS;
4333 mcp->flags = 0;
7b867cf7 4334 rval = qla2x00_mailbox_command(vha, mcp);
2c3dfe3f
SJ
4335
4336 if (rval == QLA_SUCCESS) {
4337 if (mcp->mb[0] != MBS_COMMAND_COMPLETE) {
4338 rval = BIT_1;
4339 }
4340 } else
4341 rval = BIT_1;
4342
4343 return rval;
4344}
338c9161
AV
4345
4346int
7b867cf7 4347qla2x00_dump_ram(scsi_qla_host_t *vha, dma_addr_t req_dma, uint32_t addr,
338c9161
AV
4348 uint32_t size)
4349{
4350 int rval;
4351 mbx_cmd_t mc;
4352 mbx_cmd_t *mcp = &mc;
4353
5f28d2d7
SK
4354 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1009,
4355 "Entered %s.\n", __func__);
338c9161 4356
7b867cf7 4357 if (MSW(addr) || IS_FWI2_CAPABLE(vha->hw)) {
338c9161
AV
4358 mcp->mb[0] = MBC_DUMP_RISC_RAM_EXTENDED;
4359 mcp->mb[8] = MSW(addr);
044c218b
QT
4360 mcp->mb[10] = 0;
4361 mcp->out_mb = MBX_10|MBX_8|MBX_0;
338c9161
AV
4362 } else {
4363 mcp->mb[0] = MBC_DUMP_RISC_RAM;
4364 mcp->out_mb = MBX_0;
4365 }
4366 mcp->mb[1] = LSW(addr);
4367 mcp->mb[2] = MSW(req_dma);
4368 mcp->mb[3] = LSW(req_dma);
4369 mcp->mb[6] = MSW(MSD(req_dma));
4370 mcp->mb[7] = LSW(MSD(req_dma));
4371 mcp->out_mb |= MBX_7|MBX_6|MBX_3|MBX_2|MBX_1;
7b867cf7 4372 if (IS_FWI2_CAPABLE(vha->hw)) {
338c9161
AV
4373 mcp->mb[4] = MSW(size);
4374 mcp->mb[5] = LSW(size);
4375 mcp->out_mb |= MBX_5|MBX_4;
4376 } else {
4377 mcp->mb[4] = LSW(size);
4378 mcp->out_mb |= MBX_4;
4379 }
4380
4381 mcp->in_mb = MBX_0;
b93480e3 4382 mcp->tov = MBX_TOV_SECONDS;
338c9161 4383 mcp->flags = 0;
7b867cf7 4384 rval = qla2x00_mailbox_command(vha, mcp);
338c9161
AV
4385
4386 if (rval != QLA_SUCCESS) {
7c3df132
SK
4387 ql_dbg(ql_dbg_mbx, vha, 0x1008,
4388 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
338c9161 4389 } else {
5f28d2d7
SK
4390 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1007,
4391 "Done %s.\n", __func__);
338c9161
AV
4392 }
4393
4394 return rval;
4395}
4d4df193
HK
4396/* 84XX Support **************************************************************/
4397
4398struct cs84xx_mgmt_cmd {
4399 union {
4400 struct verify_chip_entry_84xx req;
4401 struct verify_chip_rsp_84xx rsp;
4402 } p;
4403};
4404
4405int
7b867cf7 4406qla84xx_verify_chip(struct scsi_qla_host *vha, uint16_t *status)
4d4df193
HK
4407{
4408 int rval, retry;
4409 struct cs84xx_mgmt_cmd *mn;
4410 dma_addr_t mn_dma;
4411 uint16_t options;
4412 unsigned long flags;
7b867cf7 4413 struct qla_hw_data *ha = vha->hw;
4d4df193 4414
5f28d2d7
SK
4415 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10c8,
4416 "Entered %s.\n", __func__);
4d4df193
HK
4417
4418 mn = dma_pool_alloc(ha->s_dma_pool, GFP_KERNEL, &mn_dma);
4419 if (mn == NULL) {
4d4df193
HK
4420 return QLA_MEMORY_ALLOC_FAILED;
4421 }
4422
4423 /* Force Update? */
4424 options = ha->cs84xx->fw_update ? VCO_FORCE_UPDATE : 0;
4425 /* Diagnostic firmware? */
4426 /* options |= MENLO_DIAG_FW; */
4427 /* We update the firmware with only one data sequence. */
4428 options |= VCO_END_OF_DATA;
4429
4d4df193 4430 do {
c1ec1f1b 4431 retry = 0;
4d4df193
HK
4432 memset(mn, 0, sizeof(*mn));
4433 mn->p.req.entry_type = VERIFY_CHIP_IOCB_TYPE;
4434 mn->p.req.entry_count = 1;
4435 mn->p.req.options = cpu_to_le16(options);
4436
7c3df132
SK
4437 ql_dbg(ql_dbg_mbx + ql_dbg_buffer, vha, 0x111c,
4438 "Dump of Verify Request.\n");
4439 ql_dump_buffer(ql_dbg_mbx + ql_dbg_buffer, vha, 0x111e,
f8f97b0c 4440 mn, sizeof(*mn));
4d4df193 4441
7b867cf7 4442 rval = qla2x00_issue_iocb_timeout(vha, mn, mn_dma, 0, 120);
4d4df193 4443 if (rval != QLA_SUCCESS) {
7c3df132
SK
4444 ql_dbg(ql_dbg_mbx, vha, 0x10cb,
4445 "Failed to issue verify IOCB (%x).\n", rval);
4d4df193
HK
4446 goto verify_done;
4447 }
4448
7c3df132
SK
4449 ql_dbg(ql_dbg_mbx + ql_dbg_buffer, vha, 0x1110,
4450 "Dump of Verify Response.\n");
4451 ql_dump_buffer(ql_dbg_mbx + ql_dbg_buffer, vha, 0x1118,
f8f97b0c 4452 mn, sizeof(*mn));
4d4df193
HK
4453
4454 status[0] = le16_to_cpu(mn->p.rsp.comp_status);
4455 status[1] = status[0] == CS_VCS_CHIP_FAILURE ?
4456 le16_to_cpu(mn->p.rsp.failure_code) : 0;
5f28d2d7 4457 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ce,
7c3df132 4458 "cs=%x fc=%x.\n", status[0], status[1]);
4d4df193
HK
4459
4460 if (status[0] != CS_COMPLETE) {
4461 rval = QLA_FUNCTION_FAILED;
4462 if (!(options & VCO_DONT_UPDATE_FW)) {
7c3df132
SK
4463 ql_dbg(ql_dbg_mbx, vha, 0x10cf,
4464 "Firmware update failed. Retrying "
4465 "without update firmware.\n");
4d4df193
HK
4466 options |= VCO_DONT_UPDATE_FW;
4467 options &= ~VCO_FORCE_UPDATE;
4468 retry = 1;
4469 }
4470 } else {
5f28d2d7 4471 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d0,
7c3df132
SK
4472 "Firmware updated to %x.\n",
4473 le32_to_cpu(mn->p.rsp.fw_ver));
4d4df193
HK
4474
4475 /* NOTE: we only update OP firmware. */
4476 spin_lock_irqsave(&ha->cs84xx->access_lock, flags);
4477 ha->cs84xx->op_fw_version =
4478 le32_to_cpu(mn->p.rsp.fw_ver);
4479 spin_unlock_irqrestore(&ha->cs84xx->access_lock,
4480 flags);
4481 }
4482 } while (retry);
4483
4484verify_done:
4485 dma_pool_free(ha->s_dma_pool, mn, mn_dma);
4486
4487 if (rval != QLA_SUCCESS) {
5f28d2d7
SK
4488 ql_dbg(ql_dbg_mbx, vha, 0x10d1,
4489 "Failed=%x.\n", rval);
4d4df193 4490 } else {
5f28d2d7
SK
4491 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d2,
4492 "Done %s.\n", __func__);
4d4df193
HK
4493 }
4494
4495 return rval;
4496}
73208dfd
AC
4497
4498int
618a7523 4499qla25xx_init_req_que(struct scsi_qla_host *vha, struct req_que *req)
73208dfd
AC
4500{
4501 int rval;
4502 unsigned long flags;
4503 mbx_cmd_t mc;
4504 mbx_cmd_t *mcp = &mc;
73208dfd
AC
4505 struct qla_hw_data *ha = vha->hw;
4506
45235022
QT
4507 if (!ha->flags.fw_started)
4508 return QLA_SUCCESS;
4509
5f28d2d7
SK
4510 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d3,
4511 "Entered %s.\n", __func__);
7c3df132 4512
7c6300e3
JC
4513 if (IS_SHADOW_REG_CAPABLE(ha))
4514 req->options |= BIT_13;
4515
73208dfd 4516 mcp->mb[0] = MBC_INITIALIZE_MULTIQ;
618a7523 4517 mcp->mb[1] = req->options;
73208dfd
AC
4518 mcp->mb[2] = MSW(LSD(req->dma));
4519 mcp->mb[3] = LSW(LSD(req->dma));
4520 mcp->mb[6] = MSW(MSD(req->dma));
4521 mcp->mb[7] = LSW(MSD(req->dma));
4522 mcp->mb[5] = req->length;
4523 if (req->rsp)
4524 mcp->mb[10] = req->rsp->id;
4525 mcp->mb[12] = req->qos;
4526 mcp->mb[11] = req->vp_idx;
4527 mcp->mb[13] = req->rid;
ecc89f25 4528 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha))
6246b8a1 4529 mcp->mb[15] = 0;
73208dfd 4530
73208dfd
AC
4531 mcp->mb[4] = req->id;
4532 /* que in ptr index */
4533 mcp->mb[8] = 0;
4534 /* que out ptr index */
7c6300e3 4535 mcp->mb[9] = *req->out_ptr = 0;
73208dfd
AC
4536 mcp->out_mb = MBX_14|MBX_13|MBX_12|MBX_11|MBX_10|MBX_9|MBX_8|MBX_7|
4537 MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
4538 mcp->in_mb = MBX_0;
4539 mcp->flags = MBX_DMA_OUT;
6246b8a1
GM
4540 mcp->tov = MBX_TOV_SECONDS * 2;
4541
ecc89f25
JC
4542 if (IS_QLA81XX(ha) || IS_QLA83XX(ha) || IS_QLA27XX(ha) ||
4543 IS_QLA28XX(ha))
6246b8a1 4544 mcp->in_mb |= MBX_1;
ecc89f25 4545 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
6246b8a1
GM
4546 mcp->out_mb |= MBX_15;
4547 /* debug q create issue in SR-IOV */
4548 mcp->in_mb |= MBX_9 | MBX_8 | MBX_7;
4549 }
73208dfd
AC
4550
4551 spin_lock_irqsave(&ha->hardware_lock, flags);
618a7523 4552 if (!(req->options & BIT_0)) {
04474d3a 4553 wrt_reg_dword(req->req_q_in, 0);
ecc89f25 4554 if (!IS_QLA83XX(ha) && !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
04474d3a 4555 wrt_reg_dword(req->req_q_out, 0);
73208dfd
AC
4556 }
4557 spin_unlock_irqrestore(&ha->hardware_lock, flags);
4558
17d98630 4559 rval = qla2x00_mailbox_command(vha, mcp);
7c3df132
SK
4560 if (rval != QLA_SUCCESS) {
4561 ql_dbg(ql_dbg_mbx, vha, 0x10d4,
4562 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
4563 } else {
5f28d2d7
SK
4564 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d5,
4565 "Done %s.\n", __func__);
7c3df132
SK
4566 }
4567
73208dfd
AC
4568 return rval;
4569}
4570
4571int
618a7523 4572qla25xx_init_rsp_que(struct scsi_qla_host *vha, struct rsp_que *rsp)
73208dfd
AC
4573{
4574 int rval;
4575 unsigned long flags;
4576 mbx_cmd_t mc;
4577 mbx_cmd_t *mcp = &mc;
73208dfd
AC
4578 struct qla_hw_data *ha = vha->hw;
4579
45235022
QT
4580 if (!ha->flags.fw_started)
4581 return QLA_SUCCESS;
4582
5f28d2d7
SK
4583 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d6,
4584 "Entered %s.\n", __func__);
7c3df132 4585
7c6300e3
JC
4586 if (IS_SHADOW_REG_CAPABLE(ha))
4587 rsp->options |= BIT_13;
4588
73208dfd 4589 mcp->mb[0] = MBC_INITIALIZE_MULTIQ;
618a7523 4590 mcp->mb[1] = rsp->options;
73208dfd
AC
4591 mcp->mb[2] = MSW(LSD(rsp->dma));
4592 mcp->mb[3] = LSW(LSD(rsp->dma));
4593 mcp->mb[6] = MSW(MSD(rsp->dma));
4594 mcp->mb[7] = LSW(MSD(rsp->dma));
4595 mcp->mb[5] = rsp->length;
444786d7 4596 mcp->mb[14] = rsp->msix->entry;
73208dfd 4597 mcp->mb[13] = rsp->rid;
ecc89f25 4598 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha))
6246b8a1 4599 mcp->mb[15] = 0;
73208dfd 4600
73208dfd
AC
4601 mcp->mb[4] = rsp->id;
4602 /* que in ptr index */
7c6300e3 4603 mcp->mb[8] = *rsp->in_ptr = 0;
73208dfd
AC
4604 /* que out ptr index */
4605 mcp->mb[9] = 0;
2afa19a9 4606 mcp->out_mb = MBX_14|MBX_13|MBX_9|MBX_8|MBX_7
73208dfd
AC
4607 |MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
4608 mcp->in_mb = MBX_0;
4609 mcp->flags = MBX_DMA_OUT;
6246b8a1
GM
4610 mcp->tov = MBX_TOV_SECONDS * 2;
4611
4612 if (IS_QLA81XX(ha)) {
4613 mcp->out_mb |= MBX_12|MBX_11|MBX_10;
4614 mcp->in_mb |= MBX_1;
ecc89f25 4615 } else if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
6246b8a1
GM
4616 mcp->out_mb |= MBX_15|MBX_12|MBX_11|MBX_10;
4617 mcp->in_mb |= MBX_1;
4618 /* debug q create issue in SR-IOV */
4619 mcp->in_mb |= MBX_9 | MBX_8 | MBX_7;
4620 }
73208dfd
AC
4621
4622 spin_lock_irqsave(&ha->hardware_lock, flags);
618a7523 4623 if (!(rsp->options & BIT_0)) {
04474d3a 4624 wrt_reg_dword(rsp->rsp_q_out, 0);
ecc89f25 4625 if (!IS_QLA83XX(ha) && !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
04474d3a 4626 wrt_reg_dword(rsp->rsp_q_in, 0);
73208dfd
AC
4627 }
4628
4629 spin_unlock_irqrestore(&ha->hardware_lock, flags);
4630
17d98630 4631 rval = qla2x00_mailbox_command(vha, mcp);
7c3df132
SK
4632 if (rval != QLA_SUCCESS) {
4633 ql_dbg(ql_dbg_mbx, vha, 0x10d7,
4634 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
4635 } else {
5f28d2d7
SK
4636 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d8,
4637 "Done %s.\n", __func__);
7c3df132
SK
4638 }
4639
73208dfd
AC
4640 return rval;
4641}
4642
8a659571
AV
4643int
4644qla81xx_idc_ack(scsi_qla_host_t *vha, uint16_t *mb)
4645{
4646 int rval;
4647 mbx_cmd_t mc;
4648 mbx_cmd_t *mcp = &mc;
4649
5f28d2d7
SK
4650 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10d9,
4651 "Entered %s.\n", __func__);
8a659571
AV
4652
4653 mcp->mb[0] = MBC_IDC_ACK;
4654 memcpy(&mcp->mb[1], mb, QLA_IDC_ACK_REGS * sizeof(uint16_t));
4655 mcp->out_mb = MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
4656 mcp->in_mb = MBX_0;
4657 mcp->tov = MBX_TOV_SECONDS;
4658 mcp->flags = 0;
4659 rval = qla2x00_mailbox_command(vha, mcp);
4660
4661 if (rval != QLA_SUCCESS) {
7c3df132
SK
4662 ql_dbg(ql_dbg_mbx, vha, 0x10da,
4663 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
8a659571 4664 } else {
5f28d2d7
SK
4665 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10db,
4666 "Done %s.\n", __func__);
8a659571
AV
4667 }
4668
4669 return rval;
4670}
1d2874de
JC
4671
4672int
4673qla81xx_fac_get_sector_size(scsi_qla_host_t *vha, uint32_t *sector_size)
4674{
4675 int rval;
4676 mbx_cmd_t mc;
4677 mbx_cmd_t *mcp = &mc;
4678
5f28d2d7
SK
4679 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10dc,
4680 "Entered %s.\n", __func__);
7c3df132 4681
f73cb695 4682 if (!IS_QLA81XX(vha->hw) && !IS_QLA83XX(vha->hw) &&
ecc89f25 4683 !IS_QLA27XX(vha->hw) && !IS_QLA28XX(vha->hw))
1d2874de
JC
4684 return QLA_FUNCTION_FAILED;
4685
1d2874de
JC
4686 mcp->mb[0] = MBC_FLASH_ACCESS_CTRL;
4687 mcp->mb[1] = FAC_OPT_CMD_GET_SECTOR_SIZE;
4688 mcp->out_mb = MBX_1|MBX_0;
4689 mcp->in_mb = MBX_1|MBX_0;
4690 mcp->tov = MBX_TOV_SECONDS;
4691 mcp->flags = 0;
4692 rval = qla2x00_mailbox_command(vha, mcp);
4693
4694 if (rval != QLA_SUCCESS) {
7c3df132
SK
4695 ql_dbg(ql_dbg_mbx, vha, 0x10dd,
4696 "Failed=%x mb[0]=%x mb[1]=%x.\n",
4697 rval, mcp->mb[0], mcp->mb[1]);
1d2874de 4698 } else {
5f28d2d7
SK
4699 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10de,
4700 "Done %s.\n", __func__);
1d2874de
JC
4701 *sector_size = mcp->mb[1];
4702 }
4703
4704 return rval;
4705}
4706
4707int
4708qla81xx_fac_do_write_enable(scsi_qla_host_t *vha, int enable)
4709{
4710 int rval;
4711 mbx_cmd_t mc;
4712 mbx_cmd_t *mcp = &mc;
4713
f73cb695 4714 if (!IS_QLA81XX(vha->hw) && !IS_QLA83XX(vha->hw) &&
ecc89f25 4715 !IS_QLA27XX(vha->hw) && !IS_QLA28XX(vha->hw))
1d2874de
JC
4716 return QLA_FUNCTION_FAILED;
4717
5f28d2d7
SK
4718 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10df,
4719 "Entered %s.\n", __func__);
1d2874de
JC
4720
4721 mcp->mb[0] = MBC_FLASH_ACCESS_CTRL;
4722 mcp->mb[1] = enable ? FAC_OPT_CMD_WRITE_ENABLE :
4723 FAC_OPT_CMD_WRITE_PROTECT;
4724 mcp->out_mb = MBX_1|MBX_0;
4725 mcp->in_mb = MBX_1|MBX_0;
4726 mcp->tov = MBX_TOV_SECONDS;
4727 mcp->flags = 0;
4728 rval = qla2x00_mailbox_command(vha, mcp);
4729
4730 if (rval != QLA_SUCCESS) {
7c3df132
SK
4731 ql_dbg(ql_dbg_mbx, vha, 0x10e0,
4732 "Failed=%x mb[0]=%x mb[1]=%x.\n",
4733 rval, mcp->mb[0], mcp->mb[1]);
1d2874de 4734 } else {
5f28d2d7
SK
4735 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e1,
4736 "Done %s.\n", __func__);
1d2874de
JC
4737 }
4738
4739 return rval;
4740}
4741
4742int
4743qla81xx_fac_erase_sector(scsi_qla_host_t *vha, uint32_t start, uint32_t finish)
4744{
4745 int rval;
4746 mbx_cmd_t mc;
4747 mbx_cmd_t *mcp = &mc;
4748
f73cb695 4749 if (!IS_QLA81XX(vha->hw) && !IS_QLA83XX(vha->hw) &&
ecc89f25 4750 !IS_QLA27XX(vha->hw) && !IS_QLA28XX(vha->hw))
1d2874de
JC
4751 return QLA_FUNCTION_FAILED;
4752
5f28d2d7
SK
4753 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e2,
4754 "Entered %s.\n", __func__);
1d2874de
JC
4755
4756 mcp->mb[0] = MBC_FLASH_ACCESS_CTRL;
4757 mcp->mb[1] = FAC_OPT_CMD_ERASE_SECTOR;
4758 mcp->mb[2] = LSW(start);
4759 mcp->mb[3] = MSW(start);
4760 mcp->mb[4] = LSW(finish);
4761 mcp->mb[5] = MSW(finish);
4762 mcp->out_mb = MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
4763 mcp->in_mb = MBX_2|MBX_1|MBX_0;
4764 mcp->tov = MBX_TOV_SECONDS;
4765 mcp->flags = 0;
4766 rval = qla2x00_mailbox_command(vha, mcp);
4767
4768 if (rval != QLA_SUCCESS) {
7c3df132
SK
4769 ql_dbg(ql_dbg_mbx, vha, 0x10e3,
4770 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x.\n",
4771 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2]);
1d2874de 4772 } else {
5f28d2d7
SK
4773 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e4,
4774 "Done %s.\n", __func__);
1d2874de
JC
4775 }
4776
4777 return rval;
4778}
6e181be5 4779
3f006ac3
MH
4780int
4781qla81xx_fac_semaphore_access(scsi_qla_host_t *vha, int lock)
4782{
4783 int rval = QLA_SUCCESS;
4784 mbx_cmd_t mc;
4785 mbx_cmd_t *mcp = &mc;
4786 struct qla_hw_data *ha = vha->hw;
4787
4788 if (!IS_QLA81XX(ha) && !IS_QLA83XX(ha) &&
4789 !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
4790 return rval;
4791
4792 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e2,
4793 "Entered %s.\n", __func__);
4794
4795 mcp->mb[0] = MBC_FLASH_ACCESS_CTRL;
4796 mcp->mb[1] = (lock ? FAC_OPT_CMD_LOCK_SEMAPHORE :
4797 FAC_OPT_CMD_UNLOCK_SEMAPHORE);
4798 mcp->out_mb = MBX_1|MBX_0;
4799 mcp->in_mb = MBX_1|MBX_0;
4800 mcp->tov = MBX_TOV_SECONDS;
4801 mcp->flags = 0;
4802 rval = qla2x00_mailbox_command(vha, mcp);
4803
4804 if (rval != QLA_SUCCESS) {
4805 ql_dbg(ql_dbg_mbx, vha, 0x10e3,
4806 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x.\n",
4807 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2]);
4808 } else {
4809 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e4,
4810 "Done %s.\n", __func__);
4811 }
4812
4813 return rval;
4814}
4815
6e181be5
LC
4816int
4817qla81xx_restart_mpi_firmware(scsi_qla_host_t *vha)
4818{
4819 int rval = 0;
4820 mbx_cmd_t mc;
4821 mbx_cmd_t *mcp = &mc;
4822
5f28d2d7
SK
4823 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e5,
4824 "Entered %s.\n", __func__);
6e181be5
LC
4825
4826 mcp->mb[0] = MBC_RESTART_MPI_FW;
4827 mcp->out_mb = MBX_0;
4828 mcp->in_mb = MBX_0|MBX_1;
4829 mcp->tov = MBX_TOV_SECONDS;
4830 mcp->flags = 0;
4831 rval = qla2x00_mailbox_command(vha, mcp);
4832
4833 if (rval != QLA_SUCCESS) {
7c3df132
SK
4834 ql_dbg(ql_dbg_mbx, vha, 0x10e6,
4835 "Failed=%x mb[0]=%x mb[1]=%x.\n",
4836 rval, mcp->mb[0], mcp->mb[1]);
6e181be5 4837 } else {
5f28d2d7
SK
4838 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e7,
4839 "Done %s.\n", __func__);
6e181be5
LC
4840 }
4841
4842 return rval;
4843}
ad0ecd61 4844
c46e65c7
JC
4845int
4846qla82xx_set_driver_version(scsi_qla_host_t *vha, char *version)
4847{
4848 int rval;
4849 mbx_cmd_t mc;
4850 mbx_cmd_t *mcp = &mc;
4851 int i;
4852 int len;
7ffa5b93 4853 __le16 *str;
c46e65c7
JC
4854 struct qla_hw_data *ha = vha->hw;
4855
4856 if (!IS_P3P_TYPE(ha))
4857 return QLA_FUNCTION_FAILED;
4858
4859 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x117b,
4860 "Entered %s.\n", __func__);
4861
7ffa5b93 4862 str = (__force __le16 *)version;
c46e65c7
JC
4863 len = strlen(version);
4864
4865 mcp->mb[0] = MBC_SET_RNID_PARAMS;
4866 mcp->mb[1] = RNID_TYPE_SET_VERSION << 8;
4867 mcp->out_mb = MBX_1|MBX_0;
4868 for (i = 4; i < 16 && len; i++, str++, len -= 2) {
7ffa5b93 4869 mcp->mb[i] = le16_to_cpup(str);
c46e65c7
JC
4870 mcp->out_mb |= 1<<i;
4871 }
4872 for (; i < 16; i++) {
4873 mcp->mb[i] = 0;
4874 mcp->out_mb |= 1<<i;
4875 }
4876 mcp->in_mb = MBX_1|MBX_0;
4877 mcp->tov = MBX_TOV_SECONDS;
4878 mcp->flags = 0;
4879 rval = qla2x00_mailbox_command(vha, mcp);
4880
4881 if (rval != QLA_SUCCESS) {
4882 ql_dbg(ql_dbg_mbx, vha, 0x117c,
4883 "Failed=%x mb[0]=%x,%x.\n", rval, mcp->mb[0], mcp->mb[1]);
4884 } else {
4885 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x117d,
4886 "Done %s.\n", __func__);
4887 }
4888
4889 return rval;
4890}
4891
4892int
4893qla25xx_set_driver_version(scsi_qla_host_t *vha, char *version)
4894{
4895 int rval;
4896 mbx_cmd_t mc;
4897 mbx_cmd_t *mcp = &mc;
4898 int len;
4899 uint16_t dwlen;
4900 uint8_t *str;
4901 dma_addr_t str_dma;
4902 struct qla_hw_data *ha = vha->hw;
4903
4904 if (!IS_FWI2_CAPABLE(ha) || IS_QLA24XX_TYPE(ha) || IS_QLA81XX(ha) ||
4905 IS_P3P_TYPE(ha))
4906 return QLA_FUNCTION_FAILED;
4907
4908 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x117e,
4909 "Entered %s.\n", __func__);
4910
4911 str = dma_pool_alloc(ha->s_dma_pool, GFP_KERNEL, &str_dma);
4912 if (!str) {
4913 ql_log(ql_log_warn, vha, 0x117f,
4914 "Failed to allocate driver version param.\n");
4915 return QLA_MEMORY_ALLOC_FAILED;
4916 }
4917
4918 memcpy(str, "\x7\x3\x11\x0", 4);
4919 dwlen = str[0];
4920 len = dwlen * 4 - 4;
4921 memset(str + 4, 0, len);
4922 if (len > strlen(version))
4923 len = strlen(version);
4924 memcpy(str + 4, version, len);
4925
4926 mcp->mb[0] = MBC_SET_RNID_PARAMS;
4927 mcp->mb[1] = RNID_TYPE_SET_VERSION << 8 | dwlen;
4928 mcp->mb[2] = MSW(LSD(str_dma));
4929 mcp->mb[3] = LSW(LSD(str_dma));
4930 mcp->mb[6] = MSW(MSD(str_dma));
4931 mcp->mb[7] = LSW(MSD(str_dma));
4932 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
4933 mcp->in_mb = MBX_1|MBX_0;
4934 mcp->tov = MBX_TOV_SECONDS;
4935 mcp->flags = 0;
4936 rval = qla2x00_mailbox_command(vha, mcp);
4937
4938 if (rval != QLA_SUCCESS) {
4939 ql_dbg(ql_dbg_mbx, vha, 0x1180,
4940 "Failed=%x mb[0]=%x,%x.\n", rval, mcp->mb[0], mcp->mb[1]);
4941 } else {
4942 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1181,
4943 "Done %s.\n", __func__);
4944 }
4945
4946 dma_pool_free(ha->s_dma_pool, str, str_dma);
4947
4948 return rval;
4949}
4950
edd05de1
DG
4951int
4952qla24xx_get_port_login_templ(scsi_qla_host_t *vha, dma_addr_t buf_dma,
4953 void *buf, uint16_t bufsiz)
4954{
4955 int rval, i;
4956 mbx_cmd_t mc;
4957 mbx_cmd_t *mcp = &mc;
4958 uint32_t *bp;
4959
4960 if (!IS_FWI2_CAPABLE(vha->hw))
4961 return QLA_FUNCTION_FAILED;
4962
4963 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1159,
4964 "Entered %s.\n", __func__);
4965
4966 mcp->mb[0] = MBC_GET_RNID_PARAMS;
4967 mcp->mb[1] = RNID_TYPE_PORT_LOGIN << 8;
4968 mcp->mb[2] = MSW(buf_dma);
4969 mcp->mb[3] = LSW(buf_dma);
4970 mcp->mb[6] = MSW(MSD(buf_dma));
4971 mcp->mb[7] = LSW(MSD(buf_dma));
4972 mcp->mb[8] = bufsiz/4;
4973 mcp->out_mb = MBX_8|MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
4974 mcp->in_mb = MBX_1|MBX_0;
4975 mcp->tov = MBX_TOV_SECONDS;
4976 mcp->flags = 0;
4977 rval = qla2x00_mailbox_command(vha, mcp);
4978
4979 if (rval != QLA_SUCCESS) {
4980 ql_dbg(ql_dbg_mbx, vha, 0x115a,
4981 "Failed=%x mb[0]=%x,%x.\n", rval, mcp->mb[0], mcp->mb[1]);
4982 } else {
4983 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x115b,
4984 "Done %s.\n", __func__);
4985 bp = (uint32_t *) buf;
4986 for (i = 0; i < (bufsiz-4)/4; i++, bp++)
7ffa5b93 4987 *bp = le32_to_cpu((__force __le32)*bp);
edd05de1
DG
4988 }
4989
4990 return rval;
4991}
4992
44d01857 4993#define PUREX_CMD_COUNT 4
d83a80ee
JC
4994int
4995qla25xx_set_els_cmds_supported(scsi_qla_host_t *vha)
4996{
4997 int rval;
4998 mbx_cmd_t mc;
4999 mbx_cmd_t *mcp = &mc;
5000 uint8_t *els_cmd_map;
44d01857 5001 uint8_t active_cnt = 0;
d83a80ee 5002 dma_addr_t els_cmd_map_dma;
62e9dd17
SS
5003 uint8_t cmd_opcode[PUREX_CMD_COUNT];
5004 uint8_t i, index, purex_bit;
d83a80ee
JC
5005 struct qla_hw_data *ha = vha->hw;
5006
62e9dd17
SS
5007 if (!IS_QLA25XX(ha) && !IS_QLA2031(ha) &&
5008 !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
d83a80ee
JC
5009 return QLA_SUCCESS;
5010
5011 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1197,
5012 "Entered %s.\n", __func__);
5013
5014 els_cmd_map = dma_alloc_coherent(&ha->pdev->dev, ELS_CMD_MAP_SIZE,
5015 &els_cmd_map_dma, GFP_KERNEL);
5016 if (!els_cmd_map) {
5017 ql_log(ql_log_warn, vha, 0x7101,
5018 "Failed to allocate RDP els command param.\n");
5019 return QLA_MEMORY_ALLOC_FAILED;
5020 }
5021
62e9dd17 5022 /* List of Purex ELS */
44d01857
QT
5023 if (ql2xrdpenable) {
5024 cmd_opcode[active_cnt] = ELS_RDP;
5025 active_cnt++;
5026 }
5027 if (ha->flags.scm_supported_f) {
5028 cmd_opcode[active_cnt] = ELS_FPIN;
5029 active_cnt++;
5030 }
5031 if (ha->flags.edif_enabled) {
5032 cmd_opcode[active_cnt] = ELS_AUTH_ELS;
5033 active_cnt++;
5034 }
62e9dd17 5035
44d01857 5036 for (i = 0; i < active_cnt; i++) {
62e9dd17
SS
5037 index = cmd_opcode[i] / 8;
5038 purex_bit = cmd_opcode[i] % 8;
5039 els_cmd_map[index] |= 1 << purex_bit;
5040 }
d83a80ee
JC
5041
5042 mcp->mb[0] = MBC_SET_RNID_PARAMS;
5043 mcp->mb[1] = RNID_TYPE_ELS_CMD << 8;
5044 mcp->mb[2] = MSW(LSD(els_cmd_map_dma));
5045 mcp->mb[3] = LSW(LSD(els_cmd_map_dma));
5046 mcp->mb[6] = MSW(MSD(els_cmd_map_dma));
5047 mcp->mb[7] = LSW(MSD(els_cmd_map_dma));
5048 mcp->out_mb = MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
5049 mcp->in_mb = MBX_1|MBX_0;
5050 mcp->tov = MBX_TOV_SECONDS;
5051 mcp->flags = MBX_DMA_OUT;
5052 mcp->buf_size = ELS_CMD_MAP_SIZE;
5053 rval = qla2x00_mailbox_command(vha, mcp);
5054
5055 if (rval != QLA_SUCCESS) {
5056 ql_dbg(ql_dbg_mbx, vha, 0x118d,
5057 "Failed=%x (%x,%x).\n", rval, mcp->mb[0], mcp->mb[1]);
5058 } else {
5059 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x118c,
5060 "Done %s.\n", __func__);
5061 }
5062
650b323c 5063 dma_free_coherent(&ha->pdev->dev, ELS_CMD_MAP_SIZE,
d83a80ee
JC
5064 els_cmd_map, els_cmd_map_dma);
5065
5066 return rval;
5067}
5068
fe52f6e1
JC
5069static int
5070qla2x00_read_asic_temperature(scsi_qla_host_t *vha, uint16_t *temp)
5071{
5072 int rval;
5073 mbx_cmd_t mc;
5074 mbx_cmd_t *mcp = &mc;
5075
5076 if (!IS_FWI2_CAPABLE(vha->hw))
5077 return QLA_FUNCTION_FAILED;
5078
5079 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1159,
5080 "Entered %s.\n", __func__);
5081
5082 mcp->mb[0] = MBC_GET_RNID_PARAMS;
5083 mcp->mb[1] = RNID_TYPE_ASIC_TEMP << 8;
5084 mcp->out_mb = MBX_1|MBX_0;
5085 mcp->in_mb = MBX_1|MBX_0;
5086 mcp->tov = MBX_TOV_SECONDS;
5087 mcp->flags = 0;
5088 rval = qla2x00_mailbox_command(vha, mcp);
5089 *temp = mcp->mb[1];
5090
5091 if (rval != QLA_SUCCESS) {
5092 ql_dbg(ql_dbg_mbx, vha, 0x115a,
5093 "Failed=%x mb[0]=%x,%x.\n", rval, mcp->mb[0], mcp->mb[1]);
5094 } else {
5095 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x115b,
5096 "Done %s.\n", __func__);
5097 }
5098
5099 return rval;
5100}
5101
ad0ecd61 5102int
6766df9e
JC
5103qla2x00_read_sfp(scsi_qla_host_t *vha, dma_addr_t sfp_dma, uint8_t *sfp,
5104 uint16_t dev, uint16_t off, uint16_t len, uint16_t opt)
ad0ecd61
JC
5105{
5106 int rval;
5107 mbx_cmd_t mc;
5108 mbx_cmd_t *mcp = &mc;
6766df9e
JC
5109 struct qla_hw_data *ha = vha->hw;
5110
5f28d2d7
SK
5111 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e8,
5112 "Entered %s.\n", __func__);
7c3df132 5113
6766df9e
JC
5114 if (!IS_FWI2_CAPABLE(ha))
5115 return QLA_FUNCTION_FAILED;
ad0ecd61 5116
6766df9e
JC
5117 if (len == 1)
5118 opt |= BIT_0;
5119
ad0ecd61
JC
5120 mcp->mb[0] = MBC_READ_SFP;
5121 mcp->mb[1] = dev;
818c7f87
JC
5122 mcp->mb[2] = MSW(LSD(sfp_dma));
5123 mcp->mb[3] = LSW(LSD(sfp_dma));
ad0ecd61
JC
5124 mcp->mb[6] = MSW(MSD(sfp_dma));
5125 mcp->mb[7] = LSW(MSD(sfp_dma));
5126 mcp->mb[8] = len;
6766df9e 5127 mcp->mb[9] = off;
ad0ecd61
JC
5128 mcp->mb[10] = opt;
5129 mcp->out_mb = MBX_10|MBX_9|MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
1bff6cc8 5130 mcp->in_mb = MBX_1|MBX_0;
ad0ecd61
JC
5131 mcp->tov = MBX_TOV_SECONDS;
5132 mcp->flags = 0;
5133 rval = qla2x00_mailbox_command(vha, mcp);
5134
5135 if (opt & BIT_0)
6766df9e 5136 *sfp = mcp->mb[1];
ad0ecd61
JC
5137
5138 if (rval != QLA_SUCCESS) {
7c3df132
SK
5139 ql_dbg(ql_dbg_mbx, vha, 0x10e9,
5140 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
2a3192a3 5141 if (mcp->mb[0] == MBS_COMMAND_ERROR && mcp->mb[1] == 0x22) {
e4e3a2ce
QT
5142 /* sfp is not there */
5143 rval = QLA_INTERFACE_ERROR;
2a3192a3 5144 }
ad0ecd61 5145 } else {
5f28d2d7
SK
5146 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ea,
5147 "Done %s.\n", __func__);
ad0ecd61
JC
5148 }
5149
5150 return rval;
5151}
5152
5153int
6766df9e
JC
5154qla2x00_write_sfp(scsi_qla_host_t *vha, dma_addr_t sfp_dma, uint8_t *sfp,
5155 uint16_t dev, uint16_t off, uint16_t len, uint16_t opt)
ad0ecd61
JC
5156{
5157 int rval;
5158 mbx_cmd_t mc;
5159 mbx_cmd_t *mcp = &mc;
6766df9e
JC
5160 struct qla_hw_data *ha = vha->hw;
5161
5f28d2d7
SK
5162 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10eb,
5163 "Entered %s.\n", __func__);
7c3df132 5164
6766df9e
JC
5165 if (!IS_FWI2_CAPABLE(ha))
5166 return QLA_FUNCTION_FAILED;
ad0ecd61 5167
6766df9e
JC
5168 if (len == 1)
5169 opt |= BIT_0;
5170
ad0ecd61 5171 if (opt & BIT_0)
6766df9e 5172 len = *sfp;
ad0ecd61
JC
5173
5174 mcp->mb[0] = MBC_WRITE_SFP;
5175 mcp->mb[1] = dev;
818c7f87
JC
5176 mcp->mb[2] = MSW(LSD(sfp_dma));
5177 mcp->mb[3] = LSW(LSD(sfp_dma));
ad0ecd61
JC
5178 mcp->mb[6] = MSW(MSD(sfp_dma));
5179 mcp->mb[7] = LSW(MSD(sfp_dma));
5180 mcp->mb[8] = len;
6766df9e 5181 mcp->mb[9] = off;
ad0ecd61
JC
5182 mcp->mb[10] = opt;
5183 mcp->out_mb = MBX_10|MBX_9|MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
6766df9e 5184 mcp->in_mb = MBX_1|MBX_0;
ad0ecd61
JC
5185 mcp->tov = MBX_TOV_SECONDS;
5186 mcp->flags = 0;
5187 rval = qla2x00_mailbox_command(vha, mcp);
5188
5189 if (rval != QLA_SUCCESS) {
7c3df132
SK
5190 ql_dbg(ql_dbg_mbx, vha, 0x10ec,
5191 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
ad0ecd61 5192 } else {
5f28d2d7
SK
5193 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ed,
5194 "Done %s.\n", __func__);
ad0ecd61
JC
5195 }
5196
5197 return rval;
5198}
ce0423f4
AV
5199
5200int
5201qla2x00_get_xgmac_stats(scsi_qla_host_t *vha, dma_addr_t stats_dma,
5202 uint16_t size_in_bytes, uint16_t *actual_size)
5203{
5204 int rval;
5205 mbx_cmd_t mc;
5206 mbx_cmd_t *mcp = &mc;
5207
5f28d2d7
SK
5208 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ee,
5209 "Entered %s.\n", __func__);
7c3df132 5210
6246b8a1 5211 if (!IS_CNA_CAPABLE(vha->hw))
ce0423f4
AV
5212 return QLA_FUNCTION_FAILED;
5213
ce0423f4
AV
5214 mcp->mb[0] = MBC_GET_XGMAC_STATS;
5215 mcp->mb[2] = MSW(stats_dma);
5216 mcp->mb[3] = LSW(stats_dma);
5217 mcp->mb[6] = MSW(MSD(stats_dma));
5218 mcp->mb[7] = LSW(MSD(stats_dma));
5219 mcp->mb[8] = size_in_bytes >> 2;
5220 mcp->out_mb = MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_0;
5221 mcp->in_mb = MBX_2|MBX_1|MBX_0;
5222 mcp->tov = MBX_TOV_SECONDS;
5223 mcp->flags = 0;
5224 rval = qla2x00_mailbox_command(vha, mcp);
5225
5226 if (rval != QLA_SUCCESS) {
7c3df132
SK
5227 ql_dbg(ql_dbg_mbx, vha, 0x10ef,
5228 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x.\n",
5229 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2]);
ce0423f4 5230 } else {
5f28d2d7
SK
5231 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f0,
5232 "Done %s.\n", __func__);
7c3df132 5233
ce0423f4
AV
5234
5235 *actual_size = mcp->mb[2] << 2;
5236 }
5237
5238 return rval;
5239}
11bbc1d8
AV
5240
5241int
5242qla2x00_get_dcbx_params(scsi_qla_host_t *vha, dma_addr_t tlv_dma,
5243 uint16_t size)
5244{
5245 int rval;
5246 mbx_cmd_t mc;
5247 mbx_cmd_t *mcp = &mc;
5248
5f28d2d7
SK
5249 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f1,
5250 "Entered %s.\n", __func__);
7c3df132 5251
6246b8a1 5252 if (!IS_CNA_CAPABLE(vha->hw))
11bbc1d8
AV
5253 return QLA_FUNCTION_FAILED;
5254
11bbc1d8
AV
5255 mcp->mb[0] = MBC_GET_DCBX_PARAMS;
5256 mcp->mb[1] = 0;
5257 mcp->mb[2] = MSW(tlv_dma);
5258 mcp->mb[3] = LSW(tlv_dma);
5259 mcp->mb[6] = MSW(MSD(tlv_dma));
5260 mcp->mb[7] = LSW(MSD(tlv_dma));
5261 mcp->mb[8] = size;
5262 mcp->out_mb = MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
5263 mcp->in_mb = MBX_2|MBX_1|MBX_0;
5264 mcp->tov = MBX_TOV_SECONDS;
5265 mcp->flags = 0;
5266 rval = qla2x00_mailbox_command(vha, mcp);
5267
5268 if (rval != QLA_SUCCESS) {
7c3df132
SK
5269 ql_dbg(ql_dbg_mbx, vha, 0x10f2,
5270 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x.\n",
5271 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2]);
11bbc1d8 5272 } else {
5f28d2d7
SK
5273 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f3,
5274 "Done %s.\n", __func__);
11bbc1d8
AV
5275 }
5276
5277 return rval;
5278}
18e7555a
AV
5279
5280int
5281qla2x00_read_ram_word(scsi_qla_host_t *vha, uint32_t risc_addr, uint32_t *data)
5282{
5283 int rval;
5284 mbx_cmd_t mc;
5285 mbx_cmd_t *mcp = &mc;
5286
5f28d2d7
SK
5287 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f4,
5288 "Entered %s.\n", __func__);
7c3df132 5289
18e7555a
AV
5290 if (!IS_FWI2_CAPABLE(vha->hw))
5291 return QLA_FUNCTION_FAILED;
5292
18e7555a
AV
5293 mcp->mb[0] = MBC_READ_RAM_EXTENDED;
5294 mcp->mb[1] = LSW(risc_addr);
5295 mcp->mb[8] = MSW(risc_addr);
5296 mcp->out_mb = MBX_8|MBX_1|MBX_0;
5297 mcp->in_mb = MBX_3|MBX_2|MBX_0;
c314a014 5298 mcp->tov = MBX_TOV_SECONDS;
18e7555a
AV
5299 mcp->flags = 0;
5300 rval = qla2x00_mailbox_command(vha, mcp);
5301 if (rval != QLA_SUCCESS) {
7c3df132
SK
5302 ql_dbg(ql_dbg_mbx, vha, 0x10f5,
5303 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
18e7555a 5304 } else {
5f28d2d7
SK
5305 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f6,
5306 "Done %s.\n", __func__);
18e7555a
AV
5307 *data = mcp->mb[3] << 16 | mcp->mb[2];
5308 }
5309
5310 return rval;
5311}
5312
9a069e19 5313int
a9083016
GM
5314qla2x00_loopback_test(scsi_qla_host_t *vha, struct msg_echo_lb *mreq,
5315 uint16_t *mresp)
9a069e19
GM
5316{
5317 int rval;
5318 mbx_cmd_t mc;
5319 mbx_cmd_t *mcp = &mc;
9a069e19 5320
5f28d2d7
SK
5321 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f7,
5322 "Entered %s.\n", __func__);
9a069e19
GM
5323
5324 memset(mcp->mb, 0 , sizeof(mcp->mb));
5325 mcp->mb[0] = MBC_DIAGNOSTIC_LOOP_BACK;
5326 mcp->mb[1] = mreq->options | BIT_6; // BIT_6 specifies 64 bit addressing
5327
5328 /* transfer count */
5329 mcp->mb[10] = LSW(mreq->transfer_size);
5330 mcp->mb[11] = MSW(mreq->transfer_size);
5331
5332 /* send data address */
5333 mcp->mb[14] = LSW(mreq->send_dma);
5334 mcp->mb[15] = MSW(mreq->send_dma);
5335 mcp->mb[20] = LSW(MSD(mreq->send_dma));
5336 mcp->mb[21] = MSW(MSD(mreq->send_dma));
5337
25985edc 5338 /* receive data address */
9a069e19
GM
5339 mcp->mb[16] = LSW(mreq->rcv_dma);
5340 mcp->mb[17] = MSW(mreq->rcv_dma);
5341 mcp->mb[6] = LSW(MSD(mreq->rcv_dma));
5342 mcp->mb[7] = MSW(MSD(mreq->rcv_dma));
5343
5344 /* Iteration count */
1b98b421
JC
5345 mcp->mb[18] = LSW(mreq->iteration_count);
5346 mcp->mb[19] = MSW(mreq->iteration_count);
9a069e19
GM
5347
5348 mcp->out_mb = MBX_21|MBX_20|MBX_19|MBX_18|MBX_17|MBX_16|MBX_15|
5349 MBX_14|MBX_13|MBX_12|MBX_11|MBX_10|MBX_7|MBX_6|MBX_1|MBX_0;
6246b8a1 5350 if (IS_CNA_CAPABLE(vha->hw))
9a069e19
GM
5351 mcp->out_mb |= MBX_2;
5352 mcp->in_mb = MBX_19|MBX_18|MBX_3|MBX_2|MBX_1|MBX_0;
5353
5354 mcp->buf_size = mreq->transfer_size;
5355 mcp->tov = MBX_TOV_SECONDS;
5356 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN|IOCTL_CMD;
5357
5358 rval = qla2x00_mailbox_command(vha, mcp);
5359
5360 if (rval != QLA_SUCCESS) {
7c3df132
SK
5361 ql_dbg(ql_dbg_mbx, vha, 0x10f8,
5362 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x mb[3]=%x mb[18]=%x "
5363 "mb[19]=%x.\n", rval, mcp->mb[0], mcp->mb[1], mcp->mb[2],
5364 mcp->mb[3], mcp->mb[18], mcp->mb[19]);
9a069e19 5365 } else {
5f28d2d7
SK
5366 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10f9,
5367 "Done %s.\n", __func__);
9a069e19
GM
5368 }
5369
5370 /* Copy mailbox information */
5371 memcpy( mresp, mcp->mb, 64);
9a069e19
GM
5372 return rval;
5373}
5374
5375int
a9083016
GM
5376qla2x00_echo_test(scsi_qla_host_t *vha, struct msg_echo_lb *mreq,
5377 uint16_t *mresp)
9a069e19
GM
5378{
5379 int rval;
5380 mbx_cmd_t mc;
5381 mbx_cmd_t *mcp = &mc;
5382 struct qla_hw_data *ha = vha->hw;
5383
5f28d2d7
SK
5384 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10fa,
5385 "Entered %s.\n", __func__);
9a069e19
GM
5386
5387 memset(mcp->mb, 0 , sizeof(mcp->mb));
5388 mcp->mb[0] = MBC_DIAGNOSTIC_ECHO;
1d634965
JC
5389 /* BIT_6 specifies 64bit address */
5390 mcp->mb[1] = mreq->options | BIT_15 | BIT_6;
6246b8a1 5391 if (IS_CNA_CAPABLE(ha)) {
a9083016
GM
5392 mcp->mb[2] = vha->fcoe_fcf_idx;
5393 }
9a069e19
GM
5394 mcp->mb[16] = LSW(mreq->rcv_dma);
5395 mcp->mb[17] = MSW(mreq->rcv_dma);
5396 mcp->mb[6] = LSW(MSD(mreq->rcv_dma));
5397 mcp->mb[7] = MSW(MSD(mreq->rcv_dma));
5398
5399 mcp->mb[10] = LSW(mreq->transfer_size);
5400
5401 mcp->mb[14] = LSW(mreq->send_dma);
5402 mcp->mb[15] = MSW(mreq->send_dma);
5403 mcp->mb[20] = LSW(MSD(mreq->send_dma));
5404 mcp->mb[21] = MSW(MSD(mreq->send_dma));
5405
5406 mcp->out_mb = MBX_21|MBX_20|MBX_17|MBX_16|MBX_15|
5407 MBX_14|MBX_10|MBX_7|MBX_6|MBX_1|MBX_0;
6246b8a1 5408 if (IS_CNA_CAPABLE(ha))
9a069e19
GM
5409 mcp->out_mb |= MBX_2;
5410
5411 mcp->in_mb = MBX_0;
83cfd3dc
JC
5412 if (IS_CNA_CAPABLE(ha) || IS_QLA24XX_TYPE(ha) || IS_QLA25XX(ha) ||
5413 IS_QLA2031(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha))
9a069e19 5414 mcp->in_mb |= MBX_1;
83cfd3dc
JC
5415 if (IS_CNA_CAPABLE(ha) || IS_QLA2031(ha) || IS_QLA27XX(ha) ||
5416 IS_QLA28XX(ha))
9a069e19
GM
5417 mcp->in_mb |= MBX_3;
5418
5419 mcp->tov = MBX_TOV_SECONDS;
5420 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN|IOCTL_CMD;
5421 mcp->buf_size = mreq->transfer_size;
5422
5423 rval = qla2x00_mailbox_command(vha, mcp);
5424
5425 if (rval != QLA_SUCCESS) {
7c3df132
SK
5426 ql_dbg(ql_dbg_mbx, vha, 0x10fb,
5427 "Failed=%x mb[0]=%x mb[1]=%x.\n",
5428 rval, mcp->mb[0], mcp->mb[1]);
9a069e19 5429 } else {
5f28d2d7
SK
5430 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10fc,
5431 "Done %s.\n", __func__);
9a069e19
GM
5432 }
5433
5434 /* Copy mailbox information */
6dbdda4d 5435 memcpy(mresp, mcp->mb, 64);
9a069e19
GM
5436 return rval;
5437}
6dbdda4d 5438
9a069e19 5439int
7c3df132 5440qla84xx_reset_chip(scsi_qla_host_t *vha, uint16_t enable_diagnostic)
9a069e19
GM
5441{
5442 int rval;
5443 mbx_cmd_t mc;
5444 mbx_cmd_t *mcp = &mc;
5445
5f28d2d7 5446 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10fd,
7c3df132 5447 "Entered %s enable_diag=%d.\n", __func__, enable_diagnostic);
9a069e19
GM
5448
5449 mcp->mb[0] = MBC_ISP84XX_RESET;
5450 mcp->mb[1] = enable_diagnostic;
5451 mcp->out_mb = MBX_1|MBX_0;
5452 mcp->in_mb = MBX_1|MBX_0;
5453 mcp->tov = MBX_TOV_SECONDS;
5454 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN|IOCTL_CMD;
7c3df132 5455 rval = qla2x00_mailbox_command(vha, mcp);
9a069e19 5456
9a069e19 5457 if (rval != QLA_SUCCESS)
7c3df132 5458 ql_dbg(ql_dbg_mbx, vha, 0x10fe, "Failed=%x.\n", rval);
9a069e19 5459 else
5f28d2d7
SK
5460 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ff,
5461 "Done %s.\n", __func__);
9a069e19
GM
5462
5463 return rval;
5464}
5465
18e7555a
AV
5466int
5467qla2x00_write_ram_word(scsi_qla_host_t *vha, uint32_t risc_addr, uint32_t data)
5468{
5469 int rval;
5470 mbx_cmd_t mc;
5471 mbx_cmd_t *mcp = &mc;
5472
5f28d2d7
SK
5473 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1100,
5474 "Entered %s.\n", __func__);
7c3df132 5475
18e7555a 5476 if (!IS_FWI2_CAPABLE(vha->hw))
6c452a45 5477 return QLA_FUNCTION_FAILED;
18e7555a 5478
18e7555a
AV
5479 mcp->mb[0] = MBC_WRITE_RAM_WORD_EXTENDED;
5480 mcp->mb[1] = LSW(risc_addr);
5481 mcp->mb[2] = LSW(data);
5482 mcp->mb[3] = MSW(data);
5483 mcp->mb[8] = MSW(risc_addr);
5484 mcp->out_mb = MBX_8|MBX_3|MBX_2|MBX_1|MBX_0;
2a3192a3 5485 mcp->in_mb = MBX_1|MBX_0;
c314a014 5486 mcp->tov = MBX_TOV_SECONDS;
18e7555a
AV
5487 mcp->flags = 0;
5488 rval = qla2x00_mailbox_command(vha, mcp);
5489 if (rval != QLA_SUCCESS) {
7c3df132 5490 ql_dbg(ql_dbg_mbx, vha, 0x1101,
2a3192a3
JC
5491 "Failed=%x mb[0]=%x mb[1]=%x.\n",
5492 rval, mcp->mb[0], mcp->mb[1]);
18e7555a 5493 } else {
5f28d2d7
SK
5494 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1102,
5495 "Done %s.\n", __func__);
18e7555a
AV
5496 }
5497
5498 return rval;
5499}
3064ff39 5500
b1d46989
MI
5501int
5502qla81xx_write_mpi_register(scsi_qla_host_t *vha, uint16_t *mb)
5503{
5504 int rval;
5505 uint32_t stat, timer;
5506 uint16_t mb0 = 0;
5507 struct qla_hw_data *ha = vha->hw;
5508 struct device_reg_24xx __iomem *reg = &ha->iobase->isp24;
5509
5510 rval = QLA_SUCCESS;
5511
5f28d2d7
SK
5512 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1103,
5513 "Entered %s.\n", __func__);
b1d46989
MI
5514
5515 clear_bit(MBX_INTERRUPT, &ha->mbx_cmd_flags);
5516
5517 /* Write the MBC data to the registers */
04474d3a
BVA
5518 wrt_reg_word(&reg->mailbox0, MBC_WRITE_MPI_REGISTER);
5519 wrt_reg_word(&reg->mailbox1, mb[0]);
5520 wrt_reg_word(&reg->mailbox2, mb[1]);
5521 wrt_reg_word(&reg->mailbox3, mb[2]);
5522 wrt_reg_word(&reg->mailbox4, mb[3]);
b1d46989 5523
04474d3a 5524 wrt_reg_dword(&reg->hccr, HCCRX_SET_HOST_INT);
b1d46989
MI
5525
5526 /* Poll for MBC interrupt */
5527 for (timer = 6000000; timer; timer--) {
5528 /* Check for pending interrupts. */
04474d3a 5529 stat = rd_reg_dword(&reg->host_status);
b1d46989
MI
5530 if (stat & HSRX_RISC_INT) {
5531 stat &= 0xff;
5532
5533 if (stat == 0x1 || stat == 0x2 ||
5534 stat == 0x10 || stat == 0x11) {
5535 set_bit(MBX_INTERRUPT,
5536 &ha->mbx_cmd_flags);
04474d3a
BVA
5537 mb0 = rd_reg_word(&reg->mailbox0);
5538 wrt_reg_dword(&reg->hccr,
b1d46989 5539 HCCRX_CLR_RISC_INT);
04474d3a 5540 rd_reg_dword(&reg->hccr);
b1d46989
MI
5541 break;
5542 }
5543 }
5544 udelay(5);
5545 }
5546
5547 if (test_and_clear_bit(MBX_INTERRUPT, &ha->mbx_cmd_flags))
5548 rval = mb0 & MBS_MASK;
5549 else
5550 rval = QLA_FUNCTION_FAILED;
5551
5552 if (rval != QLA_SUCCESS) {
7c3df132
SK
5553 ql_dbg(ql_dbg_mbx, vha, 0x1104,
5554 "Failed=%x mb[0]=%x.\n", rval, mb[0]);
b1d46989 5555 } else {
5f28d2d7
SK
5556 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1105,
5557 "Done %s.\n", __func__);
b1d46989
MI
5558 }
5559
5560 return rval;
5561}
6246b8a1 5562
4910b524
AG
5563/* Set the specified data rate */
5564int
5565qla2x00_set_data_rate(scsi_qla_host_t *vha, uint16_t mode)
5566{
5567 int rval;
5568 mbx_cmd_t mc;
5569 mbx_cmd_t *mcp = &mc;
5570 struct qla_hw_data *ha = vha->hw;
5571 uint16_t val;
5572
5573 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1106,
5574 "Entered %s speed:0x%x mode:0x%x.\n", __func__, ha->set_data_rate,
5575 mode);
5576
5577 if (!IS_FWI2_CAPABLE(ha))
5578 return QLA_FUNCTION_FAILED;
5579
5580 memset(mcp, 0, sizeof(*mcp));
5581 switch (ha->set_data_rate) {
5582 case PORT_SPEED_AUTO:
5583 case PORT_SPEED_4GB:
5584 case PORT_SPEED_8GB:
5585 case PORT_SPEED_16GB:
5586 case PORT_SPEED_32GB:
5587 val = ha->set_data_rate;
5588 break;
5589 default:
5590 ql_log(ql_log_warn, vha, 0x1199,
5591 "Unrecognized speed setting:%d. Setting Autoneg\n",
5592 ha->set_data_rate);
5593 val = ha->set_data_rate = PORT_SPEED_AUTO;
5594 break;
5595 }
5596
5597 mcp->mb[0] = MBC_DATA_RATE;
5598 mcp->mb[1] = mode;
5599 mcp->mb[2] = val;
5600
5601 mcp->out_mb = MBX_2|MBX_1|MBX_0;
5602 mcp->in_mb = MBX_2|MBX_1|MBX_0;
ecc89f25 5603 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha))
4910b524
AG
5604 mcp->in_mb |= MBX_4|MBX_3;
5605 mcp->tov = MBX_TOV_SECONDS;
5606 mcp->flags = 0;
5607 rval = qla2x00_mailbox_command(vha, mcp);
5608 if (rval != QLA_SUCCESS) {
5609 ql_dbg(ql_dbg_mbx, vha, 0x1107,
5610 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
5611 } else {
5612 if (mcp->mb[1] != 0x7)
5613 ql_dbg(ql_dbg_mbx, vha, 0x1179,
5614 "Speed set:0x%x\n", mcp->mb[1]);
5615
5616 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1108,
5617 "Done %s.\n", __func__);
5618 }
5619
5620 return rval;
5621}
5622
3064ff39
MH
5623int
5624qla2x00_get_data_rate(scsi_qla_host_t *vha)
5625{
5626 int rval;
5627 mbx_cmd_t mc;
5628 mbx_cmd_t *mcp = &mc;
5629 struct qla_hw_data *ha = vha->hw;
5630
5f28d2d7
SK
5631 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1106,
5632 "Entered %s.\n", __func__);
7c3df132 5633
3064ff39
MH
5634 if (!IS_FWI2_CAPABLE(ha))
5635 return QLA_FUNCTION_FAILED;
5636
3064ff39 5637 mcp->mb[0] = MBC_DATA_RATE;
4910b524 5638 mcp->mb[1] = QLA_GET_DATA_RATE;
3064ff39
MH
5639 mcp->out_mb = MBX_1|MBX_0;
5640 mcp->in_mb = MBX_2|MBX_1|MBX_0;
ecc89f25 5641 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha))
0d6a536c 5642 mcp->in_mb |= MBX_4|MBX_3;
3064ff39
MH
5643 mcp->tov = MBX_TOV_SECONDS;
5644 mcp->flags = 0;
5645 rval = qla2x00_mailbox_command(vha, mcp);
5646 if (rval != QLA_SUCCESS) {
7c3df132
SK
5647 ql_dbg(ql_dbg_mbx, vha, 0x1107,
5648 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
3064ff39 5649 } else {
75666f4a
HM
5650 if (mcp->mb[1] != 0x7)
5651 ha->link_data_rate = mcp->mb[1];
5652
5653 if (IS_QLA83XX(ha) || IS_QLA27XX(ha) || IS_QLA28XX(ha)) {
5654 if (mcp->mb[4] & BIT_0)
5655 ql_log(ql_log_info, vha, 0x11a2,
5656 "FEC=enabled (data rate).\n");
5657 }
5658
5f28d2d7
SK
5659 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1108,
5660 "Done %s.\n", __func__);
3064ff39
MH
5661 if (mcp->mb[1] != 0x7)
5662 ha->link_data_rate = mcp->mb[1];
5663 }
5664
5665 return rval;
5666}
09ff701a 5667
23f2ebd1
SR
5668int
5669qla81xx_get_port_config(scsi_qla_host_t *vha, uint16_t *mb)
5670{
5671 int rval;
5672 mbx_cmd_t mc;
5673 mbx_cmd_t *mcp = &mc;
5674 struct qla_hw_data *ha = vha->hw;
5675
5f28d2d7
SK
5676 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1109,
5677 "Entered %s.\n", __func__);
23f2ebd1 5678
f73cb695 5679 if (!IS_QLA81XX(ha) && !IS_QLA83XX(ha) && !IS_QLA8044(ha) &&
ecc89f25 5680 !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
23f2ebd1
SR
5681 return QLA_FUNCTION_FAILED;
5682 mcp->mb[0] = MBC_GET_PORT_CONFIG;
5683 mcp->out_mb = MBX_0;
5684 mcp->in_mb = MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
5685 mcp->tov = MBX_TOV_SECONDS;
5686 mcp->flags = 0;
5687
5688 rval = qla2x00_mailbox_command(vha, mcp);
5689
5690 if (rval != QLA_SUCCESS) {
7c3df132
SK
5691 ql_dbg(ql_dbg_mbx, vha, 0x110a,
5692 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
23f2ebd1
SR
5693 } else {
5694 /* Copy all bits to preserve original value */
5695 memcpy(mb, &mcp->mb[1], sizeof(uint16_t) * 4);
5696
5f28d2d7
SK
5697 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x110b,
5698 "Done %s.\n", __func__);
23f2ebd1
SR
5699 }
5700 return rval;
5701}
5702
5703int
5704qla81xx_set_port_config(scsi_qla_host_t *vha, uint16_t *mb)
5705{
5706 int rval;
5707 mbx_cmd_t mc;
5708 mbx_cmd_t *mcp = &mc;
5709
5f28d2d7
SK
5710 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x110c,
5711 "Entered %s.\n", __func__);
23f2ebd1
SR
5712
5713 mcp->mb[0] = MBC_SET_PORT_CONFIG;
5714 /* Copy all bits to preserve original setting */
5715 memcpy(&mcp->mb[1], mb, sizeof(uint16_t) * 4);
5716 mcp->out_mb = MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
5717 mcp->in_mb = MBX_0;
5718 mcp->tov = MBX_TOV_SECONDS;
5719 mcp->flags = 0;
5720 rval = qla2x00_mailbox_command(vha, mcp);
5721
5722 if (rval != QLA_SUCCESS) {
7c3df132
SK
5723 ql_dbg(ql_dbg_mbx, vha, 0x110d,
5724 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
23f2ebd1 5725 } else
5f28d2d7
SK
5726 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x110e,
5727 "Done %s.\n", __func__);
23f2ebd1
SR
5728
5729 return rval;
5730}
5731
5732
09ff701a
SR
5733int
5734qla24xx_set_fcp_prio(scsi_qla_host_t *vha, uint16_t loop_id, uint16_t priority,
5735 uint16_t *mb)
5736{
5737 int rval;
5738 mbx_cmd_t mc;
5739 mbx_cmd_t *mcp = &mc;
5740 struct qla_hw_data *ha = vha->hw;
5741
5f28d2d7
SK
5742 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x110f,
5743 "Entered %s.\n", __func__);
7c3df132 5744
09ff701a
SR
5745 if (!IS_QLA24XX_TYPE(ha) && !IS_QLA25XX(ha))
5746 return QLA_FUNCTION_FAILED;
5747
09ff701a
SR
5748 mcp->mb[0] = MBC_PORT_PARAMS;
5749 mcp->mb[1] = loop_id;
5750 if (ha->flags.fcp_prio_enabled)
5751 mcp->mb[2] = BIT_1;
5752 else
5753 mcp->mb[2] = BIT_2;
5754 mcp->mb[4] = priority & 0xf;
5755 mcp->mb[9] = vha->vp_idx;
5756 mcp->out_mb = MBX_9|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
5757 mcp->in_mb = MBX_4|MBX_3|MBX_1|MBX_0;
c314a014 5758 mcp->tov = MBX_TOV_SECONDS;
09ff701a
SR
5759 mcp->flags = 0;
5760 rval = qla2x00_mailbox_command(vha, mcp);
5761 if (mb != NULL) {
5762 mb[0] = mcp->mb[0];
5763 mb[1] = mcp->mb[1];
5764 mb[3] = mcp->mb[3];
5765 mb[4] = mcp->mb[4];
5766 }
5767
5768 if (rval != QLA_SUCCESS) {
7c3df132 5769 ql_dbg(ql_dbg_mbx, vha, 0x10cd, "Failed=%x.\n", rval);
09ff701a 5770 } else {
5f28d2d7
SK
5771 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10cc,
5772 "Done %s.\n", __func__);
09ff701a
SR
5773 }
5774
5775 return rval;
5776}
a9083016 5777
794a5691 5778int
fe52f6e1 5779qla2x00_get_thermal_temp(scsi_qla_host_t *vha, uint16_t *temp)
794a5691 5780{
fe52f6e1 5781 int rval = QLA_FUNCTION_FAILED;
794a5691 5782 struct qla_hw_data *ha = vha->hw;
fe52f6e1 5783 uint8_t byte;
794a5691 5784
1ae47cf3
JC
5785 if (!IS_FWI2_CAPABLE(ha) || IS_QLA24XX_TYPE(ha) || IS_QLA81XX(ha)) {
5786 ql_dbg(ql_dbg_mbx, vha, 0x1150,
5787 "Thermal not supported by this card.\n");
5788 return rval;
5789 }
5790
5791 if (IS_QLA25XX(ha)) {
5792 if (ha->pdev->subsystem_vendor == PCI_VENDOR_ID_QLOGIC &&
5793 ha->pdev->subsystem_device == 0x0175) {
5794 rval = qla2x00_read_sfp(vha, 0, &byte,
5795 0x98, 0x1, 1, BIT_13|BIT_0);
5796 *temp = byte;
5797 return rval;
5798 }
5799 if (ha->pdev->subsystem_vendor == PCI_VENDOR_ID_HP &&
5800 ha->pdev->subsystem_device == 0x338e) {
5801 rval = qla2x00_read_sfp(vha, 0, &byte,
5802 0x98, 0x1, 1, BIT_15|BIT_14|BIT_0);
5803 *temp = byte;
5804 return rval;
5805 }
5806 ql_dbg(ql_dbg_mbx, vha, 0x10c9,
5807 "Thermal not supported by this card.\n");
5808 return rval;
794a5691 5809 }
794a5691 5810
1ae47cf3
JC
5811 if (IS_QLA82XX(ha)) {
5812 *temp = qla82xx_read_temperature(vha);
5813 rval = QLA_SUCCESS;
5814 return rval;
5815 } else if (IS_QLA8044(ha)) {
5816 *temp = qla8044_read_temperature(vha);
5817 rval = QLA_SUCCESS;
5818 return rval;
794a5691 5819 }
794a5691 5820
1ae47cf3 5821 rval = qla2x00_read_asic_temperature(vha, temp);
794a5691
AV
5822 return rval;
5823}
5824
a9083016
GM
5825int
5826qla82xx_mbx_intr_enable(scsi_qla_host_t *vha)
5827{
5828 int rval;
5829 struct qla_hw_data *ha = vha->hw;
5830 mbx_cmd_t mc;
5831 mbx_cmd_t *mcp = &mc;
5832
5f28d2d7
SK
5833 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1017,
5834 "Entered %s.\n", __func__);
7c3df132 5835
a9083016
GM
5836 if (!IS_FWI2_CAPABLE(ha))
5837 return QLA_FUNCTION_FAILED;
5838
a9083016 5839 memset(mcp, 0, sizeof(mbx_cmd_t));
3711333d 5840 mcp->mb[0] = MBC_TOGGLE_INTERRUPT;
a9083016
GM
5841 mcp->mb[1] = 1;
5842
5843 mcp->out_mb = MBX_1|MBX_0;
5844 mcp->in_mb = MBX_0;
c314a014 5845 mcp->tov = MBX_TOV_SECONDS;
a9083016
GM
5846 mcp->flags = 0;
5847
5848 rval = qla2x00_mailbox_command(vha, mcp);
5849 if (rval != QLA_SUCCESS) {
7c3df132
SK
5850 ql_dbg(ql_dbg_mbx, vha, 0x1016,
5851 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
a9083016 5852 } else {
5f28d2d7
SK
5853 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x100e,
5854 "Done %s.\n", __func__);
a9083016
GM
5855 }
5856
5857 return rval;
5858}
5859
5860int
5861qla82xx_mbx_intr_disable(scsi_qla_host_t *vha)
5862{
5863 int rval;
5864 struct qla_hw_data *ha = vha->hw;
5865 mbx_cmd_t mc;
5866 mbx_cmd_t *mcp = &mc;
5867
5f28d2d7
SK
5868 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x100d,
5869 "Entered %s.\n", __func__);
7c3df132 5870
7ec0effd 5871 if (!IS_P3P_TYPE(ha))
a9083016
GM
5872 return QLA_FUNCTION_FAILED;
5873
a9083016 5874 memset(mcp, 0, sizeof(mbx_cmd_t));
3711333d 5875 mcp->mb[0] = MBC_TOGGLE_INTERRUPT;
a9083016
GM
5876 mcp->mb[1] = 0;
5877
5878 mcp->out_mb = MBX_1|MBX_0;
5879 mcp->in_mb = MBX_0;
c314a014 5880 mcp->tov = MBX_TOV_SECONDS;
a9083016
GM
5881 mcp->flags = 0;
5882
5883 rval = qla2x00_mailbox_command(vha, mcp);
5884 if (rval != QLA_SUCCESS) {
7c3df132
SK
5885 ql_dbg(ql_dbg_mbx, vha, 0x100c,
5886 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
a9083016 5887 } else {
5f28d2d7
SK
5888 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x100b,
5889 "Done %s.\n", __func__);
a9083016
GM
5890 }
5891
5892 return rval;
5893}
08de2844
GM
5894
5895int
5896qla82xx_md_get_template_size(scsi_qla_host_t *vha)
5897{
5898 struct qla_hw_data *ha = vha->hw;
5899 mbx_cmd_t mc;
5900 mbx_cmd_t *mcp = &mc;
5901 int rval = QLA_FUNCTION_FAILED;
5902
5f28d2d7
SK
5903 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x111f,
5904 "Entered %s.\n", __func__);
08de2844
GM
5905
5906 memset(mcp->mb, 0 , sizeof(mcp->mb));
5907 mcp->mb[0] = LSW(MBC_DIAGNOSTIC_MINIDUMP_TEMPLATE);
5908 mcp->mb[1] = MSW(MBC_DIAGNOSTIC_MINIDUMP_TEMPLATE);
5909 mcp->mb[2] = LSW(RQST_TMPLT_SIZE);
5910 mcp->mb[3] = MSW(RQST_TMPLT_SIZE);
5911
5912 mcp->out_mb = MBX_3|MBX_2|MBX_1|MBX_0;
5913 mcp->in_mb = MBX_14|MBX_13|MBX_12|MBX_11|MBX_10|MBX_9|MBX_8|
5914 MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
5915
5916 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN|IOCTL_CMD;
5917 mcp->tov = MBX_TOV_SECONDS;
5918 rval = qla2x00_mailbox_command(vha, mcp);
5919
5920 /* Always copy back return mailbox values. */
5921 if (rval != QLA_SUCCESS) {
5922 ql_dbg(ql_dbg_mbx, vha, 0x1120,
5923 "mailbox command FAILED=0x%x, subcode=%x.\n",
5924 (mcp->mb[1] << 16) | mcp->mb[0],
5925 (mcp->mb[3] << 16) | mcp->mb[2]);
5926 } else {
5f28d2d7
SK
5927 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1121,
5928 "Done %s.\n", __func__);
08de2844
GM
5929 ha->md_template_size = ((mcp->mb[3] << 16) | mcp->mb[2]);
5930 if (!ha->md_template_size) {
5931 ql_dbg(ql_dbg_mbx, vha, 0x1122,
5932 "Null template size obtained.\n");
5933 rval = QLA_FUNCTION_FAILED;
5934 }
5935 }
5936 return rval;
5937}
5938
5939int
5940qla82xx_md_get_template(scsi_qla_host_t *vha)
5941{
5942 struct qla_hw_data *ha = vha->hw;
5943 mbx_cmd_t mc;
5944 mbx_cmd_t *mcp = &mc;
5945 int rval = QLA_FUNCTION_FAILED;
5946
5f28d2d7
SK
5947 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1123,
5948 "Entered %s.\n", __func__);
08de2844
GM
5949
5950 ha->md_tmplt_hdr = dma_alloc_coherent(&ha->pdev->dev,
5951 ha->md_template_size, &ha->md_tmplt_hdr_dma, GFP_KERNEL);
5952 if (!ha->md_tmplt_hdr) {
5953 ql_log(ql_log_warn, vha, 0x1124,
5954 "Unable to allocate memory for Minidump template.\n");
5955 return rval;
5956 }
5957
5958 memset(mcp->mb, 0 , sizeof(mcp->mb));
5959 mcp->mb[0] = LSW(MBC_DIAGNOSTIC_MINIDUMP_TEMPLATE);
5960 mcp->mb[1] = MSW(MBC_DIAGNOSTIC_MINIDUMP_TEMPLATE);
5961 mcp->mb[2] = LSW(RQST_TMPLT);
5962 mcp->mb[3] = MSW(RQST_TMPLT);
5963 mcp->mb[4] = LSW(LSD(ha->md_tmplt_hdr_dma));
5964 mcp->mb[5] = MSW(LSD(ha->md_tmplt_hdr_dma));
5965 mcp->mb[6] = LSW(MSD(ha->md_tmplt_hdr_dma));
5966 mcp->mb[7] = MSW(MSD(ha->md_tmplt_hdr_dma));
5967 mcp->mb[8] = LSW(ha->md_template_size);
5968 mcp->mb[9] = MSW(ha->md_template_size);
5969
5970 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN|IOCTL_CMD;
5971 mcp->tov = MBX_TOV_SECONDS;
5972 mcp->out_mb = MBX_11|MBX_10|MBX_9|MBX_8|
5973 MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
5974 mcp->in_mb = MBX_3|MBX_2|MBX_1|MBX_0;
5975 rval = qla2x00_mailbox_command(vha, mcp);
5976
5977 if (rval != QLA_SUCCESS) {
5978 ql_dbg(ql_dbg_mbx, vha, 0x1125,
5979 "mailbox command FAILED=0x%x, subcode=%x.\n",
5980 ((mcp->mb[1] << 16) | mcp->mb[0]),
5981 ((mcp->mb[3] << 16) | mcp->mb[2]));
5982 } else
5f28d2d7
SK
5983 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1126,
5984 "Done %s.\n", __func__);
08de2844
GM
5985 return rval;
5986}
999916dc 5987
7ec0effd
AD
5988int
5989qla8044_md_get_template(scsi_qla_host_t *vha)
5990{
5991 struct qla_hw_data *ha = vha->hw;
5992 mbx_cmd_t mc;
5993 mbx_cmd_t *mcp = &mc;
5994 int rval = QLA_FUNCTION_FAILED;
5995 int offset = 0, size = MINIDUMP_SIZE_36K;
bd432bb5 5996
7ec0effd
AD
5997 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0xb11f,
5998 "Entered %s.\n", __func__);
5999
6000 ha->md_tmplt_hdr = dma_alloc_coherent(&ha->pdev->dev,
6001 ha->md_template_size, &ha->md_tmplt_hdr_dma, GFP_KERNEL);
6002 if (!ha->md_tmplt_hdr) {
6003 ql_log(ql_log_warn, vha, 0xb11b,
6004 "Unable to allocate memory for Minidump template.\n");
6005 return rval;
6006 }
6007
6008 memset(mcp->mb, 0 , sizeof(mcp->mb));
6009 while (offset < ha->md_template_size) {
6010 mcp->mb[0] = LSW(MBC_DIAGNOSTIC_MINIDUMP_TEMPLATE);
6011 mcp->mb[1] = MSW(MBC_DIAGNOSTIC_MINIDUMP_TEMPLATE);
6012 mcp->mb[2] = LSW(RQST_TMPLT);
6013 mcp->mb[3] = MSW(RQST_TMPLT);
6014 mcp->mb[4] = LSW(LSD(ha->md_tmplt_hdr_dma + offset));
6015 mcp->mb[5] = MSW(LSD(ha->md_tmplt_hdr_dma + offset));
6016 mcp->mb[6] = LSW(MSD(ha->md_tmplt_hdr_dma + offset));
6017 mcp->mb[7] = MSW(MSD(ha->md_tmplt_hdr_dma + offset));
6018 mcp->mb[8] = LSW(size);
6019 mcp->mb[9] = MSW(size);
6020 mcp->mb[10] = offset & 0x0000FFFF;
6021 mcp->mb[11] = offset & 0xFFFF0000;
6022 mcp->flags = MBX_DMA_OUT|MBX_DMA_IN|IOCTL_CMD;
6023 mcp->tov = MBX_TOV_SECONDS;
6024 mcp->out_mb = MBX_11|MBX_10|MBX_9|MBX_8|
6025 MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
6026 mcp->in_mb = MBX_3|MBX_2|MBX_1|MBX_0;
6027 rval = qla2x00_mailbox_command(vha, mcp);
6028
6029 if (rval != QLA_SUCCESS) {
6030 ql_dbg(ql_dbg_mbx, vha, 0xb11c,
6031 "mailbox command FAILED=0x%x, subcode=%x.\n",
6032 ((mcp->mb[1] << 16) | mcp->mb[0]),
6033 ((mcp->mb[3] << 16) | mcp->mb[2]));
6034 return rval;
6035 } else
6036 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0xb11d,
6037 "Done %s.\n", __func__);
6038 offset = offset + size;
6039 }
6040 return rval;
6041}
6042
6246b8a1
GM
6043int
6044qla81xx_set_led_config(scsi_qla_host_t *vha, uint16_t *led_cfg)
6045{
6046 int rval;
6047 struct qla_hw_data *ha = vha->hw;
6048 mbx_cmd_t mc;
6049 mbx_cmd_t *mcp = &mc;
6050
6051 if (!IS_QLA81XX(ha) && !IS_QLA8031(ha))
6052 return QLA_FUNCTION_FAILED;
6053
5f28d2d7
SK
6054 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1133,
6055 "Entered %s.\n", __func__);
6246b8a1
GM
6056
6057 memset(mcp, 0, sizeof(mbx_cmd_t));
6058 mcp->mb[0] = MBC_SET_LED_CONFIG;
6059 mcp->mb[1] = led_cfg[0];
6060 mcp->mb[2] = led_cfg[1];
6061 if (IS_QLA8031(ha)) {
6062 mcp->mb[3] = led_cfg[2];
6063 mcp->mb[4] = led_cfg[3];
6064 mcp->mb[5] = led_cfg[4];
6065 mcp->mb[6] = led_cfg[5];
6066 }
6067
6068 mcp->out_mb = MBX_2|MBX_1|MBX_0;
6069 if (IS_QLA8031(ha))
6070 mcp->out_mb |= MBX_6|MBX_5|MBX_4|MBX_3;
6071 mcp->in_mb = MBX_0;
c314a014 6072 mcp->tov = MBX_TOV_SECONDS;
6246b8a1
GM
6073 mcp->flags = 0;
6074
6075 rval = qla2x00_mailbox_command(vha, mcp);
6076 if (rval != QLA_SUCCESS) {
6077 ql_dbg(ql_dbg_mbx, vha, 0x1134,
6078 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6079 } else {
5f28d2d7
SK
6080 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1135,
6081 "Done %s.\n", __func__);
6246b8a1
GM
6082 }
6083
6084 return rval;
6085}
6086
6087int
6088qla81xx_get_led_config(scsi_qla_host_t *vha, uint16_t *led_cfg)
6089{
6090 int rval;
6091 struct qla_hw_data *ha = vha->hw;
6092 mbx_cmd_t mc;
6093 mbx_cmd_t *mcp = &mc;
6094
6095 if (!IS_QLA81XX(ha) && !IS_QLA8031(ha))
6096 return QLA_FUNCTION_FAILED;
6097
5f28d2d7
SK
6098 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1136,
6099 "Entered %s.\n", __func__);
6246b8a1
GM
6100
6101 memset(mcp, 0, sizeof(mbx_cmd_t));
6102 mcp->mb[0] = MBC_GET_LED_CONFIG;
6103
6104 mcp->out_mb = MBX_0;
6105 mcp->in_mb = MBX_2|MBX_1|MBX_0;
6106 if (IS_QLA8031(ha))
6107 mcp->in_mb |= MBX_6|MBX_5|MBX_4|MBX_3;
c314a014 6108 mcp->tov = MBX_TOV_SECONDS;
6246b8a1
GM
6109 mcp->flags = 0;
6110
6111 rval = qla2x00_mailbox_command(vha, mcp);
6112 if (rval != QLA_SUCCESS) {
6113 ql_dbg(ql_dbg_mbx, vha, 0x1137,
6114 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6115 } else {
6116 led_cfg[0] = mcp->mb[1];
6117 led_cfg[1] = mcp->mb[2];
6118 if (IS_QLA8031(ha)) {
6119 led_cfg[2] = mcp->mb[3];
6120 led_cfg[3] = mcp->mb[4];
6121 led_cfg[4] = mcp->mb[5];
6122 led_cfg[5] = mcp->mb[6];
6123 }
5f28d2d7
SK
6124 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1138,
6125 "Done %s.\n", __func__);
6246b8a1
GM
6126 }
6127
6128 return rval;
6129}
6130
999916dc
SK
6131int
6132qla82xx_mbx_beacon_ctl(scsi_qla_host_t *vha, int enable)
6133{
6134 int rval;
6135 struct qla_hw_data *ha = vha->hw;
6136 mbx_cmd_t mc;
6137 mbx_cmd_t *mcp = &mc;
6138
7ec0effd 6139 if (!IS_P3P_TYPE(ha))
999916dc
SK
6140 return QLA_FUNCTION_FAILED;
6141
5f28d2d7 6142 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1127,
999916dc
SK
6143 "Entered %s.\n", __func__);
6144
6145 memset(mcp, 0, sizeof(mbx_cmd_t));
6146 mcp->mb[0] = MBC_SET_LED_CONFIG;
6147 if (enable)
6148 mcp->mb[7] = 0xE;
6149 else
6150 mcp->mb[7] = 0xD;
6151
6152 mcp->out_mb = MBX_7|MBX_0;
6153 mcp->in_mb = MBX_0;
6246b8a1 6154 mcp->tov = MBX_TOV_SECONDS;
999916dc
SK
6155 mcp->flags = 0;
6156
6157 rval = qla2x00_mailbox_command(vha, mcp);
6158 if (rval != QLA_SUCCESS) {
6159 ql_dbg(ql_dbg_mbx, vha, 0x1128,
6160 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6161 } else {
5f28d2d7 6162 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1129,
999916dc
SK
6163 "Done %s.\n", __func__);
6164 }
6165
6166 return rval;
6167}
6246b8a1
GM
6168
6169int
7d613ac6 6170qla83xx_wr_reg(scsi_qla_host_t *vha, uint32_t reg, uint32_t data)
6246b8a1
GM
6171{
6172 int rval;
6173 struct qla_hw_data *ha = vha->hw;
6174 mbx_cmd_t mc;
6175 mbx_cmd_t *mcp = &mc;
6176
ecc89f25 6177 if (!IS_QLA83XX(ha) && !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
6246b8a1
GM
6178 return QLA_FUNCTION_FAILED;
6179
5f28d2d7
SK
6180 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1130,
6181 "Entered %s.\n", __func__);
6246b8a1
GM
6182
6183 mcp->mb[0] = MBC_WRITE_REMOTE_REG;
6184 mcp->mb[1] = LSW(reg);
6185 mcp->mb[2] = MSW(reg);
6186 mcp->mb[3] = LSW(data);
6187 mcp->mb[4] = MSW(data);
6188 mcp->out_mb = MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
6189
6190 mcp->in_mb = MBX_1|MBX_0;
6191 mcp->tov = MBX_TOV_SECONDS;
6192 mcp->flags = 0;
6193 rval = qla2x00_mailbox_command(vha, mcp);
6194
6195 if (rval != QLA_SUCCESS) {
6196 ql_dbg(ql_dbg_mbx, vha, 0x1131,
6197 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6198 } else {
5f28d2d7 6199 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1132,
6246b8a1
GM
6200 "Done %s.\n", __func__);
6201 }
af11f64d 6202
6246b8a1
GM
6203 return rval;
6204}
af11f64d
AV
6205
6206int
6207qla2x00_port_logout(scsi_qla_host_t *vha, struct fc_port *fcport)
6208{
6209 int rval;
6210 struct qla_hw_data *ha = vha->hw;
6211 mbx_cmd_t mc;
6212 mbx_cmd_t *mcp = &mc;
6213
6214 if (IS_QLA2100(ha) || IS_QLA2200(ha)) {
5f28d2d7 6215 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x113b,
af11f64d
AV
6216 "Implicit LOGO Unsupported.\n");
6217 return QLA_FUNCTION_FAILED;
6218 }
6219
6220
5f28d2d7
SK
6221 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x113c,
6222 "Entering %s.\n", __func__);
af11f64d
AV
6223
6224 /* Perform Implicit LOGO. */
6225 mcp->mb[0] = MBC_PORT_LOGOUT;
6226 mcp->mb[1] = fcport->loop_id;
6227 mcp->mb[10] = BIT_15;
6228 mcp->out_mb = MBX_10|MBX_1|MBX_0;
6229 mcp->in_mb = MBX_0;
6230 mcp->tov = MBX_TOV_SECONDS;
6231 mcp->flags = 0;
6232 rval = qla2x00_mailbox_command(vha, mcp);
6233 if (rval != QLA_SUCCESS)
6234 ql_dbg(ql_dbg_mbx, vha, 0x113d,
6235 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6236 else
5f28d2d7
SK
6237 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x113e,
6238 "Done %s.\n", __func__);
af11f64d
AV
6239
6240 return rval;
6241}
6242
7d613ac6
SV
6243int
6244qla83xx_rd_reg(scsi_qla_host_t *vha, uint32_t reg, uint32_t *data)
6245{
6246 int rval;
6247 mbx_cmd_t mc;
6248 mbx_cmd_t *mcp = &mc;
6249 struct qla_hw_data *ha = vha->hw;
6250 unsigned long retry_max_time = jiffies + (2 * HZ);
6251
ecc89f25 6252 if (!IS_QLA83XX(ha) && !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
7d613ac6
SV
6253 return QLA_FUNCTION_FAILED;
6254
6255 ql_dbg(ql_dbg_mbx, vha, 0x114b, "Entered %s.\n", __func__);
6256
6257retry_rd_reg:
6258 mcp->mb[0] = MBC_READ_REMOTE_REG;
6259 mcp->mb[1] = LSW(reg);
6260 mcp->mb[2] = MSW(reg);
6261 mcp->out_mb = MBX_2|MBX_1|MBX_0;
6262 mcp->in_mb = MBX_4|MBX_3|MBX_1|MBX_0;
6263 mcp->tov = MBX_TOV_SECONDS;
6264 mcp->flags = 0;
6265 rval = qla2x00_mailbox_command(vha, mcp);
6266
6267 if (rval != QLA_SUCCESS) {
6268 ql_dbg(ql_dbg_mbx, vha, 0x114c,
6269 "Failed=%x mb[0]=%x mb[1]=%x.\n",
6270 rval, mcp->mb[0], mcp->mb[1]);
6271 } else {
6272 *data = (mcp->mb[3] | (mcp->mb[4] << 16));
6273 if (*data == QLA8XXX_BAD_VALUE) {
6274 /*
6275 * During soft-reset CAMRAM register reads might
6276 * return 0xbad0bad0. So retry for MAX of 2 sec
6277 * while reading camram registers.
6278 */
6279 if (time_after(jiffies, retry_max_time)) {
6280 ql_dbg(ql_dbg_mbx, vha, 0x1141,
6281 "Failure to read CAMRAM register. "
6282 "data=0x%x.\n", *data);
6283 return QLA_FUNCTION_FAILED;
6284 }
6285 msleep(100);
6286 goto retry_rd_reg;
6287 }
6288 ql_dbg(ql_dbg_mbx, vha, 0x1142, "Done %s.\n", __func__);
6289 }
6290
6291 return rval;
6292}
6293
6294int
6295qla83xx_restart_nic_firmware(scsi_qla_host_t *vha)
6296{
6297 int rval;
6298 mbx_cmd_t mc;
6299 mbx_cmd_t *mcp = &mc;
6300 struct qla_hw_data *ha = vha->hw;
6301
ecc89f25 6302 if (!IS_QLA83XX(ha))
7d613ac6
SV
6303 return QLA_FUNCTION_FAILED;
6304
6305 ql_dbg(ql_dbg_mbx, vha, 0x1143, "Entered %s.\n", __func__);
6306
6307 mcp->mb[0] = MBC_RESTART_NIC_FIRMWARE;
6308 mcp->out_mb = MBX_0;
6309 mcp->in_mb = MBX_1|MBX_0;
6310 mcp->tov = MBX_TOV_SECONDS;
6311 mcp->flags = 0;
6312 rval = qla2x00_mailbox_command(vha, mcp);
6313
6314 if (rval != QLA_SUCCESS) {
6315 ql_dbg(ql_dbg_mbx, vha, 0x1144,
6316 "Failed=%x mb[0]=%x mb[1]=%x.\n",
6317 rval, mcp->mb[0], mcp->mb[1]);
8ae17876 6318 qla2xxx_dump_fw(vha);
7d613ac6
SV
6319 } else {
6320 ql_dbg(ql_dbg_mbx, vha, 0x1145, "Done %s.\n", __func__);
6321 }
6322
6323 return rval;
6324}
6325
6326int
6327qla83xx_access_control(scsi_qla_host_t *vha, uint16_t options,
6328 uint32_t start_addr, uint32_t end_addr, uint16_t *sector_size)
6329{
6330 int rval;
6331 mbx_cmd_t mc;
6332 mbx_cmd_t *mcp = &mc;
6333 uint8_t subcode = (uint8_t)options;
6334 struct qla_hw_data *ha = vha->hw;
6335
6336 if (!IS_QLA8031(ha))
6337 return QLA_FUNCTION_FAILED;
6338
6339 ql_dbg(ql_dbg_mbx, vha, 0x1146, "Entered %s.\n", __func__);
6340
6341 mcp->mb[0] = MBC_SET_ACCESS_CONTROL;
6342 mcp->mb[1] = options;
6343 mcp->out_mb = MBX_1|MBX_0;
6344 if (subcode & BIT_2) {
6345 mcp->mb[2] = LSW(start_addr);
6346 mcp->mb[3] = MSW(start_addr);
6347 mcp->mb[4] = LSW(end_addr);
6348 mcp->mb[5] = MSW(end_addr);
6349 mcp->out_mb |= MBX_5|MBX_4|MBX_3|MBX_2;
6350 }
6351 mcp->in_mb = MBX_2|MBX_1|MBX_0;
6352 if (!(subcode & (BIT_2 | BIT_5)))
6353 mcp->in_mb |= MBX_4|MBX_3;
6354 mcp->tov = MBX_TOV_SECONDS;
6355 mcp->flags = 0;
6356 rval = qla2x00_mailbox_command(vha, mcp);
6357
6358 if (rval != QLA_SUCCESS) {
6359 ql_dbg(ql_dbg_mbx, vha, 0x1147,
6360 "Failed=%x mb[0]=%x mb[1]=%x mb[2]=%x mb[3]=%x mb[4]=%x.\n",
6361 rval, mcp->mb[0], mcp->mb[1], mcp->mb[2], mcp->mb[3],
6362 mcp->mb[4]);
8ae17876 6363 qla2xxx_dump_fw(vha);
7d613ac6
SV
6364 } else {
6365 if (subcode & BIT_5)
6366 *sector_size = mcp->mb[1];
6367 else if (subcode & (BIT_6 | BIT_7)) {
6368 ql_dbg(ql_dbg_mbx, vha, 0x1148,
6369 "Driver-lock id=%x%x", mcp->mb[4], mcp->mb[3]);
6370 } else if (subcode & (BIT_3 | BIT_4)) {
6371 ql_dbg(ql_dbg_mbx, vha, 0x1149,
6372 "Flash-lock id=%x%x", mcp->mb[4], mcp->mb[3]);
6373 }
6374 ql_dbg(ql_dbg_mbx, vha, 0x114a, "Done %s.\n", __func__);
6375 }
6376
6377 return rval;
6378}
81178772
SK
6379
6380int
6381qla2x00_dump_mctp_data(scsi_qla_host_t *vha, dma_addr_t req_dma, uint32_t addr,
6382 uint32_t size)
6383{
6384 int rval;
6385 mbx_cmd_t mc;
6386 mbx_cmd_t *mcp = &mc;
6387
6388 if (!IS_MCTP_CAPABLE(vha->hw))
6389 return QLA_FUNCTION_FAILED;
6390
6391 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x114f,
6392 "Entered %s.\n", __func__);
6393
6394 mcp->mb[0] = MBC_DUMP_RISC_RAM_EXTENDED;
6395 mcp->mb[1] = LSW(addr);
6396 mcp->mb[2] = MSW(req_dma);
6397 mcp->mb[3] = LSW(req_dma);
6398 mcp->mb[4] = MSW(size);
6399 mcp->mb[5] = LSW(size);
6400 mcp->mb[6] = MSW(MSD(req_dma));
6401 mcp->mb[7] = LSW(MSD(req_dma));
6402 mcp->mb[8] = MSW(addr);
6403 /* Setting RAM ID to valid */
81178772 6404 /* For MCTP RAM ID is 0x40 */
641e0efd 6405 mcp->mb[10] = BIT_7 | 0x40;
81178772
SK
6406
6407 mcp->out_mb |= MBX_10|MBX_8|MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|
6408 MBX_0;
6409
6410 mcp->in_mb = MBX_0;
6411 mcp->tov = MBX_TOV_SECONDS;
6412 mcp->flags = 0;
6413 rval = qla2x00_mailbox_command(vha, mcp);
6414
6415 if (rval != QLA_SUCCESS) {
6416 ql_dbg(ql_dbg_mbx, vha, 0x114e,
6417 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6418 } else {
6419 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x114d,
6420 "Done %s.\n", __func__);
6421 }
6422
6423 return rval;
6424}
ec891462
JC
6425
6426int
6427qla26xx_dport_diagnostics(scsi_qla_host_t *vha,
6428 void *dd_buf, uint size, uint options)
6429{
6430 int rval;
6431 mbx_cmd_t mc;
6432 mbx_cmd_t *mcp = &mc;
6433 dma_addr_t dd_dma;
6434
ecc89f25
JC
6435 if (!IS_QLA83XX(vha->hw) && !IS_QLA27XX(vha->hw) &&
6436 !IS_QLA28XX(vha->hw))
ec891462
JC
6437 return QLA_FUNCTION_FAILED;
6438
83548fe2 6439 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x119f,
ec891462
JC
6440 "Entered %s.\n", __func__);
6441
ec891462
JC
6442 dd_dma = dma_map_single(&vha->hw->pdev->dev,
6443 dd_buf, size, DMA_FROM_DEVICE);
0b2ce198 6444 if (dma_mapping_error(&vha->hw->pdev->dev, dd_dma)) {
ec891462
JC
6445 ql_log(ql_log_warn, vha, 0x1194, "Failed to map dma buffer.\n");
6446 return QLA_MEMORY_ALLOC_FAILED;
6447 }
6448
6449 memset(dd_buf, 0, size);
6450
6451 mcp->mb[0] = MBC_DPORT_DIAGNOSTICS;
6452 mcp->mb[1] = options;
6453 mcp->mb[2] = MSW(LSD(dd_dma));
6454 mcp->mb[3] = LSW(LSD(dd_dma));
6455 mcp->mb[6] = MSW(MSD(dd_dma));
6456 mcp->mb[7] = LSW(MSD(dd_dma));
6457 mcp->mb[8] = size;
6458 mcp->out_mb = MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_1|MBX_0;
6459 mcp->in_mb = MBX_3|MBX_2|MBX_1|MBX_0;
6460 mcp->buf_size = size;
6461 mcp->flags = MBX_DMA_IN;
6462 mcp->tov = MBX_TOV_SECONDS * 4;
6463 rval = qla2x00_mailbox_command(vha, mcp);
6464
6465 if (rval != QLA_SUCCESS) {
6466 ql_dbg(ql_dbg_mbx, vha, 0x1195, "Failed=%x.\n", rval);
6467 } else {
6468 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1196,
6469 "Done %s.\n", __func__);
6470 }
6471
6472 dma_unmap_single(&vha->hw->pdev->dev, dd_dma,
6473 size, DMA_FROM_DEVICE);
6474
6475 return rval;
6476}
15f30a57 6477
476da8fa
BH
6478int
6479qla26xx_dport_diagnostics_v2(scsi_qla_host_t *vha,
6480 struct qla_dport_diag_v2 *dd, mbx_cmd_t *mcp)
6481{
6482 int rval;
6483 dma_addr_t dd_dma;
6484 uint size = sizeof(dd->buf);
6485 uint16_t options = dd->options;
6486
6487 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x119f,
6488 "Entered %s.\n", __func__);
6489
6490 dd_dma = dma_map_single(&vha->hw->pdev->dev,
6491 dd->buf, size, DMA_FROM_DEVICE);
6492 if (dma_mapping_error(&vha->hw->pdev->dev, dd_dma)) {
6493 ql_log(ql_log_warn, vha, 0x1194,
6494 "Failed to map dma buffer.\n");
6495 return QLA_MEMORY_ALLOC_FAILED;
6496 }
6497
6498 memset(dd->buf, 0, size);
6499
6500 mcp->mb[0] = MBC_DPORT_DIAGNOSTICS;
6501 mcp->mb[1] = options;
6502 mcp->mb[2] = MSW(LSD(dd_dma));
6503 mcp->mb[3] = LSW(LSD(dd_dma));
6504 mcp->mb[6] = MSW(MSD(dd_dma));
6505 mcp->mb[7] = LSW(MSD(dd_dma));
6506 mcp->mb[8] = size;
6507 mcp->out_mb = MBX_8 | MBX_7 | MBX_6 | MBX_3 | MBX_2 | MBX_1 | MBX_0;
6508 mcp->in_mb = MBX_3 | MBX_2 | MBX_1 | MBX_0;
6509 mcp->buf_size = size;
6510 mcp->flags = MBX_DMA_IN;
6511 mcp->tov = MBX_TOV_SECONDS * 4;
6512 rval = qla2x00_mailbox_command(vha, mcp);
6513
6514 if (rval != QLA_SUCCESS) {
6515 ql_dbg(ql_dbg_mbx, vha, 0x1195, "Failed=%x.\n", rval);
6516 } else {
6517 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1196,
6518 "Done %s.\n", __func__);
6519 }
6520
6521 dma_unmap_single(&vha->hw->pdev->dev, dd_dma, size, DMA_FROM_DEVICE);
6522
6523 return rval;
6524}
6525
6c18a43e 6526static void qla2x00_async_mb_sp_done(srb_t *sp, int res)
15f30a57 6527{
15f30a57
QT
6528 sp->u.iocb_cmd.u.mbx.rc = res;
6529
6530 complete(&sp->u.iocb_cmd.u.mbx.comp);
6531 /* don't free sp here. Let the caller do the free */
6532}
6533
6534/*
6535 * This mailbox uses the iocb interface to send MB command.
6536 * This allows non-critial (non chip setup) command to go
6537 * out in parrallel.
6538 */
6539int qla24xx_send_mb_cmd(struct scsi_qla_host *vha, mbx_cmd_t *mcp)
6540{
6541 int rval = QLA_FUNCTION_FAILED;
6542 srb_t *sp;
6543 struct srb_iocb *c;
6544
6545 if (!vha->hw->flags.fw_started)
6546 goto done;
6547
31e6cdbe 6548 /* ref: INIT */
15f30a57
QT
6549 sp = qla2x00_get_sp(vha, NULL, GFP_KERNEL);
6550 if (!sp)
6551 goto done;
6552
15f30a57 6553 c = &sp->u.iocb_cmd;
15f30a57
QT
6554 init_completion(&c->u.mbx.comp);
6555
d4523bd6
DW
6556 sp->type = SRB_MB_IOCB;
6557 sp->name = mb_to_str(mcp->mb[0]);
6558 qla2x00_init_async_sp(sp, qla2x00_get_async_timeout(vha) + 2,
6559 qla2x00_async_mb_sp_done);
e74e7d95
BH
6560
6561 memcpy(sp->u.iocb_cmd.u.mbx.out_mb, mcp->mb, SIZEOF_IOCB_MB_REG);
6562
15f30a57
QT
6563 rval = qla2x00_start_sp(sp);
6564 if (rval != QLA_SUCCESS) {
83548fe2 6565 ql_dbg(ql_dbg_mbx, vha, 0x1018,
15f30a57
QT
6566 "%s: %s Failed submission. %x.\n",
6567 __func__, sp->name, rval);
6568 goto done_free_sp;
6569 }
6570
83548fe2 6571 ql_dbg(ql_dbg_mbx, vha, 0x113f, "MB:%s hndl %x submitted\n",
15f30a57
QT
6572 sp->name, sp->handle);
6573
6574 wait_for_completion(&c->u.mbx.comp);
6575 memcpy(mcp->mb, sp->u.iocb_cmd.u.mbx.in_mb, SIZEOF_IOCB_MB_REG);
6576
6577 rval = c->u.mbx.rc;
6578 switch (rval) {
6579 case QLA_FUNCTION_TIMEOUT:
83548fe2 6580 ql_dbg(ql_dbg_mbx, vha, 0x1140, "%s: %s Timeout. %x.\n",
15f30a57
QT
6581 __func__, sp->name, rval);
6582 break;
6583 case QLA_SUCCESS:
83548fe2 6584 ql_dbg(ql_dbg_mbx, vha, 0x119d, "%s: %s done.\n",
15f30a57 6585 __func__, sp->name);
15f30a57
QT
6586 break;
6587 default:
83548fe2 6588 ql_dbg(ql_dbg_mbx, vha, 0x119e, "%s: %s Failed. %x.\n",
15f30a57 6589 __func__, sp->name, rval);
15f30a57
QT
6590 break;
6591 }
6592
15f30a57 6593done_free_sp:
31e6cdbe
SK
6594 /* ref: INIT */
6595 kref_put(&sp->cmd_kref, qla2x00_sp_release);
15f30a57
QT
6596done:
6597 return rval;
6598}
6599
6600/*
6601 * qla24xx_gpdb_wait
6602 * NOTE: Do not call this routine from DPC thread
6603 */
6604int qla24xx_gpdb_wait(struct scsi_qla_host *vha, fc_port_t *fcport, u8 opt)
6605{
6606 int rval = QLA_FUNCTION_FAILED;
6607 dma_addr_t pd_dma;
6608 struct port_database_24xx *pd;
6609 struct qla_hw_data *ha = vha->hw;
6610 mbx_cmd_t mc;
6611
6612 if (!vha->hw->flags.fw_started)
6613 goto done;
6614
08eb7f45 6615 pd = dma_pool_zalloc(ha->s_dma_pool, GFP_KERNEL, &pd_dma);
15f30a57 6616 if (pd == NULL) {
83548fe2
QT
6617 ql_log(ql_log_warn, vha, 0xd047,
6618 "Failed to allocate port database structure.\n");
15f30a57
QT
6619 goto done_free_sp;
6620 }
15f30a57
QT
6621
6622 memset(&mc, 0, sizeof(mc));
6623 mc.mb[0] = MBC_GET_PORT_DATABASE;
7ffa5b93 6624 mc.mb[1] = fcport->loop_id;
15f30a57
QT
6625 mc.mb[2] = MSW(pd_dma);
6626 mc.mb[3] = LSW(pd_dma);
6627 mc.mb[6] = MSW(MSD(pd_dma));
6628 mc.mb[7] = LSW(MSD(pd_dma));
7ffa5b93
BVA
6629 mc.mb[9] = vha->vp_idx;
6630 mc.mb[10] = opt;
15f30a57
QT
6631
6632 rval = qla24xx_send_mb_cmd(vha, &mc);
6633 if (rval != QLA_SUCCESS) {
83548fe2 6634 ql_dbg(ql_dbg_mbx, vha, 0x1193,
15f30a57
QT
6635 "%s: %8phC fail\n", __func__, fcport->port_name);
6636 goto done_free_sp;
6637 }
6638
6639 rval = __qla24xx_parse_gpdb(vha, fcport, pd);
6640
83548fe2 6641 ql_dbg(ql_dbg_mbx, vha, 0x1197, "%s: %8phC done\n",
15f30a57
QT
6642 __func__, fcport->port_name);
6643
6644done_free_sp:
6645 if (pd)
6646 dma_pool_free(ha->s_dma_pool, pd, pd_dma);
6647done:
6648 return rval;
6649}
6650
6651int __qla24xx_parse_gpdb(struct scsi_qla_host *vha, fc_port_t *fcport,
6652 struct port_database_24xx *pd)
6653{
6654 int rval = QLA_SUCCESS;
6655 uint64_t zero = 0;
a5d42f4c
DG
6656 u8 current_login_state, last_login_state;
6657
84ed362a 6658 if (NVME_TARGET(vha->hw, fcport)) {
a5d42f4c
DG
6659 current_login_state = pd->current_login_state >> 4;
6660 last_login_state = pd->last_login_state >> 4;
6661 } else {
6662 current_login_state = pd->current_login_state & 0xf;
6663 last_login_state = pd->last_login_state & 0xf;
6664 }
15f30a57
QT
6665
6666 /* Check for logged in state. */
23c64559 6667 if (current_login_state != PDS_PRLI_COMPLETE) {
83548fe2
QT
6668 ql_dbg(ql_dbg_mbx, vha, 0x119a,
6669 "Unable to verify login-state (%x/%x) for loop_id %x.\n",
a5d42f4c 6670 current_login_state, last_login_state, fcport->loop_id);
15f30a57
QT
6671 rval = QLA_FUNCTION_FAILED;
6672 goto gpd_error_out;
6673 }
6674
6675 if (fcport->loop_id == FC_NO_LOOP_ID ||
6676 (memcmp(fcport->port_name, (uint8_t *)&zero, 8) &&
6677 memcmp(fcport->port_name, pd->port_name, 8))) {
6678 /* We lost the device mid way. */
6679 rval = QLA_NOT_LOGGED_IN;
6680 goto gpd_error_out;
6681 }
6682
6683 /* Names are little-endian. */
6684 memcpy(fcport->node_name, pd->node_name, WWN_SIZE);
6685 memcpy(fcport->port_name, pd->port_name, WWN_SIZE);
6686
6687 /* Get port_id of device. */
6688 fcport->d_id.b.domain = pd->port_id[0];
6689 fcport->d_id.b.area = pd->port_id[1];
6690 fcport->d_id.b.al_pa = pd->port_id[2];
6691 fcport->d_id.b.rsvd_1 = 0;
6692
9efea843
QT
6693 ql_dbg(ql_dbg_disc, vha, 0x2062,
6694 "%8phC SVC Param w3 %02x%02x",
6695 fcport->port_name,
6696 pd->prli_svc_param_word_3[1],
6697 pd->prli_svc_param_word_3[0]);
6698
84ed362a
MH
6699 if (NVME_TARGET(vha->hw, fcport)) {
6700 fcport->port_type = FCT_NVME;
a6a6d058
HR
6701 if ((pd->prli_svc_param_word_3[0] & BIT_5) == 0)
6702 fcport->port_type |= FCT_NVME_INITIATOR;
6703 if ((pd->prli_svc_param_word_3[0] & BIT_4) == 0)
6704 fcport->port_type |= FCT_NVME_TARGET;
6705 if ((pd->prli_svc_param_word_3[0] & BIT_3) == 0)
6706 fcport->port_type |= FCT_NVME_DISCOVERY;
a5d42f4c
DG
6707 } else {
6708 /* If not target must be initiator or unknown type. */
6709 if ((pd->prli_svc_param_word_3[0] & BIT_4) == 0)
6710 fcport->port_type = FCT_INITIATOR;
6711 else
6712 fcport->port_type = FCT_TARGET;
6713 }
15f30a57
QT
6714 /* Passback COS information. */
6715 fcport->supported_classes = (pd->flags & PDF_CLASS_2) ?
6716 FC_COS_CLASS2 : FC_COS_CLASS3;
6717
6718 if (pd->prli_svc_param_word_3[0] & BIT_7) {
6719 fcport->flags |= FCF_CONF_COMP_SUPPORTED;
6720 fcport->conf_compl_supported = 1;
6721 }
6722
6723gpd_error_out:
6724 return rval;
6725}
6726
6727/*
6728 * qla24xx_gidlist__wait
6729 * NOTE: don't call this routine from DPC thread.
6730 */
6731int qla24xx_gidlist_wait(struct scsi_qla_host *vha,
6732 void *id_list, dma_addr_t id_list_dma, uint16_t *entries)
6733{
6734 int rval = QLA_FUNCTION_FAILED;
6735 mbx_cmd_t mc;
6736
6737 if (!vha->hw->flags.fw_started)
6738 goto done;
6739
6740 memset(&mc, 0, sizeof(mc));
6741 mc.mb[0] = MBC_GET_ID_LIST;
6742 mc.mb[2] = MSW(id_list_dma);
6743 mc.mb[3] = LSW(id_list_dma);
6744 mc.mb[6] = MSW(MSD(id_list_dma));
6745 mc.mb[7] = LSW(MSD(id_list_dma));
6746 mc.mb[8] = 0;
7ffa5b93 6747 mc.mb[9] = vha->vp_idx;
15f30a57
QT
6748
6749 rval = qla24xx_send_mb_cmd(vha, &mc);
6750 if (rval != QLA_SUCCESS) {
83548fe2
QT
6751 ql_dbg(ql_dbg_mbx, vha, 0x119b,
6752 "%s: fail\n", __func__);
15f30a57
QT
6753 } else {
6754 *entries = mc.mb[1];
83548fe2
QT
6755 ql_dbg(ql_dbg_mbx, vha, 0x119c,
6756 "%s: done\n", __func__);
15f30a57
QT
6757 }
6758done:
6759 return rval;
6760}
deeae7a6
DG
6761
6762int qla27xx_set_zio_threshold(scsi_qla_host_t *vha, uint16_t value)
6763{
6764 int rval;
6765 mbx_cmd_t mc;
6766 mbx_cmd_t *mcp = &mc;
6767
6768 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1200,
6769 "Entered %s\n", __func__);
6770
6771 memset(mcp->mb, 0 , sizeof(mcp->mb));
6772 mcp->mb[0] = MBC_GET_SET_ZIO_THRESHOLD;
7ffa5b93
BVA
6773 mcp->mb[1] = 1;
6774 mcp->mb[2] = value;
deeae7a6
DG
6775 mcp->out_mb = MBX_2 | MBX_1 | MBX_0;
6776 mcp->in_mb = MBX_2 | MBX_0;
6777 mcp->tov = MBX_TOV_SECONDS;
6778 mcp->flags = 0;
6779
6780 rval = qla2x00_mailbox_command(vha, mcp);
6781
6782 ql_dbg(ql_dbg_mbx, vha, 0x1201, "%s %x\n",
6783 (rval != QLA_SUCCESS) ? "Failed" : "Done", rval);
6784
6785 return rval;
6786}
6787
6788int qla27xx_get_zio_threshold(scsi_qla_host_t *vha, uint16_t *value)
6789{
6790 int rval;
6791 mbx_cmd_t mc;
6792 mbx_cmd_t *mcp = &mc;
6793
6794 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x1203,
6795 "Entered %s\n", __func__);
6796
6797 memset(mcp->mb, 0, sizeof(mcp->mb));
6798 mcp->mb[0] = MBC_GET_SET_ZIO_THRESHOLD;
7ffa5b93 6799 mcp->mb[1] = 0;
deeae7a6
DG
6800 mcp->out_mb = MBX_1 | MBX_0;
6801 mcp->in_mb = MBX_2 | MBX_0;
6802 mcp->tov = MBX_TOV_SECONDS;
6803 mcp->flags = 0;
6804
6805 rval = qla2x00_mailbox_command(vha, mcp);
6806 if (rval == QLA_SUCCESS)
6807 *value = mc.mb[2];
6808
6809 ql_dbg(ql_dbg_mbx, vha, 0x1205, "%s %x\n",
6810 (rval != QLA_SUCCESS) ? "Failed" : "Done", rval);
6811
6812 return rval;
6813}
e4e3a2ce
QT
6814
6815int
6816qla2x00_read_sfp_dev(struct scsi_qla_host *vha, char *buf, int count)
6817{
6818 struct qla_hw_data *ha = vha->hw;
6819 uint16_t iter, addr, offset;
6820 dma_addr_t phys_addr;
6821 int rval, c;
6822 u8 *sfp_data;
6823
6824 memset(ha->sfp_data, 0, SFP_DEV_SIZE);
6825 addr = 0xa0;
6826 phys_addr = ha->sfp_data_dma;
6827 sfp_data = ha->sfp_data;
6828 offset = c = 0;
6829
6830 for (iter = 0; iter < SFP_DEV_SIZE / SFP_BLOCK_SIZE; iter++) {
6831 if (iter == 4) {
6832 /* Skip to next device address. */
6833 addr = 0xa2;
6834 offset = 0;
6835 }
6836
6837 rval = qla2x00_read_sfp(vha, phys_addr, sfp_data,
6838 addr, offset, SFP_BLOCK_SIZE, BIT_1);
6839 if (rval != QLA_SUCCESS) {
6840 ql_log(ql_log_warn, vha, 0x706d,
6841 "Unable to read SFP data (%x/%x/%x).\n", rval,
6842 addr, offset);
6843
6844 return rval;
6845 }
6846
6847 if (buf && (c < count)) {
6848 u16 sz;
6849
6850 if ((count - c) >= SFP_BLOCK_SIZE)
6851 sz = SFP_BLOCK_SIZE;
6852 else
6853 sz = count - c;
6854
6855 memcpy(buf, sfp_data, sz);
6856 buf += SFP_BLOCK_SIZE;
6857 c += sz;
6858 }
6859 phys_addr += SFP_BLOCK_SIZE;
6860 sfp_data += SFP_BLOCK_SIZE;
6861 offset += SFP_BLOCK_SIZE;
6862 }
6863
6864 return rval;
6865}
94d83e36
QT
6866
6867int qla24xx_res_count_wait(struct scsi_qla_host *vha,
6868 uint16_t *out_mb, int out_mb_sz)
6869{
6870 int rval = QLA_FUNCTION_FAILED;
6871 mbx_cmd_t mc;
6872
6873 if (!vha->hw->flags.fw_started)
6874 goto done;
6875
6876 memset(&mc, 0, sizeof(mc));
6877 mc.mb[0] = MBC_GET_RESOURCE_COUNTS;
6878
6879 rval = qla24xx_send_mb_cmd(vha, &mc);
6880 if (rval != QLA_SUCCESS) {
6881 ql_dbg(ql_dbg_mbx, vha, 0xffff,
6882 "%s: fail\n", __func__);
6883 } else {
6884 if (out_mb_sz <= SIZEOF_IOCB_MB_REG)
6885 memcpy(out_mb, mc.mb, out_mb_sz);
6886 else
6887 memcpy(out_mb, mc.mb, SIZEOF_IOCB_MB_REG);
6888
6889 ql_dbg(ql_dbg_mbx, vha, 0xffff,
6890 "%s: done\n", __func__);
6891 }
6892done:
6893 return rval;
6894}
3f006ac3
MH
6895
6896int qla28xx_secure_flash_update(scsi_qla_host_t *vha, uint16_t opts,
6897 uint16_t region, uint32_t len, dma_addr_t sfub_dma_addr,
6898 uint32_t sfub_len)
6899{
6900 int rval;
6901 mbx_cmd_t mc;
6902 mbx_cmd_t *mcp = &mc;
6903
6904 mcp->mb[0] = MBC_SECURE_FLASH_UPDATE;
6905 mcp->mb[1] = opts;
6906 mcp->mb[2] = region;
6907 mcp->mb[3] = MSW(len);
6908 mcp->mb[4] = LSW(len);
6909 mcp->mb[5] = MSW(sfub_dma_addr);
6910 mcp->mb[6] = LSW(sfub_dma_addr);
6911 mcp->mb[7] = MSW(MSD(sfub_dma_addr));
6912 mcp->mb[8] = LSW(MSD(sfub_dma_addr));
6913 mcp->mb[9] = sfub_len;
6914 mcp->out_mb =
6915 MBX_9|MBX_8|MBX_7|MBX_6|MBX_5|MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
6916 mcp->in_mb = MBX_2|MBX_1|MBX_0;
6917 mcp->tov = MBX_TOV_SECONDS;
6918 mcp->flags = 0;
6919 rval = qla2x00_mailbox_command(vha, mcp);
6920
6921 if (rval != QLA_SUCCESS) {
6922 ql_dbg(ql_dbg_mbx, vha, 0xffff, "%s(%ld): failed rval 0x%x, %x %x %x",
6923 __func__, vha->host_no, rval, mcp->mb[0], mcp->mb[1],
6924 mcp->mb[2]);
6925 }
6926
6927 return rval;
6928}
6929
6930int qla2xxx_write_remote_register(scsi_qla_host_t *vha, uint32_t addr,
6931 uint32_t data)
6932{
6933 int rval;
6934 mbx_cmd_t mc;
6935 mbx_cmd_t *mcp = &mc;
6936
6937 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e8,
6938 "Entered %s.\n", __func__);
6939
6940 mcp->mb[0] = MBC_WRITE_REMOTE_REG;
6941 mcp->mb[1] = LSW(addr);
6942 mcp->mb[2] = MSW(addr);
6943 mcp->mb[3] = LSW(data);
6944 mcp->mb[4] = MSW(data);
6945 mcp->out_mb = MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
6946 mcp->in_mb = MBX_1|MBX_0;
6947 mcp->tov = MBX_TOV_SECONDS;
6948 mcp->flags = 0;
6949 rval = qla2x00_mailbox_command(vha, mcp);
6950
6951 if (rval != QLA_SUCCESS) {
6952 ql_dbg(ql_dbg_mbx, vha, 0x10e9,
6953 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6954 } else {
6955 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ea,
6956 "Done %s.\n", __func__);
6957 }
6958
6959 return rval;
6960}
6961
6962int qla2xxx_read_remote_register(scsi_qla_host_t *vha, uint32_t addr,
6963 uint32_t *data)
6964{
6965 int rval;
6966 mbx_cmd_t mc;
6967 mbx_cmd_t *mcp = &mc;
6968
6969 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10e8,
6970 "Entered %s.\n", __func__);
6971
6972 mcp->mb[0] = MBC_READ_REMOTE_REG;
6973 mcp->mb[1] = LSW(addr);
6974 mcp->mb[2] = MSW(addr);
6975 mcp->out_mb = MBX_2|MBX_1|MBX_0;
6976 mcp->in_mb = MBX_4|MBX_3|MBX_2|MBX_1|MBX_0;
6977 mcp->tov = MBX_TOV_SECONDS;
6978 mcp->flags = 0;
6979 rval = qla2x00_mailbox_command(vha, mcp);
6980
6981 *data = (uint32_t)((((uint32_t)mcp->mb[4]) << 16) | mcp->mb[3]);
6982
6983 if (rval != QLA_SUCCESS) {
6984 ql_dbg(ql_dbg_mbx, vha, 0x10e9,
6985 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
6986 } else {
6987 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0x10ea,
6988 "Done %s.\n", __func__);
6989 }
6990
6991 return rval;
6992}
07553b1e
JC
6993
6994int
6995ql26xx_led_config(scsi_qla_host_t *vha, uint16_t options, uint16_t *led)
6996{
6997 struct qla_hw_data *ha = vha->hw;
6998 mbx_cmd_t mc;
6999 mbx_cmd_t *mcp = &mc;
7000 int rval;
7001
7002 if (!IS_QLA2031(ha) && !IS_QLA27XX(ha) && !IS_QLA28XX(ha))
7003 return QLA_FUNCTION_FAILED;
7004
7005 ql_dbg(ql_dbg_mbx, vha, 0x7070, "Entered %s (options=%x).\n",
7006 __func__, options);
7007
7008 mcp->mb[0] = MBC_SET_GET_FC_LED_CONFIG;
7009 mcp->mb[1] = options;
7010 mcp->out_mb = MBX_1|MBX_0;
7011 mcp->in_mb = MBX_1|MBX_0;
7012 if (options & BIT_0) {
7013 if (options & BIT_1) {
7014 mcp->mb[2] = led[2];
7015 mcp->out_mb |= MBX_2;
7016 }
7017 if (options & BIT_2) {
7018 mcp->mb[3] = led[0];
7019 mcp->out_mb |= MBX_3;
7020 }
7021 if (options & BIT_3) {
7022 mcp->mb[4] = led[1];
7023 mcp->out_mb |= MBX_4;
7024 }
7025 } else {
7026 mcp->in_mb |= MBX_4|MBX_3|MBX_2;
7027 }
7028 mcp->tov = MBX_TOV_SECONDS;
7029 mcp->flags = 0;
7030 rval = qla2x00_mailbox_command(vha, mcp);
7031 if (rval) {
7032 ql_dbg(ql_dbg_mbx, vha, 0x7071, "Failed %s %x (mb=%x,%x)\n",
7033 __func__, rval, mcp->mb[0], mcp->mb[1]);
7034 return rval;
7035 }
7036
7037 if (options & BIT_0) {
7038 ha->beacon_blink_led = 0;
7039 ql_dbg(ql_dbg_mbx, vha, 0x7072, "Done %s\n", __func__);
7040 } else {
7041 led[2] = mcp->mb[2];
7042 led[0] = mcp->mb[3];
7043 led[1] = mcp->mb[4];
7044 ql_dbg(ql_dbg_mbx, vha, 0x7073, "Done %s (led=%x,%x,%x)\n",
7045 __func__, led[0], led[1], led[2]);
7046 }
7047
7048 return rval;
7049}
d94d8158
QT
7050
7051/**
7052 * qla_no_op_mb(): This MB is used to check if FW is still alive and
7053 * able to generate an interrupt. Otherwise, a timeout will trigger
7054 * FW dump + reset
7055 * @vha: host adapter pointer
7056 * Return: None
7057 */
7058void qla_no_op_mb(struct scsi_qla_host *vha)
7059{
7060 mbx_cmd_t mc;
7061 mbx_cmd_t *mcp = &mc;
7062 int rval;
7063
7064 memset(&mc, 0, sizeof(mc));
7065 mcp->mb[0] = 0; // noop cmd= 0
7066 mcp->out_mb = MBX_0;
7067 mcp->in_mb = MBX_0;
7068 mcp->tov = 5;
7069 mcp->flags = 0;
7070 rval = qla2x00_mailbox_command(vha, mcp);
7071
7072 if (rval) {
7073 ql_dbg(ql_dbg_async, vha, 0x7071,
7074 "Failed %s %x\n", __func__, rval);
7075 }
7076}
9e1c3206
BH
7077
7078int qla_mailbox_passthru(scsi_qla_host_t *vha,
7079 uint16_t *mbx_in, uint16_t *mbx_out)
7080{
7081 mbx_cmd_t mc;
7082 mbx_cmd_t *mcp = &mc;
7083 int rval = -EINVAL;
7084
7085 memset(&mc, 0, sizeof(mc));
7086 /* Receiving all 32 register's contents */
7087 memcpy(&mcp->mb, (char *)mbx_in, (32 * sizeof(uint16_t)));
7088
7089 mcp->out_mb = 0xFFFFFFFF;
7090 mcp->in_mb = 0xFFFFFFFF;
7091
7092 mcp->tov = MBX_TOV_SECONDS;
7093 mcp->flags = 0;
7094 mcp->bufp = NULL;
7095
7096 rval = qla2x00_mailbox_command(vha, mcp);
7097
7098 if (rval != QLA_SUCCESS) {
7099 ql_dbg(ql_dbg_mbx, vha, 0xf0a2,
7100 "Failed=%x mb[0]=%x.\n", rval, mcp->mb[0]);
7101 } else {
7102 ql_dbg(ql_dbg_mbx + ql_dbg_verbose, vha, 0xf0a3, "Done %s.\n",
7103 __func__);
7104 /* passing all 32 register's contents */
7105 memcpy(mbx_out, &mcp->mb, 32 * sizeof(uint16_t));
7106 }
7107
7108 return rval;
7109}