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