xref: /linux/drivers/scsi/elx/libefc/efc_cmds.c (revision 4d5e3b06e1fc1428be14cd4ebe3b37c1bb34f95d)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (C) 2021 Broadcom. All Rights Reserved. The term
4  * “Broadcom” refers to Broadcom Inc. and/or its subsidiaries.
5  */
6 
7 #include "efclib.h"
8 #include "../libefc_sli/sli4.h"
9 #include "efc_cmds.h"
10 #include "efc_sm.h"
11 
12 static void
13 efc_nport_free_resources(struct efc_nport *nport, int evt, void *data)
14 {
15 	struct efc *efc = nport->efc;
16 
17 	/* Clear the nport attached flag */
18 	nport->attached = false;
19 
20 	/* Free the service parameters buffer */
21 	if (nport->dma.virt) {
22 		dma_free_coherent(&efc->pci->dev, nport->dma.size,
23 				  nport->dma.virt, nport->dma.phys);
24 		memset(&nport->dma, 0, sizeof(struct efc_dma));
25 	}
26 
27 	/* Free the SLI resources */
28 	sli_resource_free(efc->sli, SLI4_RSRC_VPI, nport->indicator);
29 
30 	efc_nport_cb(efc, evt, nport);
31 }
32 
33 static int
34 efc_nport_get_mbox_status(struct efc_nport *nport, u8 *mqe, int status)
35 {
36 	struct efc *efc = nport->efc;
37 	struct sli4_mbox_command_header *hdr =
38 			(struct sli4_mbox_command_header *)mqe;
39 
40 	if (status || le16_to_cpu(hdr->status)) {
41 		efc_log_debug(efc, "bad status vpi=%#x st=%x hdr=%x\n",
42 			      nport->indicator, status, le16_to_cpu(hdr->status));
43 		return -EIO;
44 	}
45 
46 	return 0;
47 }
48 
49 static int
50 efc_nport_free_unreg_vpi_cb(struct efc *efc, int status, u8 *mqe, void *arg)
51 {
52 	struct efc_nport *nport = arg;
53 	int evt = EFC_EVT_NPORT_FREE_OK;
54 	int rc;
55 
56 	rc = efc_nport_get_mbox_status(nport, mqe, status);
57 	if (rc)
58 		evt = EFC_EVT_NPORT_FREE_FAIL;
59 
60 	efc_nport_free_resources(nport, evt, mqe);
61 	return rc;
62 }
63 
64 static void
65 efc_nport_free_unreg_vpi(struct efc_nport *nport)
66 {
67 	struct efc *efc = nport->efc;
68 	int rc;
69 	u8 data[SLI4_BMBX_SIZE];
70 
71 	rc = sli_cmd_unreg_vpi(efc->sli, data, nport->indicator,
72 			       SLI4_UNREG_TYPE_PORT);
73 	if (rc) {
74 		efc_log_err(efc, "UNREG_VPI format failure\n");
75 		efc_nport_free_resources(nport, EFC_EVT_NPORT_FREE_FAIL, data);
76 		return;
77 	}
78 
79 	rc = efc->tt.issue_mbox_rqst(efc->base, data,
80 				     efc_nport_free_unreg_vpi_cb, nport);
81 	if (rc) {
82 		efc_log_err(efc, "UNREG_VPI command failure\n");
83 		efc_nport_free_resources(nport, EFC_EVT_NPORT_FREE_FAIL, data);
84 	}
85 }
86 
87 static void
88 efc_nport_send_evt(struct efc_nport *nport, int evt, void *data)
89 {
90 	struct efc *efc = nport->efc;
91 
92 	/* Now inform the registered callbacks */
93 	efc_nport_cb(efc, evt, nport);
94 
95 	/* Set the nport attached flag */
96 	if (evt == EFC_EVT_NPORT_ATTACH_OK)
97 		nport->attached = true;
98 
99 	/* If there is a pending free request, then handle it now */
100 	if (nport->free_req_pending)
101 		efc_nport_free_unreg_vpi(nport);
102 }
103 
104 static int
105 efc_nport_alloc_init_vpi_cb(struct efc *efc, int status, u8 *mqe, void *arg)
106 {
107 	struct efc_nport *nport = arg;
108 
109 	if (efc_nport_get_mbox_status(nport, mqe, status)) {
110 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, mqe);
111 		return -EIO;
112 	}
113 
114 	efc_nport_send_evt(nport, EFC_EVT_NPORT_ALLOC_OK, mqe);
115 	return 0;
116 }
117 
118 static void
119 efc_nport_alloc_init_vpi(struct efc_nport *nport)
120 {
121 	struct efc *efc = nport->efc;
122 	u8 data[SLI4_BMBX_SIZE];
123 	int rc;
124 
125 	/* If there is a pending free request, then handle it now */
126 	if (nport->free_req_pending) {
127 		efc_nport_free_resources(nport, EFC_EVT_NPORT_FREE_OK, data);
128 		return;
129 	}
130 
131 	rc = sli_cmd_init_vpi(efc->sli, data,
132 			      nport->indicator, nport->domain->indicator);
133 	if (rc) {
134 		efc_log_err(efc, "INIT_VPI format failure\n");
135 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, data);
136 		return;
137 	}
138 
139 	rc = efc->tt.issue_mbox_rqst(efc->base, data,
140 			efc_nport_alloc_init_vpi_cb, nport);
141 	if (rc) {
142 		efc_log_err(efc, "INIT_VPI command failure\n");
143 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, data);
144 	}
145 }
146 
147 static int
148 efc_nport_alloc_read_sparm64_cb(struct efc *efc, int status, u8 *mqe, void *arg)
149 {
150 	struct efc_nport *nport = arg;
151 	u8 *payload = NULL;
152 
153 	if (efc_nport_get_mbox_status(nport, mqe, status)) {
154 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, mqe);
155 		return -EIO;
156 	}
157 
158 	payload = nport->dma.virt;
159 
160 	memcpy(&nport->sli_wwpn, payload + SLI4_READ_SPARM64_WWPN_OFFSET,
161 	       sizeof(nport->sli_wwpn));
162 	memcpy(&nport->sli_wwnn, payload + SLI4_READ_SPARM64_WWNN_OFFSET,
163 	       sizeof(nport->sli_wwnn));
164 
165 	dma_free_coherent(&efc->pci->dev, nport->dma.size, nport->dma.virt,
166 			  nport->dma.phys);
167 	memset(&nport->dma, 0, sizeof(struct efc_dma));
168 	efc_nport_alloc_init_vpi(nport);
169 	return 0;
170 }
171 
172 static void
173 efc_nport_alloc_read_sparm64(struct efc *efc, struct efc_nport *nport)
174 {
175 	u8 data[SLI4_BMBX_SIZE];
176 	int rc;
177 
178 	/* Allocate memory for the service parameters */
179 	nport->dma.size = EFC_SPARAM_DMA_SZ;
180 	nport->dma.virt = dma_alloc_coherent(&efc->pci->dev,
181 					     nport->dma.size, &nport->dma.phys,
182 					     GFP_KERNEL);
183 	if (!nport->dma.virt) {
184 		efc_log_err(efc, "Failed to allocate DMA memory\n");
185 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, data);
186 		return;
187 	}
188 
189 	rc = sli_cmd_read_sparm64(efc->sli, data,
190 				  &nport->dma, nport->indicator);
191 	if (rc) {
192 		efc_log_err(efc, "READ_SPARM64 format failure\n");
193 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, data);
194 		return;
195 	}
196 
197 	rc = efc->tt.issue_mbox_rqst(efc->base, data,
198 				     efc_nport_alloc_read_sparm64_cb, nport);
199 	if (rc) {
200 		efc_log_err(efc, "READ_SPARM64 command failure\n");
201 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ALLOC_FAIL, data);
202 	}
203 }
204 
205 int
206 efc_cmd_nport_alloc(struct efc *efc, struct efc_nport *nport,
207 		    struct efc_domain *domain, u8 *wwpn)
208 {
209 	u32 index;
210 
211 	nport->indicator = U32_MAX;
212 	nport->free_req_pending = false;
213 
214 	if (wwpn)
215 		memcpy(&nport->sli_wwpn, wwpn, sizeof(nport->sli_wwpn));
216 
217 	/*
218 	 * allocate a VPI object for the port and stores it in the
219 	 * indicator field of the port object.
220 	 */
221 	if (sli_resource_alloc(efc->sli, SLI4_RSRC_VPI,
222 			       &nport->indicator, &index)) {
223 		efc_log_err(efc, "VPI allocation failure\n");
224 		return -EIO;
225 	}
226 
227 	if (domain) {
228 		/*
229 		 * If the WWPN is NULL, fetch the default
230 		 * WWPN and WWNN before initializing the VPI
231 		 */
232 		if (!wwpn)
233 			efc_nport_alloc_read_sparm64(efc, nport);
234 		else
235 			efc_nport_alloc_init_vpi(nport);
236 	} else if (!wwpn) {
237 		/* domain NULL and wwpn non-NULL */
238 		efc_log_err(efc, "need WWN for physical port\n");
239 		sli_resource_free(efc->sli, SLI4_RSRC_VPI, nport->indicator);
240 		return -EIO;
241 	}
242 
243 	return 0;
244 }
245 
246 static int
247 efc_nport_attach_reg_vpi_cb(struct efc *efc, int status, u8 *mqe,
248 			    void *arg)
249 {
250 	struct efc_nport *nport = arg;
251 
252 	nport->attaching = false;
253 	if (efc_nport_get_mbox_status(nport, mqe, status)) {
254 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ATTACH_FAIL, mqe);
255 		return -EIO;
256 	}
257 
258 	efc_nport_send_evt(nport, EFC_EVT_NPORT_ATTACH_OK, mqe);
259 	return 0;
260 }
261 
262 int
263 efc_cmd_nport_attach(struct efc *efc, struct efc_nport *nport, u32 fc_id)
264 {
265 	u8 buf[SLI4_BMBX_SIZE];
266 	int rc = 0;
267 
268 	if (!nport) {
269 		efc_log_err(efc, "bad param(s) nport=%p\n", nport);
270 		return -EIO;
271 	}
272 
273 	nport->fc_id = fc_id;
274 
275 	/* register previously-allocated VPI with the device */
276 	rc = sli_cmd_reg_vpi(efc->sli, buf, nport->fc_id,
277 			     nport->sli_wwpn, nport->indicator,
278 			    nport->domain->indicator, false);
279 	if (rc) {
280 		efc_log_err(efc, "REG_VPI format failure\n");
281 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ATTACH_FAIL, buf);
282 		return rc;
283 	}
284 
285 	rc = efc->tt.issue_mbox_rqst(efc->base, buf,
286 				     efc_nport_attach_reg_vpi_cb, nport);
287 	if (rc) {
288 		efc_log_err(efc, "REG_VPI command failure\n");
289 		efc_nport_free_resources(nport, EFC_EVT_NPORT_ATTACH_FAIL, buf);
290 	} else {
291 		nport->attaching = true;
292 	}
293 
294 	return rc;
295 }
296 
297 int
298 efc_cmd_nport_free(struct efc *efc, struct efc_nport *nport)
299 {
300 	if (!nport) {
301 		efc_log_err(efc, "bad parameter(s) nport=%p\n",	nport);
302 		return -EIO;
303 	}
304 
305 	/* Issue the UNREG_VPI command to free the assigned VPI context */
306 	if (nport->attached)
307 		efc_nport_free_unreg_vpi(nport);
308 	else if (nport->attaching)
309 		nport->free_req_pending = true;
310 	else
311 		efc_sm_post_event(&nport->sm, EFC_EVT_NPORT_FREE_OK, NULL);
312 
313 	return 0;
314 }
315 
316 static int
317 efc_domain_get_mbox_status(struct efc_domain *domain, u8 *mqe, int status)
318 {
319 	struct efc *efc = domain->efc;
320 	struct sli4_mbox_command_header *hdr =
321 			(struct sli4_mbox_command_header *)mqe;
322 
323 	if (status || le16_to_cpu(hdr->status)) {
324 		efc_log_debug(efc, "bad status vfi=%#x st=%x hdr=%x\n",
325 			      domain->indicator, status,
326 			      le16_to_cpu(hdr->status));
327 		return -EIO;
328 	}
329 
330 	return 0;
331 }
332 
333 static void
334 efc_domain_free_resources(struct efc_domain *domain, int evt, void *data)
335 {
336 	struct efc *efc = domain->efc;
337 
338 	/* Free the service parameters buffer */
339 	if (domain->dma.virt) {
340 		dma_free_coherent(&efc->pci->dev,
341 				  domain->dma.size, domain->dma.virt,
342 				  domain->dma.phys);
343 		memset(&domain->dma, 0, sizeof(struct efc_dma));
344 	}
345 
346 	/* Free the SLI resources */
347 	sli_resource_free(efc->sli, SLI4_RSRC_VFI, domain->indicator);
348 
349 	efc_domain_cb(efc, evt, domain);
350 }
351 
352 static void
353 efc_domain_send_nport_evt(struct efc_domain *domain,
354 			  int port_evt, int domain_evt, void *data)
355 {
356 	struct efc *efc = domain->efc;
357 
358 	/* Send alloc/attach ok to the physical nport */
359 	efc_nport_send_evt(domain->nport, port_evt, NULL);
360 
361 	/* Now inform the registered callbacks */
362 	efc_domain_cb(efc, domain_evt, domain);
363 }
364 
365 static int
366 efc_domain_alloc_read_sparm64_cb(struct efc *efc, int status, u8 *mqe,
367 				 void *arg)
368 {
369 	struct efc_domain *domain = arg;
370 
371 	if (efc_domain_get_mbox_status(domain, mqe, status)) {
372 		efc_domain_free_resources(domain,
373 					  EFC_HW_DOMAIN_ALLOC_FAIL, mqe);
374 		return -EIO;
375 	}
376 
377 	efc_domain_send_nport_evt(domain, EFC_EVT_NPORT_ALLOC_OK,
378 				  EFC_HW_DOMAIN_ALLOC_OK, mqe);
379 	return 0;
380 }
381 
382 static void
383 efc_domain_alloc_read_sparm64(struct efc_domain *domain)
384 {
385 	struct efc *efc = domain->efc;
386 	u8 data[SLI4_BMBX_SIZE];
387 	int rc;
388 
389 	rc = sli_cmd_read_sparm64(efc->sli, data, &domain->dma, 0);
390 	if (rc) {
391 		efc_log_err(efc, "READ_SPARM64 format failure\n");
392 		efc_domain_free_resources(domain,
393 					  EFC_HW_DOMAIN_ALLOC_FAIL, data);
394 		return;
395 	}
396 
397 	rc = efc->tt.issue_mbox_rqst(efc->base, data,
398 				     efc_domain_alloc_read_sparm64_cb, domain);
399 	if (rc) {
400 		efc_log_err(efc, "READ_SPARM64 command failure\n");
401 		efc_domain_free_resources(domain,
402 					  EFC_HW_DOMAIN_ALLOC_FAIL, data);
403 	}
404 }
405 
406 static int
407 efc_domain_alloc_init_vfi_cb(struct efc *efc, int status, u8 *mqe,
408 			     void *arg)
409 {
410 	struct efc_domain *domain = arg;
411 
412 	if (efc_domain_get_mbox_status(domain, mqe, status)) {
413 		efc_domain_free_resources(domain,
414 					  EFC_HW_DOMAIN_ALLOC_FAIL, mqe);
415 		return -EIO;
416 	}
417 
418 	efc_domain_alloc_read_sparm64(domain);
419 	return 0;
420 }
421 
422 static void
423 efc_domain_alloc_init_vfi(struct efc_domain *domain)
424 {
425 	struct efc *efc = domain->efc;
426 	struct efc_nport *nport = domain->nport;
427 	u8 data[SLI4_BMBX_SIZE];
428 	int rc;
429 
430 	/*
431 	 * For FC, the HW alread registered an FCFI.
432 	 * Copy FCF information into the domain and jump to INIT_VFI.
433 	 */
434 	domain->fcf_indicator = efc->fcfi;
435 	rc = sli_cmd_init_vfi(efc->sli, data, domain->indicator,
436 			      domain->fcf_indicator, nport->indicator);
437 	if (rc) {
438 		efc_log_err(efc, "INIT_VFI format failure\n");
439 		efc_domain_free_resources(domain,
440 					  EFC_HW_DOMAIN_ALLOC_FAIL, data);
441 		return;
442 	}
443 
444 	efc_log_err(efc, "%s issue mbox\n", __func__);
445 	rc = efc->tt.issue_mbox_rqst(efc->base, data,
446 				     efc_domain_alloc_init_vfi_cb, domain);
447 	if (rc) {
448 		efc_log_err(efc, "INIT_VFI command failure\n");
449 		efc_domain_free_resources(domain,
450 					  EFC_HW_DOMAIN_ALLOC_FAIL, data);
451 	}
452 }
453 
454 int
455 efc_cmd_domain_alloc(struct efc *efc, struct efc_domain *domain, u32 fcf)
456 {
457 	u32 index;
458 
459 	if (!domain || !domain->nport) {
460 		efc_log_err(efc, "bad parameter(s) domain=%p nport=%p\n",
461 			    domain, domain ? domain->nport : NULL);
462 		return -EIO;
463 	}
464 
465 	/* allocate memory for the service parameters */
466 	domain->dma.size = EFC_SPARAM_DMA_SZ;
467 	domain->dma.virt = dma_alloc_coherent(&efc->pci->dev,
468 					      domain->dma.size,
469 					      &domain->dma.phys, GFP_KERNEL);
470 	if (!domain->dma.virt) {
471 		efc_log_err(efc, "Failed to allocate DMA memory\n");
472 		return -EIO;
473 	}
474 
475 	domain->fcf = fcf;
476 	domain->fcf_indicator = U32_MAX;
477 	domain->indicator = U32_MAX;
478 
479 	if (sli_resource_alloc(efc->sli, SLI4_RSRC_VFI, &domain->indicator,
480 			       &index)) {
481 		efc_log_err(efc, "VFI allocation failure\n");
482 
483 		dma_free_coherent(&efc->pci->dev,
484 				  domain->dma.size, domain->dma.virt,
485 				  domain->dma.phys);
486 		memset(&domain->dma, 0, sizeof(struct efc_dma));
487 
488 		return -EIO;
489 	}
490 
491 	efc_domain_alloc_init_vfi(domain);
492 	return 0;
493 }
494 
495 static int
496 efc_domain_attach_reg_vfi_cb(struct efc *efc, int status, u8 *mqe,
497 			     void *arg)
498 {
499 	struct efc_domain *domain = arg;
500 
501 	if (efc_domain_get_mbox_status(domain, mqe, status)) {
502 		efc_domain_free_resources(domain,
503 					  EFC_HW_DOMAIN_ATTACH_FAIL, mqe);
504 		return -EIO;
505 	}
506 
507 	efc_domain_send_nport_evt(domain, EFC_EVT_NPORT_ATTACH_OK,
508 				  EFC_HW_DOMAIN_ATTACH_OK, mqe);
509 	return 0;
510 }
511 
512 int
513 efc_cmd_domain_attach(struct efc *efc, struct efc_domain *domain, u32 fc_id)
514 {
515 	u8 buf[SLI4_BMBX_SIZE];
516 	int rc = 0;
517 
518 	if (!domain) {
519 		efc_log_err(efc, "bad param(s) domain=%p\n", domain);
520 		return -EIO;
521 	}
522 
523 	domain->nport->fc_id = fc_id;
524 
525 	rc = sli_cmd_reg_vfi(efc->sli, buf, SLI4_BMBX_SIZE, domain->indicator,
526 			     domain->fcf_indicator, domain->dma,
527 			     domain->nport->indicator, domain->nport->sli_wwpn,
528 			     domain->nport->fc_id);
529 	if (rc) {
530 		efc_log_err(efc, "REG_VFI format failure\n");
531 		goto cleanup;
532 	}
533 
534 	rc = efc->tt.issue_mbox_rqst(efc->base, buf,
535 				     efc_domain_attach_reg_vfi_cb, domain);
536 	if (rc) {
537 		efc_log_err(efc, "REG_VFI command failure\n");
538 		goto cleanup;
539 	}
540 
541 	return rc;
542 
543 cleanup:
544 	efc_domain_free_resources(domain, EFC_HW_DOMAIN_ATTACH_FAIL, buf);
545 
546 	return rc;
547 }
548 
549 static int
550 efc_domain_free_unreg_vfi_cb(struct efc *efc, int status, u8 *mqe, void *arg)
551 {
552 	struct efc_domain *domain = arg;
553 	int evt = EFC_HW_DOMAIN_FREE_OK;
554 	int rc;
555 
556 	rc = efc_domain_get_mbox_status(domain, mqe, status);
557 	if (rc) {
558 		evt = EFC_HW_DOMAIN_FREE_FAIL;
559 		rc = -EIO;
560 	}
561 
562 	efc_domain_free_resources(domain, evt, mqe);
563 	return rc;
564 }
565 
566 static void
567 efc_domain_free_unreg_vfi(struct efc_domain *domain)
568 {
569 	struct efc *efc = domain->efc;
570 	int rc;
571 	u8 data[SLI4_BMBX_SIZE];
572 
573 	rc = sli_cmd_unreg_vfi(efc->sli, data, domain->indicator,
574 			       SLI4_UNREG_TYPE_DOMAIN);
575 	if (rc) {
576 		efc_log_err(efc, "UNREG_VFI format failure\n");
577 		goto cleanup;
578 	}
579 
580 	rc = efc->tt.issue_mbox_rqst(efc->base, data,
581 				     efc_domain_free_unreg_vfi_cb, domain);
582 	if (rc) {
583 		efc_log_err(efc, "UNREG_VFI command failure\n");
584 		goto cleanup;
585 	}
586 
587 	return;
588 
589 cleanup:
590 	efc_domain_free_resources(domain, EFC_HW_DOMAIN_FREE_FAIL, data);
591 }
592 
593 int
594 efc_cmd_domain_free(struct efc *efc, struct efc_domain *domain)
595 {
596 	if (!domain) {
597 		efc_log_err(efc, "bad parameter(s) domain=%p\n", domain);
598 		return -EIO;
599 	}
600 
601 	efc_domain_free_unreg_vfi(domain);
602 	return 0;
603 }
604 
605 int
606 efc_cmd_node_alloc(struct efc *efc, struct efc_remote_node *rnode, u32 fc_addr,
607 		   struct efc_nport *nport)
608 {
609 	/* Check for invalid indicator */
610 	if (rnode->indicator != U32_MAX) {
611 		efc_log_err(efc,
612 			    "RPI allocation failure addr=%#x rpi=%#x\n",
613 			    fc_addr, rnode->indicator);
614 		return -EIO;
615 	}
616 
617 	/* NULL SLI port indicates an unallocated remote node */
618 	rnode->nport = NULL;
619 
620 	if (sli_resource_alloc(efc->sli, SLI4_RSRC_RPI,
621 			       &rnode->indicator, &rnode->index)) {
622 		efc_log_err(efc, "RPI allocation failure addr=%#x\n",
623 			    fc_addr);
624 		return -EIO;
625 	}
626 
627 	rnode->fc_id = fc_addr;
628 	rnode->nport = nport;
629 
630 	return 0;
631 }
632 
633 static int
634 efc_cmd_node_attach_cb(struct efc *efc, int status, u8 *mqe, void *arg)
635 {
636 	struct efc_remote_node *rnode = arg;
637 	struct sli4_mbox_command_header *hdr =
638 				(struct sli4_mbox_command_header *)mqe;
639 	int evt = 0;
640 
641 	if (status || le16_to_cpu(hdr->status)) {
642 		efc_log_debug(efc, "bad status cqe=%#x mqe=%#x\n", status,
643 			      le16_to_cpu(hdr->status));
644 		rnode->attached = false;
645 		evt = EFC_EVT_NODE_ATTACH_FAIL;
646 	} else {
647 		rnode->attached = true;
648 		evt = EFC_EVT_NODE_ATTACH_OK;
649 	}
650 
651 	efc_remote_node_cb(efc, evt, rnode);
652 
653 	return 0;
654 }
655 
656 int
657 efc_cmd_node_attach(struct efc *efc, struct efc_remote_node *rnode,
658 		    struct efc_dma *sparms)
659 {
660 	int rc = -EIO;
661 	u8 buf[SLI4_BMBX_SIZE];
662 
663 	if (!rnode || !sparms) {
664 		efc_log_err(efc, "bad parameter(s) rnode=%p sparms=%p\n",
665 			    rnode, sparms);
666 		return -EIO;
667 	}
668 
669 	/*
670 	 * If the attach count is non-zero, this RPI has already been reg'd.
671 	 * Otherwise, register the RPI
672 	 */
673 	if (rnode->index == U32_MAX) {
674 		efc_log_err(efc, "bad parameter rnode->index invalid\n");
675 		return -EIO;
676 	}
677 
678 	/* Update a remote node object with the remote port's service params */
679 	if (!sli_cmd_reg_rpi(efc->sli, buf, rnode->indicator,
680 			     rnode->nport->indicator, rnode->fc_id, sparms, 0, 0))
681 		rc = efc->tt.issue_mbox_rqst(efc->base, buf,
682 					     efc_cmd_node_attach_cb, rnode);
683 
684 	return rc;
685 }
686 
687 int
688 efc_node_free_resources(struct efc *efc, struct efc_remote_node *rnode)
689 {
690 	int rc = 0;
691 
692 	if (!rnode) {
693 		efc_log_err(efc, "bad parameter rnode=%p\n", rnode);
694 		return -EIO;
695 	}
696 
697 	if (rnode->nport) {
698 		if (rnode->attached) {
699 			efc_log_err(efc, "rnode is still attached\n");
700 			return -EIO;
701 		}
702 		if (rnode->indicator != U32_MAX) {
703 			if (sli_resource_free(efc->sli, SLI4_RSRC_RPI,
704 					      rnode->indicator)) {
705 				efc_log_err(efc,
706 					    "RPI free fail RPI %d addr=%#x\n",
707 					    rnode->indicator, rnode->fc_id);
708 				rc = -EIO;
709 			} else {
710 				rnode->indicator = U32_MAX;
711 				rnode->index = U32_MAX;
712 			}
713 		}
714 	}
715 
716 	return rc;
717 }
718 
719 static int
720 efc_cmd_node_free_cb(struct efc *efc, int status, u8 *mqe, void *arg)
721 {
722 	struct efc_remote_node *rnode = arg;
723 	struct sli4_mbox_command_header *hdr =
724 				(struct sli4_mbox_command_header *)mqe;
725 	int evt = EFC_EVT_NODE_FREE_FAIL;
726 	int rc = 0;
727 
728 	if (status || le16_to_cpu(hdr->status)) {
729 		efc_log_debug(efc, "bad status cqe=%#x mqe=%#x\n", status,
730 			      le16_to_cpu(hdr->status));
731 
732 		/*
733 		 * In certain cases, a non-zero MQE status is OK (all must be
734 		 * true):
735 		 *   - node is attached
736 		 *   - status is 0x1400
737 		 */
738 		if (!rnode->attached ||
739 		    (le16_to_cpu(hdr->status) != SLI4_MBX_STATUS_RPI_NOT_REG))
740 			rc = -EIO;
741 	}
742 
743 	if (!rc) {
744 		rnode->attached = false;
745 		evt = EFC_EVT_NODE_FREE_OK;
746 	}
747 
748 	efc_remote_node_cb(efc, evt, rnode);
749 
750 	return rc;
751 }
752 
753 int
754 efc_cmd_node_detach(struct efc *efc, struct efc_remote_node *rnode)
755 {
756 	u8 buf[SLI4_BMBX_SIZE];
757 	int rc = -EIO;
758 
759 	if (!rnode) {
760 		efc_log_err(efc, "bad parameter rnode=%p\n", rnode);
761 		return -EIO;
762 	}
763 
764 	if (rnode->nport) {
765 		if (!rnode->attached)
766 			return -EIO;
767 
768 		rc = -EIO;
769 
770 		if (!sli_cmd_unreg_rpi(efc->sli, buf, rnode->indicator,
771 				       SLI4_RSRC_RPI, U32_MAX))
772 			rc = efc->tt.issue_mbox_rqst(efc->base, buf,
773 					efc_cmd_node_free_cb, rnode);
774 
775 		if (rc != 0) {
776 			efc_log_err(efc, "UNREG_RPI failed\n");
777 			rc = -EIO;
778 		}
779 	}
780 
781 	return rc;
782 }
783