xref: /linux/drivers/scsi/bfa/bfa_fcs_lport.c (revision 0883c2c06fb5bcf5b9e008270827e63c09a88c1e)
1 /*
2  * Copyright (c) 2005-2014 Brocade Communications Systems, Inc.
3  * Copyright (c) 2014- QLogic Corporation.
4  * All rights reserved
5  * www.qlogic.com
6  *
7  * Linux driver for QLogic BR-series Fibre Channel Host Bus Adapter.
8  *
9  * This program is free software; you can redistribute it and/or modify it
10  * under the terms of the GNU General Public License (GPL) Version 2 as
11  * published by the Free Software Foundation
12  *
13  * This program is distributed in the hope that it will be useful, but
14  * WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * General Public License for more details.
17  */
18 
19 #include "bfad_drv.h"
20 #include "bfad_im.h"
21 #include "bfa_fcs.h"
22 #include "bfa_fcbuild.h"
23 #include "bfa_fc.h"
24 
25 BFA_TRC_FILE(FCS, PORT);
26 
27 /*
28  * ALPA to LIXA bitmap mapping
29  *
30  * ALPA 0x00 (Word 0, Bit 30) is invalid for N_Ports. Also Word 0 Bit 31
31  * is for L_bit (login required) and is filled as ALPA 0x00 here.
32  */
33 static const u8 loop_alpa_map[] = {
34 	0x00, 0x00, 0x01, 0x02, 0x04, 0x08, 0x0F, 0x10, /* Word 0 Bits 31..24 */
35 	0x17, 0x18, 0x1B, 0x1D, 0x1E, 0x1F, 0x23, 0x25, /* Word 0 Bits 23..16 */
36 	0x26, 0x27, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, /* Word 0 Bits 15..08 */
37 	0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x39, 0x3A, /* Word 0 Bits 07..00 */
38 
39 	0x3C, 0x43, 0x45, 0x46, 0x47, 0x49, 0x4A, 0x4B, /* Word 1 Bits 31..24 */
40 	0x4C, 0x4D, 0x4E, 0x51, 0x52, 0x53, 0x54, 0x55, /* Word 1 Bits 23..16 */
41 	0x56, 0x59, 0x5A, 0x5C, 0x63, 0x65, 0x66, 0x67, /* Word 1 Bits 15..08 */
42 	0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x71, 0x72, /* Word 1 Bits 07..00 */
43 
44 	0x73, 0x74, 0x75, 0x76, 0x79, 0x7A, 0x7C, 0x80, /* Word 2 Bits 31..24 */
45 	0x81, 0x82, 0x84, 0x88, 0x8F, 0x90, 0x97, 0x98, /* Word 2 Bits 23..16 */
46 	0x9B, 0x9D, 0x9E, 0x9F, 0xA3, 0xA5, 0xA6, 0xA7, /* Word 2 Bits 15..08 */
47 	0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xB1, 0xB2, /* Word 2 Bits 07..00 */
48 
49 	0xB3, 0xB4, 0xB5, 0xB6, 0xB9, 0xBA, 0xBC, 0xC3, /* Word 3 Bits 31..24 */
50 	0xC5, 0xC6, 0xC7, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, /* Word 3 Bits 23..16 */
51 	0xCE, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD9, /* Word 3 Bits 15..08 */
52 	0xDA, 0xDC, 0xE0, 0xE1, 0xE2, 0xE4, 0xE8, 0xEF, /* Word 3 Bits 07..00 */
53 };
54 
55 static void     bfa_fcs_lport_send_ls_rjt(struct bfa_fcs_lport_s *port,
56 					 struct fchs_s *rx_fchs, u8 reason_code,
57 					 u8 reason_code_expl);
58 static void     bfa_fcs_lport_plogi(struct bfa_fcs_lport_s *port,
59 			struct fchs_s *rx_fchs, struct fc_logi_s *plogi);
60 static void     bfa_fcs_lport_online_actions(struct bfa_fcs_lport_s *port);
61 static void     bfa_fcs_lport_offline_actions(struct bfa_fcs_lport_s *port);
62 static void     bfa_fcs_lport_unknown_init(struct bfa_fcs_lport_s *port);
63 static void     bfa_fcs_lport_unknown_online(struct bfa_fcs_lport_s *port);
64 static void     bfa_fcs_lport_unknown_offline(struct bfa_fcs_lport_s *port);
65 static void     bfa_fcs_lport_deleted(struct bfa_fcs_lport_s *port);
66 static void     bfa_fcs_lport_echo(struct bfa_fcs_lport_s *port,
67 			struct fchs_s *rx_fchs,
68 			struct fc_echo_s *echo, u16 len);
69 static void     bfa_fcs_lport_rnid(struct bfa_fcs_lport_s *port,
70 			struct fchs_s *rx_fchs,
71 			struct fc_rnid_cmd_s *rnid, u16 len);
72 static void     bfa_fs_port_get_gen_topo_data(struct bfa_fcs_lport_s *port,
73 			struct fc_rnid_general_topology_data_s *gen_topo_data);
74 
75 static void	bfa_fcs_lport_fab_init(struct bfa_fcs_lport_s *port);
76 static void	bfa_fcs_lport_fab_online(struct bfa_fcs_lport_s *port);
77 static void	bfa_fcs_lport_fab_offline(struct bfa_fcs_lport_s *port);
78 
79 static void	bfa_fcs_lport_n2n_init(struct bfa_fcs_lport_s *port);
80 static void	bfa_fcs_lport_n2n_online(struct bfa_fcs_lport_s *port);
81 static void	bfa_fcs_lport_n2n_offline(struct bfa_fcs_lport_s *port);
82 
83 static void	bfa_fcs_lport_loop_init(struct bfa_fcs_lport_s *port);
84 static void	bfa_fcs_lport_loop_online(struct bfa_fcs_lport_s *port);
85 static void	bfa_fcs_lport_loop_offline(struct bfa_fcs_lport_s *port);
86 
87 static struct {
88 	void		(*init) (struct bfa_fcs_lport_s *port);
89 	void		(*online) (struct bfa_fcs_lport_s *port);
90 	void		(*offline) (struct bfa_fcs_lport_s *port);
91 } __port_action[] = {
92 	{
93 	bfa_fcs_lport_unknown_init, bfa_fcs_lport_unknown_online,
94 			bfa_fcs_lport_unknown_offline}, {
95 	bfa_fcs_lport_fab_init, bfa_fcs_lport_fab_online,
96 			bfa_fcs_lport_fab_offline}, {
97 	bfa_fcs_lport_n2n_init, bfa_fcs_lport_n2n_online,
98 			bfa_fcs_lport_n2n_offline}, {
99 	bfa_fcs_lport_loop_init, bfa_fcs_lport_loop_online,
100 			bfa_fcs_lport_loop_offline},
101 	};
102 
103 /*
104  *  fcs_port_sm FCS logical port state machine
105  */
106 
107 enum bfa_fcs_lport_event {
108 	BFA_FCS_PORT_SM_CREATE = 1,
109 	BFA_FCS_PORT_SM_ONLINE = 2,
110 	BFA_FCS_PORT_SM_OFFLINE = 3,
111 	BFA_FCS_PORT_SM_DELETE = 4,
112 	BFA_FCS_PORT_SM_DELRPORT = 5,
113 	BFA_FCS_PORT_SM_STOP = 6,
114 };
115 
116 static void     bfa_fcs_lport_sm_uninit(struct bfa_fcs_lport_s *port,
117 					enum bfa_fcs_lport_event event);
118 static void     bfa_fcs_lport_sm_init(struct bfa_fcs_lport_s *port,
119 					enum bfa_fcs_lport_event event);
120 static void     bfa_fcs_lport_sm_online(struct bfa_fcs_lport_s *port,
121 					enum bfa_fcs_lport_event event);
122 static void     bfa_fcs_lport_sm_offline(struct bfa_fcs_lport_s *port,
123 					enum bfa_fcs_lport_event event);
124 static void     bfa_fcs_lport_sm_deleting(struct bfa_fcs_lport_s *port,
125 					enum bfa_fcs_lport_event event);
126 static void	bfa_fcs_lport_sm_stopping(struct bfa_fcs_lport_s *port,
127 					enum bfa_fcs_lport_event event);
128 
129 static void
130 bfa_fcs_lport_sm_uninit(
131 	struct bfa_fcs_lport_s *port,
132 	enum bfa_fcs_lport_event event)
133 {
134 	bfa_trc(port->fcs, port->port_cfg.pwwn);
135 	bfa_trc(port->fcs, event);
136 
137 	switch (event) {
138 	case BFA_FCS_PORT_SM_CREATE:
139 		bfa_sm_set_state(port, bfa_fcs_lport_sm_init);
140 		break;
141 
142 	default:
143 		bfa_sm_fault(port->fcs, event);
144 	}
145 }
146 
147 static void
148 bfa_fcs_lport_sm_init(struct bfa_fcs_lport_s *port,
149 			enum bfa_fcs_lport_event event)
150 {
151 	bfa_trc(port->fcs, port->port_cfg.pwwn);
152 	bfa_trc(port->fcs, event);
153 
154 	switch (event) {
155 	case BFA_FCS_PORT_SM_ONLINE:
156 		bfa_sm_set_state(port, bfa_fcs_lport_sm_online);
157 		bfa_fcs_lport_online_actions(port);
158 		break;
159 
160 	case BFA_FCS_PORT_SM_DELETE:
161 		bfa_sm_set_state(port, bfa_fcs_lport_sm_uninit);
162 		bfa_fcs_lport_deleted(port);
163 		break;
164 
165 	case BFA_FCS_PORT_SM_STOP:
166 		/* If vport - send completion call back */
167 		if (port->vport)
168 			bfa_fcs_vport_stop_comp(port->vport);
169 		else
170 			bfa_wc_down(&(port->fabric->stop_wc));
171 		break;
172 
173 	case BFA_FCS_PORT_SM_OFFLINE:
174 		break;
175 
176 	default:
177 		bfa_sm_fault(port->fcs, event);
178 	}
179 }
180 
181 static void
182 bfa_fcs_lport_sm_online(
183 	struct bfa_fcs_lport_s *port,
184 	enum bfa_fcs_lport_event event)
185 {
186 	struct bfa_fcs_rport_s *rport;
187 	struct list_head		*qe, *qen;
188 
189 	bfa_trc(port->fcs, port->port_cfg.pwwn);
190 	bfa_trc(port->fcs, event);
191 
192 	switch (event) {
193 	case BFA_FCS_PORT_SM_OFFLINE:
194 		bfa_sm_set_state(port, bfa_fcs_lport_sm_offline);
195 		bfa_fcs_lport_offline_actions(port);
196 		break;
197 
198 	case BFA_FCS_PORT_SM_STOP:
199 		__port_action[port->fabric->fab_type].offline(port);
200 
201 		if (port->num_rports == 0) {
202 			bfa_sm_set_state(port, bfa_fcs_lport_sm_init);
203 			/* If vport - send completion call back */
204 			if (port->vport)
205 				bfa_fcs_vport_stop_comp(port->vport);
206 			else
207 				bfa_wc_down(&(port->fabric->stop_wc));
208 		} else {
209 			bfa_sm_set_state(port, bfa_fcs_lport_sm_stopping);
210 			list_for_each_safe(qe, qen, &port->rport_q) {
211 				rport = (struct bfa_fcs_rport_s *) qe;
212 				bfa_sm_send_event(rport, RPSM_EVENT_DELETE);
213 			}
214 		}
215 		break;
216 
217 	case BFA_FCS_PORT_SM_DELETE:
218 
219 		__port_action[port->fabric->fab_type].offline(port);
220 
221 		if (port->num_rports == 0) {
222 			bfa_sm_set_state(port, bfa_fcs_lport_sm_uninit);
223 			bfa_fcs_lport_deleted(port);
224 		} else {
225 			bfa_sm_set_state(port, bfa_fcs_lport_sm_deleting);
226 			list_for_each_safe(qe, qen, &port->rport_q) {
227 				rport = (struct bfa_fcs_rport_s *) qe;
228 				bfa_sm_send_event(rport, RPSM_EVENT_DELETE);
229 			}
230 		}
231 		break;
232 
233 	case BFA_FCS_PORT_SM_DELRPORT:
234 		break;
235 
236 	default:
237 		bfa_sm_fault(port->fcs, event);
238 	}
239 }
240 
241 static void
242 bfa_fcs_lport_sm_offline(
243 	struct bfa_fcs_lport_s *port,
244 	enum bfa_fcs_lport_event event)
245 {
246 	struct bfa_fcs_rport_s *rport;
247 	struct list_head		*qe, *qen;
248 
249 	bfa_trc(port->fcs, port->port_cfg.pwwn);
250 	bfa_trc(port->fcs, event);
251 
252 	switch (event) {
253 	case BFA_FCS_PORT_SM_ONLINE:
254 		bfa_sm_set_state(port, bfa_fcs_lport_sm_online);
255 		bfa_fcs_lport_online_actions(port);
256 		break;
257 
258 	case BFA_FCS_PORT_SM_STOP:
259 		if (port->num_rports == 0) {
260 			bfa_sm_set_state(port, bfa_fcs_lport_sm_init);
261 			/* If vport - send completion call back */
262 			if (port->vport)
263 				bfa_fcs_vport_stop_comp(port->vport);
264 			else
265 				bfa_wc_down(&(port->fabric->stop_wc));
266 		} else {
267 			bfa_sm_set_state(port, bfa_fcs_lport_sm_stopping);
268 			list_for_each_safe(qe, qen, &port->rport_q) {
269 				rport = (struct bfa_fcs_rport_s *) qe;
270 				bfa_sm_send_event(rport, RPSM_EVENT_DELETE);
271 			}
272 		}
273 		break;
274 
275 	case BFA_FCS_PORT_SM_DELETE:
276 		if (port->num_rports == 0) {
277 			bfa_sm_set_state(port, bfa_fcs_lport_sm_uninit);
278 			bfa_fcs_lport_deleted(port);
279 		} else {
280 			bfa_sm_set_state(port, bfa_fcs_lport_sm_deleting);
281 			list_for_each_safe(qe, qen, &port->rport_q) {
282 				rport = (struct bfa_fcs_rport_s *) qe;
283 				bfa_sm_send_event(rport, RPSM_EVENT_DELETE);
284 			}
285 		}
286 		break;
287 
288 	case BFA_FCS_PORT_SM_DELRPORT:
289 	case BFA_FCS_PORT_SM_OFFLINE:
290 		break;
291 
292 	default:
293 		bfa_sm_fault(port->fcs, event);
294 	}
295 }
296 
297 static void
298 bfa_fcs_lport_sm_stopping(struct bfa_fcs_lport_s *port,
299 			  enum bfa_fcs_lport_event event)
300 {
301 	bfa_trc(port->fcs, port->port_cfg.pwwn);
302 	bfa_trc(port->fcs, event);
303 
304 	switch (event) {
305 	case BFA_FCS_PORT_SM_DELRPORT:
306 		if (port->num_rports == 0) {
307 			bfa_sm_set_state(port, bfa_fcs_lport_sm_init);
308 			/* If vport - send completion call back */
309 			if (port->vport)
310 				bfa_fcs_vport_stop_comp(port->vport);
311 			else
312 				bfa_wc_down(&(port->fabric->stop_wc));
313 		}
314 		break;
315 
316 	default:
317 		bfa_sm_fault(port->fcs, event);
318 	}
319 }
320 
321 static void
322 bfa_fcs_lport_sm_deleting(
323 	struct bfa_fcs_lport_s *port,
324 	enum bfa_fcs_lport_event event)
325 {
326 	bfa_trc(port->fcs, port->port_cfg.pwwn);
327 	bfa_trc(port->fcs, event);
328 
329 	switch (event) {
330 	case BFA_FCS_PORT_SM_DELRPORT:
331 		if (port->num_rports == 0) {
332 			bfa_sm_set_state(port, bfa_fcs_lport_sm_uninit);
333 			bfa_fcs_lport_deleted(port);
334 		}
335 		break;
336 
337 	default:
338 		bfa_sm_fault(port->fcs, event);
339 	}
340 }
341 
342 /*
343  *  fcs_port_pvt
344  */
345 
346 /*
347  * Send AEN notification
348  */
349 static void
350 bfa_fcs_lport_aen_post(struct bfa_fcs_lport_s *port,
351 			enum bfa_lport_aen_event event)
352 {
353 	struct bfad_s *bfad = (struct bfad_s *)port->fabric->fcs->bfad;
354 	struct bfa_aen_entry_s  *aen_entry;
355 
356 	bfad_get_aen_entry(bfad, aen_entry);
357 	if (!aen_entry)
358 		return;
359 
360 	aen_entry->aen_data.lport.vf_id = port->fabric->vf_id;
361 	aen_entry->aen_data.lport.roles = port->port_cfg.roles;
362 	aen_entry->aen_data.lport.ppwwn = bfa_fcs_lport_get_pwwn(
363 					bfa_fcs_get_base_port(port->fcs));
364 	aen_entry->aen_data.lport.lpwwn = bfa_fcs_lport_get_pwwn(port);
365 
366 	/* Send the AEN notification */
367 	bfad_im_post_vendor_event(aen_entry, bfad, ++port->fcs->fcs_aen_seq,
368 				  BFA_AEN_CAT_LPORT, event);
369 }
370 
371 /*
372  * Send a LS reject
373  */
374 static void
375 bfa_fcs_lport_send_ls_rjt(struct bfa_fcs_lport_s *port, struct fchs_s *rx_fchs,
376 			 u8 reason_code, u8 reason_code_expl)
377 {
378 	struct fchs_s	fchs;
379 	struct bfa_fcxp_s *fcxp;
380 	struct bfa_rport_s *bfa_rport = NULL;
381 	int		len;
382 
383 	bfa_trc(port->fcs, rx_fchs->d_id);
384 	bfa_trc(port->fcs, rx_fchs->s_id);
385 
386 	fcxp = bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
387 	if (!fcxp)
388 		return;
389 
390 	len = fc_ls_rjt_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
391 			      rx_fchs->s_id, bfa_fcs_lport_get_fcid(port),
392 			      rx_fchs->ox_id, reason_code, reason_code_expl);
393 
394 	bfa_fcxp_send(fcxp, bfa_rport, port->fabric->vf_id, port->lp_tag,
395 			  BFA_FALSE, FC_CLASS_3, len, &fchs, NULL, NULL,
396 			  FC_MAX_PDUSZ, 0);
397 }
398 
399 /*
400  * Send a FCCT Reject
401  */
402 static void
403 bfa_fcs_lport_send_fcgs_rjt(struct bfa_fcs_lport_s *port,
404 	struct fchs_s *rx_fchs, u8 reason_code, u8 reason_code_expl)
405 {
406 	struct fchs_s   fchs;
407 	struct bfa_fcxp_s *fcxp;
408 	struct bfa_rport_s *bfa_rport = NULL;
409 	int             len;
410 	struct ct_hdr_s *rx_cthdr = (struct ct_hdr_s *)(rx_fchs + 1);
411 	struct ct_hdr_s *ct_hdr;
412 
413 	bfa_trc(port->fcs, rx_fchs->d_id);
414 	bfa_trc(port->fcs, rx_fchs->s_id);
415 
416 	fcxp = bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
417 	if (!fcxp)
418 		return;
419 
420 	ct_hdr = bfa_fcxp_get_reqbuf(fcxp);
421 	ct_hdr->gs_type = rx_cthdr->gs_type;
422 	ct_hdr->gs_sub_type = rx_cthdr->gs_sub_type;
423 
424 	len = fc_gs_rjt_build(&fchs, ct_hdr, rx_fchs->s_id,
425 			bfa_fcs_lport_get_fcid(port),
426 			rx_fchs->ox_id, reason_code, reason_code_expl);
427 
428 	bfa_fcxp_send(fcxp, bfa_rport, port->fabric->vf_id, port->lp_tag,
429 			BFA_FALSE, FC_CLASS_3, len, &fchs, NULL, NULL,
430 			FC_MAX_PDUSZ, 0);
431 }
432 
433 /*
434  * Process incoming plogi from a remote port.
435  */
436 static void
437 bfa_fcs_lport_plogi(struct bfa_fcs_lport_s *port,
438 		struct fchs_s *rx_fchs, struct fc_logi_s *plogi)
439 {
440 	struct bfa_fcs_rport_s *rport;
441 
442 	bfa_trc(port->fcs, rx_fchs->d_id);
443 	bfa_trc(port->fcs, rx_fchs->s_id);
444 
445 	/*
446 	 * If min cfg mode is enabled, drop any incoming PLOGIs
447 	 */
448 	if (__fcs_min_cfg(port->fcs)) {
449 		bfa_trc(port->fcs, rx_fchs->s_id);
450 		return;
451 	}
452 
453 	if (fc_plogi_parse(rx_fchs) != FC_PARSE_OK) {
454 		bfa_trc(port->fcs, rx_fchs->s_id);
455 		/*
456 		 * send a LS reject
457 		 */
458 		bfa_fcs_lport_send_ls_rjt(port, rx_fchs,
459 					FC_LS_RJT_RSN_PROTOCOL_ERROR,
460 					FC_LS_RJT_EXP_SPARMS_ERR_OPTIONS);
461 		return;
462 	}
463 
464 	/*
465 	 * Direct Attach P2P mode : verify address assigned by the r-port.
466 	 */
467 	if ((!bfa_fcs_fabric_is_switched(port->fabric)) &&
468 		(memcmp((void *)&bfa_fcs_lport_get_pwwn(port),
469 			   (void *)&plogi->port_name, sizeof(wwn_t)) < 0)) {
470 		if (BFA_FCS_PID_IS_WKA(rx_fchs->d_id)) {
471 			/* Address assigned to us cannot be a WKA */
472 			bfa_fcs_lport_send_ls_rjt(port, rx_fchs,
473 					FC_LS_RJT_RSN_PROTOCOL_ERROR,
474 					FC_LS_RJT_EXP_INVALID_NPORT_ID);
475 			return;
476 		}
477 		port->pid  = rx_fchs->d_id;
478 		bfa_lps_set_n2n_pid(port->fabric->lps, rx_fchs->d_id);
479 	}
480 
481 	/*
482 	 * First, check if we know the device by pwwn.
483 	 */
484 	rport = bfa_fcs_lport_get_rport_by_pwwn(port, plogi->port_name);
485 	if (rport) {
486 		/*
487 		 * Direct Attach P2P mode : handle address assigned by r-port.
488 		 */
489 		if ((!bfa_fcs_fabric_is_switched(port->fabric)) &&
490 			(memcmp((void *)&bfa_fcs_lport_get_pwwn(port),
491 			(void *)&plogi->port_name, sizeof(wwn_t)) < 0)) {
492 			port->pid  = rx_fchs->d_id;
493 			bfa_lps_set_n2n_pid(port->fabric->lps, rx_fchs->d_id);
494 			rport->pid = rx_fchs->s_id;
495 		}
496 		bfa_fcs_rport_plogi(rport, rx_fchs, plogi);
497 		return;
498 	}
499 
500 	/*
501 	 * Next, lookup rport by PID.
502 	 */
503 	rport = bfa_fcs_lport_get_rport_by_pid(port, rx_fchs->s_id);
504 	if (!rport) {
505 		/*
506 		 * Inbound PLOGI from a new device.
507 		 */
508 		bfa_fcs_rport_plogi_create(port, rx_fchs, plogi);
509 		return;
510 	}
511 
512 	/*
513 	 * Rport is known only by PID.
514 	 */
515 	if (rport->pwwn) {
516 		/*
517 		 * This is a different device with the same pid. Old device
518 		 * disappeared. Send implicit LOGO to old device.
519 		 */
520 		WARN_ON(rport->pwwn == plogi->port_name);
521 		bfa_sm_send_event(rport, RPSM_EVENT_LOGO_IMP);
522 
523 		/*
524 		 * Inbound PLOGI from a new device (with old PID).
525 		 */
526 		bfa_fcs_rport_plogi_create(port, rx_fchs, plogi);
527 		return;
528 	}
529 
530 	/*
531 	 * PLOGI crossing each other.
532 	 */
533 	WARN_ON(rport->pwwn != WWN_NULL);
534 	bfa_fcs_rport_plogi(rport, rx_fchs, plogi);
535 }
536 
537 /*
538  * Process incoming ECHO.
539  * Since it does not require a login, it is processed here.
540  */
541 static void
542 bfa_fcs_lport_echo(struct bfa_fcs_lport_s *port, struct fchs_s *rx_fchs,
543 		struct fc_echo_s *echo, u16 rx_len)
544 {
545 	struct fchs_s		fchs;
546 	struct bfa_fcxp_s	*fcxp;
547 	struct bfa_rport_s	*bfa_rport = NULL;
548 	int			len, pyld_len;
549 
550 	bfa_trc(port->fcs, rx_fchs->s_id);
551 	bfa_trc(port->fcs, rx_fchs->d_id);
552 
553 	fcxp = bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
554 	if (!fcxp)
555 		return;
556 
557 	len = fc_ls_acc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
558 				rx_fchs->s_id, bfa_fcs_lport_get_fcid(port),
559 				rx_fchs->ox_id);
560 
561 	/*
562 	 * Copy the payload (if any) from the echo frame
563 	 */
564 	pyld_len = rx_len - sizeof(struct fchs_s);
565 	bfa_trc(port->fcs, rx_len);
566 	bfa_trc(port->fcs, pyld_len);
567 
568 	if (pyld_len > len)
569 		memcpy(((u8 *) bfa_fcxp_get_reqbuf(fcxp)) +
570 			sizeof(struct fc_echo_s), (echo + 1),
571 			(pyld_len - sizeof(struct fc_echo_s)));
572 
573 	bfa_fcxp_send(fcxp, bfa_rport, port->fabric->vf_id, port->lp_tag,
574 			BFA_FALSE, FC_CLASS_3, pyld_len, &fchs, NULL, NULL,
575 			FC_MAX_PDUSZ, 0);
576 }
577 
578 /*
579  * Process incoming RNID.
580  * Since it does not require a login, it is processed here.
581  */
582 static void
583 bfa_fcs_lport_rnid(struct bfa_fcs_lport_s *port, struct fchs_s *rx_fchs,
584 		struct fc_rnid_cmd_s *rnid, u16 rx_len)
585 {
586 	struct fc_rnid_common_id_data_s common_id_data;
587 	struct fc_rnid_general_topology_data_s gen_topo_data;
588 	struct fchs_s	fchs;
589 	struct bfa_fcxp_s *fcxp;
590 	struct bfa_rport_s *bfa_rport = NULL;
591 	u16	len;
592 	u32	data_format;
593 
594 	bfa_trc(port->fcs, rx_fchs->s_id);
595 	bfa_trc(port->fcs, rx_fchs->d_id);
596 	bfa_trc(port->fcs, rx_len);
597 
598 	fcxp = bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
599 	if (!fcxp)
600 		return;
601 
602 	/*
603 	 * Check Node Indentification Data Format
604 	 * We only support General Topology Discovery Format.
605 	 * For any other requested Data Formats, we return Common Node Id Data
606 	 * only, as per FC-LS.
607 	 */
608 	bfa_trc(port->fcs, rnid->node_id_data_format);
609 	if (rnid->node_id_data_format == RNID_NODEID_DATA_FORMAT_DISCOVERY) {
610 		data_format = RNID_NODEID_DATA_FORMAT_DISCOVERY;
611 		/*
612 		 * Get General topology data for this port
613 		 */
614 		bfa_fs_port_get_gen_topo_data(port, &gen_topo_data);
615 	} else {
616 		data_format = RNID_NODEID_DATA_FORMAT_COMMON;
617 	}
618 
619 	/*
620 	 * Copy the Node Id Info
621 	 */
622 	common_id_data.port_name = bfa_fcs_lport_get_pwwn(port);
623 	common_id_data.node_name = bfa_fcs_lport_get_nwwn(port);
624 
625 	len = fc_rnid_acc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
626 				rx_fchs->s_id, bfa_fcs_lport_get_fcid(port),
627 				rx_fchs->ox_id, data_format, &common_id_data,
628 				&gen_topo_data);
629 
630 	bfa_fcxp_send(fcxp, bfa_rport, port->fabric->vf_id, port->lp_tag,
631 			BFA_FALSE, FC_CLASS_3, len, &fchs, NULL, NULL,
632 			FC_MAX_PDUSZ, 0);
633 }
634 
635 /*
636  *  Fill out General Topolpgy Discovery Data for RNID ELS.
637  */
638 static void
639 bfa_fs_port_get_gen_topo_data(struct bfa_fcs_lport_s *port,
640 			struct fc_rnid_general_topology_data_s *gen_topo_data)
641 {
642 	memset(gen_topo_data, 0,
643 		      sizeof(struct fc_rnid_general_topology_data_s));
644 
645 	gen_topo_data->asso_type = cpu_to_be32(RNID_ASSOCIATED_TYPE_HOST);
646 	gen_topo_data->phy_port_num = 0;	/* @todo */
647 	gen_topo_data->num_attached_nodes = cpu_to_be32(1);
648 }
649 
650 static void
651 bfa_fcs_lport_online_actions(struct bfa_fcs_lport_s *port)
652 {
653 	struct bfad_s *bfad = (struct bfad_s *)port->fcs->bfad;
654 	char	lpwwn_buf[BFA_STRING_32];
655 
656 	bfa_trc(port->fcs, port->fabric->oper_type);
657 
658 	__port_action[port->fabric->fab_type].init(port);
659 	__port_action[port->fabric->fab_type].online(port);
660 
661 	wwn2str(lpwwn_buf, bfa_fcs_lport_get_pwwn(port));
662 	BFA_LOG(KERN_WARNING, bfad, bfa_log_level,
663 		"Logical port online: WWN = %s Role = %s\n",
664 		lpwwn_buf, "Initiator");
665 	bfa_fcs_lport_aen_post(port, BFA_LPORT_AEN_ONLINE);
666 
667 	bfad->bfad_flags |= BFAD_PORT_ONLINE;
668 }
669 
670 static void
671 bfa_fcs_lport_offline_actions(struct bfa_fcs_lport_s *port)
672 {
673 	struct list_head	*qe, *qen;
674 	struct bfa_fcs_rport_s *rport;
675 	struct bfad_s *bfad = (struct bfad_s *)port->fcs->bfad;
676 	char    lpwwn_buf[BFA_STRING_32];
677 
678 	bfa_trc(port->fcs, port->fabric->oper_type);
679 
680 	__port_action[port->fabric->fab_type].offline(port);
681 
682 	wwn2str(lpwwn_buf, bfa_fcs_lport_get_pwwn(port));
683 	if (bfa_sm_cmp_state(port->fabric,
684 			bfa_fcs_fabric_sm_online) == BFA_TRUE) {
685 		BFA_LOG(KERN_WARNING, bfad, bfa_log_level,
686 		"Logical port lost fabric connectivity: WWN = %s Role = %s\n",
687 		lpwwn_buf, "Initiator");
688 		bfa_fcs_lport_aen_post(port, BFA_LPORT_AEN_DISCONNECT);
689 	} else {
690 		BFA_LOG(KERN_WARNING, bfad, bfa_log_level,
691 		"Logical port taken offline: WWN = %s Role = %s\n",
692 		lpwwn_buf, "Initiator");
693 		bfa_fcs_lport_aen_post(port, BFA_LPORT_AEN_OFFLINE);
694 	}
695 
696 	list_for_each_safe(qe, qen, &port->rport_q) {
697 		rport = (struct bfa_fcs_rport_s *) qe;
698 		bfa_sm_send_event(rport, RPSM_EVENT_LOGO_IMP);
699 	}
700 }
701 
702 static void
703 bfa_fcs_lport_unknown_init(struct bfa_fcs_lport_s *port)
704 {
705 	WARN_ON(1);
706 }
707 
708 static void
709 bfa_fcs_lport_unknown_online(struct bfa_fcs_lport_s *port)
710 {
711 	WARN_ON(1);
712 }
713 
714 static void
715 bfa_fcs_lport_unknown_offline(struct bfa_fcs_lport_s *port)
716 {
717 	WARN_ON(1);
718 }
719 
720 static void
721 bfa_fcs_lport_abts_acc(struct bfa_fcs_lport_s *port, struct fchs_s *rx_fchs)
722 {
723 	struct fchs_s fchs;
724 	struct bfa_fcxp_s *fcxp;
725 	int		len;
726 
727 	bfa_trc(port->fcs, rx_fchs->d_id);
728 	bfa_trc(port->fcs, rx_fchs->s_id);
729 
730 	fcxp = bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
731 	if (!fcxp)
732 		return;
733 
734 	len = fc_ba_acc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
735 			rx_fchs->s_id, bfa_fcs_lport_get_fcid(port),
736 			rx_fchs->ox_id, 0);
737 
738 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag,
739 			  BFA_FALSE, FC_CLASS_3, len, &fchs, NULL, NULL,
740 			  FC_MAX_PDUSZ, 0);
741 }
742 static void
743 bfa_fcs_lport_deleted(struct bfa_fcs_lport_s *port)
744 {
745 	struct bfad_s *bfad = (struct bfad_s *)port->fcs->bfad;
746 	char    lpwwn_buf[BFA_STRING_32];
747 
748 	wwn2str(lpwwn_buf, bfa_fcs_lport_get_pwwn(port));
749 	BFA_LOG(KERN_INFO, bfad, bfa_log_level,
750 		"Logical port deleted: WWN = %s Role = %s\n",
751 		lpwwn_buf, "Initiator");
752 	bfa_fcs_lport_aen_post(port, BFA_LPORT_AEN_DELETE);
753 
754 	/* Base port will be deleted by the OS driver */
755 	if (port->vport)
756 		bfa_fcs_vport_delete_comp(port->vport);
757 	else
758 		bfa_wc_down(&port->fabric->wc);
759 }
760 
761 
762 /*
763  * Unsolicited frame receive handling.
764  */
765 void
766 bfa_fcs_lport_uf_recv(struct bfa_fcs_lport_s *lport,
767 			struct fchs_s *fchs, u16 len)
768 {
769 	u32	pid = fchs->s_id;
770 	struct bfa_fcs_rport_s *rport = NULL;
771 	struct fc_els_cmd_s *els_cmd = (struct fc_els_cmd_s *) (fchs + 1);
772 
773 	bfa_stats(lport, uf_recvs);
774 	bfa_trc(lport->fcs, fchs->type);
775 
776 	if (!bfa_fcs_lport_is_online(lport)) {
777 		/*
778 		 * In direct attach topology, it is possible to get a PLOGI
779 		 * before the lport is online due to port feature
780 		 * (QoS/Trunk/FEC/CR), so send a rjt
781 		 */
782 		if ((fchs->type == FC_TYPE_ELS) &&
783 			(els_cmd->els_code == FC_ELS_PLOGI)) {
784 			bfa_fcs_lport_send_ls_rjt(lport, fchs,
785 				FC_LS_RJT_RSN_UNABLE_TO_PERF_CMD,
786 				FC_LS_RJT_EXP_NO_ADDL_INFO);
787 			bfa_stats(lport, plogi_rcvd);
788 		} else
789 			bfa_stats(lport, uf_recv_drops);
790 
791 		return;
792 	}
793 
794 	/*
795 	 * First, handle ELSs that donot require a login.
796 	 */
797 	/*
798 	 * Handle PLOGI first
799 	 */
800 	if ((fchs->type == FC_TYPE_ELS) &&
801 		(els_cmd->els_code == FC_ELS_PLOGI)) {
802 		bfa_fcs_lport_plogi(lport, fchs, (struct fc_logi_s *) els_cmd);
803 		return;
804 	}
805 
806 	/*
807 	 * Handle ECHO separately.
808 	 */
809 	if ((fchs->type == FC_TYPE_ELS) && (els_cmd->els_code == FC_ELS_ECHO)) {
810 		bfa_fcs_lport_echo(lport, fchs,
811 				(struct fc_echo_s *)els_cmd, len);
812 		return;
813 	}
814 
815 	/*
816 	 * Handle RNID separately.
817 	 */
818 	if ((fchs->type == FC_TYPE_ELS) && (els_cmd->els_code == FC_ELS_RNID)) {
819 		bfa_fcs_lport_rnid(lport, fchs,
820 			(struct fc_rnid_cmd_s *) els_cmd, len);
821 		return;
822 	}
823 
824 	if (fchs->type == FC_TYPE_BLS) {
825 		if ((fchs->routing == FC_RTG_BASIC_LINK) &&
826 				(fchs->cat_info == FC_CAT_ABTS))
827 			bfa_fcs_lport_abts_acc(lport, fchs);
828 		return;
829 	}
830 
831 	if (fchs->type == FC_TYPE_SERVICES) {
832 		/*
833 		 * Unhandled FC-GS frames. Send a FC-CT Reject
834 		 */
835 		bfa_fcs_lport_send_fcgs_rjt(lport, fchs, CT_RSN_NOT_SUPP,
836 				CT_NS_EXP_NOADDITIONAL);
837 		return;
838 	}
839 
840 	/*
841 	 * look for a matching remote port ID
842 	 */
843 	rport = bfa_fcs_lport_get_rport_by_pid(lport, pid);
844 	if (rport) {
845 		bfa_trc(rport->fcs, fchs->s_id);
846 		bfa_trc(rport->fcs, fchs->d_id);
847 		bfa_trc(rport->fcs, fchs->type);
848 
849 		bfa_fcs_rport_uf_recv(rport, fchs, len);
850 		return;
851 	}
852 
853 	/*
854 	 * Only handles ELS frames for now.
855 	 */
856 	if (fchs->type != FC_TYPE_ELS) {
857 		bfa_trc(lport->fcs, fchs->s_id);
858 		bfa_trc(lport->fcs, fchs->d_id);
859 		/* ignore type FC_TYPE_FC_FSS */
860 		if (fchs->type != FC_TYPE_FC_FSS)
861 			bfa_sm_fault(lport->fcs, fchs->type);
862 		return;
863 	}
864 
865 	bfa_trc(lport->fcs, els_cmd->els_code);
866 	if (els_cmd->els_code == FC_ELS_RSCN) {
867 		bfa_fcs_lport_scn_process_rscn(lport, fchs, len);
868 		return;
869 	}
870 
871 	if (els_cmd->els_code == FC_ELS_LOGO) {
872 		/*
873 		 * @todo Handle LOGO frames received.
874 		 */
875 		return;
876 	}
877 
878 	if (els_cmd->els_code == FC_ELS_PRLI) {
879 		/*
880 		 * @todo Handle PRLI frames received.
881 		 */
882 		return;
883 	}
884 
885 	/*
886 	 * Unhandled ELS frames. Send a LS_RJT.
887 	 */
888 	bfa_fcs_lport_send_ls_rjt(lport, fchs, FC_LS_RJT_RSN_CMD_NOT_SUPP,
889 				 FC_LS_RJT_EXP_NO_ADDL_INFO);
890 
891 }
892 
893 /*
894  *   PID based Lookup for a R-Port in the Port R-Port Queue
895  */
896 struct bfa_fcs_rport_s *
897 bfa_fcs_lport_get_rport_by_pid(struct bfa_fcs_lport_s *port, u32 pid)
898 {
899 	struct bfa_fcs_rport_s *rport;
900 	struct list_head	*qe;
901 
902 	list_for_each(qe, &port->rport_q) {
903 		rport = (struct bfa_fcs_rport_s *) qe;
904 		if (rport->pid == pid)
905 			return rport;
906 	}
907 
908 	bfa_trc(port->fcs, pid);
909 	return NULL;
910 }
911 
912 /*
913  * OLD_PID based Lookup for a R-Port in the Port R-Port Queue
914  */
915 struct bfa_fcs_rport_s *
916 bfa_fcs_lport_get_rport_by_old_pid(struct bfa_fcs_lport_s *port, u32 pid)
917 {
918 	struct bfa_fcs_rport_s *rport;
919 	struct list_head	*qe;
920 
921 	list_for_each(qe, &port->rport_q) {
922 		rport = (struct bfa_fcs_rport_s *) qe;
923 		if (rport->old_pid == pid)
924 			return rport;
925 	}
926 
927 	bfa_trc(port->fcs, pid);
928 	return NULL;
929 }
930 
931 /*
932  *   PWWN based Lookup for a R-Port in the Port R-Port Queue
933  */
934 struct bfa_fcs_rport_s *
935 bfa_fcs_lport_get_rport_by_pwwn(struct bfa_fcs_lport_s *port, wwn_t pwwn)
936 {
937 	struct bfa_fcs_rport_s *rport;
938 	struct list_head	*qe;
939 
940 	list_for_each(qe, &port->rport_q) {
941 		rport = (struct bfa_fcs_rport_s *) qe;
942 		if (wwn_is_equal(rport->pwwn, pwwn))
943 			return rport;
944 	}
945 
946 	bfa_trc(port->fcs, pwwn);
947 	return NULL;
948 }
949 
950 /*
951  *   NWWN based Lookup for a R-Port in the Port R-Port Queue
952  */
953 struct bfa_fcs_rport_s *
954 bfa_fcs_lport_get_rport_by_nwwn(struct bfa_fcs_lport_s *port, wwn_t nwwn)
955 {
956 	struct bfa_fcs_rport_s *rport;
957 	struct list_head	*qe;
958 
959 	list_for_each(qe, &port->rport_q) {
960 		rport = (struct bfa_fcs_rport_s *) qe;
961 		if (wwn_is_equal(rport->nwwn, nwwn))
962 			return rport;
963 	}
964 
965 	bfa_trc(port->fcs, nwwn);
966 	return NULL;
967 }
968 
969 /*
970  * PWWN & PID based Lookup for a R-Port in the Port R-Port Queue
971  */
972 struct bfa_fcs_rport_s *
973 bfa_fcs_lport_get_rport_by_qualifier(struct bfa_fcs_lport_s *port,
974 				     wwn_t pwwn, u32 pid)
975 {
976 	struct bfa_fcs_rport_s *rport;
977 	struct list_head	*qe;
978 
979 	list_for_each(qe, &port->rport_q) {
980 		rport = (struct bfa_fcs_rport_s *) qe;
981 		if (wwn_is_equal(rport->pwwn, pwwn) && rport->pid == pid)
982 			return rport;
983 	}
984 
985 	bfa_trc(port->fcs, pwwn);
986 	return NULL;
987 }
988 
989 /*
990  * Called by rport module when new rports are discovered.
991  */
992 void
993 bfa_fcs_lport_add_rport(
994 	struct bfa_fcs_lport_s *port,
995 	struct bfa_fcs_rport_s *rport)
996 {
997 	list_add_tail(&rport->qe, &port->rport_q);
998 	port->num_rports++;
999 }
1000 
1001 /*
1002  * Called by rport module to when rports are deleted.
1003  */
1004 void
1005 bfa_fcs_lport_del_rport(
1006 	struct bfa_fcs_lport_s *port,
1007 	struct bfa_fcs_rport_s *rport)
1008 {
1009 	WARN_ON(!bfa_q_is_on_q(&port->rport_q, rport));
1010 	list_del(&rport->qe);
1011 	port->num_rports--;
1012 
1013 	bfa_sm_send_event(port, BFA_FCS_PORT_SM_DELRPORT);
1014 }
1015 
1016 /*
1017  * Called by fabric for base port when fabric login is complete.
1018  * Called by vport for virtual ports when FDISC is complete.
1019  */
1020 void
1021 bfa_fcs_lport_online(struct bfa_fcs_lport_s *port)
1022 {
1023 	bfa_sm_send_event(port, BFA_FCS_PORT_SM_ONLINE);
1024 }
1025 
1026 /*
1027  * Called by fabric for base port when fabric goes offline.
1028  * Called by vport for virtual ports when virtual port becomes offline.
1029  */
1030 void
1031 bfa_fcs_lport_offline(struct bfa_fcs_lport_s *port)
1032 {
1033 	bfa_sm_send_event(port, BFA_FCS_PORT_SM_OFFLINE);
1034 }
1035 
1036 /*
1037  * Called by fabric for base port and by vport for virtual ports
1038  * when target mode driver is unloaded.
1039  */
1040 void
1041 bfa_fcs_lport_stop(struct bfa_fcs_lport_s *port)
1042 {
1043 	bfa_sm_send_event(port, BFA_FCS_PORT_SM_STOP);
1044 }
1045 
1046 /*
1047  * Called by fabric to delete base lport and associated resources.
1048  *
1049  * Called by vport to delete lport and associated resources. Should call
1050  * bfa_fcs_vport_delete_comp() for vports on completion.
1051  */
1052 void
1053 bfa_fcs_lport_delete(struct bfa_fcs_lport_s *port)
1054 {
1055 	bfa_sm_send_event(port, BFA_FCS_PORT_SM_DELETE);
1056 }
1057 
1058 /*
1059  * Return TRUE if port is online, else return FALSE
1060  */
1061 bfa_boolean_t
1062 bfa_fcs_lport_is_online(struct bfa_fcs_lport_s *port)
1063 {
1064 	return bfa_sm_cmp_state(port, bfa_fcs_lport_sm_online);
1065 }
1066 
1067 /*
1068   * Attach time initialization of logical ports.
1069  */
1070 void
1071 bfa_fcs_lport_attach(struct bfa_fcs_lport_s *lport, struct bfa_fcs_s *fcs,
1072 		   u16 vf_id, struct bfa_fcs_vport_s *vport)
1073 {
1074 	lport->fcs = fcs;
1075 	lport->fabric = bfa_fcs_vf_lookup(fcs, vf_id);
1076 	lport->vport = vport;
1077 	lport->lp_tag = (vport) ? vport->lps->bfa_tag :
1078 				  lport->fabric->lps->bfa_tag;
1079 
1080 	INIT_LIST_HEAD(&lport->rport_q);
1081 	lport->num_rports = 0;
1082 }
1083 
1084 /*
1085  * Logical port initialization of base or virtual port.
1086  * Called by fabric for base port or by vport for virtual ports.
1087  */
1088 
1089 void
1090 bfa_fcs_lport_init(struct bfa_fcs_lport_s *lport,
1091 	struct bfa_lport_cfg_s *port_cfg)
1092 {
1093 	struct bfa_fcs_vport_s *vport = lport->vport;
1094 	struct bfad_s *bfad = (struct bfad_s *)lport->fcs->bfad;
1095 	char    lpwwn_buf[BFA_STRING_32];
1096 
1097 	lport->port_cfg = *port_cfg;
1098 
1099 	lport->bfad_port = bfa_fcb_lport_new(lport->fcs->bfad, lport,
1100 					lport->port_cfg.roles,
1101 					lport->fabric->vf_drv,
1102 					vport ? vport->vport_drv : NULL);
1103 
1104 	wwn2str(lpwwn_buf, bfa_fcs_lport_get_pwwn(lport));
1105 	BFA_LOG(KERN_INFO, bfad, bfa_log_level,
1106 		"New logical port created: WWN = %s Role = %s\n",
1107 		lpwwn_buf, "Initiator");
1108 	bfa_fcs_lport_aen_post(lport, BFA_LPORT_AEN_NEW);
1109 
1110 	bfa_sm_set_state(lport, bfa_fcs_lport_sm_uninit);
1111 	bfa_sm_send_event(lport, BFA_FCS_PORT_SM_CREATE);
1112 }
1113 
1114 void
1115 bfa_fcs_lport_set_symname(struct bfa_fcs_lport_s *port,
1116 				char *symname)
1117 {
1118 	strcpy(port->port_cfg.sym_name.symname, symname);
1119 
1120 	if (bfa_sm_cmp_state(port, bfa_fcs_lport_sm_online))
1121 		bfa_fcs_lport_ns_util_send_rspn_id(
1122 			BFA_FCS_GET_NS_FROM_PORT(port), NULL);
1123 }
1124 
1125 /*
1126  *  fcs_lport_api
1127  */
1128 
1129 void
1130 bfa_fcs_lport_get_attr(
1131 	struct bfa_fcs_lport_s *port,
1132 	struct bfa_lport_attr_s *port_attr)
1133 {
1134 	if (bfa_sm_cmp_state(port, bfa_fcs_lport_sm_online))
1135 		port_attr->pid = port->pid;
1136 	else
1137 		port_attr->pid = 0;
1138 
1139 	port_attr->port_cfg = port->port_cfg;
1140 
1141 	if (port->fabric) {
1142 		port_attr->port_type = port->fabric->oper_type;
1143 		port_attr->loopback = bfa_sm_cmp_state(port->fabric,
1144 				bfa_fcs_fabric_sm_loopback);
1145 		port_attr->authfail =
1146 			bfa_sm_cmp_state(port->fabric,
1147 				bfa_fcs_fabric_sm_auth_failed);
1148 		port_attr->fabric_name  = bfa_fcs_lport_get_fabric_name(port);
1149 		memcpy(port_attr->fabric_ip_addr,
1150 			bfa_fcs_lport_get_fabric_ipaddr(port),
1151 			BFA_FCS_FABRIC_IPADDR_SZ);
1152 
1153 		if (port->vport != NULL) {
1154 			port_attr->port_type = BFA_PORT_TYPE_VPORT;
1155 			port_attr->fpma_mac =
1156 				port->vport->lps->lp_mac;
1157 		} else {
1158 			port_attr->fpma_mac =
1159 				port->fabric->lps->lp_mac;
1160 		}
1161 	} else {
1162 		port_attr->port_type = BFA_PORT_TYPE_UNKNOWN;
1163 		port_attr->state = BFA_LPORT_UNINIT;
1164 	}
1165 }
1166 
1167 /*
1168  *  bfa_fcs_lport_fab port fab functions
1169  */
1170 
1171 /*
1172  *   Called by port to initialize fabric services of the base port.
1173  */
1174 static void
1175 bfa_fcs_lport_fab_init(struct bfa_fcs_lport_s *port)
1176 {
1177 	bfa_fcs_lport_ns_init(port);
1178 	bfa_fcs_lport_scn_init(port);
1179 	bfa_fcs_lport_ms_init(port);
1180 }
1181 
1182 /*
1183  *   Called by port to notify transition to online state.
1184  */
1185 static void
1186 bfa_fcs_lport_fab_online(struct bfa_fcs_lport_s *port)
1187 {
1188 	bfa_fcs_lport_ns_online(port);
1189 	bfa_fcs_lport_fab_scn_online(port);
1190 }
1191 
1192 /*
1193  *   Called by port to notify transition to offline state.
1194  */
1195 static void
1196 bfa_fcs_lport_fab_offline(struct bfa_fcs_lport_s *port)
1197 {
1198 	bfa_fcs_lport_ns_offline(port);
1199 	bfa_fcs_lport_scn_offline(port);
1200 	bfa_fcs_lport_ms_offline(port);
1201 }
1202 
1203 /*
1204  *  bfa_fcs_lport_n2n  functions
1205  */
1206 
1207 /*
1208  *   Called by fcs/port to initialize N2N topology.
1209  */
1210 static void
1211 bfa_fcs_lport_n2n_init(struct bfa_fcs_lport_s *port)
1212 {
1213 }
1214 
1215 /*
1216  *   Called by fcs/port to notify transition to online state.
1217  */
1218 static void
1219 bfa_fcs_lport_n2n_online(struct bfa_fcs_lport_s *port)
1220 {
1221 	struct bfa_fcs_lport_n2n_s *n2n_port = &port->port_topo.pn2n;
1222 	struct bfa_lport_cfg_s *pcfg = &port->port_cfg;
1223 	struct bfa_fcs_rport_s *rport;
1224 
1225 	bfa_trc(port->fcs, pcfg->pwwn);
1226 
1227 	/*
1228 	 * If our PWWN is > than that of the r-port, we have to initiate PLOGI
1229 	 * and assign an Address. if not, we need to wait for its PLOGI.
1230 	 *
1231 	 * If our PWWN is < than that of the remote port, it will send a PLOGI
1232 	 * with the PIDs assigned. The rport state machine take care of this
1233 	 * incoming PLOGI.
1234 	 */
1235 	if (memcmp
1236 	    ((void *)&pcfg->pwwn, (void *)&n2n_port->rem_port_wwn,
1237 	     sizeof(wwn_t)) > 0) {
1238 		port->pid = N2N_LOCAL_PID;
1239 		bfa_lps_set_n2n_pid(port->fabric->lps, N2N_LOCAL_PID);
1240 		/*
1241 		 * First, check if we know the device by pwwn.
1242 		 */
1243 		rport = bfa_fcs_lport_get_rport_by_pwwn(port,
1244 							n2n_port->rem_port_wwn);
1245 		if (rport) {
1246 			bfa_trc(port->fcs, rport->pid);
1247 			bfa_trc(port->fcs, rport->pwwn);
1248 			rport->pid = N2N_REMOTE_PID;
1249 			bfa_sm_send_event(rport, RPSM_EVENT_PLOGI_SEND);
1250 			return;
1251 		}
1252 
1253 		/*
1254 		 * In n2n there can be only one rport. Delete the old one
1255 		 * whose pid should be zero, because it is offline.
1256 		 */
1257 		if (port->num_rports > 0) {
1258 			rport = bfa_fcs_lport_get_rport_by_pid(port, 0);
1259 			WARN_ON(rport == NULL);
1260 			if (rport) {
1261 				bfa_trc(port->fcs, rport->pwwn);
1262 				bfa_sm_send_event(rport, RPSM_EVENT_DELETE);
1263 			}
1264 		}
1265 		bfa_fcs_rport_create(port, N2N_REMOTE_PID);
1266 	}
1267 }
1268 
1269 /*
1270  *   Called by fcs/port to notify transition to offline state.
1271  */
1272 static void
1273 bfa_fcs_lport_n2n_offline(struct bfa_fcs_lport_s *port)
1274 {
1275 	struct bfa_fcs_lport_n2n_s *n2n_port = &port->port_topo.pn2n;
1276 
1277 	bfa_trc(port->fcs, port->pid);
1278 	port->pid = 0;
1279 	n2n_port->rem_port_wwn = 0;
1280 	n2n_port->reply_oxid = 0;
1281 }
1282 
1283 void
1284 bfa_fcport_get_loop_attr(struct bfa_fcs_lport_s *port)
1285 {
1286 	int i = 0, j = 0, bit = 0, alpa_bit = 0;
1287 	u8 k = 0;
1288 	struct bfa_fcport_s *fcport = BFA_FCPORT_MOD(port->fcs->bfa);
1289 
1290 	port->port_topo.ploop.alpabm_valid = fcport->alpabm_valid;
1291 	port->pid = fcport->myalpa;
1292 	port->pid = bfa_hton3b(port->pid);
1293 
1294 	for (i = 0; i < (FC_ALPA_MAX / 8); i++) {
1295 		for (j = 0, alpa_bit = 0; j < 8; j++, alpa_bit++) {
1296 			bfa_trc(port->fcs->bfa, fcport->alpabm.alpa_bm[i]);
1297 			bit = (fcport->alpabm.alpa_bm[i] & (1 << (7 - j)));
1298 			if (bit) {
1299 				port->port_topo.ploop.alpa_pos_map[k] =
1300 					loop_alpa_map[(i * 8) + alpa_bit];
1301 				k++;
1302 				bfa_trc(port->fcs->bfa, k);
1303 				bfa_trc(port->fcs->bfa,
1304 					 port->port_topo.ploop.alpa_pos_map[k]);
1305 			}
1306 		}
1307 	}
1308 	port->port_topo.ploop.num_alpa = k;
1309 }
1310 
1311 /*
1312  * Called by fcs/port to initialize Loop topology.
1313  */
1314 static void
1315 bfa_fcs_lport_loop_init(struct bfa_fcs_lport_s *port)
1316 {
1317 }
1318 
1319 /*
1320  * Called by fcs/port to notify transition to online state.
1321  */
1322 static void
1323 bfa_fcs_lport_loop_online(struct bfa_fcs_lport_s *port)
1324 {
1325 	u8 num_alpa = 0, alpabm_valid = 0;
1326 	struct bfa_fcs_rport_s *rport;
1327 	u8 *alpa_map = NULL;
1328 	int i = 0;
1329 	u32 pid;
1330 
1331 	bfa_fcport_get_loop_attr(port);
1332 
1333 	num_alpa = port->port_topo.ploop.num_alpa;
1334 	alpabm_valid = port->port_topo.ploop.alpabm_valid;
1335 	alpa_map = port->port_topo.ploop.alpa_pos_map;
1336 
1337 	bfa_trc(port->fcs->bfa, port->pid);
1338 	bfa_trc(port->fcs->bfa, num_alpa);
1339 	if (alpabm_valid == 1) {
1340 		for (i = 0; i < num_alpa; i++) {
1341 			bfa_trc(port->fcs->bfa, alpa_map[i]);
1342 			if (alpa_map[i] != bfa_hton3b(port->pid)) {
1343 				pid = alpa_map[i];
1344 				bfa_trc(port->fcs->bfa, pid);
1345 				rport = bfa_fcs_lport_get_rport_by_pid(port,
1346 						bfa_hton3b(pid));
1347 				if (!rport)
1348 					rport = bfa_fcs_rport_create(port,
1349 						bfa_hton3b(pid));
1350 			}
1351 		}
1352 	} else {
1353 		for (i = 0; i < MAX_ALPA_COUNT; i++) {
1354 			if (alpa_map[i] != port->pid) {
1355 				pid = loop_alpa_map[i];
1356 				bfa_trc(port->fcs->bfa, pid);
1357 				rport = bfa_fcs_lport_get_rport_by_pid(port,
1358 						bfa_hton3b(pid));
1359 				if (!rport)
1360 					rport = bfa_fcs_rport_create(port,
1361 						bfa_hton3b(pid));
1362 			}
1363 		}
1364 	}
1365 }
1366 
1367 /*
1368  * Called by fcs/port to notify transition to offline state.
1369  */
1370 static void
1371 bfa_fcs_lport_loop_offline(struct bfa_fcs_lport_s *port)
1372 {
1373 }
1374 
1375 #define BFA_FCS_FDMI_CMD_MAX_RETRIES 2
1376 
1377 /*
1378  * forward declarations
1379  */
1380 static void     bfa_fcs_lport_fdmi_send_rhba(void *fdmi_cbarg,
1381 					    struct bfa_fcxp_s *fcxp_alloced);
1382 static void     bfa_fcs_lport_fdmi_send_rprt(void *fdmi_cbarg,
1383 					    struct bfa_fcxp_s *fcxp_alloced);
1384 static void     bfa_fcs_lport_fdmi_send_rpa(void *fdmi_cbarg,
1385 					   struct bfa_fcxp_s *fcxp_alloced);
1386 static void     bfa_fcs_lport_fdmi_rhba_response(void *fcsarg,
1387 						struct bfa_fcxp_s *fcxp,
1388 						void *cbarg,
1389 						bfa_status_t req_status,
1390 						u32 rsp_len,
1391 						u32 resid_len,
1392 						struct fchs_s *rsp_fchs);
1393 static void     bfa_fcs_lport_fdmi_rprt_response(void *fcsarg,
1394 						struct bfa_fcxp_s *fcxp,
1395 						void *cbarg,
1396 						bfa_status_t req_status,
1397 						u32 rsp_len,
1398 						u32 resid_len,
1399 						struct fchs_s *rsp_fchs);
1400 static void     bfa_fcs_lport_fdmi_rpa_response(void *fcsarg,
1401 					       struct bfa_fcxp_s *fcxp,
1402 					       void *cbarg,
1403 					       bfa_status_t req_status,
1404 					       u32 rsp_len,
1405 					       u32 resid_len,
1406 					       struct fchs_s *rsp_fchs);
1407 static void     bfa_fcs_lport_fdmi_timeout(void *arg);
1408 static u16 bfa_fcs_lport_fdmi_build_rhba_pyld(struct bfa_fcs_lport_fdmi_s *fdmi,
1409 						  u8 *pyld);
1410 static u16 bfa_fcs_lport_fdmi_build_rprt_pyld(struct bfa_fcs_lport_fdmi_s *fdmi,
1411 						  u8 *pyld);
1412 static u16 bfa_fcs_lport_fdmi_build_rpa_pyld(struct bfa_fcs_lport_fdmi_s *fdmi,
1413 						 u8 *pyld);
1414 static u16 bfa_fcs_lport_fdmi_build_portattr_block(struct bfa_fcs_lport_fdmi_s *
1415 						       fdmi, u8 *pyld);
1416 static void	bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_lport_fdmi_s *fdmi,
1417 				 struct bfa_fcs_fdmi_hba_attr_s *hba_attr);
1418 static void	bfa_fcs_fdmi_get_portattr(struct bfa_fcs_lport_fdmi_s *fdmi,
1419 				  struct bfa_fcs_fdmi_port_attr_s *port_attr);
1420 u32	bfa_fcs_fdmi_convert_speed(enum bfa_port_speed pport_speed);
1421 
1422 /*
1423  *  fcs_fdmi_sm FCS FDMI state machine
1424  */
1425 
1426 /*
1427  *  FDMI State Machine events
1428  */
1429 enum port_fdmi_event {
1430 	FDMISM_EVENT_PORT_ONLINE = 1,
1431 	FDMISM_EVENT_PORT_OFFLINE = 2,
1432 	FDMISM_EVENT_RSP_OK = 4,
1433 	FDMISM_EVENT_RSP_ERROR = 5,
1434 	FDMISM_EVENT_TIMEOUT = 6,
1435 	FDMISM_EVENT_RHBA_SENT = 7,
1436 	FDMISM_EVENT_RPRT_SENT = 8,
1437 	FDMISM_EVENT_RPA_SENT = 9,
1438 };
1439 
1440 static void     bfa_fcs_lport_fdmi_sm_offline(struct bfa_fcs_lport_fdmi_s *fdmi,
1441 					     enum port_fdmi_event event);
1442 static void     bfa_fcs_lport_fdmi_sm_sending_rhba(
1443 				struct bfa_fcs_lport_fdmi_s *fdmi,
1444 				enum port_fdmi_event event);
1445 static void     bfa_fcs_lport_fdmi_sm_rhba(struct bfa_fcs_lport_fdmi_s *fdmi,
1446 					  enum port_fdmi_event event);
1447 static void     bfa_fcs_lport_fdmi_sm_rhba_retry(
1448 				struct bfa_fcs_lport_fdmi_s *fdmi,
1449 				enum port_fdmi_event event);
1450 static void     bfa_fcs_lport_fdmi_sm_sending_rprt(
1451 				struct bfa_fcs_lport_fdmi_s *fdmi,
1452 				enum port_fdmi_event event);
1453 static void     bfa_fcs_lport_fdmi_sm_rprt(struct bfa_fcs_lport_fdmi_s *fdmi,
1454 					  enum port_fdmi_event event);
1455 static void     bfa_fcs_lport_fdmi_sm_rprt_retry(
1456 				struct bfa_fcs_lport_fdmi_s *fdmi,
1457 				enum port_fdmi_event event);
1458 static void     bfa_fcs_lport_fdmi_sm_sending_rpa(
1459 				struct bfa_fcs_lport_fdmi_s *fdmi,
1460 				enum port_fdmi_event event);
1461 static void     bfa_fcs_lport_fdmi_sm_rpa(struct bfa_fcs_lport_fdmi_s *fdmi,
1462 					 enum port_fdmi_event event);
1463 static void     bfa_fcs_lport_fdmi_sm_rpa_retry(
1464 				struct bfa_fcs_lport_fdmi_s *fdmi,
1465 				enum port_fdmi_event event);
1466 static void     bfa_fcs_lport_fdmi_sm_online(struct bfa_fcs_lport_fdmi_s *fdmi,
1467 					    enum port_fdmi_event event);
1468 static void     bfa_fcs_lport_fdmi_sm_disabled(
1469 				struct bfa_fcs_lport_fdmi_s *fdmi,
1470 				enum port_fdmi_event event);
1471 /*
1472  *	Start in offline state - awaiting MS to send start.
1473  */
1474 static void
1475 bfa_fcs_lport_fdmi_sm_offline(struct bfa_fcs_lport_fdmi_s *fdmi,
1476 			     enum port_fdmi_event event)
1477 {
1478 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1479 
1480 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1481 	bfa_trc(port->fcs, event);
1482 
1483 	fdmi->retry_cnt = 0;
1484 
1485 	switch (event) {
1486 	case FDMISM_EVENT_PORT_ONLINE:
1487 		if (port->vport) {
1488 			/*
1489 			 * For Vports, register a new port.
1490 			 */
1491 			bfa_sm_set_state(fdmi,
1492 					 bfa_fcs_lport_fdmi_sm_sending_rprt);
1493 			bfa_fcs_lport_fdmi_send_rprt(fdmi, NULL);
1494 		} else {
1495 			/*
1496 			 * For a base port, we should first register the HBA
1497 			 * attribute. The HBA attribute also contains the base
1498 			 *  port registration.
1499 			 */
1500 			bfa_sm_set_state(fdmi,
1501 					 bfa_fcs_lport_fdmi_sm_sending_rhba);
1502 			bfa_fcs_lport_fdmi_send_rhba(fdmi, NULL);
1503 		}
1504 		break;
1505 
1506 	case FDMISM_EVENT_PORT_OFFLINE:
1507 		break;
1508 
1509 	default:
1510 		bfa_sm_fault(port->fcs, event);
1511 	}
1512 }
1513 
1514 static void
1515 bfa_fcs_lport_fdmi_sm_sending_rhba(struct bfa_fcs_lport_fdmi_s *fdmi,
1516 				  enum port_fdmi_event event)
1517 {
1518 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1519 
1520 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1521 	bfa_trc(port->fcs, event);
1522 
1523 	switch (event) {
1524 	case FDMISM_EVENT_RHBA_SENT:
1525 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_rhba);
1526 		break;
1527 
1528 	case FDMISM_EVENT_PORT_OFFLINE:
1529 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1530 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(port),
1531 					   &fdmi->fcxp_wqe);
1532 		break;
1533 
1534 	default:
1535 		bfa_sm_fault(port->fcs, event);
1536 	}
1537 }
1538 
1539 static void
1540 bfa_fcs_lport_fdmi_sm_rhba(struct bfa_fcs_lport_fdmi_s *fdmi,
1541 			enum port_fdmi_event event)
1542 {
1543 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1544 
1545 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1546 	bfa_trc(port->fcs, event);
1547 
1548 	switch (event) {
1549 	case FDMISM_EVENT_RSP_ERROR:
1550 		/*
1551 		 * if max retries have not been reached, start timer for a
1552 		 * delayed retry
1553 		 */
1554 		if (fdmi->retry_cnt++ < BFA_FCS_FDMI_CMD_MAX_RETRIES) {
1555 			bfa_sm_set_state(fdmi,
1556 					bfa_fcs_lport_fdmi_sm_rhba_retry);
1557 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(port),
1558 					    &fdmi->timer,
1559 					    bfa_fcs_lport_fdmi_timeout, fdmi,
1560 					    BFA_FCS_RETRY_TIMEOUT);
1561 		} else {
1562 			/*
1563 			 * set state to offline
1564 			 */
1565 			bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1566 		}
1567 		break;
1568 
1569 	case FDMISM_EVENT_RSP_OK:
1570 		/*
1571 		 * Initiate Register Port Attributes
1572 		 */
1573 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_sending_rpa);
1574 		fdmi->retry_cnt = 0;
1575 		bfa_fcs_lport_fdmi_send_rpa(fdmi, NULL);
1576 		break;
1577 
1578 	case FDMISM_EVENT_PORT_OFFLINE:
1579 		bfa_fcxp_discard(fdmi->fcxp);
1580 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1581 		break;
1582 
1583 	default:
1584 		bfa_sm_fault(port->fcs, event);
1585 	}
1586 }
1587 
1588 static void
1589 bfa_fcs_lport_fdmi_sm_rhba_retry(struct bfa_fcs_lport_fdmi_s *fdmi,
1590 				enum port_fdmi_event event)
1591 {
1592 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1593 
1594 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1595 	bfa_trc(port->fcs, event);
1596 
1597 	switch (event) {
1598 	case FDMISM_EVENT_TIMEOUT:
1599 		/*
1600 		 * Retry Timer Expired. Re-send
1601 		 */
1602 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_sending_rhba);
1603 		bfa_fcs_lport_fdmi_send_rhba(fdmi, NULL);
1604 		break;
1605 
1606 	case FDMISM_EVENT_PORT_OFFLINE:
1607 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1608 		bfa_timer_stop(&fdmi->timer);
1609 		break;
1610 
1611 	default:
1612 		bfa_sm_fault(port->fcs, event);
1613 	}
1614 }
1615 
1616 /*
1617 * RPRT : Register Port
1618  */
1619 static void
1620 bfa_fcs_lport_fdmi_sm_sending_rprt(struct bfa_fcs_lport_fdmi_s *fdmi,
1621 				  enum port_fdmi_event event)
1622 {
1623 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1624 
1625 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1626 	bfa_trc(port->fcs, event);
1627 
1628 	switch (event) {
1629 	case FDMISM_EVENT_RPRT_SENT:
1630 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_rprt);
1631 		break;
1632 
1633 	case FDMISM_EVENT_PORT_OFFLINE:
1634 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1635 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(port),
1636 					   &fdmi->fcxp_wqe);
1637 		break;
1638 
1639 	default:
1640 		bfa_sm_fault(port->fcs, event);
1641 	}
1642 }
1643 
1644 static void
1645 bfa_fcs_lport_fdmi_sm_rprt(struct bfa_fcs_lport_fdmi_s *fdmi,
1646 			enum port_fdmi_event event)
1647 {
1648 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1649 
1650 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1651 	bfa_trc(port->fcs, event);
1652 
1653 	switch (event) {
1654 	case FDMISM_EVENT_RSP_ERROR:
1655 		/*
1656 		 * if max retries have not been reached, start timer for a
1657 		 * delayed retry
1658 		 */
1659 		if (fdmi->retry_cnt++ < BFA_FCS_FDMI_CMD_MAX_RETRIES) {
1660 			bfa_sm_set_state(fdmi,
1661 					bfa_fcs_lport_fdmi_sm_rprt_retry);
1662 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(port),
1663 					    &fdmi->timer,
1664 					    bfa_fcs_lport_fdmi_timeout, fdmi,
1665 					    BFA_FCS_RETRY_TIMEOUT);
1666 
1667 		} else {
1668 			/*
1669 			 * set state to offline
1670 			 */
1671 			bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1672 			fdmi->retry_cnt = 0;
1673 		}
1674 		break;
1675 
1676 	case FDMISM_EVENT_RSP_OK:
1677 		fdmi->retry_cnt = 0;
1678 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_online);
1679 		break;
1680 
1681 	case FDMISM_EVENT_PORT_OFFLINE:
1682 		bfa_fcxp_discard(fdmi->fcxp);
1683 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1684 		break;
1685 
1686 	default:
1687 		bfa_sm_fault(port->fcs, event);
1688 	}
1689 }
1690 
1691 static void
1692 bfa_fcs_lport_fdmi_sm_rprt_retry(struct bfa_fcs_lport_fdmi_s *fdmi,
1693 				enum port_fdmi_event event)
1694 {
1695 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1696 
1697 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1698 	bfa_trc(port->fcs, event);
1699 
1700 	switch (event) {
1701 	case FDMISM_EVENT_TIMEOUT:
1702 		/*
1703 		 * Retry Timer Expired. Re-send
1704 		 */
1705 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_sending_rprt);
1706 		bfa_fcs_lport_fdmi_send_rprt(fdmi, NULL);
1707 		break;
1708 
1709 	case FDMISM_EVENT_PORT_OFFLINE:
1710 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1711 		bfa_timer_stop(&fdmi->timer);
1712 		break;
1713 
1714 	default:
1715 		bfa_sm_fault(port->fcs, event);
1716 	}
1717 }
1718 
1719 /*
1720  * Register Port Attributes
1721  */
1722 static void
1723 bfa_fcs_lport_fdmi_sm_sending_rpa(struct bfa_fcs_lport_fdmi_s *fdmi,
1724 				 enum port_fdmi_event event)
1725 {
1726 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1727 
1728 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1729 	bfa_trc(port->fcs, event);
1730 
1731 	switch (event) {
1732 	case FDMISM_EVENT_RPA_SENT:
1733 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_rpa);
1734 		break;
1735 
1736 	case FDMISM_EVENT_PORT_OFFLINE:
1737 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1738 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(port),
1739 					   &fdmi->fcxp_wqe);
1740 		break;
1741 
1742 	default:
1743 		bfa_sm_fault(port->fcs, event);
1744 	}
1745 }
1746 
1747 static void
1748 bfa_fcs_lport_fdmi_sm_rpa(struct bfa_fcs_lport_fdmi_s *fdmi,
1749 			enum port_fdmi_event event)
1750 {
1751 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1752 
1753 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1754 	bfa_trc(port->fcs, event);
1755 
1756 	switch (event) {
1757 	case FDMISM_EVENT_RSP_ERROR:
1758 		/*
1759 		 * if max retries have not been reached, start timer for a
1760 		 * delayed retry
1761 		 */
1762 		if (fdmi->retry_cnt++ < BFA_FCS_FDMI_CMD_MAX_RETRIES) {
1763 			bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_rpa_retry);
1764 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(port),
1765 					    &fdmi->timer,
1766 					    bfa_fcs_lport_fdmi_timeout, fdmi,
1767 					    BFA_FCS_RETRY_TIMEOUT);
1768 		} else {
1769 			/*
1770 			 * set state to offline
1771 			 */
1772 			bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1773 			fdmi->retry_cnt = 0;
1774 		}
1775 		break;
1776 
1777 	case FDMISM_EVENT_RSP_OK:
1778 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_online);
1779 		fdmi->retry_cnt = 0;
1780 		break;
1781 
1782 	case FDMISM_EVENT_PORT_OFFLINE:
1783 		bfa_fcxp_discard(fdmi->fcxp);
1784 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1785 		break;
1786 
1787 	default:
1788 		bfa_sm_fault(port->fcs, event);
1789 	}
1790 }
1791 
1792 static void
1793 bfa_fcs_lport_fdmi_sm_rpa_retry(struct bfa_fcs_lport_fdmi_s *fdmi,
1794 			       enum port_fdmi_event event)
1795 {
1796 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1797 
1798 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1799 	bfa_trc(port->fcs, event);
1800 
1801 	switch (event) {
1802 	case FDMISM_EVENT_TIMEOUT:
1803 		/*
1804 		 * Retry Timer Expired. Re-send
1805 		 */
1806 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_sending_rpa);
1807 		bfa_fcs_lport_fdmi_send_rpa(fdmi, NULL);
1808 		break;
1809 
1810 	case FDMISM_EVENT_PORT_OFFLINE:
1811 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1812 		bfa_timer_stop(&fdmi->timer);
1813 		break;
1814 
1815 	default:
1816 		bfa_sm_fault(port->fcs, event);
1817 	}
1818 }
1819 
1820 static void
1821 bfa_fcs_lport_fdmi_sm_online(struct bfa_fcs_lport_fdmi_s *fdmi,
1822 				enum port_fdmi_event event)
1823 {
1824 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1825 
1826 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1827 	bfa_trc(port->fcs, event);
1828 
1829 	switch (event) {
1830 	case FDMISM_EVENT_PORT_OFFLINE:
1831 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
1832 		break;
1833 
1834 	default:
1835 		bfa_sm_fault(port->fcs, event);
1836 	}
1837 }
1838 /*
1839  *  FDMI is disabled state.
1840  */
1841 static void
1842 bfa_fcs_lport_fdmi_sm_disabled(struct bfa_fcs_lport_fdmi_s *fdmi,
1843 			     enum port_fdmi_event event)
1844 {
1845 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1846 
1847 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1848 	bfa_trc(port->fcs, event);
1849 
1850 	/* No op State. It can only be enabled at Driver Init. */
1851 }
1852 
1853 /*
1854 *  RHBA : Register HBA Attributes.
1855  */
1856 static void
1857 bfa_fcs_lport_fdmi_send_rhba(void *fdmi_cbarg, struct bfa_fcxp_s *fcxp_alloced)
1858 {
1859 	struct bfa_fcs_lport_fdmi_s *fdmi = fdmi_cbarg;
1860 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1861 	struct fchs_s fchs;
1862 	int             len, attr_len;
1863 	struct bfa_fcxp_s *fcxp;
1864 	u8        *pyld;
1865 
1866 	bfa_trc(port->fcs, port->port_cfg.pwwn);
1867 
1868 	fcxp = fcxp_alloced ? fcxp_alloced :
1869 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
1870 	if (!fcxp) {
1871 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &fdmi->fcxp_wqe,
1872 				bfa_fcs_lport_fdmi_send_rhba, fdmi, BFA_TRUE);
1873 		return;
1874 	}
1875 	fdmi->fcxp = fcxp;
1876 
1877 	pyld = bfa_fcxp_get_reqbuf(fcxp);
1878 	memset(pyld, 0, FC_MAX_PDUSZ);
1879 
1880 	len = fc_fdmi_reqhdr_build(&fchs, pyld, bfa_fcs_lport_get_fcid(port),
1881 				   FDMI_RHBA);
1882 
1883 	attr_len =
1884 		bfa_fcs_lport_fdmi_build_rhba_pyld(fdmi,
1885 					  (u8 *) ((struct ct_hdr_s *) pyld
1886 						       + 1));
1887 
1888 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
1889 			  FC_CLASS_3, (len + attr_len), &fchs,
1890 			  bfa_fcs_lport_fdmi_rhba_response, (void *)fdmi,
1891 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
1892 
1893 	bfa_sm_send_event(fdmi, FDMISM_EVENT_RHBA_SENT);
1894 }
1895 
1896 static          u16
1897 bfa_fcs_lport_fdmi_build_rhba_pyld(struct bfa_fcs_lport_fdmi_s *fdmi, u8 *pyld)
1898 {
1899 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
1900 	struct bfa_fcs_fdmi_hba_attr_s hba_attr;
1901 	struct bfa_fcs_fdmi_hba_attr_s *fcs_hba_attr = &hba_attr;
1902 	struct fdmi_rhba_s *rhba = (struct fdmi_rhba_s *) pyld;
1903 	struct fdmi_attr_s *attr;
1904 	u8        *curr_ptr;
1905 	u16        len, count;
1906 	u16	templen;
1907 
1908 	/*
1909 	 * get hba attributes
1910 	 */
1911 	bfa_fcs_fdmi_get_hbaattr(fdmi, fcs_hba_attr);
1912 
1913 	rhba->hba_id = bfa_fcs_lport_get_pwwn(port);
1914 	rhba->port_list.num_ports = cpu_to_be32(1);
1915 	rhba->port_list.port_entry = bfa_fcs_lport_get_pwwn(port);
1916 
1917 	len = sizeof(rhba->hba_id) + sizeof(rhba->port_list);
1918 
1919 	count = 0;
1920 	len += sizeof(rhba->hba_attr_blk.attr_count);
1921 
1922 	/*
1923 	 * fill out the invididual entries of the HBA attrib Block
1924 	 */
1925 	curr_ptr = (u8 *) &rhba->hba_attr_blk.hba_attr;
1926 
1927 	/*
1928 	 * Node Name
1929 	 */
1930 	attr = (struct fdmi_attr_s *) curr_ptr;
1931 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_NODENAME);
1932 	templen = sizeof(wwn_t);
1933 	memcpy(attr->value, &bfa_fcs_lport_get_nwwn(port), templen);
1934 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
1935 	len += templen;
1936 	count++;
1937 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
1938 			     sizeof(templen));
1939 
1940 	/*
1941 	 * Manufacturer
1942 	 */
1943 	attr = (struct fdmi_attr_s *) curr_ptr;
1944 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_MANUFACTURER);
1945 	templen = (u16) strlen(fcs_hba_attr->manufacturer);
1946 	memcpy(attr->value, fcs_hba_attr->manufacturer, templen);
1947 	templen = fc_roundup(templen, sizeof(u32));
1948 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
1949 	len += templen;
1950 	count++;
1951 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
1952 			     sizeof(templen));
1953 
1954 	/*
1955 	 * Serial Number
1956 	 */
1957 	attr = (struct fdmi_attr_s *) curr_ptr;
1958 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_SERIALNUM);
1959 	templen = (u16) strlen(fcs_hba_attr->serial_num);
1960 	memcpy(attr->value, fcs_hba_attr->serial_num, templen);
1961 	templen = fc_roundup(templen, sizeof(u32));
1962 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
1963 	len += templen;
1964 	count++;
1965 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
1966 			     sizeof(templen));
1967 
1968 	/*
1969 	 * Model
1970 	 */
1971 	attr = (struct fdmi_attr_s *) curr_ptr;
1972 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_MODEL);
1973 	templen = (u16) strlen(fcs_hba_attr->model);
1974 	memcpy(attr->value, fcs_hba_attr->model, templen);
1975 	templen = fc_roundup(templen, sizeof(u32));
1976 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
1977 	len += templen;
1978 	count++;
1979 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
1980 			     sizeof(templen));
1981 
1982 	/*
1983 	 * Model Desc
1984 	 */
1985 	attr = (struct fdmi_attr_s *) curr_ptr;
1986 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_MODEL_DESC);
1987 	templen = (u16) strlen(fcs_hba_attr->model_desc);
1988 	memcpy(attr->value, fcs_hba_attr->model_desc, templen);
1989 	templen = fc_roundup(templen, sizeof(u32));
1990 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
1991 	len += templen;
1992 	count++;
1993 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
1994 			     sizeof(templen));
1995 
1996 	/*
1997 	 * H/W Version
1998 	 */
1999 	if (fcs_hba_attr->hw_version[0] != '\0') {
2000 		attr = (struct fdmi_attr_s *) curr_ptr;
2001 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_HW_VERSION);
2002 		templen = (u16) strlen(fcs_hba_attr->hw_version);
2003 		memcpy(attr->value, fcs_hba_attr->hw_version, templen);
2004 		templen = fc_roundup(templen, sizeof(u32));
2005 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2006 		len += templen;
2007 		count++;
2008 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2009 					 sizeof(templen));
2010 	}
2011 
2012 	/*
2013 	 * Driver Version
2014 	 */
2015 	attr = (struct fdmi_attr_s *) curr_ptr;
2016 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_DRIVER_VERSION);
2017 	templen = (u16) strlen(fcs_hba_attr->driver_version);
2018 	memcpy(attr->value, fcs_hba_attr->driver_version, templen);
2019 	templen = fc_roundup(templen, sizeof(u32));
2020 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2021 	len += templen;
2022 	count++;
2023 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2024 			     sizeof(templen));
2025 
2026 	/*
2027 	 * Option Rom Version
2028 	 */
2029 	if (fcs_hba_attr->option_rom_ver[0] != '\0') {
2030 		attr = (struct fdmi_attr_s *) curr_ptr;
2031 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_ROM_VERSION);
2032 		templen = (u16) strlen(fcs_hba_attr->option_rom_ver);
2033 		memcpy(attr->value, fcs_hba_attr->option_rom_ver, templen);
2034 		templen = fc_roundup(templen, sizeof(u32));
2035 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2036 		len += templen;
2037 		count++;
2038 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2039 					 sizeof(templen));
2040 	}
2041 
2042 	attr = (struct fdmi_attr_s *) curr_ptr;
2043 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_FW_VERSION);
2044 	templen = (u16) strlen(fcs_hba_attr->fw_version);
2045 	memcpy(attr->value, fcs_hba_attr->fw_version, templen);
2046 	templen = fc_roundup(templen, sizeof(u32));
2047 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2048 	len += templen;
2049 	count++;
2050 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2051 			     sizeof(templen));
2052 
2053 	/*
2054 	 * OS Name
2055 	 */
2056 	if (fcs_hba_attr->os_name[0] != '\0') {
2057 		attr = (struct fdmi_attr_s *) curr_ptr;
2058 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_OS_NAME);
2059 		templen = (u16) strlen(fcs_hba_attr->os_name);
2060 		memcpy(attr->value, fcs_hba_attr->os_name, templen);
2061 		templen = fc_roundup(templen, sizeof(u32));
2062 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2063 		len += templen;
2064 		count++;
2065 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2066 					sizeof(templen));
2067 	}
2068 
2069 	/*
2070 	 * MAX_CT_PAYLOAD
2071 	 */
2072 	attr = (struct fdmi_attr_s *) curr_ptr;
2073 	attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_MAX_CT);
2074 	templen = sizeof(fcs_hba_attr->max_ct_pyld);
2075 	memcpy(attr->value, &fcs_hba_attr->max_ct_pyld, templen);
2076 	templen = fc_roundup(templen, sizeof(u32));
2077 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2078 	len += templen;
2079 	count++;
2080 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2081 			     sizeof(templen));
2082 	/*
2083 	 * Send extended attributes ( FOS 7.1 support )
2084 	 */
2085 	if (fdmi->retry_cnt == 0) {
2086 		attr = (struct fdmi_attr_s *) curr_ptr;
2087 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_NODE_SYM_NAME);
2088 		templen = sizeof(fcs_hba_attr->node_sym_name);
2089 		memcpy(attr->value, &fcs_hba_attr->node_sym_name, templen);
2090 		templen = fc_roundup(templen, sizeof(u32));
2091 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2092 		len += templen;
2093 		count++;
2094 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2095 					sizeof(templen));
2096 
2097 		attr = (struct fdmi_attr_s *) curr_ptr;
2098 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_VENDOR_ID);
2099 		templen = sizeof(fcs_hba_attr->vendor_info);
2100 		memcpy(attr->value, &fcs_hba_attr->vendor_info, templen);
2101 		templen = fc_roundup(templen, sizeof(u32));
2102 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2103 		len += templen;
2104 		count++;
2105 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2106 					sizeof(templen));
2107 
2108 		attr = (struct fdmi_attr_s *) curr_ptr;
2109 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_NUM_PORTS);
2110 		templen = sizeof(fcs_hba_attr->num_ports);
2111 		memcpy(attr->value, &fcs_hba_attr->num_ports, templen);
2112 		templen = fc_roundup(templen, sizeof(u32));
2113 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2114 		len += templen;
2115 		count++;
2116 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2117 					sizeof(templen));
2118 
2119 		attr = (struct fdmi_attr_s *) curr_ptr;
2120 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_FABRIC_NAME);
2121 		templen = sizeof(fcs_hba_attr->fabric_name);
2122 		memcpy(attr->value, &fcs_hba_attr->fabric_name, templen);
2123 		templen = fc_roundup(templen, sizeof(u32));
2124 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2125 		len += templen;
2126 		count++;
2127 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2128 					sizeof(templen));
2129 
2130 		attr = (struct fdmi_attr_s *) curr_ptr;
2131 		attr->type = cpu_to_be16(FDMI_HBA_ATTRIB_BIOS_VER);
2132 		templen = sizeof(fcs_hba_attr->bios_ver);
2133 		memcpy(attr->value, &fcs_hba_attr->bios_ver, templen);
2134 		templen = fc_roundup(attr->len, sizeof(u32));
2135 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2136 		len += templen;
2137 		count++;
2138 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2139 					sizeof(templen));
2140 	}
2141 
2142 	/*
2143 	 * Update size of payload
2144 	 */
2145 	len += ((sizeof(attr->type) + sizeof(attr->len)) * count);
2146 
2147 	rhba->hba_attr_blk.attr_count = cpu_to_be32(count);
2148 	return len;
2149 }
2150 
2151 static void
2152 bfa_fcs_lport_fdmi_rhba_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
2153 				void *cbarg, bfa_status_t req_status,
2154 				u32 rsp_len, u32 resid_len,
2155 				struct fchs_s *rsp_fchs)
2156 {
2157 	struct bfa_fcs_lport_fdmi_s *fdmi =
2158 				(struct bfa_fcs_lport_fdmi_s *) cbarg;
2159 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2160 	struct ct_hdr_s *cthdr = NULL;
2161 
2162 	bfa_trc(port->fcs, port->port_cfg.pwwn);
2163 
2164 	/*
2165 	 * Sanity Checks
2166 	 */
2167 	if (req_status != BFA_STATUS_OK) {
2168 		bfa_trc(port->fcs, req_status);
2169 		bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_ERROR);
2170 		return;
2171 	}
2172 
2173 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
2174 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
2175 
2176 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
2177 		bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_OK);
2178 		return;
2179 	}
2180 
2181 	bfa_trc(port->fcs, cthdr->reason_code);
2182 	bfa_trc(port->fcs, cthdr->exp_code);
2183 	bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_ERROR);
2184 }
2185 
2186 /*
2187 *  RPRT : Register Port
2188  */
2189 static void
2190 bfa_fcs_lport_fdmi_send_rprt(void *fdmi_cbarg, struct bfa_fcxp_s *fcxp_alloced)
2191 {
2192 	struct bfa_fcs_lport_fdmi_s *fdmi = fdmi_cbarg;
2193 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2194 	struct fchs_s fchs;
2195 	u16        len, attr_len;
2196 	struct bfa_fcxp_s *fcxp;
2197 	u8        *pyld;
2198 
2199 	bfa_trc(port->fcs, port->port_cfg.pwwn);
2200 
2201 	fcxp = fcxp_alloced ? fcxp_alloced :
2202 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
2203 	if (!fcxp) {
2204 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &fdmi->fcxp_wqe,
2205 				bfa_fcs_lport_fdmi_send_rprt, fdmi, BFA_TRUE);
2206 		return;
2207 	}
2208 	fdmi->fcxp = fcxp;
2209 
2210 	pyld = bfa_fcxp_get_reqbuf(fcxp);
2211 	memset(pyld, 0, FC_MAX_PDUSZ);
2212 
2213 	len = fc_fdmi_reqhdr_build(&fchs, pyld, bfa_fcs_lport_get_fcid(port),
2214 				   FDMI_RPRT);
2215 
2216 	attr_len =
2217 		bfa_fcs_lport_fdmi_build_rprt_pyld(fdmi,
2218 					  (u8 *) ((struct ct_hdr_s *) pyld
2219 						       + 1));
2220 
2221 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
2222 			  FC_CLASS_3, len + attr_len, &fchs,
2223 			  bfa_fcs_lport_fdmi_rprt_response, (void *)fdmi,
2224 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
2225 
2226 	bfa_sm_send_event(fdmi, FDMISM_EVENT_RPRT_SENT);
2227 }
2228 
2229 /*
2230  * This routine builds Port Attribute Block that used in RPA, RPRT commands.
2231  */
2232 static          u16
2233 bfa_fcs_lport_fdmi_build_portattr_block(struct bfa_fcs_lport_fdmi_s *fdmi,
2234 				       u8 *pyld)
2235 {
2236 	struct bfa_fcs_fdmi_port_attr_s fcs_port_attr;
2237 	struct fdmi_port_attr_s *port_attrib = (struct fdmi_port_attr_s *) pyld;
2238 	struct fdmi_attr_s *attr;
2239 	u8        *curr_ptr;
2240 	u16        len;
2241 	u8	count = 0;
2242 	u16	templen;
2243 
2244 	/*
2245 	 * get port attributes
2246 	 */
2247 	bfa_fcs_fdmi_get_portattr(fdmi, &fcs_port_attr);
2248 
2249 	len = sizeof(port_attrib->attr_count);
2250 
2251 	/*
2252 	 * fill out the invididual entries
2253 	 */
2254 	curr_ptr = (u8 *) &port_attrib->port_attr;
2255 
2256 	/*
2257 	 * FC4 Types
2258 	 */
2259 	attr = (struct fdmi_attr_s *) curr_ptr;
2260 	attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_FC4_TYPES);
2261 	templen = sizeof(fcs_port_attr.supp_fc4_types);
2262 	memcpy(attr->value, fcs_port_attr.supp_fc4_types, templen);
2263 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2264 	len += templen;
2265 	++count;
2266 	attr->len =
2267 		cpu_to_be16(templen + sizeof(attr->type) +
2268 			     sizeof(templen));
2269 
2270 	/*
2271 	 * Supported Speed
2272 	 */
2273 	attr = (struct fdmi_attr_s *) curr_ptr;
2274 	attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_SUPP_SPEED);
2275 	templen = sizeof(fcs_port_attr.supp_speed);
2276 	memcpy(attr->value, &fcs_port_attr.supp_speed, templen);
2277 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2278 	len += templen;
2279 	++count;
2280 	attr->len =
2281 		cpu_to_be16(templen + sizeof(attr->type) +
2282 			     sizeof(templen));
2283 
2284 	/*
2285 	 * current Port Speed
2286 	 */
2287 	attr = (struct fdmi_attr_s *) curr_ptr;
2288 	attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_SPEED);
2289 	templen = sizeof(fcs_port_attr.curr_speed);
2290 	memcpy(attr->value, &fcs_port_attr.curr_speed, templen);
2291 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2292 	len += templen;
2293 	++count;
2294 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2295 			     sizeof(templen));
2296 
2297 	/*
2298 	 * max frame size
2299 	 */
2300 	attr = (struct fdmi_attr_s *) curr_ptr;
2301 	attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_FRAME_SIZE);
2302 	templen = sizeof(fcs_port_attr.max_frm_size);
2303 	memcpy(attr->value, &fcs_port_attr.max_frm_size, templen);
2304 	curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2305 	len += templen;
2306 	++count;
2307 	attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2308 			     sizeof(templen));
2309 
2310 	/*
2311 	 * OS Device Name
2312 	 */
2313 	if (fcs_port_attr.os_device_name[0] != '\0') {
2314 		attr = (struct fdmi_attr_s *) curr_ptr;
2315 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_DEV_NAME);
2316 		templen = (u16) strlen(fcs_port_attr.os_device_name);
2317 		memcpy(attr->value, fcs_port_attr.os_device_name, templen);
2318 		templen = fc_roundup(templen, sizeof(u32));
2319 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2320 		len += templen;
2321 		++count;
2322 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2323 					sizeof(templen));
2324 	}
2325 	/*
2326 	 * Host Name
2327 	 */
2328 	if (fcs_port_attr.host_name[0] != '\0') {
2329 		attr = (struct fdmi_attr_s *) curr_ptr;
2330 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_HOST_NAME);
2331 		templen = (u16) strlen(fcs_port_attr.host_name);
2332 		memcpy(attr->value, fcs_port_attr.host_name, templen);
2333 		templen = fc_roundup(templen, sizeof(u32));
2334 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2335 		len += templen;
2336 		++count;
2337 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2338 				sizeof(templen));
2339 	}
2340 
2341 	if (fdmi->retry_cnt == 0) {
2342 		attr = (struct fdmi_attr_s *) curr_ptr;
2343 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_NODE_NAME);
2344 		templen = sizeof(fcs_port_attr.node_name);
2345 		memcpy(attr->value, &fcs_port_attr.node_name, templen);
2346 		templen = fc_roundup(templen, sizeof(u32));
2347 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2348 		len += templen;
2349 		++count;
2350 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2351 				 sizeof(templen));
2352 
2353 		attr = (struct fdmi_attr_s *) curr_ptr;
2354 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_NAME);
2355 		templen = sizeof(fcs_port_attr.port_name);
2356 		memcpy(attr->value, &fcs_port_attr.port_name, templen);
2357 		templen = fc_roundup(templen, sizeof(u32));
2358 		curr_ptr += sizeof(attr->type) + sizeof(attr->len) + templen;
2359 		len += templen;
2360 		++count;
2361 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2362 				 sizeof(templen));
2363 
2364 		if (fcs_port_attr.port_sym_name.symname[0] != '\0') {
2365 			attr = (struct fdmi_attr_s *) curr_ptr;
2366 			attr->type =
2367 				cpu_to_be16(FDMI_PORT_ATTRIB_PORT_SYM_NAME);
2368 			templen = sizeof(fcs_port_attr.port_sym_name);
2369 			memcpy(attr->value,
2370 				&fcs_port_attr.port_sym_name, templen);
2371 			templen = fc_roundup(templen, sizeof(u32));
2372 			curr_ptr += sizeof(attr->type) +
2373 					sizeof(templen) + templen;
2374 			len += templen;
2375 			++count;
2376 			attr->len = cpu_to_be16(templen +
2377 				sizeof(attr->type) + sizeof(templen));
2378 		}
2379 
2380 		attr = (struct fdmi_attr_s *) curr_ptr;
2381 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_TYPE);
2382 		templen = sizeof(fcs_port_attr.port_type);
2383 		memcpy(attr->value, &fcs_port_attr.port_type, templen);
2384 		templen = fc_roundup(templen, sizeof(u32));
2385 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2386 		len += templen;
2387 		++count;
2388 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2389 				 sizeof(templen));
2390 
2391 		attr = (struct fdmi_attr_s *) curr_ptr;
2392 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_SUPP_COS);
2393 		templen = sizeof(fcs_port_attr.scos);
2394 		memcpy(attr->value, &fcs_port_attr.scos, templen);
2395 		templen = fc_roundup(templen, sizeof(u32));
2396 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2397 		len += templen;
2398 		++count;
2399 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2400 				 sizeof(templen));
2401 
2402 		attr = (struct fdmi_attr_s *) curr_ptr;
2403 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_FAB_NAME);
2404 		templen = sizeof(fcs_port_attr.port_fabric_name);
2405 		memcpy(attr->value, &fcs_port_attr.port_fabric_name, templen);
2406 		templen = fc_roundup(templen, sizeof(u32));
2407 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2408 		len += templen;
2409 		++count;
2410 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2411 				 sizeof(templen));
2412 
2413 		attr = (struct fdmi_attr_s *) curr_ptr;
2414 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_FC4_TYPE);
2415 		templen = sizeof(fcs_port_attr.port_act_fc4_type);
2416 		memcpy(attr->value, fcs_port_attr.port_act_fc4_type,
2417 				templen);
2418 		templen = fc_roundup(templen, sizeof(u32));
2419 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2420 		len += templen;
2421 		++count;
2422 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2423 				 sizeof(templen));
2424 
2425 		attr = (struct fdmi_attr_s *) curr_ptr;
2426 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_STATE);
2427 		templen = sizeof(fcs_port_attr.port_state);
2428 		memcpy(attr->value, &fcs_port_attr.port_state, templen);
2429 		templen = fc_roundup(templen, sizeof(u32));
2430 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2431 		len += templen;
2432 		++count;
2433 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2434 				 sizeof(templen));
2435 
2436 		attr = (struct fdmi_attr_s *) curr_ptr;
2437 		attr->type = cpu_to_be16(FDMI_PORT_ATTRIB_PORT_NUM_RPRT);
2438 		templen = sizeof(fcs_port_attr.num_ports);
2439 		memcpy(attr->value, &fcs_port_attr.num_ports, templen);
2440 		templen = fc_roundup(templen, sizeof(u32));
2441 		curr_ptr += sizeof(attr->type) + sizeof(templen) + templen;
2442 		len += templen;
2443 		++count;
2444 		attr->len = cpu_to_be16(templen + sizeof(attr->type) +
2445 				sizeof(templen));
2446 	}
2447 
2448 	/*
2449 	 * Update size of payload
2450 	 */
2451 	port_attrib->attr_count = cpu_to_be32(count);
2452 	len += ((sizeof(attr->type) + sizeof(attr->len)) * count);
2453 	return len;
2454 }
2455 
2456 static          u16
2457 bfa_fcs_lport_fdmi_build_rprt_pyld(struct bfa_fcs_lport_fdmi_s *fdmi, u8 *pyld)
2458 {
2459 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2460 	struct fdmi_rprt_s *rprt = (struct fdmi_rprt_s *) pyld;
2461 	u16        len;
2462 
2463 	rprt->hba_id = bfa_fcs_lport_get_pwwn(bfa_fcs_get_base_port(port->fcs));
2464 	rprt->port_name = bfa_fcs_lport_get_pwwn(port);
2465 
2466 	len = bfa_fcs_lport_fdmi_build_portattr_block(fdmi,
2467 				(u8 *) &rprt->port_attr_blk);
2468 
2469 	len += sizeof(rprt->hba_id) + sizeof(rprt->port_name);
2470 
2471 	return len;
2472 }
2473 
2474 static void
2475 bfa_fcs_lport_fdmi_rprt_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
2476 				void *cbarg, bfa_status_t req_status,
2477 				u32 rsp_len, u32 resid_len,
2478 				struct fchs_s *rsp_fchs)
2479 {
2480 	struct bfa_fcs_lport_fdmi_s *fdmi =
2481 			(struct bfa_fcs_lport_fdmi_s *) cbarg;
2482 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2483 	struct ct_hdr_s *cthdr = NULL;
2484 
2485 	bfa_trc(port->fcs, port->port_cfg.pwwn);
2486 
2487 	/*
2488 	 * Sanity Checks
2489 	 */
2490 	if (req_status != BFA_STATUS_OK) {
2491 		bfa_trc(port->fcs, req_status);
2492 		bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_ERROR);
2493 		return;
2494 	}
2495 
2496 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
2497 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
2498 
2499 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
2500 		bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_OK);
2501 		return;
2502 	}
2503 
2504 	bfa_trc(port->fcs, cthdr->reason_code);
2505 	bfa_trc(port->fcs, cthdr->exp_code);
2506 	bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_ERROR);
2507 }
2508 
2509 /*
2510 *  RPA : Register Port Attributes.
2511  */
2512 static void
2513 bfa_fcs_lport_fdmi_send_rpa(void *fdmi_cbarg, struct bfa_fcxp_s *fcxp_alloced)
2514 {
2515 	struct bfa_fcs_lport_fdmi_s *fdmi = fdmi_cbarg;
2516 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2517 	struct fchs_s fchs;
2518 	u16        len, attr_len;
2519 	struct bfa_fcxp_s *fcxp;
2520 	u8        *pyld;
2521 
2522 	bfa_trc(port->fcs, port->port_cfg.pwwn);
2523 
2524 	fcxp = fcxp_alloced ? fcxp_alloced :
2525 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
2526 	if (!fcxp) {
2527 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &fdmi->fcxp_wqe,
2528 				bfa_fcs_lport_fdmi_send_rpa, fdmi, BFA_TRUE);
2529 		return;
2530 	}
2531 	fdmi->fcxp = fcxp;
2532 
2533 	pyld = bfa_fcxp_get_reqbuf(fcxp);
2534 	memset(pyld, 0, FC_MAX_PDUSZ);
2535 
2536 	len = fc_fdmi_reqhdr_build(&fchs, pyld, bfa_fcs_lport_get_fcid(port),
2537 				   FDMI_RPA);
2538 
2539 	attr_len = bfa_fcs_lport_fdmi_build_rpa_pyld(fdmi,
2540 				(u8 *) ((struct ct_hdr_s *) pyld + 1));
2541 
2542 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
2543 			  FC_CLASS_3, len + attr_len, &fchs,
2544 			  bfa_fcs_lport_fdmi_rpa_response, (void *)fdmi,
2545 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
2546 
2547 	bfa_sm_send_event(fdmi, FDMISM_EVENT_RPA_SENT);
2548 }
2549 
2550 static          u16
2551 bfa_fcs_lport_fdmi_build_rpa_pyld(struct bfa_fcs_lport_fdmi_s *fdmi, u8 *pyld)
2552 {
2553 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2554 	struct fdmi_rpa_s *rpa = (struct fdmi_rpa_s *) pyld;
2555 	u16        len;
2556 
2557 	rpa->port_name = bfa_fcs_lport_get_pwwn(port);
2558 
2559 	len = bfa_fcs_lport_fdmi_build_portattr_block(fdmi,
2560 				(u8 *) &rpa->port_attr_blk);
2561 
2562 	len += sizeof(rpa->port_name);
2563 
2564 	return len;
2565 }
2566 
2567 static void
2568 bfa_fcs_lport_fdmi_rpa_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
2569 			void *cbarg, bfa_status_t req_status, u32 rsp_len,
2570 			u32 resid_len, struct fchs_s *rsp_fchs)
2571 {
2572 	struct bfa_fcs_lport_fdmi_s *fdmi =
2573 				(struct bfa_fcs_lport_fdmi_s *) cbarg;
2574 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2575 	struct ct_hdr_s *cthdr = NULL;
2576 
2577 	bfa_trc(port->fcs, port->port_cfg.pwwn);
2578 
2579 	/*
2580 	 * Sanity Checks
2581 	 */
2582 	if (req_status != BFA_STATUS_OK) {
2583 		bfa_trc(port->fcs, req_status);
2584 		bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_ERROR);
2585 		return;
2586 	}
2587 
2588 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
2589 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
2590 
2591 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
2592 		bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_OK);
2593 		return;
2594 	}
2595 
2596 	bfa_trc(port->fcs, cthdr->reason_code);
2597 	bfa_trc(port->fcs, cthdr->exp_code);
2598 	bfa_sm_send_event(fdmi, FDMISM_EVENT_RSP_ERROR);
2599 }
2600 
2601 static void
2602 bfa_fcs_lport_fdmi_timeout(void *arg)
2603 {
2604 	struct bfa_fcs_lport_fdmi_s *fdmi = (struct bfa_fcs_lport_fdmi_s *) arg;
2605 
2606 	bfa_sm_send_event(fdmi, FDMISM_EVENT_TIMEOUT);
2607 }
2608 
2609 static void
2610 bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_lport_fdmi_s *fdmi,
2611 			 struct bfa_fcs_fdmi_hba_attr_s *hba_attr)
2612 {
2613 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2614 	struct bfa_fcs_driver_info_s  *driver_info = &port->fcs->driver_info;
2615 	struct bfa_fcs_fdmi_port_attr_s fcs_port_attr;
2616 
2617 	memset(hba_attr, 0, sizeof(struct bfa_fcs_fdmi_hba_attr_s));
2618 
2619 	bfa_ioc_get_adapter_manufacturer(&port->fcs->bfa->ioc,
2620 					hba_attr->manufacturer);
2621 	bfa_ioc_get_adapter_serial_num(&port->fcs->bfa->ioc,
2622 					hba_attr->serial_num);
2623 	bfa_ioc_get_adapter_model(&port->fcs->bfa->ioc,
2624 					hba_attr->model);
2625 	bfa_ioc_get_adapter_model(&port->fcs->bfa->ioc,
2626 					hba_attr->model_desc);
2627 	bfa_ioc_get_pci_chip_rev(&port->fcs->bfa->ioc,
2628 					hba_attr->hw_version);
2629 	bfa_ioc_get_adapter_optrom_ver(&port->fcs->bfa->ioc,
2630 					hba_attr->option_rom_ver);
2631 	bfa_ioc_get_adapter_fw_ver(&port->fcs->bfa->ioc,
2632 					hba_attr->fw_version);
2633 
2634 	strncpy(hba_attr->driver_version, (char *)driver_info->version,
2635 		sizeof(hba_attr->driver_version));
2636 
2637 	strncpy(hba_attr->os_name, driver_info->host_os_name,
2638 		sizeof(hba_attr->os_name));
2639 
2640 	/*
2641 	 * If there is a patch level, append it
2642 	 * to the os name along with a separator
2643 	 */
2644 	if (driver_info->host_os_patch[0] != '\0') {
2645 		strncat(hba_attr->os_name, BFA_FCS_PORT_SYMBNAME_SEPARATOR,
2646 			sizeof(BFA_FCS_PORT_SYMBNAME_SEPARATOR));
2647 		strncat(hba_attr->os_name, driver_info->host_os_patch,
2648 				sizeof(driver_info->host_os_patch));
2649 	}
2650 
2651 	/* Retrieve the max frame size from the port attr */
2652 	bfa_fcs_fdmi_get_portattr(fdmi, &fcs_port_attr);
2653 	hba_attr->max_ct_pyld = fcs_port_attr.max_frm_size;
2654 
2655 	strncpy(hba_attr->node_sym_name.symname,
2656 		port->port_cfg.node_sym_name.symname, BFA_SYMNAME_MAXLEN);
2657 	strcpy(hba_attr->vendor_info, "QLogic");
2658 	hba_attr->num_ports =
2659 		cpu_to_be32(bfa_ioc_get_nports(&port->fcs->bfa->ioc));
2660 	hba_attr->fabric_name = port->fabric->lps->pr_nwwn;
2661 	strncpy(hba_attr->bios_ver, hba_attr->option_rom_ver, BFA_VERSION_LEN);
2662 
2663 }
2664 
2665 static void
2666 bfa_fcs_fdmi_get_portattr(struct bfa_fcs_lport_fdmi_s *fdmi,
2667 			  struct bfa_fcs_fdmi_port_attr_s *port_attr)
2668 {
2669 	struct bfa_fcs_lport_s *port = fdmi->ms->port;
2670 	struct bfa_fcs_driver_info_s  *driver_info = &port->fcs->driver_info;
2671 	struct bfa_port_attr_s pport_attr;
2672 	struct bfa_lport_attr_s lport_attr;
2673 
2674 	memset(port_attr, 0, sizeof(struct bfa_fcs_fdmi_port_attr_s));
2675 
2676 	/*
2677 	 * get pport attributes from hal
2678 	 */
2679 	bfa_fcport_get_attr(port->fcs->bfa, &pport_attr);
2680 
2681 	/*
2682 	 * get FC4 type Bitmask
2683 	 */
2684 	fc_get_fc4type_bitmask(FC_TYPE_FCP, port_attr->supp_fc4_types);
2685 
2686 	/*
2687 	 * Supported Speeds
2688 	 */
2689 	switch (pport_attr.speed_supported) {
2690 	case BFA_PORT_SPEED_16GBPS:
2691 		port_attr->supp_speed =
2692 			cpu_to_be32(BFA_FCS_FDMI_SUPP_SPEEDS_16G);
2693 		break;
2694 
2695 	case BFA_PORT_SPEED_10GBPS:
2696 		port_attr->supp_speed =
2697 			cpu_to_be32(BFA_FCS_FDMI_SUPP_SPEEDS_10G);
2698 		break;
2699 
2700 	case BFA_PORT_SPEED_8GBPS:
2701 		port_attr->supp_speed =
2702 			cpu_to_be32(BFA_FCS_FDMI_SUPP_SPEEDS_8G);
2703 		break;
2704 
2705 	case BFA_PORT_SPEED_4GBPS:
2706 		port_attr->supp_speed =
2707 			cpu_to_be32(BFA_FCS_FDMI_SUPP_SPEEDS_4G);
2708 		break;
2709 
2710 	default:
2711 		bfa_sm_fault(port->fcs, pport_attr.speed_supported);
2712 	}
2713 
2714 	/*
2715 	 * Current Speed
2716 	 */
2717 	port_attr->curr_speed = cpu_to_be32(
2718 				bfa_fcs_fdmi_convert_speed(pport_attr.speed));
2719 
2720 	/*
2721 	 * Max PDU Size.
2722 	 */
2723 	port_attr->max_frm_size = cpu_to_be32(pport_attr.pport_cfg.maxfrsize);
2724 
2725 	/*
2726 	 * OS device Name
2727 	 */
2728 	strncpy(port_attr->os_device_name, (char *)driver_info->os_device_name,
2729 		sizeof(port_attr->os_device_name));
2730 
2731 	/*
2732 	 * Host name
2733 	 */
2734 	strncpy(port_attr->host_name, (char *)driver_info->host_machine_name,
2735 		sizeof(port_attr->host_name));
2736 
2737 	port_attr->node_name = bfa_fcs_lport_get_nwwn(port);
2738 	port_attr->port_name = bfa_fcs_lport_get_pwwn(port);
2739 
2740 	strncpy(port_attr->port_sym_name.symname,
2741 		(char *)&bfa_fcs_lport_get_psym_name(port), BFA_SYMNAME_MAXLEN);
2742 	bfa_fcs_lport_get_attr(port, &lport_attr);
2743 	port_attr->port_type = cpu_to_be32(lport_attr.port_type);
2744 	port_attr->scos = pport_attr.cos_supported;
2745 	port_attr->port_fabric_name = port->fabric->lps->pr_nwwn;
2746 	fc_get_fc4type_bitmask(FC_TYPE_FCP, port_attr->port_act_fc4_type);
2747 	port_attr->port_state = cpu_to_be32(pport_attr.port_state);
2748 	port_attr->num_ports = cpu_to_be32(port->num_rports);
2749 }
2750 
2751 /*
2752  * Convert BFA speed to FDMI format.
2753  */
2754 u32
2755 bfa_fcs_fdmi_convert_speed(bfa_port_speed_t pport_speed)
2756 {
2757 	u32	ret;
2758 
2759 	switch (pport_speed) {
2760 	case BFA_PORT_SPEED_1GBPS:
2761 	case BFA_PORT_SPEED_2GBPS:
2762 		ret = pport_speed;
2763 		break;
2764 
2765 	case BFA_PORT_SPEED_4GBPS:
2766 		ret = FDMI_TRANS_SPEED_4G;
2767 		break;
2768 
2769 	case BFA_PORT_SPEED_8GBPS:
2770 		ret = FDMI_TRANS_SPEED_8G;
2771 		break;
2772 
2773 	case BFA_PORT_SPEED_10GBPS:
2774 		ret = FDMI_TRANS_SPEED_10G;
2775 		break;
2776 
2777 	case BFA_PORT_SPEED_16GBPS:
2778 		ret = FDMI_TRANS_SPEED_16G;
2779 		break;
2780 
2781 	default:
2782 		ret = FDMI_TRANS_SPEED_UNKNOWN;
2783 	}
2784 	return ret;
2785 }
2786 
2787 void
2788 bfa_fcs_lport_fdmi_init(struct bfa_fcs_lport_ms_s *ms)
2789 {
2790 	struct bfa_fcs_lport_fdmi_s *fdmi = &ms->fdmi;
2791 
2792 	fdmi->ms = ms;
2793 	if (ms->port->fcs->fdmi_enabled)
2794 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_offline);
2795 	else
2796 		bfa_sm_set_state(fdmi, bfa_fcs_lport_fdmi_sm_disabled);
2797 }
2798 
2799 void
2800 bfa_fcs_lport_fdmi_offline(struct bfa_fcs_lport_ms_s *ms)
2801 {
2802 	struct bfa_fcs_lport_fdmi_s *fdmi = &ms->fdmi;
2803 
2804 	fdmi->ms = ms;
2805 	bfa_sm_send_event(fdmi, FDMISM_EVENT_PORT_OFFLINE);
2806 }
2807 
2808 void
2809 bfa_fcs_lport_fdmi_online(struct bfa_fcs_lport_ms_s *ms)
2810 {
2811 	struct bfa_fcs_lport_fdmi_s *fdmi = &ms->fdmi;
2812 
2813 	fdmi->ms = ms;
2814 	bfa_sm_send_event(fdmi, FDMISM_EVENT_PORT_ONLINE);
2815 }
2816 
2817 #define BFA_FCS_MS_CMD_MAX_RETRIES  2
2818 
2819 /*
2820  * forward declarations
2821  */
2822 static void     bfa_fcs_lport_ms_send_plogi(void *ms_cbarg,
2823 					   struct bfa_fcxp_s *fcxp_alloced);
2824 static void     bfa_fcs_lport_ms_timeout(void *arg);
2825 static void     bfa_fcs_lport_ms_plogi_response(void *fcsarg,
2826 					       struct bfa_fcxp_s *fcxp,
2827 					       void *cbarg,
2828 					       bfa_status_t req_status,
2829 					       u32 rsp_len,
2830 					       u32 resid_len,
2831 					       struct fchs_s *rsp_fchs);
2832 
2833 static void	bfa_fcs_lport_ms_send_gmal(void *ms_cbarg,
2834 					struct bfa_fcxp_s *fcxp_alloced);
2835 static void     bfa_fcs_lport_ms_gmal_response(void *fcsarg,
2836 					       struct bfa_fcxp_s *fcxp,
2837 					       void *cbarg,
2838 					       bfa_status_t req_status,
2839 					       u32 rsp_len,
2840 					       u32 resid_len,
2841 					       struct fchs_s *rsp_fchs);
2842 static void	bfa_fcs_lport_ms_send_gfn(void *ms_cbarg,
2843 					struct bfa_fcxp_s *fcxp_alloced);
2844 static void     bfa_fcs_lport_ms_gfn_response(void *fcsarg,
2845 					       struct bfa_fcxp_s *fcxp,
2846 					       void *cbarg,
2847 					       bfa_status_t req_status,
2848 					       u32 rsp_len,
2849 					       u32 resid_len,
2850 					       struct fchs_s *rsp_fchs);
2851 /*
2852  *  fcs_ms_sm FCS MS state machine
2853  */
2854 
2855 /*
2856  *  MS State Machine events
2857  */
2858 enum port_ms_event {
2859 	MSSM_EVENT_PORT_ONLINE = 1,
2860 	MSSM_EVENT_PORT_OFFLINE = 2,
2861 	MSSM_EVENT_RSP_OK = 3,
2862 	MSSM_EVENT_RSP_ERROR = 4,
2863 	MSSM_EVENT_TIMEOUT = 5,
2864 	MSSM_EVENT_FCXP_SENT = 6,
2865 	MSSM_EVENT_PORT_FABRIC_RSCN = 7
2866 };
2867 
2868 static void     bfa_fcs_lport_ms_sm_offline(struct bfa_fcs_lport_ms_s *ms,
2869 					   enum port_ms_event event);
2870 static void     bfa_fcs_lport_ms_sm_plogi_sending(struct bfa_fcs_lport_ms_s *ms,
2871 						 enum port_ms_event event);
2872 static void     bfa_fcs_lport_ms_sm_plogi(struct bfa_fcs_lport_ms_s *ms,
2873 					 enum port_ms_event event);
2874 static void     bfa_fcs_lport_ms_sm_plogi_retry(struct bfa_fcs_lport_ms_s *ms,
2875 					       enum port_ms_event event);
2876 static void     bfa_fcs_lport_ms_sm_gmal_sending(struct bfa_fcs_lport_ms_s *ms,
2877 						 enum port_ms_event event);
2878 static void     bfa_fcs_lport_ms_sm_gmal(struct bfa_fcs_lport_ms_s *ms,
2879 					 enum port_ms_event event);
2880 static void     bfa_fcs_lport_ms_sm_gmal_retry(struct bfa_fcs_lport_ms_s *ms,
2881 					       enum port_ms_event event);
2882 static void     bfa_fcs_lport_ms_sm_gfn_sending(struct bfa_fcs_lport_ms_s *ms,
2883 						 enum port_ms_event event);
2884 static void     bfa_fcs_lport_ms_sm_gfn(struct bfa_fcs_lport_ms_s *ms,
2885 					 enum port_ms_event event);
2886 static void     bfa_fcs_lport_ms_sm_gfn_retry(struct bfa_fcs_lport_ms_s *ms,
2887 					       enum port_ms_event event);
2888 static void     bfa_fcs_lport_ms_sm_online(struct bfa_fcs_lport_ms_s *ms,
2889 					  enum port_ms_event event);
2890 /*
2891  *	Start in offline state - awaiting NS to send start.
2892  */
2893 static void
2894 bfa_fcs_lport_ms_sm_offline(struct bfa_fcs_lport_ms_s *ms,
2895 				enum port_ms_event event)
2896 {
2897 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
2898 	bfa_trc(ms->port->fcs, event);
2899 
2900 	switch (event) {
2901 	case MSSM_EVENT_PORT_ONLINE:
2902 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_plogi_sending);
2903 		bfa_fcs_lport_ms_send_plogi(ms, NULL);
2904 		break;
2905 
2906 	case MSSM_EVENT_PORT_OFFLINE:
2907 		break;
2908 
2909 	default:
2910 		bfa_sm_fault(ms->port->fcs, event);
2911 	}
2912 }
2913 
2914 static void
2915 bfa_fcs_lport_ms_sm_plogi_sending(struct bfa_fcs_lport_ms_s *ms,
2916 				enum port_ms_event event)
2917 {
2918 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
2919 	bfa_trc(ms->port->fcs, event);
2920 
2921 	switch (event) {
2922 	case MSSM_EVENT_FCXP_SENT:
2923 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_plogi);
2924 		break;
2925 
2926 	case MSSM_EVENT_PORT_OFFLINE:
2927 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
2928 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ms->port),
2929 					   &ms->fcxp_wqe);
2930 		break;
2931 
2932 	default:
2933 		bfa_sm_fault(ms->port->fcs, event);
2934 	}
2935 }
2936 
2937 static void
2938 bfa_fcs_lport_ms_sm_plogi(struct bfa_fcs_lport_ms_s *ms,
2939 			enum port_ms_event event)
2940 {
2941 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
2942 	bfa_trc(ms->port->fcs, event);
2943 
2944 	switch (event) {
2945 	case MSSM_EVENT_RSP_ERROR:
2946 		/*
2947 		 * Start timer for a delayed retry
2948 		 */
2949 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_plogi_retry);
2950 		ms->port->stats.ms_retries++;
2951 		bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ms->port),
2952 				    &ms->timer, bfa_fcs_lport_ms_timeout, ms,
2953 				    BFA_FCS_RETRY_TIMEOUT);
2954 		break;
2955 
2956 	case MSSM_EVENT_RSP_OK:
2957 		/*
2958 		 * since plogi is done, now invoke MS related sub-modules
2959 		 */
2960 		bfa_fcs_lport_fdmi_online(ms);
2961 
2962 		/*
2963 		 * if this is a Vport, go to online state.
2964 		 */
2965 		if (ms->port->vport) {
2966 			bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_online);
2967 			break;
2968 		}
2969 
2970 		/*
2971 		 * For a base port we need to get the
2972 		 * switch's IP address.
2973 		 */
2974 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gmal_sending);
2975 		bfa_fcs_lport_ms_send_gmal(ms, NULL);
2976 		break;
2977 
2978 	case MSSM_EVENT_PORT_OFFLINE:
2979 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
2980 		bfa_fcxp_discard(ms->fcxp);
2981 		break;
2982 
2983 	default:
2984 		bfa_sm_fault(ms->port->fcs, event);
2985 	}
2986 }
2987 
2988 static void
2989 bfa_fcs_lport_ms_sm_plogi_retry(struct bfa_fcs_lport_ms_s *ms,
2990 			enum port_ms_event event)
2991 {
2992 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
2993 	bfa_trc(ms->port->fcs, event);
2994 
2995 	switch (event) {
2996 	case MSSM_EVENT_TIMEOUT:
2997 		/*
2998 		 * Retry Timer Expired. Re-send
2999 		 */
3000 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_plogi_sending);
3001 		bfa_fcs_lport_ms_send_plogi(ms, NULL);
3002 		break;
3003 
3004 	case MSSM_EVENT_PORT_OFFLINE:
3005 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3006 		bfa_timer_stop(&ms->timer);
3007 		break;
3008 
3009 	default:
3010 		bfa_sm_fault(ms->port->fcs, event);
3011 	}
3012 }
3013 
3014 static void
3015 bfa_fcs_lport_ms_sm_online(struct bfa_fcs_lport_ms_s *ms,
3016 			enum port_ms_event event)
3017 {
3018 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3019 	bfa_trc(ms->port->fcs, event);
3020 
3021 	switch (event) {
3022 	case MSSM_EVENT_PORT_OFFLINE:
3023 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3024 		break;
3025 
3026 	case MSSM_EVENT_PORT_FABRIC_RSCN:
3027 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gfn_sending);
3028 		ms->retry_cnt = 0;
3029 		bfa_fcs_lport_ms_send_gfn(ms, NULL);
3030 		break;
3031 
3032 	default:
3033 		bfa_sm_fault(ms->port->fcs, event);
3034 	}
3035 }
3036 
3037 static void
3038 bfa_fcs_lport_ms_sm_gmal_sending(struct bfa_fcs_lport_ms_s *ms,
3039 				enum port_ms_event event)
3040 {
3041 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3042 	bfa_trc(ms->port->fcs, event);
3043 
3044 	switch (event) {
3045 	case MSSM_EVENT_FCXP_SENT:
3046 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gmal);
3047 		break;
3048 
3049 	case MSSM_EVENT_PORT_OFFLINE:
3050 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3051 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ms->port),
3052 					   &ms->fcxp_wqe);
3053 		break;
3054 
3055 	default:
3056 		bfa_sm_fault(ms->port->fcs, event);
3057 	}
3058 }
3059 
3060 static void
3061 bfa_fcs_lport_ms_sm_gmal(struct bfa_fcs_lport_ms_s *ms,
3062 				enum port_ms_event event)
3063 {
3064 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3065 	bfa_trc(ms->port->fcs, event);
3066 
3067 	switch (event) {
3068 	case MSSM_EVENT_RSP_ERROR:
3069 		/*
3070 		 * Start timer for a delayed retry
3071 		 */
3072 		if (ms->retry_cnt++ < BFA_FCS_MS_CMD_MAX_RETRIES) {
3073 			bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gmal_retry);
3074 			ms->port->stats.ms_retries++;
3075 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ms->port),
3076 				&ms->timer, bfa_fcs_lport_ms_timeout, ms,
3077 				BFA_FCS_RETRY_TIMEOUT);
3078 		} else {
3079 			bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gfn_sending);
3080 			bfa_fcs_lport_ms_send_gfn(ms, NULL);
3081 			ms->retry_cnt = 0;
3082 		}
3083 		break;
3084 
3085 	case MSSM_EVENT_RSP_OK:
3086 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gfn_sending);
3087 		bfa_fcs_lport_ms_send_gfn(ms, NULL);
3088 		break;
3089 
3090 	case MSSM_EVENT_PORT_OFFLINE:
3091 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3092 		bfa_fcxp_discard(ms->fcxp);
3093 		break;
3094 
3095 	default:
3096 		bfa_sm_fault(ms->port->fcs, event);
3097 	}
3098 }
3099 
3100 static void
3101 bfa_fcs_lport_ms_sm_gmal_retry(struct bfa_fcs_lport_ms_s *ms,
3102 				enum port_ms_event event)
3103 {
3104 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3105 	bfa_trc(ms->port->fcs, event);
3106 
3107 	switch (event) {
3108 	case MSSM_EVENT_TIMEOUT:
3109 		/*
3110 		 * Retry Timer Expired. Re-send
3111 		 */
3112 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gmal_sending);
3113 		bfa_fcs_lport_ms_send_gmal(ms, NULL);
3114 		break;
3115 
3116 	case MSSM_EVENT_PORT_OFFLINE:
3117 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3118 		bfa_timer_stop(&ms->timer);
3119 		break;
3120 
3121 	default:
3122 		bfa_sm_fault(ms->port->fcs, event);
3123 	}
3124 }
3125 /*
3126  *  ms_pvt MS local functions
3127  */
3128 
3129 static void
3130 bfa_fcs_lport_ms_send_gmal(void *ms_cbarg, struct bfa_fcxp_s *fcxp_alloced)
3131 {
3132 	struct bfa_fcs_lport_ms_s *ms = ms_cbarg;
3133 	bfa_fcs_lport_t *port = ms->port;
3134 	struct fchs_s	fchs;
3135 	int		len;
3136 	struct bfa_fcxp_s *fcxp;
3137 
3138 	bfa_trc(port->fcs, port->pid);
3139 
3140 	fcxp = fcxp_alloced ? fcxp_alloced :
3141 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
3142 	if (!fcxp) {
3143 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ms->fcxp_wqe,
3144 				bfa_fcs_lport_ms_send_gmal, ms, BFA_TRUE);
3145 		return;
3146 	}
3147 	ms->fcxp = fcxp;
3148 
3149 	len = fc_gmal_req_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
3150 			     bfa_fcs_lport_get_fcid(port),
3151 				 port->fabric->lps->pr_nwwn);
3152 
3153 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
3154 			  FC_CLASS_3, len, &fchs,
3155 			  bfa_fcs_lport_ms_gmal_response, (void *)ms,
3156 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
3157 
3158 	bfa_sm_send_event(ms, MSSM_EVENT_FCXP_SENT);
3159 }
3160 
3161 static void
3162 bfa_fcs_lport_ms_gmal_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
3163 				void *cbarg, bfa_status_t req_status,
3164 				u32 rsp_len, u32 resid_len,
3165 				struct fchs_s *rsp_fchs)
3166 {
3167 	struct bfa_fcs_lport_ms_s *ms = (struct bfa_fcs_lport_ms_s *) cbarg;
3168 	bfa_fcs_lport_t *port = ms->port;
3169 	struct ct_hdr_s		*cthdr = NULL;
3170 	struct fcgs_gmal_resp_s *gmal_resp;
3171 	struct fcgs_gmal_entry_s *gmal_entry;
3172 	u32		num_entries;
3173 	u8			*rsp_str;
3174 
3175 	bfa_trc(port->fcs, req_status);
3176 	bfa_trc(port->fcs, port->port_cfg.pwwn);
3177 
3178 	/*
3179 	 * Sanity Checks
3180 	 */
3181 	if (req_status != BFA_STATUS_OK) {
3182 		bfa_trc(port->fcs, req_status);
3183 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3184 		return;
3185 	}
3186 
3187 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
3188 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
3189 
3190 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
3191 		gmal_resp = (struct fcgs_gmal_resp_s *)(cthdr + 1);
3192 
3193 		num_entries = be32_to_cpu(gmal_resp->ms_len);
3194 		if (num_entries == 0) {
3195 			bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3196 			return;
3197 		}
3198 		/*
3199 		* The response could contain multiple Entries.
3200 		* Entries for SNMP interface, etc.
3201 		* We look for the entry with a telnet prefix.
3202 		* First "http://" entry refers to IP addr
3203 		*/
3204 
3205 		gmal_entry = (struct fcgs_gmal_entry_s *)gmal_resp->ms_ma;
3206 		while (num_entries > 0) {
3207 			if (strncmp(gmal_entry->prefix,
3208 				CT_GMAL_RESP_PREFIX_HTTP,
3209 				sizeof(gmal_entry->prefix)) == 0) {
3210 
3211 				/*
3212 				* if the IP address is terminating with a '/',
3213 				* remove it.
3214 				* Byte 0 consists of the length of the string.
3215 				*/
3216 				rsp_str = &(gmal_entry->prefix[0]);
3217 				if (rsp_str[gmal_entry->len-1] == '/')
3218 					rsp_str[gmal_entry->len-1] = 0;
3219 
3220 				/* copy IP Address to fabric */
3221 				strncpy(bfa_fcs_lport_get_fabric_ipaddr(port),
3222 					gmal_entry->ip_addr,
3223 					BFA_FCS_FABRIC_IPADDR_SZ);
3224 				break;
3225 			} else {
3226 				--num_entries;
3227 				++gmal_entry;
3228 			}
3229 		}
3230 
3231 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_OK);
3232 		return;
3233 	}
3234 
3235 	bfa_trc(port->fcs, cthdr->reason_code);
3236 	bfa_trc(port->fcs, cthdr->exp_code);
3237 	bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3238 }
3239 
3240 static void
3241 bfa_fcs_lport_ms_sm_gfn_sending(struct bfa_fcs_lport_ms_s *ms,
3242 			enum port_ms_event event)
3243 {
3244 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3245 	bfa_trc(ms->port->fcs, event);
3246 
3247 	switch (event) {
3248 	case MSSM_EVENT_FCXP_SENT:
3249 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gfn);
3250 		break;
3251 
3252 	case MSSM_EVENT_PORT_OFFLINE:
3253 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3254 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ms->port),
3255 					   &ms->fcxp_wqe);
3256 		break;
3257 
3258 	default:
3259 		bfa_sm_fault(ms->port->fcs, event);
3260 	}
3261 }
3262 
3263 static void
3264 bfa_fcs_lport_ms_sm_gfn(struct bfa_fcs_lport_ms_s *ms,
3265 			enum port_ms_event event)
3266 {
3267 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3268 	bfa_trc(ms->port->fcs, event);
3269 
3270 	switch (event) {
3271 	case MSSM_EVENT_RSP_ERROR:
3272 		/*
3273 		 * Start timer for a delayed retry
3274 		 */
3275 		if (ms->retry_cnt++ < BFA_FCS_MS_CMD_MAX_RETRIES) {
3276 			bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gfn_retry);
3277 			ms->port->stats.ms_retries++;
3278 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ms->port),
3279 				&ms->timer, bfa_fcs_lport_ms_timeout, ms,
3280 				BFA_FCS_RETRY_TIMEOUT);
3281 		} else {
3282 			bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_online);
3283 			ms->retry_cnt = 0;
3284 		}
3285 		break;
3286 
3287 	case MSSM_EVENT_RSP_OK:
3288 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_online);
3289 		break;
3290 
3291 	case MSSM_EVENT_PORT_OFFLINE:
3292 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3293 		bfa_fcxp_discard(ms->fcxp);
3294 		break;
3295 
3296 	default:
3297 		bfa_sm_fault(ms->port->fcs, event);
3298 	}
3299 }
3300 
3301 static void
3302 bfa_fcs_lport_ms_sm_gfn_retry(struct bfa_fcs_lport_ms_s *ms,
3303 				enum port_ms_event event)
3304 {
3305 	bfa_trc(ms->port->fcs, ms->port->port_cfg.pwwn);
3306 	bfa_trc(ms->port->fcs, event);
3307 
3308 	switch (event) {
3309 	case MSSM_EVENT_TIMEOUT:
3310 		/*
3311 		 * Retry Timer Expired. Re-send
3312 		 */
3313 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_gfn_sending);
3314 		bfa_fcs_lport_ms_send_gfn(ms, NULL);
3315 		break;
3316 
3317 	case MSSM_EVENT_PORT_OFFLINE:
3318 		bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3319 		bfa_timer_stop(&ms->timer);
3320 		break;
3321 
3322 	default:
3323 		bfa_sm_fault(ms->port->fcs, event);
3324 	}
3325 }
3326 /*
3327  *  ms_pvt MS local functions
3328  */
3329 
3330 static void
3331 bfa_fcs_lport_ms_send_gfn(void *ms_cbarg, struct bfa_fcxp_s *fcxp_alloced)
3332 {
3333 	struct bfa_fcs_lport_ms_s *ms = ms_cbarg;
3334 	bfa_fcs_lport_t *port = ms->port;
3335 	struct fchs_s		fchs;
3336 	int			len;
3337 	struct bfa_fcxp_s *fcxp;
3338 
3339 	bfa_trc(port->fcs, port->pid);
3340 
3341 	fcxp = fcxp_alloced ? fcxp_alloced :
3342 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
3343 	if (!fcxp) {
3344 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ms->fcxp_wqe,
3345 				bfa_fcs_lport_ms_send_gfn, ms, BFA_TRUE);
3346 		return;
3347 	}
3348 	ms->fcxp = fcxp;
3349 
3350 	len = fc_gfn_req_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
3351 			     bfa_fcs_lport_get_fcid(port),
3352 				 port->fabric->lps->pr_nwwn);
3353 
3354 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
3355 			  FC_CLASS_3, len, &fchs,
3356 			  bfa_fcs_lport_ms_gfn_response, (void *)ms,
3357 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
3358 
3359 	bfa_sm_send_event(ms, MSSM_EVENT_FCXP_SENT);
3360 }
3361 
3362 static void
3363 bfa_fcs_lport_ms_gfn_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
3364 			void *cbarg, bfa_status_t req_status, u32 rsp_len,
3365 			u32 resid_len, struct fchs_s *rsp_fchs)
3366 {
3367 	struct bfa_fcs_lport_ms_s *ms = (struct bfa_fcs_lport_ms_s *) cbarg;
3368 	bfa_fcs_lport_t *port = ms->port;
3369 	struct ct_hdr_s	*cthdr = NULL;
3370 	wwn_t	       *gfn_resp;
3371 
3372 	bfa_trc(port->fcs, req_status);
3373 	bfa_trc(port->fcs, port->port_cfg.pwwn);
3374 
3375 	/*
3376 	 * Sanity Checks
3377 	 */
3378 	if (req_status != BFA_STATUS_OK) {
3379 		bfa_trc(port->fcs, req_status);
3380 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3381 		return;
3382 	}
3383 
3384 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
3385 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
3386 
3387 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
3388 		gfn_resp = (wwn_t *)(cthdr + 1);
3389 		/* check if it has actually changed */
3390 		if ((memcmp((void *)&bfa_fcs_lport_get_fabric_name(port),
3391 				gfn_resp, sizeof(wwn_t)) != 0)) {
3392 			bfa_fcs_fabric_set_fabric_name(port->fabric, *gfn_resp);
3393 		}
3394 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_OK);
3395 		return;
3396 	}
3397 
3398 	bfa_trc(port->fcs, cthdr->reason_code);
3399 	bfa_trc(port->fcs, cthdr->exp_code);
3400 	bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3401 }
3402 
3403 /*
3404  *  ms_pvt MS local functions
3405  */
3406 
3407 static void
3408 bfa_fcs_lport_ms_send_plogi(void *ms_cbarg, struct bfa_fcxp_s *fcxp_alloced)
3409 {
3410 	struct bfa_fcs_lport_ms_s *ms = ms_cbarg;
3411 	struct bfa_fcs_lport_s *port = ms->port;
3412 	struct fchs_s	fchs;
3413 	int	len;
3414 	struct bfa_fcxp_s *fcxp;
3415 
3416 	bfa_trc(port->fcs, port->pid);
3417 
3418 	fcxp = fcxp_alloced ? fcxp_alloced :
3419 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
3420 	if (!fcxp) {
3421 		port->stats.ms_plogi_alloc_wait++;
3422 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ms->fcxp_wqe,
3423 				bfa_fcs_lport_ms_send_plogi, ms, BFA_TRUE);
3424 		return;
3425 	}
3426 	ms->fcxp = fcxp;
3427 
3428 	len = fc_plogi_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
3429 			     bfa_hton3b(FC_MGMT_SERVER),
3430 			     bfa_fcs_lport_get_fcid(port), 0,
3431 			     port->port_cfg.pwwn, port->port_cfg.nwwn,
3432 			     bfa_fcport_get_maxfrsize(port->fcs->bfa),
3433 			     bfa_fcport_get_rx_bbcredit(port->fcs->bfa));
3434 
3435 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
3436 			  FC_CLASS_3, len, &fchs,
3437 			  bfa_fcs_lport_ms_plogi_response, (void *)ms,
3438 			  FC_MAX_PDUSZ, FC_ELS_TOV);
3439 
3440 	port->stats.ms_plogi_sent++;
3441 	bfa_sm_send_event(ms, MSSM_EVENT_FCXP_SENT);
3442 }
3443 
3444 static void
3445 bfa_fcs_lport_ms_plogi_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
3446 			void *cbarg, bfa_status_t req_status,
3447 			u32 rsp_len, u32 resid_len, struct fchs_s *rsp_fchs)
3448 {
3449 	struct bfa_fcs_lport_ms_s *ms = (struct bfa_fcs_lport_ms_s *) cbarg;
3450 	struct bfa_fcs_lport_s *port = ms->port;
3451 	struct fc_els_cmd_s *els_cmd;
3452 	struct fc_ls_rjt_s *ls_rjt;
3453 
3454 	bfa_trc(port->fcs, req_status);
3455 	bfa_trc(port->fcs, port->port_cfg.pwwn);
3456 
3457 	/*
3458 	 * Sanity Checks
3459 	 */
3460 	if (req_status != BFA_STATUS_OK) {
3461 		port->stats.ms_plogi_rsp_err++;
3462 		bfa_trc(port->fcs, req_status);
3463 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3464 		return;
3465 	}
3466 
3467 	els_cmd = (struct fc_els_cmd_s *) BFA_FCXP_RSP_PLD(fcxp);
3468 
3469 	switch (els_cmd->els_code) {
3470 
3471 	case FC_ELS_ACC:
3472 		if (rsp_len < sizeof(struct fc_logi_s)) {
3473 			bfa_trc(port->fcs, rsp_len);
3474 			port->stats.ms_plogi_acc_err++;
3475 			bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3476 			break;
3477 		}
3478 		port->stats.ms_plogi_accepts++;
3479 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_OK);
3480 		break;
3481 
3482 	case FC_ELS_LS_RJT:
3483 		ls_rjt = (struct fc_ls_rjt_s *) BFA_FCXP_RSP_PLD(fcxp);
3484 
3485 		bfa_trc(port->fcs, ls_rjt->reason_code);
3486 		bfa_trc(port->fcs, ls_rjt->reason_code_expl);
3487 
3488 		port->stats.ms_rejects++;
3489 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3490 		break;
3491 
3492 	default:
3493 		port->stats.ms_plogi_unknown_rsp++;
3494 		bfa_trc(port->fcs, els_cmd->els_code);
3495 		bfa_sm_send_event(ms, MSSM_EVENT_RSP_ERROR);
3496 	}
3497 }
3498 
3499 static void
3500 bfa_fcs_lport_ms_timeout(void *arg)
3501 {
3502 	struct bfa_fcs_lport_ms_s *ms = (struct bfa_fcs_lport_ms_s *) arg;
3503 
3504 	ms->port->stats.ms_timeouts++;
3505 	bfa_sm_send_event(ms, MSSM_EVENT_TIMEOUT);
3506 }
3507 
3508 
3509 void
3510 bfa_fcs_lport_ms_init(struct bfa_fcs_lport_s *port)
3511 {
3512 	struct bfa_fcs_lport_ms_s *ms = BFA_FCS_GET_MS_FROM_PORT(port);
3513 
3514 	ms->port = port;
3515 	bfa_sm_set_state(ms, bfa_fcs_lport_ms_sm_offline);
3516 
3517 	/*
3518 	 * Invoke init routines of sub modules.
3519 	 */
3520 	bfa_fcs_lport_fdmi_init(ms);
3521 }
3522 
3523 void
3524 bfa_fcs_lport_ms_offline(struct bfa_fcs_lport_s *port)
3525 {
3526 	struct bfa_fcs_lport_ms_s *ms = BFA_FCS_GET_MS_FROM_PORT(port);
3527 
3528 	ms->port = port;
3529 	bfa_sm_send_event(ms, MSSM_EVENT_PORT_OFFLINE);
3530 	bfa_fcs_lport_fdmi_offline(ms);
3531 }
3532 
3533 void
3534 bfa_fcs_lport_ms_online(struct bfa_fcs_lport_s *port)
3535 {
3536 	struct bfa_fcs_lport_ms_s *ms = BFA_FCS_GET_MS_FROM_PORT(port);
3537 
3538 	ms->port = port;
3539 	bfa_sm_send_event(ms, MSSM_EVENT_PORT_ONLINE);
3540 }
3541 void
3542 bfa_fcs_lport_ms_fabric_rscn(struct bfa_fcs_lport_s *port)
3543 {
3544 	struct bfa_fcs_lport_ms_s *ms = BFA_FCS_GET_MS_FROM_PORT(port);
3545 
3546 	/* todo.  Handle this only  when in Online state */
3547 	if (bfa_sm_cmp_state(ms, bfa_fcs_lport_ms_sm_online))
3548 		bfa_sm_send_event(ms, MSSM_EVENT_PORT_FABRIC_RSCN);
3549 }
3550 
3551 /*
3552  * @page ns_sm_info VPORT NS State Machine
3553  *
3554  * @section ns_sm_interactions VPORT NS State Machine Interactions
3555  *
3556  * @section ns_sm VPORT NS State Machine
3557  * img ns_sm.jpg
3558  */
3559 
3560 /*
3561  * forward declarations
3562  */
3563 static void     bfa_fcs_lport_ns_send_plogi(void *ns_cbarg,
3564 					   struct bfa_fcxp_s *fcxp_alloced);
3565 static void     bfa_fcs_lport_ns_send_rspn_id(void *ns_cbarg,
3566 					     struct bfa_fcxp_s *fcxp_alloced);
3567 static void     bfa_fcs_lport_ns_send_rft_id(void *ns_cbarg,
3568 					    struct bfa_fcxp_s *fcxp_alloced);
3569 static void     bfa_fcs_lport_ns_send_rff_id(void *ns_cbarg,
3570 					    struct bfa_fcxp_s *fcxp_alloced);
3571 static void     bfa_fcs_lport_ns_send_gid_ft(void *ns_cbarg,
3572 					    struct bfa_fcxp_s *fcxp_alloced);
3573 static void	bfa_fcs_lport_ns_send_rnn_id(void *ns_cbarg,
3574 					struct bfa_fcxp_s *fcxp_alloced);
3575 static void	bfa_fcs_lport_ns_send_rsnn_nn(void *ns_cbarg,
3576 					struct bfa_fcxp_s *fcxp_alloced);
3577 static void     bfa_fcs_lport_ns_timeout(void *arg);
3578 static void     bfa_fcs_lport_ns_plogi_response(void *fcsarg,
3579 					       struct bfa_fcxp_s *fcxp,
3580 					       void *cbarg,
3581 					       bfa_status_t req_status,
3582 					       u32 rsp_len,
3583 					       u32 resid_len,
3584 					       struct fchs_s *rsp_fchs);
3585 static void     bfa_fcs_lport_ns_rspn_id_response(void *fcsarg,
3586 						 struct bfa_fcxp_s *fcxp,
3587 						 void *cbarg,
3588 						 bfa_status_t req_status,
3589 						 u32 rsp_len,
3590 						 u32 resid_len,
3591 						 struct fchs_s *rsp_fchs);
3592 static void     bfa_fcs_lport_ns_rft_id_response(void *fcsarg,
3593 						struct bfa_fcxp_s *fcxp,
3594 						void *cbarg,
3595 						bfa_status_t req_status,
3596 						u32 rsp_len,
3597 						u32 resid_len,
3598 						struct fchs_s *rsp_fchs);
3599 static void     bfa_fcs_lport_ns_rff_id_response(void *fcsarg,
3600 						struct bfa_fcxp_s *fcxp,
3601 						void *cbarg,
3602 						bfa_status_t req_status,
3603 						u32 rsp_len,
3604 						u32 resid_len,
3605 						struct fchs_s *rsp_fchs);
3606 static void     bfa_fcs_lport_ns_gid_ft_response(void *fcsarg,
3607 						struct bfa_fcxp_s *fcxp,
3608 						void *cbarg,
3609 						bfa_status_t req_status,
3610 						u32 rsp_len,
3611 						u32 resid_len,
3612 						struct fchs_s *rsp_fchs);
3613 static void     bfa_fcs_lport_ns_rnn_id_response(void *fcsarg,
3614 						struct bfa_fcxp_s *fcxp,
3615 						void *cbarg,
3616 						bfa_status_t req_status,
3617 						u32 rsp_len,
3618 						u32 resid_len,
3619 						struct fchs_s *rsp_fchs);
3620 static void     bfa_fcs_lport_ns_rsnn_nn_response(void *fcsarg,
3621 						struct bfa_fcxp_s *fcxp,
3622 						void *cbarg,
3623 						bfa_status_t req_status,
3624 						u32 rsp_len,
3625 						u32 resid_len,
3626 						struct fchs_s *rsp_fchs);
3627 static void     bfa_fcs_lport_ns_process_gidft_pids(
3628 				struct bfa_fcs_lport_s *port,
3629 				u32 *pid_buf, u32 n_pids);
3630 
3631 static void bfa_fcs_lport_ns_boot_target_disc(bfa_fcs_lport_t *port);
3632 /*
3633  *  fcs_ns_sm FCS nameserver interface state machine
3634  */
3635 
3636 /*
3637  * VPort NS State Machine events
3638  */
3639 enum vport_ns_event {
3640 	NSSM_EVENT_PORT_ONLINE = 1,
3641 	NSSM_EVENT_PORT_OFFLINE = 2,
3642 	NSSM_EVENT_PLOGI_SENT = 3,
3643 	NSSM_EVENT_RSP_OK = 4,
3644 	NSSM_EVENT_RSP_ERROR = 5,
3645 	NSSM_EVENT_TIMEOUT = 6,
3646 	NSSM_EVENT_NS_QUERY = 7,
3647 	NSSM_EVENT_RSPNID_SENT = 8,
3648 	NSSM_EVENT_RFTID_SENT = 9,
3649 	NSSM_EVENT_RFFID_SENT = 10,
3650 	NSSM_EVENT_GIDFT_SENT = 11,
3651 	NSSM_EVENT_RNNID_SENT = 12,
3652 	NSSM_EVENT_RSNN_NN_SENT = 13,
3653 };
3654 
3655 static void     bfa_fcs_lport_ns_sm_offline(struct bfa_fcs_lport_ns_s *ns,
3656 					   enum vport_ns_event event);
3657 static void     bfa_fcs_lport_ns_sm_plogi_sending(struct bfa_fcs_lport_ns_s *ns,
3658 						 enum vport_ns_event event);
3659 static void     bfa_fcs_lport_ns_sm_plogi(struct bfa_fcs_lport_ns_s *ns,
3660 					 enum vport_ns_event event);
3661 static void     bfa_fcs_lport_ns_sm_plogi_retry(struct bfa_fcs_lport_ns_s *ns,
3662 					       enum vport_ns_event event);
3663 static void     bfa_fcs_lport_ns_sm_sending_rspn_id(
3664 					struct bfa_fcs_lport_ns_s *ns,
3665 					enum vport_ns_event event);
3666 static void     bfa_fcs_lport_ns_sm_rspn_id(struct bfa_fcs_lport_ns_s *ns,
3667 					   enum vport_ns_event event);
3668 static void     bfa_fcs_lport_ns_sm_rspn_id_retry(struct bfa_fcs_lport_ns_s *ns,
3669 						 enum vport_ns_event event);
3670 static void     bfa_fcs_lport_ns_sm_sending_rft_id(
3671 					struct bfa_fcs_lport_ns_s *ns,
3672 					enum vport_ns_event event);
3673 static void     bfa_fcs_lport_ns_sm_rft_id_retry(struct bfa_fcs_lport_ns_s *ns,
3674 						enum vport_ns_event event);
3675 static void     bfa_fcs_lport_ns_sm_rft_id(struct bfa_fcs_lport_ns_s *ns,
3676 					  enum vport_ns_event event);
3677 static void     bfa_fcs_lport_ns_sm_sending_rff_id(
3678 					struct bfa_fcs_lport_ns_s *ns,
3679 					enum vport_ns_event event);
3680 static void     bfa_fcs_lport_ns_sm_rff_id_retry(struct bfa_fcs_lport_ns_s *ns,
3681 						enum vport_ns_event event);
3682 static void     bfa_fcs_lport_ns_sm_rff_id(struct bfa_fcs_lport_ns_s *ns,
3683 					  enum vport_ns_event event);
3684 static void     bfa_fcs_lport_ns_sm_sending_gid_ft(
3685 					struct bfa_fcs_lport_ns_s *ns,
3686 					enum vport_ns_event event);
3687 static void     bfa_fcs_lport_ns_sm_gid_ft(struct bfa_fcs_lport_ns_s *ns,
3688 					  enum vport_ns_event event);
3689 static void     bfa_fcs_lport_ns_sm_gid_ft_retry(struct bfa_fcs_lport_ns_s *ns,
3690 						enum vport_ns_event event);
3691 static void     bfa_fcs_lport_ns_sm_online(struct bfa_fcs_lport_ns_s *ns,
3692 					  enum vport_ns_event event);
3693 static void     bfa_fcs_lport_ns_sm_sending_rnn_id(
3694 					struct bfa_fcs_lport_ns_s *ns,
3695 					enum vport_ns_event event);
3696 static void     bfa_fcs_lport_ns_sm_rnn_id(struct bfa_fcs_lport_ns_s *ns,
3697 					enum vport_ns_event event);
3698 static void     bfa_fcs_lport_ns_sm_rnn_id_retry(struct bfa_fcs_lport_ns_s *ns,
3699 						enum vport_ns_event event);
3700 static void     bfa_fcs_lport_ns_sm_sending_rsnn_nn(
3701 					struct bfa_fcs_lport_ns_s *ns,
3702 					enum vport_ns_event event);
3703 static void     bfa_fcs_lport_ns_sm_rsnn_nn(struct bfa_fcs_lport_ns_s *ns,
3704 						enum vport_ns_event event);
3705 static void     bfa_fcs_lport_ns_sm_rsnn_nn_retry(
3706 					struct bfa_fcs_lport_ns_s *ns,
3707 					enum vport_ns_event event);
3708 /*
3709  *	Start in offline state - awaiting linkup
3710  */
3711 static void
3712 bfa_fcs_lport_ns_sm_offline(struct bfa_fcs_lport_ns_s *ns,
3713 			enum vport_ns_event event)
3714 {
3715 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3716 	bfa_trc(ns->port->fcs, event);
3717 
3718 	switch (event) {
3719 	case NSSM_EVENT_PORT_ONLINE:
3720 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_plogi_sending);
3721 		bfa_fcs_lport_ns_send_plogi(ns, NULL);
3722 		break;
3723 
3724 	case NSSM_EVENT_PORT_OFFLINE:
3725 		break;
3726 
3727 	default:
3728 		bfa_sm_fault(ns->port->fcs, event);
3729 	}
3730 }
3731 
3732 static void
3733 bfa_fcs_lport_ns_sm_plogi_sending(struct bfa_fcs_lport_ns_s *ns,
3734 			enum vport_ns_event event)
3735 {
3736 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3737 	bfa_trc(ns->port->fcs, event);
3738 
3739 	switch (event) {
3740 	case NSSM_EVENT_PLOGI_SENT:
3741 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_plogi);
3742 		break;
3743 
3744 	case NSSM_EVENT_PORT_OFFLINE:
3745 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3746 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
3747 					   &ns->fcxp_wqe);
3748 		break;
3749 
3750 	default:
3751 		bfa_sm_fault(ns->port->fcs, event);
3752 	}
3753 }
3754 
3755 static void
3756 bfa_fcs_lport_ns_sm_plogi(struct bfa_fcs_lport_ns_s *ns,
3757 			enum vport_ns_event event)
3758 {
3759 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3760 	bfa_trc(ns->port->fcs, event);
3761 
3762 	switch (event) {
3763 	case NSSM_EVENT_RSP_ERROR:
3764 		/*
3765 		 * Start timer for a delayed retry
3766 		 */
3767 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_plogi_retry);
3768 		ns->port->stats.ns_retries++;
3769 		bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
3770 				    &ns->timer, bfa_fcs_lport_ns_timeout, ns,
3771 				    BFA_FCS_RETRY_TIMEOUT);
3772 		break;
3773 
3774 	case NSSM_EVENT_RSP_OK:
3775 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rnn_id);
3776 		ns->num_rnnid_retries = 0;
3777 		bfa_fcs_lport_ns_send_rnn_id(ns, NULL);
3778 		break;
3779 
3780 	case NSSM_EVENT_PORT_OFFLINE:
3781 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3782 		bfa_fcxp_discard(ns->fcxp);
3783 		break;
3784 
3785 	default:
3786 		bfa_sm_fault(ns->port->fcs, event);
3787 	}
3788 }
3789 
3790 static void
3791 bfa_fcs_lport_ns_sm_plogi_retry(struct bfa_fcs_lport_ns_s *ns,
3792 				enum vport_ns_event event)
3793 {
3794 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3795 	bfa_trc(ns->port->fcs, event);
3796 
3797 	switch (event) {
3798 	case NSSM_EVENT_TIMEOUT:
3799 		/*
3800 		 * Retry Timer Expired. Re-send
3801 		 */
3802 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_plogi_sending);
3803 		bfa_fcs_lport_ns_send_plogi(ns, NULL);
3804 		break;
3805 
3806 	case NSSM_EVENT_PORT_OFFLINE:
3807 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3808 		bfa_timer_stop(&ns->timer);
3809 		break;
3810 
3811 	default:
3812 		bfa_sm_fault(ns->port->fcs, event);
3813 	}
3814 }
3815 
3816 static void
3817 bfa_fcs_lport_ns_sm_sending_rnn_id(struct bfa_fcs_lport_ns_s *ns,
3818 					enum vport_ns_event event)
3819 {
3820 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3821 	bfa_trc(ns->port->fcs, event);
3822 
3823 	switch (event) {
3824 	case NSSM_EVENT_RNNID_SENT:
3825 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rnn_id);
3826 		break;
3827 
3828 	case NSSM_EVENT_PORT_OFFLINE:
3829 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3830 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
3831 						&ns->fcxp_wqe);
3832 		break;
3833 	default:
3834 		bfa_sm_fault(ns->port->fcs, event);
3835 	}
3836 }
3837 
3838 static void
3839 bfa_fcs_lport_ns_sm_rnn_id(struct bfa_fcs_lport_ns_s *ns,
3840 				enum vport_ns_event event)
3841 {
3842 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3843 	bfa_trc(ns->port->fcs, event);
3844 
3845 	switch (event) {
3846 	case NSSM_EVENT_RSP_OK:
3847 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rsnn_nn);
3848 		ns->num_rnnid_retries = 0;
3849 		ns->num_rsnn_nn_retries = 0;
3850 		bfa_fcs_lport_ns_send_rsnn_nn(ns, NULL);
3851 		break;
3852 
3853 	case NSSM_EVENT_RSP_ERROR:
3854 		if (ns->num_rnnid_retries < BFA_FCS_MAX_NS_RETRIES) {
3855 			bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rnn_id_retry);
3856 			ns->port->stats.ns_retries++;
3857 			ns->num_rnnid_retries++;
3858 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
3859 				&ns->timer, bfa_fcs_lport_ns_timeout, ns,
3860 				BFA_FCS_RETRY_TIMEOUT);
3861 		} else {
3862 			bfa_sm_set_state(ns,
3863 				bfa_fcs_lport_ns_sm_sending_rspn_id);
3864 			bfa_fcs_lport_ns_send_rspn_id(ns, NULL);
3865 		}
3866 		break;
3867 
3868 	case NSSM_EVENT_PORT_OFFLINE:
3869 		bfa_fcxp_discard(ns->fcxp);
3870 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3871 		break;
3872 
3873 	default:
3874 		bfa_sm_fault(ns->port->fcs, event);
3875 	}
3876 }
3877 
3878 static void
3879 bfa_fcs_lport_ns_sm_rnn_id_retry(struct bfa_fcs_lport_ns_s *ns,
3880 				enum vport_ns_event event)
3881 {
3882 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3883 	bfa_trc(ns->port->fcs, event);
3884 
3885 	switch (event) {
3886 	case NSSM_EVENT_TIMEOUT:
3887 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rnn_id);
3888 		bfa_fcs_lport_ns_send_rnn_id(ns, NULL);
3889 		break;
3890 
3891 	case NSSM_EVENT_PORT_OFFLINE:
3892 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3893 		bfa_timer_stop(&ns->timer);
3894 		break;
3895 
3896 	default:
3897 		bfa_sm_fault(ns->port->fcs, event);
3898 	}
3899 }
3900 
3901 static void
3902 bfa_fcs_lport_ns_sm_sending_rsnn_nn(struct bfa_fcs_lport_ns_s *ns,
3903 					enum vport_ns_event event)
3904 {
3905 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3906 	bfa_trc(ns->port->fcs, event);
3907 
3908 	switch (event) {
3909 	case NSSM_EVENT_RSNN_NN_SENT:
3910 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rsnn_nn);
3911 		break;
3912 
3913 	case NSSM_EVENT_PORT_OFFLINE:
3914 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3915 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
3916 			&ns->fcxp_wqe);
3917 		break;
3918 
3919 	default:
3920 		bfa_sm_fault(ns->port->fcs, event);
3921 	}
3922 }
3923 
3924 static void
3925 bfa_fcs_lport_ns_sm_rsnn_nn(struct bfa_fcs_lport_ns_s *ns,
3926 				enum vport_ns_event event)
3927 {
3928 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3929 	bfa_trc(ns->port->fcs, event);
3930 
3931 	switch (event) {
3932 	case NSSM_EVENT_RSP_OK:
3933 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rspn_id);
3934 		ns->num_rsnn_nn_retries = 0;
3935 		bfa_fcs_lport_ns_send_rspn_id(ns, NULL);
3936 		break;
3937 
3938 	case NSSM_EVENT_RSP_ERROR:
3939 		if (ns->num_rsnn_nn_retries < BFA_FCS_MAX_NS_RETRIES) {
3940 			bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rsnn_nn_retry);
3941 			ns->port->stats.ns_retries++;
3942 			ns->num_rsnn_nn_retries++;
3943 			bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
3944 					&ns->timer, bfa_fcs_lport_ns_timeout,
3945 					ns, BFA_FCS_RETRY_TIMEOUT);
3946 		} else {
3947 			bfa_sm_set_state(ns,
3948 				bfa_fcs_lport_ns_sm_sending_rspn_id);
3949 			bfa_fcs_lport_ns_send_rspn_id(ns, NULL);
3950 		}
3951 		break;
3952 
3953 	case NSSM_EVENT_PORT_OFFLINE:
3954 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3955 		bfa_fcxp_discard(ns->fcxp);
3956 		break;
3957 
3958 	default:
3959 		bfa_sm_fault(ns->port->fcs, event);
3960 	}
3961 }
3962 
3963 static void
3964 bfa_fcs_lport_ns_sm_rsnn_nn_retry(struct bfa_fcs_lport_ns_s *ns,
3965 					enum vport_ns_event event)
3966 {
3967 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3968 	bfa_trc(ns->port->fcs, event);
3969 
3970 	switch (event) {
3971 	case NSSM_EVENT_TIMEOUT:
3972 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rsnn_nn);
3973 		bfa_fcs_lport_ns_send_rsnn_nn(ns, NULL);
3974 		break;
3975 
3976 	case NSSM_EVENT_PORT_OFFLINE:
3977 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
3978 		bfa_timer_stop(&ns->timer);
3979 		break;
3980 
3981 	default:
3982 		bfa_sm_fault(ns->port->fcs, event);
3983 	}
3984 }
3985 
3986 static void
3987 bfa_fcs_lport_ns_sm_sending_rspn_id(struct bfa_fcs_lport_ns_s *ns,
3988 				   enum vport_ns_event event)
3989 {
3990 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
3991 	bfa_trc(ns->port->fcs, event);
3992 
3993 	switch (event) {
3994 	case NSSM_EVENT_RSPNID_SENT:
3995 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rspn_id);
3996 		break;
3997 
3998 	case NSSM_EVENT_PORT_OFFLINE:
3999 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4000 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4001 					   &ns->fcxp_wqe);
4002 		break;
4003 
4004 	default:
4005 		bfa_sm_fault(ns->port->fcs, event);
4006 	}
4007 }
4008 
4009 static void
4010 bfa_fcs_lport_ns_sm_rspn_id(struct bfa_fcs_lport_ns_s *ns,
4011 			enum vport_ns_event event)
4012 {
4013 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4014 	bfa_trc(ns->port->fcs, event);
4015 
4016 	switch (event) {
4017 	case NSSM_EVENT_RSP_ERROR:
4018 		/*
4019 		 * Start timer for a delayed retry
4020 		 */
4021 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rspn_id_retry);
4022 		ns->port->stats.ns_retries++;
4023 		bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4024 				    &ns->timer, bfa_fcs_lport_ns_timeout, ns,
4025 				    BFA_FCS_RETRY_TIMEOUT);
4026 		break;
4027 
4028 	case NSSM_EVENT_RSP_OK:
4029 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rft_id);
4030 		bfa_fcs_lport_ns_send_rft_id(ns, NULL);
4031 		break;
4032 
4033 	case NSSM_EVENT_PORT_OFFLINE:
4034 		bfa_fcxp_discard(ns->fcxp);
4035 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4036 		break;
4037 
4038 	default:
4039 		bfa_sm_fault(ns->port->fcs, event);
4040 	}
4041 }
4042 
4043 static void
4044 bfa_fcs_lport_ns_sm_rspn_id_retry(struct bfa_fcs_lport_ns_s *ns,
4045 				enum vport_ns_event event)
4046 {
4047 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4048 	bfa_trc(ns->port->fcs, event);
4049 
4050 	switch (event) {
4051 	case NSSM_EVENT_TIMEOUT:
4052 		/*
4053 		 * Retry Timer Expired. Re-send
4054 		 */
4055 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rspn_id);
4056 		bfa_fcs_lport_ns_send_rspn_id(ns, NULL);
4057 		break;
4058 
4059 	case NSSM_EVENT_PORT_OFFLINE:
4060 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4061 		bfa_timer_stop(&ns->timer);
4062 		break;
4063 
4064 	default:
4065 		bfa_sm_fault(ns->port->fcs, event);
4066 	}
4067 }
4068 
4069 static void
4070 bfa_fcs_lport_ns_sm_sending_rft_id(struct bfa_fcs_lport_ns_s *ns,
4071 				  enum vport_ns_event event)
4072 {
4073 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4074 	bfa_trc(ns->port->fcs, event);
4075 
4076 	switch (event) {
4077 	case NSSM_EVENT_RFTID_SENT:
4078 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rft_id);
4079 		break;
4080 
4081 	case NSSM_EVENT_PORT_OFFLINE:
4082 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4083 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4084 					   &ns->fcxp_wqe);
4085 		break;
4086 
4087 	default:
4088 		bfa_sm_fault(ns->port->fcs, event);
4089 	}
4090 }
4091 
4092 static void
4093 bfa_fcs_lport_ns_sm_rft_id(struct bfa_fcs_lport_ns_s *ns,
4094 			enum vport_ns_event event)
4095 {
4096 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4097 	bfa_trc(ns->port->fcs, event);
4098 
4099 	switch (event) {
4100 	case NSSM_EVENT_RSP_OK:
4101 		/* Now move to register FC4 Features */
4102 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rff_id);
4103 		bfa_fcs_lport_ns_send_rff_id(ns, NULL);
4104 		break;
4105 
4106 	case NSSM_EVENT_RSP_ERROR:
4107 		/*
4108 		 * Start timer for a delayed retry
4109 		 */
4110 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rft_id_retry);
4111 		ns->port->stats.ns_retries++;
4112 		bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4113 				    &ns->timer, bfa_fcs_lport_ns_timeout, ns,
4114 				    BFA_FCS_RETRY_TIMEOUT);
4115 		break;
4116 
4117 	case NSSM_EVENT_PORT_OFFLINE:
4118 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4119 		bfa_fcxp_discard(ns->fcxp);
4120 		break;
4121 
4122 	default:
4123 		bfa_sm_fault(ns->port->fcs, event);
4124 	}
4125 }
4126 
4127 static void
4128 bfa_fcs_lport_ns_sm_rft_id_retry(struct bfa_fcs_lport_ns_s *ns,
4129 				enum vport_ns_event event)
4130 {
4131 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4132 	bfa_trc(ns->port->fcs, event);
4133 
4134 	switch (event) {
4135 	case NSSM_EVENT_TIMEOUT:
4136 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rft_id);
4137 		bfa_fcs_lport_ns_send_rft_id(ns, NULL);
4138 		break;
4139 
4140 	case NSSM_EVENT_PORT_OFFLINE:
4141 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4142 		bfa_timer_stop(&ns->timer);
4143 		break;
4144 
4145 	default:
4146 		bfa_sm_fault(ns->port->fcs, event);
4147 	}
4148 }
4149 
4150 static void
4151 bfa_fcs_lport_ns_sm_sending_rff_id(struct bfa_fcs_lport_ns_s *ns,
4152 				  enum vport_ns_event event)
4153 {
4154 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4155 	bfa_trc(ns->port->fcs, event);
4156 
4157 	switch (event) {
4158 	case NSSM_EVENT_RFFID_SENT:
4159 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rff_id);
4160 		break;
4161 
4162 	case NSSM_EVENT_PORT_OFFLINE:
4163 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4164 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4165 					   &ns->fcxp_wqe);
4166 		break;
4167 
4168 	default:
4169 		bfa_sm_fault(ns->port->fcs, event);
4170 	}
4171 }
4172 
4173 static void
4174 bfa_fcs_lport_ns_sm_rff_id(struct bfa_fcs_lport_ns_s *ns,
4175 			enum vport_ns_event event)
4176 {
4177 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4178 	bfa_trc(ns->port->fcs, event);
4179 
4180 	switch (event) {
4181 	case NSSM_EVENT_RSP_OK:
4182 
4183 		/*
4184 		 * If min cfg mode is enabled, we donot initiate rport
4185 		 * discovery with the fabric. Instead, we will retrieve the
4186 		 * boot targets from HAL/FW.
4187 		 */
4188 		if (__fcs_min_cfg(ns->port->fcs)) {
4189 			bfa_fcs_lport_ns_boot_target_disc(ns->port);
4190 			bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_online);
4191 			return;
4192 		}
4193 
4194 		/*
4195 		 * If the port role is Initiator Mode issue NS query.
4196 		 * If it is Target Mode, skip this and go to online.
4197 		 */
4198 		if (BFA_FCS_VPORT_IS_INITIATOR_MODE(ns->port)) {
4199 			bfa_sm_set_state(ns,
4200 				bfa_fcs_lport_ns_sm_sending_gid_ft);
4201 			bfa_fcs_lport_ns_send_gid_ft(ns, NULL);
4202 		}
4203 		/*
4204 		 * kick off mgmt srvr state machine
4205 		 */
4206 		bfa_fcs_lport_ms_online(ns->port);
4207 		break;
4208 
4209 	case NSSM_EVENT_RSP_ERROR:
4210 		/*
4211 		 * Start timer for a delayed retry
4212 		 */
4213 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_rff_id_retry);
4214 		ns->port->stats.ns_retries++;
4215 		bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4216 				    &ns->timer, bfa_fcs_lport_ns_timeout, ns,
4217 				    BFA_FCS_RETRY_TIMEOUT);
4218 		break;
4219 
4220 	case NSSM_EVENT_PORT_OFFLINE:
4221 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4222 		bfa_fcxp_discard(ns->fcxp);
4223 		break;
4224 
4225 	default:
4226 		bfa_sm_fault(ns->port->fcs, event);
4227 	}
4228 }
4229 
4230 static void
4231 bfa_fcs_lport_ns_sm_rff_id_retry(struct bfa_fcs_lport_ns_s *ns,
4232 				enum vport_ns_event event)
4233 {
4234 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4235 	bfa_trc(ns->port->fcs, event);
4236 
4237 	switch (event) {
4238 	case NSSM_EVENT_TIMEOUT:
4239 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_rff_id);
4240 		bfa_fcs_lport_ns_send_rff_id(ns, NULL);
4241 		break;
4242 
4243 	case NSSM_EVENT_PORT_OFFLINE:
4244 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4245 		bfa_timer_stop(&ns->timer);
4246 		break;
4247 
4248 	default:
4249 		bfa_sm_fault(ns->port->fcs, event);
4250 	}
4251 }
4252 static void
4253 bfa_fcs_lport_ns_sm_sending_gid_ft(struct bfa_fcs_lport_ns_s *ns,
4254 				  enum vport_ns_event event)
4255 {
4256 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4257 	bfa_trc(ns->port->fcs, event);
4258 
4259 	switch (event) {
4260 	case NSSM_EVENT_GIDFT_SENT:
4261 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_gid_ft);
4262 		break;
4263 
4264 	case NSSM_EVENT_PORT_OFFLINE:
4265 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4266 		bfa_fcxp_walloc_cancel(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4267 					   &ns->fcxp_wqe);
4268 		break;
4269 
4270 	default:
4271 		bfa_sm_fault(ns->port->fcs, event);
4272 	}
4273 }
4274 
4275 static void
4276 bfa_fcs_lport_ns_sm_gid_ft(struct bfa_fcs_lport_ns_s *ns,
4277 			enum vport_ns_event event)
4278 {
4279 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4280 	bfa_trc(ns->port->fcs, event);
4281 
4282 	switch (event) {
4283 	case NSSM_EVENT_RSP_OK:
4284 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_online);
4285 		break;
4286 
4287 	case NSSM_EVENT_RSP_ERROR:
4288 		/*
4289 		 * TBD: for certain reject codes, we don't need to retry
4290 		 */
4291 		/*
4292 		 * Start timer for a delayed retry
4293 		 */
4294 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_gid_ft_retry);
4295 		ns->port->stats.ns_retries++;
4296 		bfa_timer_start(BFA_FCS_GET_HAL_FROM_PORT(ns->port),
4297 				    &ns->timer, bfa_fcs_lport_ns_timeout, ns,
4298 				    BFA_FCS_RETRY_TIMEOUT);
4299 		break;
4300 
4301 	case NSSM_EVENT_PORT_OFFLINE:
4302 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4303 		bfa_fcxp_discard(ns->fcxp);
4304 		break;
4305 
4306 	case  NSSM_EVENT_NS_QUERY:
4307 		break;
4308 
4309 	default:
4310 		bfa_sm_fault(ns->port->fcs, event);
4311 	}
4312 }
4313 
4314 static void
4315 bfa_fcs_lport_ns_sm_gid_ft_retry(struct bfa_fcs_lport_ns_s *ns,
4316 				enum vport_ns_event event)
4317 {
4318 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4319 	bfa_trc(ns->port->fcs, event);
4320 
4321 	switch (event) {
4322 	case NSSM_EVENT_TIMEOUT:
4323 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_sending_gid_ft);
4324 		bfa_fcs_lport_ns_send_gid_ft(ns, NULL);
4325 		break;
4326 
4327 	case NSSM_EVENT_PORT_OFFLINE:
4328 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4329 		bfa_timer_stop(&ns->timer);
4330 		break;
4331 
4332 	default:
4333 		bfa_sm_fault(ns->port->fcs, event);
4334 	}
4335 }
4336 
4337 static void
4338 bfa_fcs_lport_ns_sm_online(struct bfa_fcs_lport_ns_s *ns,
4339 			enum vport_ns_event event)
4340 {
4341 	bfa_trc(ns->port->fcs, ns->port->port_cfg.pwwn);
4342 	bfa_trc(ns->port->fcs, event);
4343 
4344 	switch (event) {
4345 	case NSSM_EVENT_PORT_OFFLINE:
4346 		bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
4347 		break;
4348 
4349 	case NSSM_EVENT_NS_QUERY:
4350 		/*
4351 		 * If the port role is Initiator Mode issue NS query.
4352 		 * If it is Target Mode, skip this and go to online.
4353 		 */
4354 		if (BFA_FCS_VPORT_IS_INITIATOR_MODE(ns->port)) {
4355 			bfa_sm_set_state(ns,
4356 				bfa_fcs_lport_ns_sm_sending_gid_ft);
4357 			bfa_fcs_lport_ns_send_gid_ft(ns, NULL);
4358 		};
4359 		break;
4360 
4361 	default:
4362 		bfa_sm_fault(ns->port->fcs, event);
4363 	}
4364 }
4365 
4366 
4367 
4368 /*
4369  *  ns_pvt Nameserver local functions
4370  */
4371 
4372 static void
4373 bfa_fcs_lport_ns_send_plogi(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4374 {
4375 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4376 	struct bfa_fcs_lport_s *port = ns->port;
4377 	struct fchs_s fchs;
4378 	int             len;
4379 	struct bfa_fcxp_s *fcxp;
4380 
4381 	bfa_trc(port->fcs, port->pid);
4382 
4383 	fcxp = fcxp_alloced ? fcxp_alloced :
4384 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4385 	if (!fcxp) {
4386 		port->stats.ns_plogi_alloc_wait++;
4387 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4388 				bfa_fcs_lport_ns_send_plogi, ns, BFA_TRUE);
4389 		return;
4390 	}
4391 	ns->fcxp = fcxp;
4392 
4393 	len = fc_plogi_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4394 			     bfa_hton3b(FC_NAME_SERVER),
4395 			     bfa_fcs_lport_get_fcid(port), 0,
4396 			     port->port_cfg.pwwn, port->port_cfg.nwwn,
4397 			     bfa_fcport_get_maxfrsize(port->fcs->bfa),
4398 			     bfa_fcport_get_rx_bbcredit(port->fcs->bfa));
4399 
4400 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4401 			  FC_CLASS_3, len, &fchs,
4402 			  bfa_fcs_lport_ns_plogi_response, (void *)ns,
4403 			  FC_MAX_PDUSZ, FC_ELS_TOV);
4404 	port->stats.ns_plogi_sent++;
4405 
4406 	bfa_sm_send_event(ns, NSSM_EVENT_PLOGI_SENT);
4407 }
4408 
4409 static void
4410 bfa_fcs_lport_ns_plogi_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4411 			void *cbarg, bfa_status_t req_status, u32 rsp_len,
4412 		       u32 resid_len, struct fchs_s *rsp_fchs)
4413 {
4414 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4415 	struct bfa_fcs_lport_s *port = ns->port;
4416 	/* struct fc_logi_s *plogi_resp; */
4417 	struct fc_els_cmd_s *els_cmd;
4418 	struct fc_ls_rjt_s *ls_rjt;
4419 
4420 	bfa_trc(port->fcs, req_status);
4421 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4422 
4423 	/*
4424 	 * Sanity Checks
4425 	 */
4426 	if (req_status != BFA_STATUS_OK) {
4427 		bfa_trc(port->fcs, req_status);
4428 		port->stats.ns_plogi_rsp_err++;
4429 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4430 		return;
4431 	}
4432 
4433 	els_cmd = (struct fc_els_cmd_s *) BFA_FCXP_RSP_PLD(fcxp);
4434 
4435 	switch (els_cmd->els_code) {
4436 
4437 	case FC_ELS_ACC:
4438 		if (rsp_len < sizeof(struct fc_logi_s)) {
4439 			bfa_trc(port->fcs, rsp_len);
4440 			port->stats.ns_plogi_acc_err++;
4441 			bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4442 			break;
4443 		}
4444 		port->stats.ns_plogi_accepts++;
4445 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4446 		break;
4447 
4448 	case FC_ELS_LS_RJT:
4449 		ls_rjt = (struct fc_ls_rjt_s *) BFA_FCXP_RSP_PLD(fcxp);
4450 
4451 		bfa_trc(port->fcs, ls_rjt->reason_code);
4452 		bfa_trc(port->fcs, ls_rjt->reason_code_expl);
4453 
4454 		port->stats.ns_rejects++;
4455 
4456 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4457 		break;
4458 
4459 	default:
4460 		port->stats.ns_plogi_unknown_rsp++;
4461 		bfa_trc(port->fcs, els_cmd->els_code);
4462 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4463 	}
4464 }
4465 
4466 /*
4467  * Register node name for port_id
4468  */
4469 static void
4470 bfa_fcs_lport_ns_send_rnn_id(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4471 {
4472 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4473 	struct bfa_fcs_lport_s *port = ns->port;
4474 	struct fchs_s  fchs;
4475 	int	len;
4476 	struct bfa_fcxp_s *fcxp;
4477 
4478 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4479 
4480 	fcxp = fcxp_alloced ? fcxp_alloced :
4481 			bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4482 	if (!fcxp) {
4483 		port->stats.ns_rnnid_alloc_wait++;
4484 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4485 				bfa_fcs_lport_ns_send_rnn_id, ns, BFA_TRUE);
4486 		return;
4487 	}
4488 
4489 	ns->fcxp = fcxp;
4490 
4491 	len = fc_rnnid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4492 				bfa_fcs_lport_get_fcid(port),
4493 				bfa_fcs_lport_get_fcid(port),
4494 				bfa_fcs_lport_get_nwwn(port));
4495 
4496 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4497 			  FC_CLASS_3, len, &fchs,
4498 			  bfa_fcs_lport_ns_rnn_id_response, (void *)ns,
4499 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
4500 
4501 	port->stats.ns_rnnid_sent++;
4502 	bfa_sm_send_event(ns, NSSM_EVENT_RNNID_SENT);
4503 }
4504 
4505 static void
4506 bfa_fcs_lport_ns_rnn_id_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4507 				void *cbarg, bfa_status_t req_status,
4508 				u32 rsp_len, u32 resid_len,
4509 				struct fchs_s *rsp_fchs)
4510 
4511 {
4512 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4513 	struct bfa_fcs_lport_s *port = ns->port;
4514 	struct ct_hdr_s	*cthdr = NULL;
4515 
4516 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4517 
4518 	/*
4519 	 * Sanity Checks
4520 	 */
4521 	if (req_status != BFA_STATUS_OK) {
4522 		bfa_trc(port->fcs, req_status);
4523 		port->stats.ns_rnnid_rsp_err++;
4524 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4525 		return;
4526 	}
4527 
4528 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
4529 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
4530 
4531 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
4532 		port->stats.ns_rnnid_accepts++;
4533 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4534 		return;
4535 	}
4536 
4537 	port->stats.ns_rnnid_rejects++;
4538 	bfa_trc(port->fcs, cthdr->reason_code);
4539 	bfa_trc(port->fcs, cthdr->exp_code);
4540 	bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4541 }
4542 
4543 /*
4544  * Register the symbolic node name for a given node name.
4545  */
4546 static void
4547 bfa_fcs_lport_ns_send_rsnn_nn(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4548 {
4549 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4550 	struct bfa_fcs_lport_s *port = ns->port;
4551 	struct fchs_s  fchs;
4552 	int     len;
4553 	struct bfa_fcxp_s *fcxp;
4554 	u8 *nsymbl;
4555 
4556 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4557 
4558 	fcxp = fcxp_alloced ? fcxp_alloced :
4559 			bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4560 	if (!fcxp) {
4561 		port->stats.ns_rsnn_nn_alloc_wait++;
4562 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4563 				bfa_fcs_lport_ns_send_rsnn_nn, ns, BFA_TRUE);
4564 		return;
4565 	}
4566 	ns->fcxp = fcxp;
4567 
4568 	nsymbl = (u8 *) &(bfa_fcs_lport_get_nsym_name(
4569 					bfa_fcs_get_base_port(port->fcs)));
4570 
4571 	len = fc_rsnn_nn_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4572 				bfa_fcs_lport_get_fcid(port),
4573 				bfa_fcs_lport_get_nwwn(port), nsymbl);
4574 
4575 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4576 			  FC_CLASS_3, len, &fchs,
4577 			  bfa_fcs_lport_ns_rsnn_nn_response, (void *)ns,
4578 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
4579 
4580 	port->stats.ns_rsnn_nn_sent++;
4581 
4582 	bfa_sm_send_event(ns, NSSM_EVENT_RSNN_NN_SENT);
4583 }
4584 
4585 static void
4586 bfa_fcs_lport_ns_rsnn_nn_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4587 				void *cbarg, bfa_status_t req_status,
4588 				u32 rsp_len, u32 resid_len,
4589 				struct fchs_s *rsp_fchs)
4590 {
4591 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4592 	struct bfa_fcs_lport_s *port = ns->port;
4593 	struct ct_hdr_s	*cthdr = NULL;
4594 
4595 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4596 
4597 	/*
4598 	 * Sanity Checks
4599 	 */
4600 	if (req_status != BFA_STATUS_OK) {
4601 		bfa_trc(port->fcs, req_status);
4602 		port->stats.ns_rsnn_nn_rsp_err++;
4603 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4604 		return;
4605 	}
4606 
4607 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
4608 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
4609 
4610 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
4611 		port->stats.ns_rsnn_nn_accepts++;
4612 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4613 		return;
4614 	}
4615 
4616 	port->stats.ns_rsnn_nn_rejects++;
4617 	bfa_trc(port->fcs, cthdr->reason_code);
4618 	bfa_trc(port->fcs, cthdr->exp_code);
4619 	bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4620 }
4621 
4622 /*
4623  * Register the symbolic port name.
4624  */
4625 static void
4626 bfa_fcs_lport_ns_send_rspn_id(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4627 {
4628 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4629 	struct bfa_fcs_lport_s *port = ns->port;
4630 	struct fchs_s fchs;
4631 	int             len;
4632 	struct bfa_fcxp_s *fcxp;
4633 	u8         symbl[256];
4634 	u8         *psymbl = &symbl[0];
4635 
4636 	memset(symbl, 0, sizeof(symbl));
4637 
4638 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4639 
4640 	fcxp = fcxp_alloced ? fcxp_alloced :
4641 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4642 	if (!fcxp) {
4643 		port->stats.ns_rspnid_alloc_wait++;
4644 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4645 				bfa_fcs_lport_ns_send_rspn_id, ns, BFA_TRUE);
4646 		return;
4647 	}
4648 	ns->fcxp = fcxp;
4649 
4650 	/*
4651 	 * for V-Port, form a Port Symbolic Name
4652 	 */
4653 	if (port->vport) {
4654 		/*
4655 		 * For Vports, we append the vport's port symbolic name
4656 		 * to that of the base port.
4657 		 */
4658 
4659 		strncpy((char *)psymbl,
4660 			(char *) &
4661 			(bfa_fcs_lport_get_psym_name
4662 			 (bfa_fcs_get_base_port(port->fcs))),
4663 			strlen((char *) &
4664 			       bfa_fcs_lport_get_psym_name(bfa_fcs_get_base_port
4665 							  (port->fcs))));
4666 
4667 		/* Ensure we have a null terminating string. */
4668 		((char *)psymbl)[strlen((char *) &
4669 			bfa_fcs_lport_get_psym_name(bfa_fcs_get_base_port
4670 						(port->fcs)))] = 0;
4671 		strncat((char *)psymbl,
4672 			(char *) &(bfa_fcs_lport_get_psym_name(port)),
4673 		strlen((char *) &bfa_fcs_lport_get_psym_name(port)));
4674 	} else {
4675 		psymbl = (u8 *) &(bfa_fcs_lport_get_psym_name(port));
4676 	}
4677 
4678 	len = fc_rspnid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4679 			      bfa_fcs_lport_get_fcid(port), 0, psymbl);
4680 
4681 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4682 			  FC_CLASS_3, len, &fchs,
4683 			  bfa_fcs_lport_ns_rspn_id_response, (void *)ns,
4684 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
4685 
4686 	port->stats.ns_rspnid_sent++;
4687 
4688 	bfa_sm_send_event(ns, NSSM_EVENT_RSPNID_SENT);
4689 }
4690 
4691 static void
4692 bfa_fcs_lport_ns_rspn_id_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4693 				 void *cbarg, bfa_status_t req_status,
4694 				 u32 rsp_len, u32 resid_len,
4695 				 struct fchs_s *rsp_fchs)
4696 {
4697 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4698 	struct bfa_fcs_lport_s *port = ns->port;
4699 	struct ct_hdr_s *cthdr = NULL;
4700 
4701 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4702 
4703 	/*
4704 	 * Sanity Checks
4705 	 */
4706 	if (req_status != BFA_STATUS_OK) {
4707 		bfa_trc(port->fcs, req_status);
4708 		port->stats.ns_rspnid_rsp_err++;
4709 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4710 		return;
4711 	}
4712 
4713 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
4714 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
4715 
4716 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
4717 		port->stats.ns_rspnid_accepts++;
4718 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4719 		return;
4720 	}
4721 
4722 	port->stats.ns_rspnid_rejects++;
4723 	bfa_trc(port->fcs, cthdr->reason_code);
4724 	bfa_trc(port->fcs, cthdr->exp_code);
4725 	bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4726 }
4727 
4728 /*
4729  * Register FC4-Types
4730  */
4731 static void
4732 bfa_fcs_lport_ns_send_rft_id(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4733 {
4734 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4735 	struct bfa_fcs_lport_s *port = ns->port;
4736 	struct fchs_s fchs;
4737 	int             len;
4738 	struct bfa_fcxp_s *fcxp;
4739 
4740 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4741 
4742 	fcxp = fcxp_alloced ? fcxp_alloced :
4743 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4744 	if (!fcxp) {
4745 		port->stats.ns_rftid_alloc_wait++;
4746 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4747 				bfa_fcs_lport_ns_send_rft_id, ns, BFA_TRUE);
4748 		return;
4749 	}
4750 	ns->fcxp = fcxp;
4751 
4752 	len = fc_rftid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4753 		     bfa_fcs_lport_get_fcid(port), 0, port->port_cfg.roles);
4754 
4755 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4756 			  FC_CLASS_3, len, &fchs,
4757 			  bfa_fcs_lport_ns_rft_id_response, (void *)ns,
4758 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
4759 
4760 	port->stats.ns_rftid_sent++;
4761 	bfa_sm_send_event(ns, NSSM_EVENT_RFTID_SENT);
4762 }
4763 
4764 static void
4765 bfa_fcs_lport_ns_rft_id_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4766 				void *cbarg, bfa_status_t req_status,
4767 				u32 rsp_len, u32 resid_len,
4768 				struct fchs_s *rsp_fchs)
4769 {
4770 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4771 	struct bfa_fcs_lport_s *port = ns->port;
4772 	struct ct_hdr_s *cthdr = NULL;
4773 
4774 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4775 
4776 	/*
4777 	 * Sanity Checks
4778 	 */
4779 	if (req_status != BFA_STATUS_OK) {
4780 		bfa_trc(port->fcs, req_status);
4781 		port->stats.ns_rftid_rsp_err++;
4782 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4783 		return;
4784 	}
4785 
4786 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
4787 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
4788 
4789 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
4790 		port->stats.ns_rftid_accepts++;
4791 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4792 		return;
4793 	}
4794 
4795 	port->stats.ns_rftid_rejects++;
4796 	bfa_trc(port->fcs, cthdr->reason_code);
4797 	bfa_trc(port->fcs, cthdr->exp_code);
4798 	bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4799 }
4800 
4801 /*
4802  * Register FC4-Features : Should be done after RFT_ID
4803  */
4804 static void
4805 bfa_fcs_lport_ns_send_rff_id(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4806 {
4807 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4808 	struct bfa_fcs_lport_s *port = ns->port;
4809 	struct fchs_s fchs;
4810 	int             len;
4811 	struct bfa_fcxp_s *fcxp;
4812 	u8			fc4_ftrs = 0;
4813 
4814 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4815 
4816 	fcxp = fcxp_alloced ? fcxp_alloced :
4817 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4818 	if (!fcxp) {
4819 		port->stats.ns_rffid_alloc_wait++;
4820 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4821 				bfa_fcs_lport_ns_send_rff_id, ns, BFA_TRUE);
4822 		return;
4823 	}
4824 	ns->fcxp = fcxp;
4825 
4826 	if (BFA_FCS_VPORT_IS_INITIATOR_MODE(ns->port))
4827 		fc4_ftrs = FC_GS_FCP_FC4_FEATURE_INITIATOR;
4828 
4829 	len = fc_rffid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4830 			     bfa_fcs_lport_get_fcid(port), 0,
4831 				 FC_TYPE_FCP, fc4_ftrs);
4832 
4833 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4834 			  FC_CLASS_3, len, &fchs,
4835 			  bfa_fcs_lport_ns_rff_id_response, (void *)ns,
4836 			  FC_MAX_PDUSZ, FC_FCCT_TOV);
4837 
4838 	port->stats.ns_rffid_sent++;
4839 	bfa_sm_send_event(ns, NSSM_EVENT_RFFID_SENT);
4840 }
4841 
4842 static void
4843 bfa_fcs_lport_ns_rff_id_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4844 				void *cbarg, bfa_status_t req_status,
4845 				u32 rsp_len, u32 resid_len,
4846 				struct fchs_s *rsp_fchs)
4847 {
4848 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4849 	struct bfa_fcs_lport_s *port = ns->port;
4850 	struct ct_hdr_s *cthdr = NULL;
4851 
4852 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4853 
4854 	/*
4855 	 * Sanity Checks
4856 	 */
4857 	if (req_status != BFA_STATUS_OK) {
4858 		bfa_trc(port->fcs, req_status);
4859 		port->stats.ns_rffid_rsp_err++;
4860 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4861 		return;
4862 	}
4863 
4864 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
4865 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
4866 
4867 	if (cthdr->cmd_rsp_code == CT_RSP_ACCEPT) {
4868 		port->stats.ns_rffid_accepts++;
4869 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4870 		return;
4871 	}
4872 
4873 	port->stats.ns_rffid_rejects++;
4874 	bfa_trc(port->fcs, cthdr->reason_code);
4875 	bfa_trc(port->fcs, cthdr->exp_code);
4876 
4877 	if (cthdr->reason_code == CT_RSN_NOT_SUPP) {
4878 		/* if this command is not supported, we don't retry */
4879 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4880 	} else
4881 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4882 }
4883 /*
4884  * Query Fabric for FC4-Types Devices.
4885  *
4886 * TBD : Need to use a local (FCS private) response buffer, since the response
4887  * can be larger than 2K.
4888  */
4889 static void
4890 bfa_fcs_lport_ns_send_gid_ft(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
4891 {
4892 	struct bfa_fcs_lport_ns_s *ns = ns_cbarg;
4893 	struct bfa_fcs_lport_s *port = ns->port;
4894 	struct fchs_s fchs;
4895 	int             len;
4896 	struct bfa_fcxp_s *fcxp;
4897 
4898 	bfa_trc(port->fcs, port->pid);
4899 
4900 	fcxp = fcxp_alloced ? fcxp_alloced :
4901 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
4902 	if (!fcxp) {
4903 		port->stats.ns_gidft_alloc_wait++;
4904 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
4905 				bfa_fcs_lport_ns_send_gid_ft, ns, BFA_TRUE);
4906 		return;
4907 	}
4908 	ns->fcxp = fcxp;
4909 
4910 	/*
4911 	 * This query is only initiated for FCP initiator mode.
4912 	 */
4913 	len = fc_gid_ft_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
4914 			      ns->port->pid, FC_TYPE_FCP);
4915 
4916 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
4917 			  FC_CLASS_3, len, &fchs,
4918 			  bfa_fcs_lport_ns_gid_ft_response, (void *)ns,
4919 			  bfa_fcxp_get_maxrsp(port->fcs->bfa), FC_FCCT_TOV);
4920 
4921 	port->stats.ns_gidft_sent++;
4922 
4923 	bfa_sm_send_event(ns, NSSM_EVENT_GIDFT_SENT);
4924 }
4925 
4926 static void
4927 bfa_fcs_lport_ns_gid_ft_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
4928 				void *cbarg, bfa_status_t req_status,
4929 				u32 rsp_len, u32 resid_len,
4930 				struct fchs_s *rsp_fchs)
4931 {
4932 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) cbarg;
4933 	struct bfa_fcs_lport_s *port = ns->port;
4934 	struct ct_hdr_s *cthdr = NULL;
4935 	u32        n_pids;
4936 
4937 	bfa_trc(port->fcs, port->port_cfg.pwwn);
4938 
4939 	/*
4940 	 * Sanity Checks
4941 	 */
4942 	if (req_status != BFA_STATUS_OK) {
4943 		bfa_trc(port->fcs, req_status);
4944 		port->stats.ns_gidft_rsp_err++;
4945 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4946 		return;
4947 	}
4948 
4949 	if (resid_len != 0) {
4950 		/*
4951 		 * TBD : we will need to allocate a larger buffer & retry the
4952 		 * command
4953 		 */
4954 		bfa_trc(port->fcs, rsp_len);
4955 		bfa_trc(port->fcs, resid_len);
4956 		return;
4957 	}
4958 
4959 	cthdr = (struct ct_hdr_s *) BFA_FCXP_RSP_PLD(fcxp);
4960 	cthdr->cmd_rsp_code = be16_to_cpu(cthdr->cmd_rsp_code);
4961 
4962 	switch (cthdr->cmd_rsp_code) {
4963 
4964 	case CT_RSP_ACCEPT:
4965 
4966 		port->stats.ns_gidft_accepts++;
4967 		n_pids = (fc_get_ctresp_pyld_len(rsp_len) / sizeof(u32));
4968 		bfa_trc(port->fcs, n_pids);
4969 		bfa_fcs_lport_ns_process_gidft_pids(port,
4970 						   (u32 *) (cthdr + 1),
4971 						   n_pids);
4972 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4973 		break;
4974 
4975 	case CT_RSP_REJECT:
4976 
4977 		/*
4978 		 * Check the reason code  & explanation.
4979 		 * There may not have been any FC4 devices in the fabric
4980 		 */
4981 		port->stats.ns_gidft_rejects++;
4982 		bfa_trc(port->fcs, cthdr->reason_code);
4983 		bfa_trc(port->fcs, cthdr->exp_code);
4984 
4985 		if ((cthdr->reason_code == CT_RSN_UNABLE_TO_PERF)
4986 		    && (cthdr->exp_code == CT_NS_EXP_FT_NOT_REG)) {
4987 
4988 			bfa_sm_send_event(ns, NSSM_EVENT_RSP_OK);
4989 		} else {
4990 			/*
4991 			 * for all other errors, retry
4992 			 */
4993 			bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
4994 		}
4995 		break;
4996 
4997 	default:
4998 		port->stats.ns_gidft_unknown_rsp++;
4999 		bfa_trc(port->fcs, cthdr->cmd_rsp_code);
5000 		bfa_sm_send_event(ns, NSSM_EVENT_RSP_ERROR);
5001 	}
5002 }
5003 
5004 /*
5005  *     This routine will be called by bfa_timer on timer timeouts.
5006  *
5007  *	param[in]	port - pointer to bfa_fcs_lport_t.
5008  *
5009  *	return
5010  *		void
5011  *
5012  *	Special Considerations:
5013  *
5014  *	note
5015  */
5016 static void
5017 bfa_fcs_lport_ns_timeout(void *arg)
5018 {
5019 	struct bfa_fcs_lport_ns_s *ns = (struct bfa_fcs_lport_ns_s *) arg;
5020 
5021 	ns->port->stats.ns_timeouts++;
5022 	bfa_sm_send_event(ns, NSSM_EVENT_TIMEOUT);
5023 }
5024 
5025 /*
5026  * Process the PID list in GID_FT response
5027  */
5028 static void
5029 bfa_fcs_lport_ns_process_gidft_pids(struct bfa_fcs_lport_s *port, u32 *pid_buf,
5030 				   u32 n_pids)
5031 {
5032 	struct fcgs_gidft_resp_s *gidft_entry;
5033 	struct bfa_fcs_rport_s *rport;
5034 	u32        ii;
5035 	struct bfa_fcs_fabric_s *fabric = port->fabric;
5036 	struct bfa_fcs_vport_s *vport;
5037 	struct list_head *qe;
5038 	u8 found = 0;
5039 
5040 	for (ii = 0; ii < n_pids; ii++) {
5041 		gidft_entry = (struct fcgs_gidft_resp_s *) &pid_buf[ii];
5042 
5043 		if (gidft_entry->pid == port->pid)
5044 			continue;
5045 
5046 		/*
5047 		 * Ignore PID if it is of base port
5048 		 * (Avoid vports discovering base port as remote port)
5049 		 */
5050 		if (gidft_entry->pid == fabric->bport.pid)
5051 			continue;
5052 
5053 		/*
5054 		 * Ignore PID if it is of vport created on the same base port
5055 		 * (Avoid vport discovering every other vport created on the
5056 		 * same port as remote port)
5057 		 */
5058 		list_for_each(qe, &fabric->vport_q) {
5059 			vport = (struct bfa_fcs_vport_s *) qe;
5060 			if (vport->lport.pid == gidft_entry->pid)
5061 				found = 1;
5062 		}
5063 
5064 		if (found) {
5065 			found = 0;
5066 			continue;
5067 		}
5068 
5069 		/*
5070 		 * Check if this rport already exists
5071 		 */
5072 		rport = bfa_fcs_lport_get_rport_by_pid(port, gidft_entry->pid);
5073 		if (rport == NULL) {
5074 			/*
5075 			 * this is a new device. create rport
5076 			 */
5077 			rport = bfa_fcs_rport_create(port, gidft_entry->pid);
5078 		} else {
5079 			/*
5080 			 * this rport already exists
5081 			 */
5082 			bfa_fcs_rport_scn(rport);
5083 		}
5084 
5085 		bfa_trc(port->fcs, gidft_entry->pid);
5086 
5087 		/*
5088 		 * if the last entry bit is set, bail out.
5089 		 */
5090 		if (gidft_entry->last)
5091 			return;
5092 	}
5093 }
5094 
5095 /*
5096  *  fcs_ns_public FCS nameserver public interfaces
5097  */
5098 
5099 /*
5100  * Functions called by port/fab.
5101  * These will send relevant Events to the ns state machine.
5102  */
5103 void
5104 bfa_fcs_lport_ns_init(struct bfa_fcs_lport_s *port)
5105 {
5106 	struct bfa_fcs_lport_ns_s *ns = BFA_FCS_GET_NS_FROM_PORT(port);
5107 
5108 	ns->port = port;
5109 	bfa_sm_set_state(ns, bfa_fcs_lport_ns_sm_offline);
5110 }
5111 
5112 void
5113 bfa_fcs_lport_ns_offline(struct bfa_fcs_lport_s *port)
5114 {
5115 	struct bfa_fcs_lport_ns_s *ns = BFA_FCS_GET_NS_FROM_PORT(port);
5116 
5117 	ns->port = port;
5118 	bfa_sm_send_event(ns, NSSM_EVENT_PORT_OFFLINE);
5119 }
5120 
5121 void
5122 bfa_fcs_lport_ns_online(struct bfa_fcs_lport_s *port)
5123 {
5124 	struct bfa_fcs_lport_ns_s *ns = BFA_FCS_GET_NS_FROM_PORT(port);
5125 
5126 	ns->port = port;
5127 	bfa_sm_send_event(ns, NSSM_EVENT_PORT_ONLINE);
5128 }
5129 
5130 void
5131 bfa_fcs_lport_ns_query(struct bfa_fcs_lport_s *port)
5132 {
5133 	struct bfa_fcs_lport_ns_s *ns = BFA_FCS_GET_NS_FROM_PORT(port);
5134 
5135 	bfa_trc(port->fcs, port->pid);
5136 	if (bfa_sm_cmp_state(ns, bfa_fcs_lport_ns_sm_online))
5137 		bfa_sm_send_event(ns, NSSM_EVENT_NS_QUERY);
5138 }
5139 
5140 static void
5141 bfa_fcs_lport_ns_boot_target_disc(bfa_fcs_lport_t *port)
5142 {
5143 
5144 	struct bfa_fcs_rport_s *rport;
5145 	u8 nwwns;
5146 	wwn_t  wwns[BFA_PREBOOT_BOOTLUN_MAX];
5147 	int ii;
5148 
5149 	bfa_iocfc_get_bootwwns(port->fcs->bfa, &nwwns, wwns);
5150 
5151 	for (ii = 0 ; ii < nwwns; ++ii) {
5152 		rport = bfa_fcs_rport_create_by_wwn(port, wwns[ii]);
5153 		WARN_ON(!rport);
5154 	}
5155 }
5156 
5157 void
5158 bfa_fcs_lport_ns_util_send_rspn_id(void *cbarg, struct bfa_fcxp_s *fcxp_alloced)
5159 {
5160 	struct bfa_fcs_lport_ns_s *ns = cbarg;
5161 	struct bfa_fcs_lport_s *port = ns->port;
5162 	struct fchs_s fchs;
5163 	struct bfa_fcxp_s *fcxp;
5164 	u8 symbl[256];
5165 	u8 *psymbl = &symbl[0];
5166 	int len;
5167 
5168 	/* Avoid sending RSPN in the following states. */
5169 	if (bfa_sm_cmp_state(ns, bfa_fcs_lport_ns_sm_offline) ||
5170 	    bfa_sm_cmp_state(ns, bfa_fcs_lport_ns_sm_plogi_sending) ||
5171 	    bfa_sm_cmp_state(ns, bfa_fcs_lport_ns_sm_plogi) ||
5172 	    bfa_sm_cmp_state(ns, bfa_fcs_lport_ns_sm_plogi_retry) ||
5173 	    bfa_sm_cmp_state(ns, bfa_fcs_lport_ns_sm_rspn_id_retry))
5174 		return;
5175 
5176 	memset(symbl, 0, sizeof(symbl));
5177 	bfa_trc(port->fcs, port->port_cfg.pwwn);
5178 
5179 	fcxp = fcxp_alloced ? fcxp_alloced :
5180 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
5181 	if (!fcxp) {
5182 		port->stats.ns_rspnid_alloc_wait++;
5183 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &ns->fcxp_wqe,
5184 			bfa_fcs_lport_ns_util_send_rspn_id, ns, BFA_FALSE);
5185 		return;
5186 	}
5187 
5188 	ns->fcxp = fcxp;
5189 
5190 	if (port->vport) {
5191 		/*
5192 		 * For Vports, we append the vport's port symbolic name
5193 		 * to that of the base port.
5194 		 */
5195 		strncpy((char *)psymbl, (char *)&(bfa_fcs_lport_get_psym_name
5196 			(bfa_fcs_get_base_port(port->fcs))),
5197 			strlen((char *)&bfa_fcs_lport_get_psym_name(
5198 			bfa_fcs_get_base_port(port->fcs))));
5199 
5200 		/* Ensure we have a null terminating string. */
5201 		((char *)psymbl)[strlen((char *)&bfa_fcs_lport_get_psym_name(
5202 		 bfa_fcs_get_base_port(port->fcs)))] = 0;
5203 
5204 		strncat((char *)psymbl,
5205 			(char *)&(bfa_fcs_lport_get_psym_name(port)),
5206 			strlen((char *)&bfa_fcs_lport_get_psym_name(port)));
5207 	}
5208 
5209 	len = fc_rspnid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
5210 			      bfa_fcs_lport_get_fcid(port), 0, psymbl);
5211 
5212 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
5213 		      FC_CLASS_3, len, &fchs, NULL, NULL, FC_MAX_PDUSZ, 0);
5214 
5215 	port->stats.ns_rspnid_sent++;
5216 }
5217 
5218 /*
5219  * FCS SCN
5220  */
5221 
5222 #define FC_QOS_RSCN_EVENT		0x0c
5223 #define FC_FABRIC_NAME_RSCN_EVENT	0x0d
5224 
5225 /*
5226  * forward declarations
5227  */
5228 static void     bfa_fcs_lport_scn_send_scr(void *scn_cbarg,
5229 					  struct bfa_fcxp_s *fcxp_alloced);
5230 static void     bfa_fcs_lport_scn_scr_response(void *fcsarg,
5231 					      struct bfa_fcxp_s *fcxp,
5232 					      void *cbarg,
5233 					      bfa_status_t req_status,
5234 					      u32 rsp_len,
5235 					      u32 resid_len,
5236 					      struct fchs_s *rsp_fchs);
5237 static void     bfa_fcs_lport_scn_send_ls_acc(struct bfa_fcs_lport_s *port,
5238 					     struct fchs_s *rx_fchs);
5239 static void     bfa_fcs_lport_scn_timeout(void *arg);
5240 
5241 /*
5242  *  fcs_scm_sm FCS SCN state machine
5243  */
5244 
5245 /*
5246  * VPort SCN State Machine events
5247  */
5248 enum port_scn_event {
5249 	SCNSM_EVENT_PORT_ONLINE = 1,
5250 	SCNSM_EVENT_PORT_OFFLINE = 2,
5251 	SCNSM_EVENT_RSP_OK = 3,
5252 	SCNSM_EVENT_RSP_ERROR = 4,
5253 	SCNSM_EVENT_TIMEOUT = 5,
5254 	SCNSM_EVENT_SCR_SENT = 6,
5255 };
5256 
5257 static void     bfa_fcs_lport_scn_sm_offline(struct bfa_fcs_lport_scn_s *scn,
5258 					    enum port_scn_event event);
5259 static void     bfa_fcs_lport_scn_sm_sending_scr(
5260 					struct bfa_fcs_lport_scn_s *scn,
5261 					enum port_scn_event event);
5262 static void     bfa_fcs_lport_scn_sm_scr(struct bfa_fcs_lport_scn_s *scn,
5263 					enum port_scn_event event);
5264 static void     bfa_fcs_lport_scn_sm_scr_retry(struct bfa_fcs_lport_scn_s *scn,
5265 					      enum port_scn_event event);
5266 static void     bfa_fcs_lport_scn_sm_online(struct bfa_fcs_lport_scn_s *scn,
5267 					   enum port_scn_event event);
5268 
5269 /*
5270  *	Starting state - awaiting link up.
5271  */
5272 static void
5273 bfa_fcs_lport_scn_sm_offline(struct bfa_fcs_lport_scn_s *scn,
5274 			enum port_scn_event event)
5275 {
5276 	switch (event) {
5277 	case SCNSM_EVENT_PORT_ONLINE:
5278 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_sending_scr);
5279 		bfa_fcs_lport_scn_send_scr(scn, NULL);
5280 		break;
5281 
5282 	case SCNSM_EVENT_PORT_OFFLINE:
5283 		break;
5284 
5285 	default:
5286 		bfa_sm_fault(scn->port->fcs, event);
5287 	}
5288 }
5289 
5290 static void
5291 bfa_fcs_lport_scn_sm_sending_scr(struct bfa_fcs_lport_scn_s *scn,
5292 				enum port_scn_event event)
5293 {
5294 	switch (event) {
5295 	case SCNSM_EVENT_SCR_SENT:
5296 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_scr);
5297 		break;
5298 
5299 	case SCNSM_EVENT_PORT_OFFLINE:
5300 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_offline);
5301 		bfa_fcxp_walloc_cancel(scn->port->fcs->bfa, &scn->fcxp_wqe);
5302 		break;
5303 
5304 	default:
5305 		bfa_sm_fault(scn->port->fcs, event);
5306 	}
5307 }
5308 
5309 static void
5310 bfa_fcs_lport_scn_sm_scr(struct bfa_fcs_lport_scn_s *scn,
5311 			enum port_scn_event event)
5312 {
5313 	struct bfa_fcs_lport_s *port = scn->port;
5314 
5315 	switch (event) {
5316 	case SCNSM_EVENT_RSP_OK:
5317 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_online);
5318 		break;
5319 
5320 	case SCNSM_EVENT_RSP_ERROR:
5321 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_scr_retry);
5322 		bfa_timer_start(port->fcs->bfa, &scn->timer,
5323 				    bfa_fcs_lport_scn_timeout, scn,
5324 				    BFA_FCS_RETRY_TIMEOUT);
5325 		break;
5326 
5327 	case SCNSM_EVENT_PORT_OFFLINE:
5328 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_offline);
5329 		bfa_fcxp_discard(scn->fcxp);
5330 		break;
5331 
5332 	default:
5333 		bfa_sm_fault(port->fcs, event);
5334 	}
5335 }
5336 
5337 static void
5338 bfa_fcs_lport_scn_sm_scr_retry(struct bfa_fcs_lport_scn_s *scn,
5339 				enum port_scn_event event)
5340 {
5341 	switch (event) {
5342 	case SCNSM_EVENT_TIMEOUT:
5343 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_sending_scr);
5344 		bfa_fcs_lport_scn_send_scr(scn, NULL);
5345 		break;
5346 
5347 	case SCNSM_EVENT_PORT_OFFLINE:
5348 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_offline);
5349 		bfa_timer_stop(&scn->timer);
5350 		break;
5351 
5352 	default:
5353 		bfa_sm_fault(scn->port->fcs, event);
5354 	}
5355 }
5356 
5357 static void
5358 bfa_fcs_lport_scn_sm_online(struct bfa_fcs_lport_scn_s *scn,
5359 			enum port_scn_event event)
5360 {
5361 	switch (event) {
5362 	case SCNSM_EVENT_PORT_OFFLINE:
5363 		bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_offline);
5364 		break;
5365 
5366 	default:
5367 		bfa_sm_fault(scn->port->fcs, event);
5368 	}
5369 }
5370 
5371 
5372 
5373 /*
5374  *  fcs_scn_private FCS SCN private functions
5375  */
5376 
5377 /*
5378  * This routine will be called to send a SCR command.
5379  */
5380 static void
5381 bfa_fcs_lport_scn_send_scr(void *scn_cbarg, struct bfa_fcxp_s *fcxp_alloced)
5382 {
5383 	struct bfa_fcs_lport_scn_s *scn = scn_cbarg;
5384 	struct bfa_fcs_lport_s *port = scn->port;
5385 	struct fchs_s fchs;
5386 	int             len;
5387 	struct bfa_fcxp_s *fcxp;
5388 
5389 	bfa_trc(port->fcs, port->pid);
5390 	bfa_trc(port->fcs, port->port_cfg.pwwn);
5391 
5392 	fcxp = fcxp_alloced ? fcxp_alloced :
5393 	       bfa_fcs_fcxp_alloc(port->fcs, BFA_TRUE);
5394 	if (!fcxp) {
5395 		bfa_fcs_fcxp_alloc_wait(port->fcs->bfa, &scn->fcxp_wqe,
5396 				bfa_fcs_lport_scn_send_scr, scn, BFA_TRUE);
5397 		return;
5398 	}
5399 	scn->fcxp = fcxp;
5400 
5401 	/* Handle VU registrations for Base port only */
5402 	if ((!port->vport) && bfa_ioc_get_fcmode(&port->fcs->bfa->ioc)) {
5403 		len = fc_scr_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
5404 				port->fabric->lps->brcd_switch,
5405 				port->pid, 0);
5406 	} else {
5407 	    len = fc_scr_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
5408 				    BFA_FALSE,
5409 				    port->pid, 0);
5410 	}
5411 
5412 	bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
5413 			  FC_CLASS_3, len, &fchs,
5414 			  bfa_fcs_lport_scn_scr_response,
5415 			  (void *)scn, FC_MAX_PDUSZ, FC_ELS_TOV);
5416 
5417 	bfa_sm_send_event(scn, SCNSM_EVENT_SCR_SENT);
5418 }
5419 
5420 static void
5421 bfa_fcs_lport_scn_scr_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
5422 			void *cbarg, bfa_status_t req_status, u32 rsp_len,
5423 			      u32 resid_len, struct fchs_s *rsp_fchs)
5424 {
5425 	struct bfa_fcs_lport_scn_s *scn = (struct bfa_fcs_lport_scn_s *) cbarg;
5426 	struct bfa_fcs_lport_s *port = scn->port;
5427 	struct fc_els_cmd_s *els_cmd;
5428 	struct fc_ls_rjt_s *ls_rjt;
5429 
5430 	bfa_trc(port->fcs, port->port_cfg.pwwn);
5431 
5432 	/*
5433 	 * Sanity Checks
5434 	 */
5435 	if (req_status != BFA_STATUS_OK) {
5436 		bfa_trc(port->fcs, req_status);
5437 		bfa_sm_send_event(scn, SCNSM_EVENT_RSP_ERROR);
5438 		return;
5439 	}
5440 
5441 	els_cmd = (struct fc_els_cmd_s *) BFA_FCXP_RSP_PLD(fcxp);
5442 
5443 	switch (els_cmd->els_code) {
5444 
5445 	case FC_ELS_ACC:
5446 		bfa_sm_send_event(scn, SCNSM_EVENT_RSP_OK);
5447 		break;
5448 
5449 	case FC_ELS_LS_RJT:
5450 
5451 		ls_rjt = (struct fc_ls_rjt_s *) BFA_FCXP_RSP_PLD(fcxp);
5452 
5453 		bfa_trc(port->fcs, ls_rjt->reason_code);
5454 		bfa_trc(port->fcs, ls_rjt->reason_code_expl);
5455 
5456 		bfa_sm_send_event(scn, SCNSM_EVENT_RSP_ERROR);
5457 		break;
5458 
5459 	default:
5460 		bfa_sm_send_event(scn, SCNSM_EVENT_RSP_ERROR);
5461 	}
5462 }
5463 
5464 /*
5465  * Send a LS Accept
5466  */
5467 static void
5468 bfa_fcs_lport_scn_send_ls_acc(struct bfa_fcs_lport_s *port,
5469 				struct fchs_s *rx_fchs)
5470 {
5471 	struct fchs_s fchs;
5472 	struct bfa_fcxp_s *fcxp;
5473 	struct bfa_rport_s *bfa_rport = NULL;
5474 	int             len;
5475 
5476 	bfa_trc(port->fcs, rx_fchs->s_id);
5477 
5478 	fcxp = bfa_fcs_fcxp_alloc(port->fcs, BFA_FALSE);
5479 	if (!fcxp)
5480 		return;
5481 
5482 	len = fc_ls_acc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
5483 			      rx_fchs->s_id, bfa_fcs_lport_get_fcid(port),
5484 			      rx_fchs->ox_id);
5485 
5486 	bfa_fcxp_send(fcxp, bfa_rport, port->fabric->vf_id, port->lp_tag,
5487 			  BFA_FALSE, FC_CLASS_3, len, &fchs, NULL, NULL,
5488 			  FC_MAX_PDUSZ, 0);
5489 }
5490 
5491 /*
5492  *     This routine will be called by bfa_timer on timer timeouts.
5493  *
5494  *	param[in]	vport		- pointer to bfa_fcs_lport_t.
5495  *	param[out]	vport_status	- pointer to return vport status in
5496  *
5497  *	return
5498  *		void
5499  *
5500  *	Special Considerations:
5501  *
5502  *	note
5503  */
5504 static void
5505 bfa_fcs_lport_scn_timeout(void *arg)
5506 {
5507 	struct bfa_fcs_lport_scn_s *scn = (struct bfa_fcs_lport_scn_s *) arg;
5508 
5509 	bfa_sm_send_event(scn, SCNSM_EVENT_TIMEOUT);
5510 }
5511 
5512 
5513 
5514 /*
5515  *  fcs_scn_public FCS state change notification public interfaces
5516  */
5517 
5518 /*
5519  * Functions called by port/fab
5520  */
5521 void
5522 bfa_fcs_lport_scn_init(struct bfa_fcs_lport_s *port)
5523 {
5524 	struct bfa_fcs_lport_scn_s *scn = BFA_FCS_GET_SCN_FROM_PORT(port);
5525 
5526 	scn->port = port;
5527 	bfa_sm_set_state(scn, bfa_fcs_lport_scn_sm_offline);
5528 }
5529 
5530 void
5531 bfa_fcs_lport_scn_offline(struct bfa_fcs_lport_s *port)
5532 {
5533 	struct bfa_fcs_lport_scn_s *scn = BFA_FCS_GET_SCN_FROM_PORT(port);
5534 
5535 	scn->port = port;
5536 	bfa_sm_send_event(scn, SCNSM_EVENT_PORT_OFFLINE);
5537 }
5538 
5539 void
5540 bfa_fcs_lport_fab_scn_online(struct bfa_fcs_lport_s *port)
5541 {
5542 	struct bfa_fcs_lport_scn_s *scn = BFA_FCS_GET_SCN_FROM_PORT(port);
5543 
5544 	scn->port = port;
5545 	bfa_sm_send_event(scn, SCNSM_EVENT_PORT_ONLINE);
5546 }
5547 
5548 static void
5549 bfa_fcs_lport_scn_portid_rscn(struct bfa_fcs_lport_s *port, u32 rpid)
5550 {
5551 	struct bfa_fcs_rport_s *rport;
5552 	struct bfa_fcs_fabric_s *fabric = port->fabric;
5553 	struct bfa_fcs_vport_s *vport;
5554 	struct list_head *qe;
5555 
5556 	bfa_trc(port->fcs, rpid);
5557 
5558 	/*
5559 	 * Ignore PID if it is of base port or of vports created on the
5560 	 * same base port. It is to avoid vports discovering base port or
5561 	 * other vports created on same base port as remote port
5562 	 */
5563 	if (rpid == fabric->bport.pid)
5564 		return;
5565 
5566 	list_for_each(qe, &fabric->vport_q) {
5567 		vport = (struct bfa_fcs_vport_s *) qe;
5568 		if (vport->lport.pid == rpid)
5569 			return;
5570 	}
5571 	/*
5572 	 * If this is an unknown device, then it just came online.
5573 	 * Otherwise let rport handle the RSCN event.
5574 	 */
5575 	rport = bfa_fcs_lport_get_rport_by_pid(port, rpid);
5576 	if (!rport)
5577 		rport = bfa_fcs_lport_get_rport_by_old_pid(port, rpid);
5578 
5579 	if (rport == NULL) {
5580 		/*
5581 		 * If min cfg mode is enabled, we donot need to
5582 		 * discover any new rports.
5583 		 */
5584 		if (!__fcs_min_cfg(port->fcs))
5585 			rport = bfa_fcs_rport_create(port, rpid);
5586 	} else
5587 		bfa_fcs_rport_scn(rport);
5588 }
5589 
5590 /*
5591  * rscn format based PID comparison
5592  */
5593 #define __fc_pid_match(__c0, __c1, __fmt)		\
5594 	(((__fmt) == FC_RSCN_FORMAT_FABRIC) ||		\
5595 	 (((__fmt) == FC_RSCN_FORMAT_DOMAIN) &&		\
5596 	  ((__c0)[0] == (__c1)[0])) ||				\
5597 	 (((__fmt) == FC_RSCN_FORMAT_AREA) &&		\
5598 	  ((__c0)[0] == (__c1)[0]) &&				\
5599 	  ((__c0)[1] == (__c1)[1])))
5600 
5601 static void
5602 bfa_fcs_lport_scn_multiport_rscn(struct bfa_fcs_lport_s *port,
5603 				enum fc_rscn_format format,
5604 				u32 rscn_pid)
5605 {
5606 	struct bfa_fcs_rport_s *rport;
5607 	struct list_head        *qe, *qe_next;
5608 	u8        *c0, *c1;
5609 
5610 	bfa_trc(port->fcs, format);
5611 	bfa_trc(port->fcs, rscn_pid);
5612 
5613 	c0 = (u8 *) &rscn_pid;
5614 
5615 	list_for_each_safe(qe, qe_next, &port->rport_q) {
5616 		rport = (struct bfa_fcs_rport_s *) qe;
5617 		c1 = (u8 *) &rport->pid;
5618 		if (__fc_pid_match(c0, c1, format))
5619 			bfa_fcs_rport_scn(rport);
5620 	}
5621 }
5622 
5623 
5624 void
5625 bfa_fcs_lport_scn_process_rscn(struct bfa_fcs_lport_s *port,
5626 			struct fchs_s *fchs, u32 len)
5627 {
5628 	struct fc_rscn_pl_s *rscn = (struct fc_rscn_pl_s *) (fchs + 1);
5629 	int             num_entries;
5630 	u32        rscn_pid;
5631 	bfa_boolean_t   nsquery = BFA_FALSE, found;
5632 	int             i = 0, j;
5633 
5634 	num_entries =
5635 		(be16_to_cpu(rscn->payldlen) -
5636 		 sizeof(u32)) / sizeof(rscn->event[0]);
5637 
5638 	bfa_trc(port->fcs, num_entries);
5639 
5640 	port->stats.num_rscn++;
5641 
5642 	bfa_fcs_lport_scn_send_ls_acc(port, fchs);
5643 
5644 	for (i = 0; i < num_entries; i++) {
5645 		rscn_pid = rscn->event[i].portid;
5646 
5647 		bfa_trc(port->fcs, rscn->event[i].format);
5648 		bfa_trc(port->fcs, rscn_pid);
5649 
5650 		/* check for duplicate entries in the list */
5651 		found = BFA_FALSE;
5652 		for (j = 0; j < i; j++) {
5653 			if (rscn->event[j].portid == rscn_pid) {
5654 				found = BFA_TRUE;
5655 				break;
5656 			}
5657 		}
5658 
5659 		/* if found in down the list, pid has been already processed */
5660 		if (found) {
5661 			bfa_trc(port->fcs, rscn_pid);
5662 			continue;
5663 		}
5664 
5665 		switch (rscn->event[i].format) {
5666 		case FC_RSCN_FORMAT_PORTID:
5667 			if (rscn->event[i].qualifier == FC_QOS_RSCN_EVENT) {
5668 				/*
5669 				 * Ignore this event.
5670 				 * f/w would have processed it
5671 				 */
5672 				bfa_trc(port->fcs, rscn_pid);
5673 			} else {
5674 				port->stats.num_portid_rscn++;
5675 				bfa_fcs_lport_scn_portid_rscn(port, rscn_pid);
5676 			}
5677 		break;
5678 
5679 		case FC_RSCN_FORMAT_FABRIC:
5680 			if (rscn->event[i].qualifier ==
5681 					FC_FABRIC_NAME_RSCN_EVENT) {
5682 				bfa_fcs_lport_ms_fabric_rscn(port);
5683 				break;
5684 			}
5685 			/* !!!!!!!!! Fall Through !!!!!!!!!!!!! */
5686 
5687 		case FC_RSCN_FORMAT_AREA:
5688 		case FC_RSCN_FORMAT_DOMAIN:
5689 			nsquery = BFA_TRUE;
5690 			bfa_fcs_lport_scn_multiport_rscn(port,
5691 							rscn->event[i].format,
5692 							rscn_pid);
5693 			break;
5694 
5695 
5696 		default:
5697 			WARN_ON(1);
5698 			nsquery = BFA_TRUE;
5699 		}
5700 	}
5701 
5702 	/*
5703 	 * If any of area, domain or fabric RSCN is received, do a fresh
5704 	 * discovery to find new devices.
5705 	 */
5706 	if (nsquery)
5707 		bfa_fcs_lport_ns_query(port);
5708 }
5709 
5710 /*
5711  * BFA FCS port
5712  */
5713 /*
5714  *  fcs_port_api BFA FCS port API
5715  */
5716 struct bfa_fcs_lport_s *
5717 bfa_fcs_get_base_port(struct bfa_fcs_s *fcs)
5718 {
5719 	return &fcs->fabric.bport;
5720 }
5721 
5722 wwn_t
5723 bfa_fcs_lport_get_rport(struct bfa_fcs_lport_s *port, wwn_t wwn, int index,
5724 		int nrports, bfa_boolean_t bwwn)
5725 {
5726 	struct list_head	*qh, *qe;
5727 	struct bfa_fcs_rport_s *rport = NULL;
5728 	int	i;
5729 	struct bfa_fcs_s	*fcs;
5730 
5731 	if (port == NULL || nrports == 0)
5732 		return (wwn_t) 0;
5733 
5734 	fcs = port->fcs;
5735 	bfa_trc(fcs, (u32) nrports);
5736 
5737 	i = 0;
5738 	qh = &port->rport_q;
5739 	qe = bfa_q_first(qh);
5740 
5741 	while ((qe != qh) && (i < nrports)) {
5742 		rport = (struct bfa_fcs_rport_s *) qe;
5743 		if (bfa_ntoh3b(rport->pid) > 0xFFF000) {
5744 			qe = bfa_q_next(qe);
5745 			bfa_trc(fcs, (u32) rport->pwwn);
5746 			bfa_trc(fcs, rport->pid);
5747 			bfa_trc(fcs, i);
5748 			continue;
5749 		}
5750 
5751 		if (bwwn) {
5752 			if (!memcmp(&wwn, &rport->pwwn, 8))
5753 				break;
5754 		} else {
5755 			if (i == index)
5756 				break;
5757 		}
5758 
5759 		i++;
5760 		qe = bfa_q_next(qe);
5761 	}
5762 
5763 	bfa_trc(fcs, i);
5764 	if (rport)
5765 		return rport->pwwn;
5766 	else
5767 		return (wwn_t) 0;
5768 }
5769 
5770 void
5771 bfa_fcs_lport_get_rport_quals(struct bfa_fcs_lport_s *port,
5772 		struct bfa_rport_qualifier_s rports[], int *nrports)
5773 {
5774 	struct list_head	*qh, *qe;
5775 	struct bfa_fcs_rport_s *rport = NULL;
5776 	int	i;
5777 	struct bfa_fcs_s	*fcs;
5778 
5779 	if (port == NULL || rports == NULL || *nrports == 0)
5780 		return;
5781 
5782 	fcs = port->fcs;
5783 	bfa_trc(fcs, (u32) *nrports);
5784 
5785 	i = 0;
5786 	qh = &port->rport_q;
5787 	qe = bfa_q_first(qh);
5788 
5789 	while ((qe != qh) && (i < *nrports)) {
5790 		rport = (struct bfa_fcs_rport_s *) qe;
5791 		if (bfa_ntoh3b(rport->pid) > 0xFFF000) {
5792 			qe = bfa_q_next(qe);
5793 			bfa_trc(fcs, (u32) rport->pwwn);
5794 			bfa_trc(fcs, rport->pid);
5795 			bfa_trc(fcs, i);
5796 			continue;
5797 		}
5798 
5799 		if (!rport->pwwn && !rport->pid) {
5800 			qe = bfa_q_next(qe);
5801 			continue;
5802 		}
5803 
5804 		rports[i].pwwn = rport->pwwn;
5805 		rports[i].pid = rport->pid;
5806 
5807 		i++;
5808 		qe = bfa_q_next(qe);
5809 	}
5810 
5811 	bfa_trc(fcs, i);
5812 	*nrports = i;
5813 }
5814 
5815 /*
5816  * Iterate's through all the rport's in the given port to
5817  * determine the maximum operating speed.
5818  *
5819  * !!!! To be used in TRL Functionality only !!!!
5820  */
5821 bfa_port_speed_t
5822 bfa_fcs_lport_get_rport_max_speed(bfa_fcs_lport_t *port)
5823 {
5824 	struct list_head *qh, *qe;
5825 	struct bfa_fcs_rport_s *rport = NULL;
5826 	struct bfa_fcs_s	*fcs;
5827 	bfa_port_speed_t max_speed = 0;
5828 	struct bfa_port_attr_s port_attr;
5829 	bfa_port_speed_t port_speed, rport_speed;
5830 	bfa_boolean_t trl_enabled = bfa_fcport_is_ratelim(port->fcs->bfa);
5831 
5832 
5833 	if (port == NULL)
5834 		return 0;
5835 
5836 	fcs = port->fcs;
5837 
5838 	/* Get Physical port's current speed */
5839 	bfa_fcport_get_attr(port->fcs->bfa, &port_attr);
5840 	port_speed = port_attr.speed;
5841 	bfa_trc(fcs, port_speed);
5842 
5843 	qh = &port->rport_q;
5844 	qe = bfa_q_first(qh);
5845 
5846 	while (qe != qh) {
5847 		rport = (struct bfa_fcs_rport_s *) qe;
5848 		if ((bfa_ntoh3b(rport->pid) > 0xFFF000) ||
5849 			(bfa_fcs_rport_get_state(rport) == BFA_RPORT_OFFLINE) ||
5850 			(rport->scsi_function != BFA_RPORT_TARGET)) {
5851 			qe = bfa_q_next(qe);
5852 			continue;
5853 		}
5854 
5855 		rport_speed = rport->rpf.rpsc_speed;
5856 		if ((trl_enabled) && (rport_speed ==
5857 			BFA_PORT_SPEED_UNKNOWN)) {
5858 			/* Use default ratelim speed setting */
5859 			rport_speed =
5860 				bfa_fcport_get_ratelim_speed(port->fcs->bfa);
5861 		}
5862 
5863 		if (rport_speed > max_speed)
5864 			max_speed = rport_speed;
5865 
5866 		qe = bfa_q_next(qe);
5867 	}
5868 
5869 	if (max_speed > port_speed)
5870 		max_speed = port_speed;
5871 
5872 	bfa_trc(fcs, max_speed);
5873 	return max_speed;
5874 }
5875 
5876 struct bfa_fcs_lport_s *
5877 bfa_fcs_lookup_port(struct bfa_fcs_s *fcs, u16 vf_id, wwn_t lpwwn)
5878 {
5879 	struct bfa_fcs_vport_s *vport;
5880 	bfa_fcs_vf_t   *vf;
5881 
5882 	WARN_ON(fcs == NULL);
5883 
5884 	vf = bfa_fcs_vf_lookup(fcs, vf_id);
5885 	if (vf == NULL) {
5886 		bfa_trc(fcs, vf_id);
5887 		return NULL;
5888 	}
5889 
5890 	if (!lpwwn || (vf->bport.port_cfg.pwwn == lpwwn))
5891 		return &vf->bport;
5892 
5893 	vport = bfa_fcs_fabric_vport_lookup(vf, lpwwn);
5894 	if (vport)
5895 		return &vport->lport;
5896 
5897 	return NULL;
5898 }
5899 
5900 /*
5901  *  API corresponding to NPIV_VPORT_GETINFO.
5902  */
5903 void
5904 bfa_fcs_lport_get_info(struct bfa_fcs_lport_s *port,
5905 	 struct bfa_lport_info_s *port_info)
5906 {
5907 
5908 	bfa_trc(port->fcs, port->fabric->fabric_name);
5909 
5910 	if (port->vport == NULL) {
5911 		/*
5912 		 * This is a Physical port
5913 		 */
5914 		port_info->port_type = BFA_LPORT_TYPE_PHYSICAL;
5915 
5916 		/*
5917 		 * @todo : need to fix the state & reason
5918 		 */
5919 		port_info->port_state = 0;
5920 		port_info->offline_reason = 0;
5921 
5922 		port_info->port_wwn = bfa_fcs_lport_get_pwwn(port);
5923 		port_info->node_wwn = bfa_fcs_lport_get_nwwn(port);
5924 
5925 		port_info->max_vports_supp =
5926 			bfa_lps_get_max_vport(port->fcs->bfa);
5927 		port_info->num_vports_inuse =
5928 			port->fabric->num_vports;
5929 		port_info->max_rports_supp = BFA_FCS_MAX_RPORTS_SUPP;
5930 		port_info->num_rports_inuse = port->num_rports;
5931 	} else {
5932 		/*
5933 		 * This is a virtual port
5934 		 */
5935 		port_info->port_type = BFA_LPORT_TYPE_VIRTUAL;
5936 
5937 		/*
5938 		 * @todo : need to fix the state & reason
5939 		 */
5940 		port_info->port_state = 0;
5941 		port_info->offline_reason = 0;
5942 
5943 		port_info->port_wwn = bfa_fcs_lport_get_pwwn(port);
5944 		port_info->node_wwn = bfa_fcs_lport_get_nwwn(port);
5945 	}
5946 }
5947 
5948 void
5949 bfa_fcs_lport_get_stats(struct bfa_fcs_lport_s *fcs_port,
5950 	 struct bfa_lport_stats_s *port_stats)
5951 {
5952 	*port_stats = fcs_port->stats;
5953 }
5954 
5955 void
5956 bfa_fcs_lport_clear_stats(struct bfa_fcs_lport_s *fcs_port)
5957 {
5958 	memset(&fcs_port->stats, 0, sizeof(struct bfa_lport_stats_s));
5959 }
5960 
5961 /*
5962  * Let new loop map create missing rports
5963  */
5964 void
5965 bfa_fcs_lport_lip_scn_online(struct bfa_fcs_lport_s *port)
5966 {
5967 	bfa_fcs_lport_loop_online(port);
5968 }
5969 
5970 /*
5971  * FCS virtual port state machine
5972  */
5973 
5974 #define __vport_fcs(__vp)       ((__vp)->lport.fcs)
5975 #define __vport_pwwn(__vp)      ((__vp)->lport.port_cfg.pwwn)
5976 #define __vport_nwwn(__vp)      ((__vp)->lport.port_cfg.nwwn)
5977 #define __vport_bfa(__vp)       ((__vp)->lport.fcs->bfa)
5978 #define __vport_fcid(__vp)      ((__vp)->lport.pid)
5979 #define __vport_fabric(__vp)    ((__vp)->lport.fabric)
5980 #define __vport_vfid(__vp)      ((__vp)->lport.fabric->vf_id)
5981 
5982 #define BFA_FCS_VPORT_MAX_RETRIES  5
5983 /*
5984  * Forward declarations
5985  */
5986 static void     bfa_fcs_vport_do_fdisc(struct bfa_fcs_vport_s *vport);
5987 static void     bfa_fcs_vport_timeout(void *vport_arg);
5988 static void     bfa_fcs_vport_do_logo(struct bfa_fcs_vport_s *vport);
5989 static void     bfa_fcs_vport_free(struct bfa_fcs_vport_s *vport);
5990 
5991 /*
5992  *  fcs_vport_sm FCS virtual port state machine
5993  */
5994 
5995 /*
5996  * VPort State Machine events
5997  */
5998 enum bfa_fcs_vport_event {
5999 	BFA_FCS_VPORT_SM_CREATE = 1,	/*  vport create event */
6000 	BFA_FCS_VPORT_SM_DELETE = 2,	/*  vport delete event */
6001 	BFA_FCS_VPORT_SM_START = 3,	/*  vport start request */
6002 	BFA_FCS_VPORT_SM_STOP = 4,	/*  stop: unsupported */
6003 	BFA_FCS_VPORT_SM_ONLINE = 5,	/*  fabric online */
6004 	BFA_FCS_VPORT_SM_OFFLINE = 6,	/*  fabric offline event */
6005 	BFA_FCS_VPORT_SM_FRMSENT = 7,	/*  fdisc/logo sent events */
6006 	BFA_FCS_VPORT_SM_RSP_OK = 8,	/*  good response */
6007 	BFA_FCS_VPORT_SM_RSP_ERROR = 9,	/*  error/bad response */
6008 	BFA_FCS_VPORT_SM_TIMEOUT = 10,	/*  delay timer event */
6009 	BFA_FCS_VPORT_SM_DELCOMP = 11,	/*  lport delete completion */
6010 	BFA_FCS_VPORT_SM_RSP_DUP_WWN = 12,	/*  Dup wnn error*/
6011 	BFA_FCS_VPORT_SM_RSP_FAILED = 13,	/*  non-retryable failure */
6012 	BFA_FCS_VPORT_SM_STOPCOMP = 14,	/* vport delete completion */
6013 	BFA_FCS_VPORT_SM_FABRIC_MAX = 15, /* max vports on fabric */
6014 };
6015 
6016 static void     bfa_fcs_vport_sm_uninit(struct bfa_fcs_vport_s *vport,
6017 					enum bfa_fcs_vport_event event);
6018 static void     bfa_fcs_vport_sm_created(struct bfa_fcs_vport_s *vport,
6019 					 enum bfa_fcs_vport_event event);
6020 static void     bfa_fcs_vport_sm_offline(struct bfa_fcs_vport_s *vport,
6021 					 enum bfa_fcs_vport_event event);
6022 static void     bfa_fcs_vport_sm_fdisc(struct bfa_fcs_vport_s *vport,
6023 				       enum bfa_fcs_vport_event event);
6024 static void     bfa_fcs_vport_sm_fdisc_retry(struct bfa_fcs_vport_s *vport,
6025 					     enum bfa_fcs_vport_event event);
6026 static void	bfa_fcs_vport_sm_fdisc_rsp_wait(struct bfa_fcs_vport_s *vport,
6027 					enum bfa_fcs_vport_event event);
6028 static void     bfa_fcs_vport_sm_online(struct bfa_fcs_vport_s *vport,
6029 					enum bfa_fcs_vport_event event);
6030 static void     bfa_fcs_vport_sm_deleting(struct bfa_fcs_vport_s *vport,
6031 					  enum bfa_fcs_vport_event event);
6032 static void     bfa_fcs_vport_sm_cleanup(struct bfa_fcs_vport_s *vport,
6033 					 enum bfa_fcs_vport_event event);
6034 static void     bfa_fcs_vport_sm_logo(struct bfa_fcs_vport_s *vport,
6035 				      enum bfa_fcs_vport_event event);
6036 static void     bfa_fcs_vport_sm_error(struct bfa_fcs_vport_s *vport,
6037 				      enum bfa_fcs_vport_event event);
6038 static void	bfa_fcs_vport_sm_stopping(struct bfa_fcs_vport_s *vport,
6039 					enum bfa_fcs_vport_event event);
6040 static void	bfa_fcs_vport_sm_logo_for_stop(struct bfa_fcs_vport_s *vport,
6041 					enum bfa_fcs_vport_event event);
6042 
6043 static struct bfa_sm_table_s  vport_sm_table[] = {
6044 	{BFA_SM(bfa_fcs_vport_sm_uninit), BFA_FCS_VPORT_UNINIT},
6045 	{BFA_SM(bfa_fcs_vport_sm_created), BFA_FCS_VPORT_CREATED},
6046 	{BFA_SM(bfa_fcs_vport_sm_offline), BFA_FCS_VPORT_OFFLINE},
6047 	{BFA_SM(bfa_fcs_vport_sm_fdisc), BFA_FCS_VPORT_FDISC},
6048 	{BFA_SM(bfa_fcs_vport_sm_fdisc_retry), BFA_FCS_VPORT_FDISC_RETRY},
6049 	{BFA_SM(bfa_fcs_vport_sm_fdisc_rsp_wait), BFA_FCS_VPORT_FDISC_RSP_WAIT},
6050 	{BFA_SM(bfa_fcs_vport_sm_online), BFA_FCS_VPORT_ONLINE},
6051 	{BFA_SM(bfa_fcs_vport_sm_deleting), BFA_FCS_VPORT_DELETING},
6052 	{BFA_SM(bfa_fcs_vport_sm_cleanup), BFA_FCS_VPORT_CLEANUP},
6053 	{BFA_SM(bfa_fcs_vport_sm_logo), BFA_FCS_VPORT_LOGO},
6054 	{BFA_SM(bfa_fcs_vport_sm_error), BFA_FCS_VPORT_ERROR}
6055 };
6056 
6057 /*
6058  * Beginning state.
6059  */
6060 static void
6061 bfa_fcs_vport_sm_uninit(struct bfa_fcs_vport_s *vport,
6062 			enum bfa_fcs_vport_event event)
6063 {
6064 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6065 	bfa_trc(__vport_fcs(vport), event);
6066 
6067 	switch (event) {
6068 	case BFA_FCS_VPORT_SM_CREATE:
6069 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_created);
6070 		bfa_fcs_fabric_addvport(__vport_fabric(vport), vport);
6071 		break;
6072 
6073 	default:
6074 		bfa_sm_fault(__vport_fcs(vport), event);
6075 	}
6076 }
6077 
6078 /*
6079  * Created state - a start event is required to start up the state machine.
6080  */
6081 static void
6082 bfa_fcs_vport_sm_created(struct bfa_fcs_vport_s *vport,
6083 			enum bfa_fcs_vport_event event)
6084 {
6085 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6086 	bfa_trc(__vport_fcs(vport), event);
6087 
6088 	switch (event) {
6089 	case BFA_FCS_VPORT_SM_START:
6090 		if (bfa_sm_cmp_state(__vport_fabric(vport),
6091 					bfa_fcs_fabric_sm_online)
6092 		    && bfa_fcs_fabric_npiv_capable(__vport_fabric(vport))) {
6093 			bfa_sm_set_state(vport, bfa_fcs_vport_sm_fdisc);
6094 			bfa_fcs_vport_do_fdisc(vport);
6095 		} else {
6096 			/*
6097 			 * Fabric is offline or not NPIV capable, stay in
6098 			 * offline state.
6099 			 */
6100 			vport->vport_stats.fab_no_npiv++;
6101 			bfa_sm_set_state(vport, bfa_fcs_vport_sm_offline);
6102 		}
6103 		break;
6104 
6105 	case BFA_FCS_VPORT_SM_DELETE:
6106 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6107 		bfa_fcs_lport_delete(&vport->lport);
6108 		break;
6109 
6110 	case BFA_FCS_VPORT_SM_ONLINE:
6111 	case BFA_FCS_VPORT_SM_OFFLINE:
6112 		/*
6113 		 * Ignore ONLINE/OFFLINE events from fabric
6114 		 * till vport is started.
6115 		 */
6116 		break;
6117 
6118 	default:
6119 		bfa_sm_fault(__vport_fcs(vport), event);
6120 	}
6121 }
6122 
6123 /*
6124  * Offline state - awaiting ONLINE event from fabric SM.
6125  */
6126 static void
6127 bfa_fcs_vport_sm_offline(struct bfa_fcs_vport_s *vport,
6128 			enum bfa_fcs_vport_event event)
6129 {
6130 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6131 	bfa_trc(__vport_fcs(vport), event);
6132 
6133 	switch (event) {
6134 	case BFA_FCS_VPORT_SM_DELETE:
6135 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6136 		bfa_fcs_lport_delete(&vport->lport);
6137 		break;
6138 
6139 	case BFA_FCS_VPORT_SM_ONLINE:
6140 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_fdisc);
6141 		vport->fdisc_retries = 0;
6142 		bfa_fcs_vport_do_fdisc(vport);
6143 		break;
6144 
6145 	case BFA_FCS_VPORT_SM_STOP:
6146 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6147 		bfa_sm_send_event(&vport->lport, BFA_FCS_PORT_SM_STOP);
6148 		break;
6149 
6150 	case BFA_FCS_VPORT_SM_OFFLINE:
6151 		/*
6152 		 * This can happen if the vport couldn't be initialzied
6153 		 * due the fact that the npiv was not enabled on the switch.
6154 		 * In that case we will put the vport in offline state.
6155 		 * However, the link can go down and cause the this event to
6156 		 * be sent when we are already offline. Ignore it.
6157 		 */
6158 		break;
6159 
6160 	default:
6161 		bfa_sm_fault(__vport_fcs(vport), event);
6162 	}
6163 }
6164 
6165 
6166 /*
6167  * FDISC is sent and awaiting reply from fabric.
6168  */
6169 static void
6170 bfa_fcs_vport_sm_fdisc(struct bfa_fcs_vport_s *vport,
6171 			enum bfa_fcs_vport_event event)
6172 {
6173 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6174 	bfa_trc(__vport_fcs(vport), event);
6175 
6176 	switch (event) {
6177 	case BFA_FCS_VPORT_SM_DELETE:
6178 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_fdisc_rsp_wait);
6179 		break;
6180 
6181 	case BFA_FCS_VPORT_SM_OFFLINE:
6182 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_offline);
6183 		bfa_sm_send_event(vport->lps, BFA_LPS_SM_OFFLINE);
6184 		break;
6185 
6186 	case BFA_FCS_VPORT_SM_RSP_OK:
6187 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_online);
6188 		bfa_fcs_lport_online(&vport->lport);
6189 		break;
6190 
6191 	case BFA_FCS_VPORT_SM_RSP_ERROR:
6192 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_fdisc_retry);
6193 		bfa_timer_start(__vport_bfa(vport), &vport->timer,
6194 				    bfa_fcs_vport_timeout, vport,
6195 				    BFA_FCS_RETRY_TIMEOUT);
6196 		break;
6197 
6198 	case BFA_FCS_VPORT_SM_RSP_FAILED:
6199 	case BFA_FCS_VPORT_SM_FABRIC_MAX:
6200 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_offline);
6201 		break;
6202 
6203 	case BFA_FCS_VPORT_SM_RSP_DUP_WWN:
6204 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_error);
6205 		break;
6206 
6207 	default:
6208 		bfa_sm_fault(__vport_fcs(vport), event);
6209 	}
6210 }
6211 
6212 /*
6213  * FDISC attempt failed - a timer is active to retry FDISC.
6214  */
6215 static void
6216 bfa_fcs_vport_sm_fdisc_retry(struct bfa_fcs_vport_s *vport,
6217 			     enum bfa_fcs_vport_event event)
6218 {
6219 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6220 	bfa_trc(__vport_fcs(vport), event);
6221 
6222 	switch (event) {
6223 	case BFA_FCS_VPORT_SM_DELETE:
6224 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6225 		bfa_timer_stop(&vport->timer);
6226 		bfa_fcs_lport_delete(&vport->lport);
6227 		break;
6228 
6229 	case BFA_FCS_VPORT_SM_OFFLINE:
6230 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_offline);
6231 		bfa_timer_stop(&vport->timer);
6232 		break;
6233 
6234 	case BFA_FCS_VPORT_SM_TIMEOUT:
6235 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_fdisc);
6236 		vport->vport_stats.fdisc_retries++;
6237 		vport->fdisc_retries++;
6238 		bfa_fcs_vport_do_fdisc(vport);
6239 		break;
6240 
6241 	default:
6242 		bfa_sm_fault(__vport_fcs(vport), event);
6243 	}
6244 }
6245 
6246 /*
6247  * FDISC is in progress and we got a vport delete request -
6248  * this is a wait state while we wait for fdisc response and
6249  * we will transition to the appropriate state - on rsp status.
6250  */
6251 static void
6252 bfa_fcs_vport_sm_fdisc_rsp_wait(struct bfa_fcs_vport_s *vport,
6253 				enum bfa_fcs_vport_event event)
6254 {
6255 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6256 	bfa_trc(__vport_fcs(vport), event);
6257 
6258 	switch (event) {
6259 	case BFA_FCS_VPORT_SM_RSP_OK:
6260 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_deleting);
6261 		bfa_fcs_lport_delete(&vport->lport);
6262 		break;
6263 
6264 	case BFA_FCS_VPORT_SM_DELETE:
6265 		break;
6266 
6267 	case BFA_FCS_VPORT_SM_OFFLINE:
6268 	case BFA_FCS_VPORT_SM_RSP_ERROR:
6269 	case BFA_FCS_VPORT_SM_RSP_FAILED:
6270 	case BFA_FCS_VPORT_SM_FABRIC_MAX:
6271 	case BFA_FCS_VPORT_SM_RSP_DUP_WWN:
6272 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6273 		bfa_sm_send_event(vport->lps, BFA_LPS_SM_OFFLINE);
6274 		bfa_fcs_lport_delete(&vport->lport);
6275 		break;
6276 
6277 	default:
6278 		bfa_sm_fault(__vport_fcs(vport), event);
6279 	}
6280 }
6281 
6282 /*
6283  * Vport is online (FDISC is complete).
6284  */
6285 static void
6286 bfa_fcs_vport_sm_online(struct bfa_fcs_vport_s *vport,
6287 			enum bfa_fcs_vport_event event)
6288 {
6289 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6290 	bfa_trc(__vport_fcs(vport), event);
6291 
6292 	switch (event) {
6293 	case BFA_FCS_VPORT_SM_DELETE:
6294 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_deleting);
6295 		bfa_fcs_lport_delete(&vport->lport);
6296 		break;
6297 
6298 	case BFA_FCS_VPORT_SM_STOP:
6299 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_stopping);
6300 		bfa_sm_send_event(&vport->lport, BFA_FCS_PORT_SM_STOP);
6301 		break;
6302 
6303 	case BFA_FCS_VPORT_SM_OFFLINE:
6304 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_offline);
6305 		bfa_sm_send_event(vport->lps, BFA_LPS_SM_OFFLINE);
6306 		bfa_fcs_lport_offline(&vport->lport);
6307 		break;
6308 
6309 	default:
6310 		bfa_sm_fault(__vport_fcs(vport), event);
6311 	}
6312 }
6313 
6314 /*
6315  * Vport is being stopped - awaiting lport stop completion to send
6316  * LOGO to fabric.
6317  */
6318 static void
6319 bfa_fcs_vport_sm_stopping(struct bfa_fcs_vport_s *vport,
6320 			  enum bfa_fcs_vport_event event)
6321 {
6322 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6323 	bfa_trc(__vport_fcs(vport), event);
6324 
6325 	switch (event) {
6326 	case BFA_FCS_VPORT_SM_STOPCOMP:
6327 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_logo_for_stop);
6328 		bfa_fcs_vport_do_logo(vport);
6329 		break;
6330 
6331 	case BFA_FCS_VPORT_SM_OFFLINE:
6332 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6333 		break;
6334 
6335 	default:
6336 		bfa_sm_fault(__vport_fcs(vport), event);
6337 	}
6338 }
6339 
6340 /*
6341  * Vport is being deleted - awaiting lport delete completion to send
6342  * LOGO to fabric.
6343  */
6344 static void
6345 bfa_fcs_vport_sm_deleting(struct bfa_fcs_vport_s *vport,
6346 			enum bfa_fcs_vport_event event)
6347 {
6348 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6349 	bfa_trc(__vport_fcs(vport), event);
6350 
6351 	switch (event) {
6352 	case BFA_FCS_VPORT_SM_DELETE:
6353 		break;
6354 
6355 	case BFA_FCS_VPORT_SM_DELCOMP:
6356 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_logo);
6357 		bfa_fcs_vport_do_logo(vport);
6358 		break;
6359 
6360 	case BFA_FCS_VPORT_SM_OFFLINE:
6361 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6362 		break;
6363 
6364 	default:
6365 		bfa_sm_fault(__vport_fcs(vport), event);
6366 	}
6367 }
6368 
6369 /*
6370  * Error State.
6371  * This state will be set when the Vport Creation fails due
6372  * to errors like Dup WWN. In this state only operation allowed
6373  * is a Vport Delete.
6374  */
6375 static void
6376 bfa_fcs_vport_sm_error(struct bfa_fcs_vport_s *vport,
6377 			enum bfa_fcs_vport_event event)
6378 {
6379 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6380 	bfa_trc(__vport_fcs(vport), event);
6381 
6382 	switch (event) {
6383 	case BFA_FCS_VPORT_SM_DELETE:
6384 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_cleanup);
6385 		bfa_fcs_lport_delete(&vport->lport);
6386 		break;
6387 
6388 	default:
6389 		bfa_trc(__vport_fcs(vport), event);
6390 	}
6391 }
6392 
6393 /*
6394  * Lport cleanup is in progress since vport is being deleted. Fabric is
6395  * offline, so no LOGO is needed to complete vport deletion.
6396  */
6397 static void
6398 bfa_fcs_vport_sm_cleanup(struct bfa_fcs_vport_s *vport,
6399 			enum bfa_fcs_vport_event event)
6400 {
6401 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6402 	bfa_trc(__vport_fcs(vport), event);
6403 
6404 	switch (event) {
6405 	case BFA_FCS_VPORT_SM_DELCOMP:
6406 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_uninit);
6407 		bfa_fcs_vport_free(vport);
6408 		break;
6409 
6410 	case BFA_FCS_VPORT_SM_STOPCOMP:
6411 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_created);
6412 		break;
6413 
6414 	case BFA_FCS_VPORT_SM_DELETE:
6415 		break;
6416 
6417 	default:
6418 		bfa_sm_fault(__vport_fcs(vport), event);
6419 	}
6420 }
6421 
6422 /*
6423  * LOGO is sent to fabric. Vport stop is in progress. Lport stop cleanup
6424  * is done.
6425  */
6426 static void
6427 bfa_fcs_vport_sm_logo_for_stop(struct bfa_fcs_vport_s *vport,
6428 			       enum bfa_fcs_vport_event event)
6429 {
6430 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6431 	bfa_trc(__vport_fcs(vport), event);
6432 
6433 	switch (event) {
6434 	case BFA_FCS_VPORT_SM_OFFLINE:
6435 		bfa_sm_send_event(vport->lps, BFA_LPS_SM_OFFLINE);
6436 		/*
6437 		 * !!! fall through !!!
6438 		 */
6439 
6440 	case BFA_FCS_VPORT_SM_RSP_OK:
6441 	case BFA_FCS_VPORT_SM_RSP_ERROR:
6442 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_created);
6443 		break;
6444 
6445 	default:
6446 		bfa_sm_fault(__vport_fcs(vport), event);
6447 	}
6448 }
6449 
6450 /*
6451  * LOGO is sent to fabric. Vport delete is in progress. Lport delete cleanup
6452  * is done.
6453  */
6454 static void
6455 bfa_fcs_vport_sm_logo(struct bfa_fcs_vport_s *vport,
6456 			enum bfa_fcs_vport_event event)
6457 {
6458 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6459 	bfa_trc(__vport_fcs(vport), event);
6460 
6461 	switch (event) {
6462 	case BFA_FCS_VPORT_SM_OFFLINE:
6463 		bfa_sm_send_event(vport->lps, BFA_LPS_SM_OFFLINE);
6464 		/*
6465 		 * !!! fall through !!!
6466 		 */
6467 
6468 	case BFA_FCS_VPORT_SM_RSP_OK:
6469 	case BFA_FCS_VPORT_SM_RSP_ERROR:
6470 		bfa_sm_set_state(vport, bfa_fcs_vport_sm_uninit);
6471 		bfa_fcs_vport_free(vport);
6472 		break;
6473 
6474 	case BFA_FCS_VPORT_SM_DELETE:
6475 		break;
6476 
6477 	default:
6478 		bfa_sm_fault(__vport_fcs(vport), event);
6479 	}
6480 }
6481 
6482 
6483 
6484 /*
6485  *  fcs_vport_private FCS virtual port private functions
6486  */
6487 /*
6488  * Send AEN notification
6489  */
6490 static void
6491 bfa_fcs_vport_aen_post(struct bfa_fcs_lport_s *port,
6492 		       enum bfa_lport_aen_event event)
6493 {
6494 	struct bfad_s *bfad = (struct bfad_s *)port->fabric->fcs->bfad;
6495 	struct bfa_aen_entry_s  *aen_entry;
6496 
6497 	bfad_get_aen_entry(bfad, aen_entry);
6498 	if (!aen_entry)
6499 		return;
6500 
6501 	aen_entry->aen_data.lport.vf_id = port->fabric->vf_id;
6502 	aen_entry->aen_data.lport.roles = port->port_cfg.roles;
6503 	aen_entry->aen_data.lport.ppwwn = bfa_fcs_lport_get_pwwn(
6504 					bfa_fcs_get_base_port(port->fcs));
6505 	aen_entry->aen_data.lport.lpwwn = bfa_fcs_lport_get_pwwn(port);
6506 
6507 	/* Send the AEN notification */
6508 	bfad_im_post_vendor_event(aen_entry, bfad, ++port->fcs->fcs_aen_seq,
6509 				  BFA_AEN_CAT_LPORT, event);
6510 }
6511 
6512 /*
6513  * This routine will be called to send a FDISC command.
6514  */
6515 static void
6516 bfa_fcs_vport_do_fdisc(struct bfa_fcs_vport_s *vport)
6517 {
6518 	bfa_lps_fdisc(vport->lps, vport,
6519 		bfa_fcport_get_maxfrsize(__vport_bfa(vport)),
6520 		__vport_pwwn(vport), __vport_nwwn(vport));
6521 	vport->vport_stats.fdisc_sent++;
6522 }
6523 
6524 static void
6525 bfa_fcs_vport_fdisc_rejected(struct bfa_fcs_vport_s *vport)
6526 {
6527 	u8		lsrjt_rsn = vport->lps->lsrjt_rsn;
6528 	u8		lsrjt_expl = vport->lps->lsrjt_expl;
6529 
6530 	bfa_trc(__vport_fcs(vport), lsrjt_rsn);
6531 	bfa_trc(__vport_fcs(vport), lsrjt_expl);
6532 
6533 	/* For certain reason codes, we don't want to retry. */
6534 	switch (vport->lps->lsrjt_expl) {
6535 	case FC_LS_RJT_EXP_INV_PORT_NAME: /* by brocade */
6536 	case FC_LS_RJT_EXP_INVALID_NPORT_ID: /* by Cisco */
6537 		if (vport->fdisc_retries < BFA_FCS_VPORT_MAX_RETRIES)
6538 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6539 		else {
6540 			bfa_fcs_vport_aen_post(&vport->lport,
6541 					BFA_LPORT_AEN_NPIV_DUP_WWN);
6542 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_DUP_WWN);
6543 		}
6544 		break;
6545 
6546 	case FC_LS_RJT_EXP_INSUFF_RES:
6547 		/*
6548 		 * This means max logins per port/switch setting on the
6549 		 * switch was exceeded.
6550 		 */
6551 		if (vport->fdisc_retries < BFA_FCS_VPORT_MAX_RETRIES)
6552 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6553 		else {
6554 			bfa_fcs_vport_aen_post(&vport->lport,
6555 					BFA_LPORT_AEN_NPIV_FABRIC_MAX);
6556 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_FABRIC_MAX);
6557 		}
6558 		break;
6559 
6560 	default:
6561 		if (vport->fdisc_retries == 0)
6562 			bfa_fcs_vport_aen_post(&vport->lport,
6563 					BFA_LPORT_AEN_NPIV_UNKNOWN);
6564 		bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6565 	}
6566 }
6567 
6568 /*
6569  *	Called to send a logout to the fabric. Used when a V-Port is
6570  *	deleted/stopped.
6571  */
6572 static void
6573 bfa_fcs_vport_do_logo(struct bfa_fcs_vport_s *vport)
6574 {
6575 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6576 
6577 	vport->vport_stats.logo_sent++;
6578 	bfa_lps_fdisclogo(vport->lps);
6579 }
6580 
6581 
6582 /*
6583  *     This routine will be called by bfa_timer on timer timeouts.
6584  *
6585  *	param[in]	vport		- pointer to bfa_fcs_vport_t.
6586  *	param[out]	vport_status	- pointer to return vport status in
6587  *
6588  *	return
6589  *		void
6590  *
6591  *	Special Considerations:
6592  *
6593  *	note
6594  */
6595 static void
6596 bfa_fcs_vport_timeout(void *vport_arg)
6597 {
6598 	struct bfa_fcs_vport_s *vport = (struct bfa_fcs_vport_s *) vport_arg;
6599 
6600 	vport->vport_stats.fdisc_timeouts++;
6601 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_TIMEOUT);
6602 }
6603 
6604 static void
6605 bfa_fcs_vport_free(struct bfa_fcs_vport_s *vport)
6606 {
6607 	struct bfad_vport_s *vport_drv =
6608 			(struct bfad_vport_s *)vport->vport_drv;
6609 
6610 	bfa_fcs_fabric_delvport(__vport_fabric(vport), vport);
6611 	bfa_lps_delete(vport->lps);
6612 
6613 	if (vport_drv->comp_del) {
6614 		complete(vport_drv->comp_del);
6615 		return;
6616 	}
6617 
6618 	/*
6619 	 * We queue the vport delete work to the IM work_q from here.
6620 	 * The memory for the bfad_vport_s is freed from the FC function
6621 	 * template vport_delete entry point.
6622 	 */
6623 	bfad_im_port_delete(vport_drv->drv_port.bfad, &vport_drv->drv_port);
6624 }
6625 
6626 /*
6627  *  fcs_vport_public FCS virtual port public interfaces
6628  */
6629 
6630 /*
6631  * Online notification from fabric SM.
6632  */
6633 void
6634 bfa_fcs_vport_online(struct bfa_fcs_vport_s *vport)
6635 {
6636 	vport->vport_stats.fab_online++;
6637 	if (bfa_fcs_fabric_npiv_capable(__vport_fabric(vport)))
6638 		bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_ONLINE);
6639 	else
6640 		vport->vport_stats.fab_no_npiv++;
6641 }
6642 
6643 /*
6644  * Offline notification from fabric SM.
6645  */
6646 void
6647 bfa_fcs_vport_offline(struct bfa_fcs_vport_s *vport)
6648 {
6649 	vport->vport_stats.fab_offline++;
6650 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_OFFLINE);
6651 }
6652 
6653 /*
6654  * Cleanup notification from fabric SM on link timer expiry.
6655  */
6656 void
6657 bfa_fcs_vport_cleanup(struct bfa_fcs_vport_s *vport)
6658 {
6659 	vport->vport_stats.fab_cleanup++;
6660 }
6661 
6662 /*
6663  * Stop notification from fabric SM. To be invoked from within FCS.
6664  */
6665 void
6666 bfa_fcs_vport_fcs_stop(struct bfa_fcs_vport_s *vport)
6667 {
6668 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_STOP);
6669 }
6670 
6671 /*
6672  * delete notification from fabric SM. To be invoked from within FCS.
6673  */
6674 void
6675 bfa_fcs_vport_fcs_delete(struct bfa_fcs_vport_s *vport)
6676 {
6677 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_DELETE);
6678 }
6679 
6680 /*
6681  * Stop completion callback from associated lport
6682  */
6683 void
6684 bfa_fcs_vport_stop_comp(struct bfa_fcs_vport_s *vport)
6685 {
6686 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_STOPCOMP);
6687 }
6688 
6689 /*
6690  * Delete completion callback from associated lport
6691  */
6692 void
6693 bfa_fcs_vport_delete_comp(struct bfa_fcs_vport_s *vport)
6694 {
6695 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_DELCOMP);
6696 }
6697 
6698 
6699 
6700 /*
6701  *  fcs_vport_api Virtual port API
6702  */
6703 
6704 /*
6705  *	Use this function to instantiate a new FCS vport object. This
6706  *	function will not trigger any HW initialization process (which will be
6707  *	done in vport_start() call)
6708  *
6709  *	param[in] vport	-		pointer to bfa_fcs_vport_t. This space
6710  *					needs to be allocated by the driver.
6711  *	param[in] fcs		-	FCS instance
6712  *	param[in] vport_cfg	-	vport configuration
6713  *	param[in] vf_id		-	VF_ID if vport is created within a VF.
6714  *					FC_VF_ID_NULL to specify base fabric.
6715  *	param[in] vport_drv	-	Opaque handle back to the driver's vport
6716  *					structure
6717  *
6718  *	retval BFA_STATUS_OK - on success.
6719  *	retval BFA_STATUS_FAILED - on failure.
6720  */
6721 bfa_status_t
6722 bfa_fcs_vport_create(struct bfa_fcs_vport_s *vport, struct bfa_fcs_s *fcs,
6723 		u16 vf_id, struct bfa_lport_cfg_s *vport_cfg,
6724 		struct bfad_vport_s *vport_drv)
6725 {
6726 	if (vport_cfg->pwwn == 0)
6727 		return BFA_STATUS_INVALID_WWN;
6728 
6729 	if (bfa_fcs_lport_get_pwwn(&fcs->fabric.bport) == vport_cfg->pwwn)
6730 		return BFA_STATUS_VPORT_WWN_BP;
6731 
6732 	if (bfa_fcs_vport_lookup(fcs, vf_id, vport_cfg->pwwn) != NULL)
6733 		return BFA_STATUS_VPORT_EXISTS;
6734 
6735 	if (fcs->fabric.num_vports ==
6736 			bfa_lps_get_max_vport(fcs->bfa))
6737 		return BFA_STATUS_VPORT_MAX;
6738 
6739 	vport->lps = bfa_lps_alloc(fcs->bfa);
6740 	if (!vport->lps)
6741 		return BFA_STATUS_VPORT_MAX;
6742 
6743 	vport->vport_drv = vport_drv;
6744 	vport_cfg->preboot_vp = BFA_FALSE;
6745 
6746 	bfa_sm_set_state(vport, bfa_fcs_vport_sm_uninit);
6747 	bfa_fcs_lport_attach(&vport->lport, fcs, vf_id, vport);
6748 	bfa_fcs_lport_init(&vport->lport, vport_cfg);
6749 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_CREATE);
6750 
6751 	return BFA_STATUS_OK;
6752 }
6753 
6754 /*
6755  *	Use this function to instantiate a new FCS PBC vport object. This
6756  *	function will not trigger any HW initialization process (which will be
6757  *	done in vport_start() call)
6758  *
6759  *	param[in] vport	-	pointer to bfa_fcs_vport_t. This space
6760  *				needs to be allocated by the driver.
6761  *	param[in] fcs	-	FCS instance
6762  *	param[in] vport_cfg	-	vport configuration
6763  *	param[in] vf_id		-	VF_ID if vport is created within a VF.
6764  *					FC_VF_ID_NULL to specify base fabric.
6765  *	param[in] vport_drv	-	Opaque handle back to the driver's vport
6766  *					structure
6767  *
6768  *	retval BFA_STATUS_OK - on success.
6769  *	retval BFA_STATUS_FAILED - on failure.
6770  */
6771 bfa_status_t
6772 bfa_fcs_pbc_vport_create(struct bfa_fcs_vport_s *vport, struct bfa_fcs_s *fcs,
6773 			u16 vf_id, struct bfa_lport_cfg_s *vport_cfg,
6774 			struct bfad_vport_s *vport_drv)
6775 {
6776 	bfa_status_t rc;
6777 
6778 	rc = bfa_fcs_vport_create(vport, fcs, vf_id, vport_cfg, vport_drv);
6779 	vport->lport.port_cfg.preboot_vp = BFA_TRUE;
6780 
6781 	return rc;
6782 }
6783 
6784 /*
6785  *	Use this function to findout if this is a pbc vport or not.
6786  *
6787  * @param[in] vport - pointer to bfa_fcs_vport_t.
6788  *
6789  * @returns None
6790  */
6791 bfa_boolean_t
6792 bfa_fcs_is_pbc_vport(struct bfa_fcs_vport_s *vport)
6793 {
6794 
6795 	if (vport && (vport->lport.port_cfg.preboot_vp == BFA_TRUE))
6796 		return BFA_TRUE;
6797 	else
6798 		return BFA_FALSE;
6799 
6800 }
6801 
6802 /*
6803  * Use this function initialize the vport.
6804  *
6805  * @param[in] vport - pointer to bfa_fcs_vport_t.
6806  *
6807  * @returns None
6808  */
6809 bfa_status_t
6810 bfa_fcs_vport_start(struct bfa_fcs_vport_s *vport)
6811 {
6812 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_START);
6813 
6814 	return BFA_STATUS_OK;
6815 }
6816 
6817 /*
6818  *	Use this function quiese the vport object. This function will return
6819  *	immediately, when the vport is actually stopped, the
6820  *	bfa_drv_vport_stop_cb() will be called.
6821  *
6822  *	param[in] vport - pointer to bfa_fcs_vport_t.
6823  *
6824  *	return None
6825  */
6826 bfa_status_t
6827 bfa_fcs_vport_stop(struct bfa_fcs_vport_s *vport)
6828 {
6829 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_STOP);
6830 
6831 	return BFA_STATUS_OK;
6832 }
6833 
6834 /*
6835  *	Use this function to delete a vport object. Fabric object should
6836  *	be stopped before this function call.
6837  *
6838  *	!!!!!!! Donot invoke this from within FCS  !!!!!!!
6839  *
6840  *	param[in] vport - pointer to bfa_fcs_vport_t.
6841  *
6842  *	return     None
6843  */
6844 bfa_status_t
6845 bfa_fcs_vport_delete(struct bfa_fcs_vport_s *vport)
6846 {
6847 
6848 	if (vport->lport.port_cfg.preboot_vp)
6849 		return BFA_STATUS_PBC;
6850 
6851 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_DELETE);
6852 
6853 	return BFA_STATUS_OK;
6854 }
6855 
6856 /*
6857  *	Use this function to get vport's current status info.
6858  *
6859  *	param[in] vport		pointer to bfa_fcs_vport_t.
6860  *	param[out] attr		pointer to return vport attributes
6861  *
6862  *	return None
6863  */
6864 void
6865 bfa_fcs_vport_get_attr(struct bfa_fcs_vport_s *vport,
6866 			struct bfa_vport_attr_s *attr)
6867 {
6868 	if (vport == NULL || attr == NULL)
6869 		return;
6870 
6871 	memset(attr, 0, sizeof(struct bfa_vport_attr_s));
6872 
6873 	bfa_fcs_lport_get_attr(&vport->lport, &attr->port_attr);
6874 	attr->vport_state = bfa_sm_to_state(vport_sm_table, vport->sm);
6875 }
6876 
6877 
6878 /*
6879  *	Lookup a virtual port. Excludes base port from lookup.
6880  */
6881 struct bfa_fcs_vport_s *
6882 bfa_fcs_vport_lookup(struct bfa_fcs_s *fcs, u16 vf_id, wwn_t vpwwn)
6883 {
6884 	struct bfa_fcs_vport_s *vport;
6885 	struct bfa_fcs_fabric_s *fabric;
6886 
6887 	bfa_trc(fcs, vf_id);
6888 	bfa_trc(fcs, vpwwn);
6889 
6890 	fabric = bfa_fcs_vf_lookup(fcs, vf_id);
6891 	if (!fabric) {
6892 		bfa_trc(fcs, vf_id);
6893 		return NULL;
6894 	}
6895 
6896 	vport = bfa_fcs_fabric_vport_lookup(fabric, vpwwn);
6897 	return vport;
6898 }
6899 
6900 /*
6901  * FDISC Response
6902  */
6903 void
6904 bfa_cb_lps_fdisc_comp(void *bfad, void *uarg, bfa_status_t status)
6905 {
6906 	struct bfa_fcs_vport_s *vport = uarg;
6907 
6908 	bfa_trc(__vport_fcs(vport), __vport_pwwn(vport));
6909 	bfa_trc(__vport_fcs(vport), status);
6910 
6911 	switch (status) {
6912 	case BFA_STATUS_OK:
6913 		/*
6914 		 * Initialize the V-Port fields
6915 		 */
6916 		__vport_fcid(vport) = vport->lps->lp_pid;
6917 		vport->vport_stats.fdisc_accepts++;
6918 		bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_OK);
6919 		break;
6920 
6921 	case BFA_STATUS_INVALID_MAC:
6922 		/* Only for CNA */
6923 		vport->vport_stats.fdisc_acc_bad++;
6924 		bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6925 
6926 		break;
6927 
6928 	case BFA_STATUS_EPROTOCOL:
6929 		switch (vport->lps->ext_status) {
6930 		case BFA_EPROTO_BAD_ACCEPT:
6931 			vport->vport_stats.fdisc_acc_bad++;
6932 			break;
6933 
6934 		case BFA_EPROTO_UNKNOWN_RSP:
6935 			vport->vport_stats.fdisc_unknown_rsp++;
6936 			break;
6937 
6938 		default:
6939 			break;
6940 		}
6941 
6942 		if (vport->fdisc_retries < BFA_FCS_VPORT_MAX_RETRIES)
6943 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6944 		else
6945 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_FAILED);
6946 
6947 		break;
6948 
6949 	case BFA_STATUS_ETIMER:
6950 		vport->vport_stats.fdisc_timeouts++;
6951 		if (vport->fdisc_retries < BFA_FCS_VPORT_MAX_RETRIES)
6952 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6953 		else
6954 			bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_FAILED);
6955 		break;
6956 
6957 	case BFA_STATUS_FABRIC_RJT:
6958 		vport->vport_stats.fdisc_rejects++;
6959 		bfa_fcs_vport_fdisc_rejected(vport);
6960 		break;
6961 
6962 	default:
6963 		vport->vport_stats.fdisc_rsp_err++;
6964 		bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_ERROR);
6965 	}
6966 }
6967 
6968 /*
6969  * LOGO response
6970  */
6971 void
6972 bfa_cb_lps_fdisclogo_comp(void *bfad, void *uarg)
6973 {
6974 	struct bfa_fcs_vport_s *vport = uarg;
6975 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_RSP_OK);
6976 }
6977 
6978 /*
6979  * Received clear virtual link
6980  */
6981 void
6982 bfa_cb_lps_cvl_event(void *bfad, void *uarg)
6983 {
6984 	struct bfa_fcs_vport_s *vport = uarg;
6985 
6986 	/* Send an Offline followed by an ONLINE */
6987 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_OFFLINE);
6988 	bfa_sm_send_event(vport, BFA_FCS_VPORT_SM_ONLINE);
6989 }
6990