xref: /illumos-gate/usr/src/lib/cfgadm_plugins/fp/common/cfga_cs.c (revision 45ede40b2394db7967e59f19288fae9b62efd4aa)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 
27 #include	"cfga_fp.h"
28 
29 /* define */
30 #define	ALL_APID_LUNS_UNUSABLE	0x10
31 
32 #define	DEFAULT_LUN_COUNT	1024
33 #define	LUN_SIZE		8
34 #define	LUN_HEADER_SIZE		8
35 #define	DEFAULT_LUN_LENGTH	DEFAULT_LUN_COUNT   *	\
36 				LUN_SIZE	    +	\
37 				LUN_HEADER_SIZE
38 
39 /* Some forward declarations */
40 static fpcfga_ret_t do_devctl_dev_create(apid_t *, char *, int,
41 							uchar_t, char **);
42 static fpcfga_ret_t dev_rcm_online(apid_t *, int, cfga_flags_t, char **);
43 static void dev_rcm_online_nonoperationalpath(apid_t *, cfga_flags_t, char **);
44 static fpcfga_ret_t dev_rcm_offline(apid_t *, cfga_flags_t, char **);
45 static fpcfga_ret_t dev_rcm_remove(apid_t *, cfga_flags_t, char **);
46 static fpcfga_ret_t lun_unconf(char *, int, char *, char *, char **);
47 static fpcfga_ret_t dev_unconf(apid_t *, char **, uchar_t *);
48 static fpcfga_ret_t is_xport_phys_in_pathlist(apid_t *, char **);
49 static void copy_pwwn_data_to_str(char *, const uchar_t *);
50 static fpcfga_ret_t unconf_vhci_nodes(di_path_t, di_node_t, char *,
51 			char *, int, int *, int *, char **, cfga_flags_t);
52 static fpcfga_ret_t unconf_non_vhci_nodes(di_node_t, char *, char *,
53 			int, int *, int *, char **, cfga_flags_t);
54 static fpcfga_ret_t unconf_any_devinfo_nodes(apid_t *, cfga_flags_t, char **,
55 			int *, int *);
56 static fpcfga_ret_t handle_devs(cfga_cmd_t, apid_t *, cfga_flags_t,
57 			char **, HBA_HANDLE, int, HBA_PORTATTRIBUTES);
58 
59 
60 /*
61  * This function initiates the creation of the new device node for a given
62  * port WWN.
63  * So, apidt->dyncomp CANNOT be NULL
64  */
65 static fpcfga_ret_t
66 do_devctl_dev_create(apid_t *apidt, char *dev_path, int pathlen,
67 					uchar_t dev_dtype, char **errstring)
68 {
69 	devctl_ddef_t	ddef_hdl;
70 	devctl_hdl_t	bus_hdl, dev_hdl;
71 	char		*drvr_name = "dummy";
72 	la_wwn_t	pwwn;
73 
74 	*dev_path = '\0';
75 	if ((ddef_hdl = devctl_ddef_alloc(drvr_name, 0)) == NULL) {
76 		cfga_err(errstring, errno, ERRARG_DC_DDEF_ALLOC, drvr_name, 0);
77 		return (FPCFGA_LIB_ERR);
78 	}
79 
80 	if (cvt_dyncomp_to_lawwn(apidt->dyncomp, &pwwn)) {
81 		devctl_ddef_free(ddef_hdl);
82 		cfga_err(errstring, 0, ERR_APID_INVAL, 0);
83 		return (FPCFGA_LIB_ERR);
84 	}
85 
86 	if (devctl_ddef_byte_array(ddef_hdl, PORT_WWN_PROP, FC_WWN_SIZE,
87 							pwwn.raw_wwn) == -1) {
88 		devctl_ddef_free(ddef_hdl);
89 		cfga_err(errstring, errno, ERRARG_DC_BYTE_ARRAY,
90 							PORT_WWN_PROP, 0);
91 		return (FPCFGA_LIB_ERR);
92 	}
93 
94 	if ((bus_hdl = devctl_bus_acquire(apidt->xport_phys, 0)) == NULL) {
95 		devctl_ddef_free(ddef_hdl);
96 		cfga_err(errstring, errno, ERRARG_DC_BUS_ACQUIRE,
97 							apidt->xport_phys, 0);
98 		return (FPCFGA_LIB_ERR);
99 	}
100 
101 	/* Let driver handle creation of the new path */
102 	if (devctl_bus_dev_create(bus_hdl, ddef_hdl, 0, &dev_hdl)) {
103 		devctl_ddef_free(ddef_hdl);
104 		devctl_release(bus_hdl);
105 		if (dev_dtype == DTYPE_UNKNOWN) {
106 			/*
107 			 * Unknown DTYPES are devices such as another system's
108 			 * FC HBA port. We have tried to configure it but
109 			 * have failed. Since devices with no device type
110 			 * or an unknown dtype cannot be configured, we will
111 			 * return an appropriate error message.
112 			 */
113 			cfga_err(errstring, errno,
114 			    ERRARG_BUS_DEV_CREATE_UNKNOWN, apidt->dyncomp, 0);
115 		} else {
116 			cfga_err(errstring, errno, ERRARG_BUS_DEV_CREATE,
117 			    apidt->dyncomp, 0);
118 		}
119 		return (FPCFGA_LIB_ERR);
120 	}
121 	devctl_release(bus_hdl);
122 	devctl_ddef_free(ddef_hdl);
123 
124 	devctl_get_pathname(dev_hdl, dev_path, pathlen);
125 	devctl_release(dev_hdl);
126 
127 	return (FPCFGA_OK);
128 }
129 
130 /*
131  * Online, in RCM, all the LUNs for a particular device.
132  * Caller can specify the # of luns in the lunlist that have to be onlined
133  * by passing a count that is not -ve.
134  *
135  * INPUT :
136  * apidt - this is expected to have the list of luns for the device and so
137  *         is assumed to be filled in prior to this call
138  * count - # of LUNs in the list that have to be onlined.
139  * errstring - If non-NULL, it will hold any error messages
140  *
141  * RETURNS :
142  * 0 on success
143  * non-zero otherwise
144  */
145 static fpcfga_ret_t
146 dev_rcm_online(apid_t *apidt, int count, cfga_flags_t flags, char **errstring)
147 {
148 	luninfo_list_t	*lunlistp;
149 	int		i = 0, ret = 0;
150 	fpcfga_ret_t	retval = FPCFGA_OK;
151 
152 	/* This check may be redundant, but safer this way */
153 	if ((apidt->flags & FLAG_DISABLE_RCM) != 0) {
154 		/* User has requested not to notify RCM framework */
155 		return (FPCFGA_OK);
156 	}
157 
158 	lunlistp = apidt->lunlist;
159 
160 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
161 					i++, lunlistp = lunlistp->next) {
162 		if ((count >= 0) && (i >= count))
163 			break;
164 		if (fp_rcm_online(lunlistp->path, errstring, flags) !=
165 								FPCFGA_OK) {
166 			ret++;
167 		}
168 	}
169 
170 	if (ret > 0)
171 		retval = FPCFGA_LIB_ERR;
172 
173 	return (retval);
174 }
175 
176 /*
177  * Online in RCM for devices which only have paths
178  * not in ONLINE/STANDBY state
179  */
180 void
181 dev_rcm_online_nonoperationalpath(apid_t *apidt, cfga_flags_t flags,
182     char **errstring)
183 {
184 	luninfo_list_t	*lunlistp;
185 
186 	if ((apidt->flags & FLAG_DISABLE_RCM) != 0) {
187 		return;
188 	}
189 
190 	lunlistp = apidt->lunlist;
191 
192 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
193 	    lunlistp = lunlistp->next) {
194 		if ((lunlistp->lun_flag & FLAG_SKIP_ONLINEOTHERS) != 0) {
195 			continue;
196 		}
197 		(void) fp_rcm_online(lunlistp->path, errstring, flags);
198 	}
199 }
200 
201 /*
202  * Offline, in RCM, all the LUNs for a particular device.
203  * This function should not be called for the MPXIO case.
204  *
205  * INPUT :
206  * apidt - this is expected to have the list of luns for the device and so
207  *         is assumed to be filled in prior to this call
208  * errstring - If non-NULL, it will hold any error messages
209  *
210  * RETURNS :
211  * FPCFGA_OK on success
212  * error code otherwise
213  */
214 static fpcfga_ret_t
215 dev_rcm_offline(apid_t *apidt, cfga_flags_t flags, char **errstring)
216 {
217 	int		count = 0;
218 	luninfo_list_t	*lunlistp;
219 
220 	if ((apidt->flags & FLAG_DISABLE_RCM) != 0) {
221 		/* User has requested not to notify RCM framework */
222 		return (FPCFGA_OK);
223 	}
224 
225 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
226 						lunlistp = lunlistp->next) {
227 		if ((lunlistp->lun_flag & FLAG_SKIP_RCMOFFLINE) != 0) {
228 			continue;
229 		}
230 		if ((apidt->flags & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
231 		    FLAG_REMOVE_UNUSABLE_FCP_DEV) {
232 			int ret = strncmp(lunlistp->path, SCSI_VHCI_ROOT,
233 				strlen(SCSI_VHCI_ROOT));
234 
235 			if (((ret == 0) &&
236 			    (lunlistp->node_state == DI_PATH_STATE_OFFLINE)) ||
237 			    ((ret != 0) &&
238 			    ((lunlistp->node_state & DI_DEVICE_OFFLINE) ==
239 			    DI_DEVICE_OFFLINE))) {
240 				/* Offline the device through RCM */
241 				if (fp_rcm_offline(lunlistp->path, errstring,
242 					    flags) != 0) {
243 					/*
244 					 * Bring everything back online in
245 					 * rcm and return
246 					 */
247 					(void) dev_rcm_online(apidt, count,
248 								flags, NULL);
249 					return (FPCFGA_LIB_ERR);
250 				}
251 				count++;
252 			}
253 		} else {
254 			/* Offline the device through RCM */
255 			if (fp_rcm_offline(lunlistp->path, errstring,
256 				    flags) != 0) {
257 				/*
258 				 * Bring everything back online in
259 				 * rcm and return
260 				 */
261 				(void) dev_rcm_online(apidt, count, flags,
262 									NULL);
263 				return (FPCFGA_LIB_ERR);
264 			}
265 			count++;
266 		}
267 	}
268 	return (FPCFGA_OK);
269 }
270 
271 /*
272  * Remove, in RCM, all the LUNs for a particular device.
273  * This function should not be called for the MPXIO case.
274  *
275  * INPUT :
276  * apidt - this is expected to have the list of luns for the device and so
277  *         is assumed to be filled in prior to this call
278  * errstring - If non-NULL, it will hold any error messages
279  *
280  * RETURNS :
281  * FPCFGA_OK on success
282  * error code otherwise
283  */
284 static fpcfga_ret_t
285 dev_rcm_remove(apid_t *apidt, cfga_flags_t flags, char **errstring)
286 {
287 	int		count = 0;
288 	luninfo_list_t	*lunlistp;
289 
290 	if ((apidt->flags & FLAG_DISABLE_RCM) != 0) {
291 		/* User has requested not to notify RCM framework */
292 		return (FPCFGA_OK);
293 	}
294 
295 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
296 						lunlistp = lunlistp->next) {
297 		if ((lunlistp->lun_flag & FLAG_SKIP_RCMREMOVE) != 0)
298 			continue;
299 		if ((apidt->flags & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
300 		    FLAG_REMOVE_UNUSABLE_FCP_DEV) {
301 			int ret = strncmp(lunlistp->path, SCSI_VHCI_ROOT,
302 				strlen(SCSI_VHCI_ROOT));
303 
304 			if (((ret == 0) &&
305 			    (lunlistp->node_state == DI_PATH_STATE_OFFLINE)) ||
306 			    ((ret != 0) &&
307 			    ((lunlistp->node_state & DI_DEVICE_OFFLINE) ==
308 			    DI_DEVICE_OFFLINE))) {
309 				/* remove the device through RCM */
310 				if (fp_rcm_remove(lunlistp->path, errstring,
311 					    flags) != 0) {
312 					/*
313 					 * Bring everything back online in
314 					 * rcm and return
315 					 */
316 					(void) dev_rcm_online(apidt, count,
317 								flags, NULL);
318 					return (FPCFGA_LIB_ERR);
319 				}
320 				count++;
321 			}
322 		} else {
323 			/* remove the device through RCM */
324 			if (fp_rcm_remove(lunlistp->path, errstring,
325 				flags) != 0) {
326 				/*
327 				 * Bring everything back online in rcm and
328 				 * return
329 				 */
330 				(void) dev_rcm_online(apidt, count, flags,
331 									NULL);
332 				return (FPCFGA_LIB_ERR);
333 			}
334 			count++;
335 		}
336 	}
337 	return (FPCFGA_OK);
338 }
339 
340 static fpcfga_ret_t
341 lun_unconf(char *path, int lunnum, char *xport_phys, char *dyncomp,
342 						char **errstring)
343 {
344 	devctl_hdl_t	hdl;
345 	char		*ptr;		/* To use as scratch/temp pointer */
346 	char		pathname[MAXPATHLEN];
347 
348 	if (path == NULL)
349 		return (FPCFGA_OK);
350 
351 	if (strncmp(path, SCSI_VHCI_ROOT, strlen(SCSI_VHCI_ROOT)) == 0) {
352 		/*
353 		 * We have an MPXIO managed device here.
354 		 * So, we have to concoct a path for the device.
355 		 *
356 		 * xport_phys looks like :
357 		 * /devices/pci@b,2000/pci@1/SUNW,qlc@5/fp@0,0:fc
358 		 */
359 		(void) strlcpy(pathname, xport_phys, MAXPATHLEN);
360 		if ((ptr = strrchr(pathname, ':')) != NULL) {
361 			*ptr = '\0';
362 		}
363 
364 		/*
365 		 * Get pointer to driver name from VHCI path
366 		 * So, if lunlistp->path is
367 		 * /devices/scsi_vhci/ssd@g220000203707a417,
368 		 * we need a pointer to the last '/'
369 		 *
370 		 * Assumption:
371 		 * With MPXIO there will be only one entry per lun
372 		 * So, there will only be one entry in the linked list
373 		 * apidt->lunlist
374 		 */
375 		if ((ptr = strrchr(path, '/')) == NULL) {
376 			/* This shouldn't happen, but anyways ... */
377 			cfga_err(errstring, 0, ERRARG_INVALID_PATH, path, 0);
378 			return (FPCFGA_LIB_ERR);
379 		}
380 
381 		/*
382 		 * Make pathname to look something like :
383 		 * /devices/pci@x,xxxx/pci@x/SUNW,qlc@x/fp@x,x/ssd@w...
384 		 */
385 		strcat(pathname, ptr);
386 
387 		/*
388 		 * apidt_create() will make sure that lunlist->path
389 		 * has a "@<something>" at the end even if the driver
390 		 * state is "detached"
391 		 */
392 		if ((ptr = strrchr(pathname, '@')) == NULL) {
393 			/* This shouldn't happen, but anyways ... */
394 			cfga_err(errstring, 0, ERRARG_INVALID_PATH,
395 						pathname, 0);
396 			return (FPCFGA_LIB_ERR);
397 		}
398 		*ptr = '\0';
399 
400 		/* Now, concoct the path */
401 		sprintf(&pathname[strlen(pathname)], "@w%s,%x",
402 							dyncomp, lunnum);
403 		ptr = pathname;
404 	} else {
405 		/*
406 		 * non-MPXIO path, use the path that is passed in
407 		 */
408 		ptr = path;
409 	}
410 
411 	if ((hdl = devctl_device_acquire(ptr,  0)) == NULL) {
412 		cfga_err(errstring, errno, ERRARG_DEV_ACQUIRE, ptr, 0);
413 		return (FPCFGA_LIB_ERR);
414 	}
415 
416 	if (devctl_device_remove(hdl) != 0) {
417 		devctl_release(hdl);
418 		cfga_err(errstring, errno, ERRARG_DEV_REMOVE, ptr, 0);
419 		return (FPCFGA_LIB_ERR);
420 	}
421 	devctl_release(hdl);
422 
423 	return (FPCFGA_OK);
424 }
425 
426 static fpcfga_ret_t
427 dev_unconf(apid_t *apidt, char **errstring, uchar_t *flag)
428 {
429 	luninfo_list_t	*lunlistp;
430 	fpcfga_ret_t	ret = FPCFGA_OK;
431 	int lun_cnt = 0, unusable_lun_cnt = 0;
432 
433 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
434 	    lunlistp = lunlistp->next) {
435 		lun_cnt++;
436 		/*
437 		 * Unconfigure each LUN.
438 		 * Note that for MPXIO devices, lunlistp->path will be a
439 		 * vHCI path
440 		 */
441 		if ((apidt->flags & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
442 			FLAG_REMOVE_UNUSABLE_FCP_DEV) {
443 		    if (strncmp(lunlistp->path, SCSI_VHCI_ROOT,
444 			strlen(SCSI_VHCI_ROOT)) == 0) {
445 			if (lunlistp->node_state ==
446 				DI_PATH_STATE_OFFLINE) {
447 			    unusable_lun_cnt++;
448 			    if ((ret = lun_unconf(lunlistp->path,
449 				lunlistp->lunnum, apidt->xport_phys,
450 				apidt->dyncomp, errstring)) != FPCFGA_OK) {
451 				return (ret);
452 			    }
453 			}
454 		    } else {
455 			if ((lunlistp->node_state & DI_DEVICE_OFFLINE) ==
456 				DI_DEVICE_OFFLINE) {
457 			    unusable_lun_cnt++;
458 			    if ((ret = lun_unconf(lunlistp->path,
459 				lunlistp->lunnum, apidt->xport_phys,
460 				apidt->dyncomp, errstring)) != FPCFGA_OK) {
461 				return (ret);
462 			    }
463 			}
464 		    }
465 		} else {
466 		/*
467 		 * Unconfigure each LUN.
468 		 * Note that for MPXIO devices, lunlistp->path will be a
469 		 * vHCI path
470 		 */
471 		    if ((ret = lun_unconf(lunlistp->path, lunlistp->lunnum,
472 				apidt->xport_phys, apidt->dyncomp,
473 				errstring)) != FPCFGA_OK) {
474 			return (ret);
475 		    }
476 		}
477 	}
478 
479 	if ((apidt->flags & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
480 			FLAG_REMOVE_UNUSABLE_FCP_DEV) {
481 		/*
482 		 * when all luns are unconfigured
483 		 * indicate to remove repository entry.
484 		 */
485 		if (lun_cnt == unusable_lun_cnt) {
486 			*flag = ALL_APID_LUNS_UNUSABLE;
487 		}
488 	}
489 
490 	return (ret);
491 }
492 
493 /*
494  * Check if the given physical path (the xport_phys) is part of the
495  * pHCI list and if the RCM should be done for a particular pHCI.
496  * Skip non-MPxIO dev node if any.
497  */
498 static fpcfga_ret_t
499 is_xport_phys_in_pathlist(apid_t *apidt, char **errstring)
500 {
501 	di_node_t	root, vhci, node, phci;
502 	di_path_t	path = DI_PATH_NIL;
503 	int		num_active_paths, found = 0;
504 	char		*vhci_path_ptr, *pathname_ptr, pathname[MAXPATHLEN];
505 	char		*phci_path, *node_path;
506 	char		phci_addr[MAXPATHLEN];
507 	char		*xport_phys, *vhci_path, *dyncomp;
508 	luninfo_list_t	*lunlistp, *temp;
509 	int		non_operational_path_count;
510 
511 	/* a safety check */
512 	if ((apidt->dyncomp == NULL) || (*apidt->dyncomp == '\0')) {
513 		return (FPCFGA_LIB_ERR);
514 	}
515 
516 	xport_phys = apidt->xport_phys;
517 	dyncomp = apidt->dyncomp;
518 
519 	lunlistp = apidt->lunlist;
520 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
521 	    lunlistp = lunlistp->next) {
522 
523 		if (strncmp(lunlistp->path, SCSI_VHCI_ROOT,
524 		    strlen(SCSI_VHCI_ROOT)) != 0) {
525 			lunlistp->lun_flag |= FLAG_SKIP_ONLINEOTHERS;
526 			continue;
527 		}
528 
529 		vhci_path = lunlistp->path;
530 
531 		num_active_paths = 0;	/* # of paths in ONLINE/STANDBY */
532 		non_operational_path_count = 0;
533 
534 		if (xport_phys == NULL || vhci_path == NULL) {
535 		    cfga_err(errstring, 0, ERRARG_XPORT_NOT_IN_PHCI_LIST,
536 		    xport_phys, 0);
537 			return (FPCFGA_LIB_ERR);
538 		}
539 
540 		(void) strlcpy(pathname, xport_phys, MAXPATHLEN);
541 		if ((pathname_ptr = strrchr(pathname, ':')) != NULL) {
542 			*pathname_ptr = '\0';
543 		}
544 		/* strip off the /devices/from the path */
545 		pathname_ptr = pathname + strlen(DEVICES_DIR);
546 
547 		root = di_init("/", DINFOCPYALL|DINFOPATH);
548 
549 		if (root == DI_NODE_NIL) {
550 			return (FPCFGA_LIB_ERR);
551 		}
552 
553 		vhci_path_ptr = vhci_path + strlen(DEVICES_DIR);
554 		if ((vhci = di_drv_first_node(SCSI_VHCI_DRVR, root)) ==
555 		    DI_NODE_NIL) {
556 			return (FPCFGA_LIB_ERR);
557 		}
558 		found = 0;
559 		for (node = di_child_node(vhci); node != DI_NODE_NIL;
560 		    node = di_sibling_node(node)) {
561 			if ((node_path = di_devfs_path(node)) != NULL) {
562 				if (strncmp(vhci_path_ptr, node_path,
563 				    strlen(node_path)) != 0) {
564 					di_devfs_path_free(node_path);
565 				} else {
566 					found = 1;
567 					break;
568 				}
569 			}
570 		}
571 		if (found == 0) {
572 			cfga_err(errstring, 0, ERRARG_XPORT_NOT_IN_PHCI_LIST,
573 			    xport_phys, 0);
574 			di_fini(root);
575 			return (FPCFGA_LIB_ERR);
576 		}
577 		/* found vhci_path we are looking for */
578 		di_devfs_path_free(node_path);
579 		found = 0;
580 		for (path = di_path_next_phci(node, DI_PATH_NIL);
581 		    path != DI_PATH_NIL;
582 		    path = di_path_next_phci(node, path)) {
583 			if ((phci = di_path_phci_node(path)) == DI_NODE_NIL) {
584 				cfga_err(errstring, 0,
585 				    ERRARG_XPORT_NOT_IN_PHCI_LIST,
586 				    xport_phys, 0);
587 				di_fini(root);
588 				return (FPCFGA_LIB_ERR);
589 			}
590 			if ((phci_path = di_devfs_path(phci)) == NULL) {
591 				cfga_err(errstring, 0,
592 				    ERRARG_XPORT_NOT_IN_PHCI_LIST,
593 				    xport_phys, 0);
594 				di_fini(root);
595 				return (FPCFGA_LIB_ERR);
596 			}
597 			(void) di_path_addr(path, (char *)phci_addr);
598 			if ((phci_addr == NULL) || (*phci_addr == '\0')) {
599 				cfga_err(errstring, 0,
600 				    ERRARG_XPORT_NOT_IN_PHCI_LIST,
601 				    xport_phys, 0);
602 				di_devfs_path_free(phci_path);
603 				di_fini(root);
604 				return (FPCFGA_LIB_ERR);
605 			}
606 			/*
607 			 * Check if the phci path has the same
608 			 * xport addr and the target addr with current lun
609 			 */
610 			if ((strncmp(phci_path, pathname_ptr,
611 			    strlen(pathname_ptr)) == 0) &&
612 			    (strstr(phci_addr, dyncomp) != NULL)) {
613 				/* SUCCESS Found xport_phys */
614 				found = 1;
615 			} else if ((di_path_state(path) ==
616 			    DI_PATH_STATE_ONLINE) ||
617 			    (di_path_state(path) == DI_PATH_STATE_STANDBY)) {
618 				num_active_paths++;
619 			} else {
620 				/*
621 				 * We have another path not in ONLINE/STANDBY
622 				 * state now, so should do a RCM online after
623 				 * the unconfiguration of current path.
624 				 */
625 				non_operational_path_count++;
626 			}
627 			di_devfs_path_free(phci_path);
628 		}
629 		di_fini(root);
630 		if (found == 1) {
631 			if (num_active_paths != 0) {
632 				/*
633 				 * There are other ONLINE/STANDBY paths,
634 				 * so no need to do the RCM
635 				 */
636 				lunlistp->lun_flag |= FLAG_SKIP_RCMREMOVE;
637 				lunlistp->lun_flag |= FLAG_SKIP_RCMOFFLINE;
638 			}
639 			if (non_operational_path_count == 0) {
640 				lunlistp->lun_flag |= FLAG_SKIP_ONLINEOTHERS;
641 			}
642 		} else {
643 			/*
644 			 * Fail all operations here
645 			 */
646 			cfga_err(errstring, 0, ERRARG_XPORT_NOT_IN_PHCI_LIST,
647 			    xport_phys, 0);
648 			return (FPCFGA_APID_NOEXIST);
649 		}
650 	}
651 
652 	/* Mark duplicated paths for same vhci in the list */
653 	for (lunlistp = apidt->lunlist; lunlistp != NULL;
654 	    lunlistp = lunlistp->next) {
655 		if (strncmp(lunlistp->path, SCSI_VHCI_ROOT,
656 		    strlen(SCSI_VHCI_ROOT)) != 0) {
657 			continue;
658 		}
659 		for (temp = lunlistp->next; temp != NULL;
660 		    temp = temp->next) {
661 			if (strcmp(lunlistp->path, temp->path) == 0) {
662 				/*
663 				 * don't do RCM for dup
664 				 */
665 				lunlistp->lun_flag |= FLAG_SKIP_RCMREMOVE;
666 				lunlistp->lun_flag |= FLAG_SKIP_RCMOFFLINE;
667 				lunlistp->lun_flag |= FLAG_SKIP_ONLINEOTHERS;
668 			}
669 		}
670 	}
671 	return (FPCFGA_OK);
672 }
673 /*
674  * apidt->dyncomp has to be non-NULL by the time this routine is called
675  */
676 fpcfga_ret_t
677 dev_change_state(cfga_cmd_t state_change_cmd, apid_t *apidt, la_wwn_t *pwwn,
678 		cfga_flags_t flags, char **errstring, HBA_HANDLE handle,
679 		HBA_PORTATTRIBUTES portAttrs)
680 {
681 	char			dev_path[MAXPATHLEN];
682 	char			*update_str, *t_apid;
683 	int			optflag = apidt->flags;
684 	int			no_config_attempt = 0;
685 	fpcfga_ret_t		ret;
686 	apid_t			my_apidt;
687 	uchar_t			unconf_flag = 0, peri_qual;
688 	HBA_STATUS		status;
689 	HBA_PORTATTRIBUTES	discPortAttrs;
690 	uint64_t		lun = 0;
691 	struct scsi_inquiry	inq;
692 	struct scsi_extended_sense sense;
693 	HBA_UINT8		scsiStatus;
694 	uint32_t		inquirySize = sizeof (inq),
695 				senseSize = sizeof (sense);
696 	report_lun_resp_t	*resp_buf;
697 	int			i, l_errno, num_luns = 0;
698 	uchar_t			*lun_string;
699 
700 	if ((apidt->dyncomp == NULL) || (*apidt->dyncomp == '\0')) {
701 		/*
702 		 * No dynamic component specified. Just return success.
703 		 * Should not see this case. Just a safety check.
704 		 */
705 		return (FPCFGA_OK);
706 	}
707 
708 	/* Now construct the string we are going to put in the repository */
709 	if ((update_str = calloc(1, (strlen(apidt->xport_phys) +
710 		strlen(DYN_SEP) + strlen(apidt->dyncomp) + 1))) == NULL) {
711 		cfga_err(errstring, errno, ERR_MEM_ALLOC, 0);
712 		return (FPCFGA_LIB_ERR);
713 	}
714 	strcpy(update_str, apidt->xport_phys);
715 	strcat(update_str, DYN_SEP);
716 	strcat(update_str, apidt->dyncomp);
717 
718 	/* If force update of repository is sought, do it first */
719 	if (optflag & FLAG_FORCE_UPDATE_REP) {
720 		/* Ignore any failure in rep update */
721 		(void) update_fabric_wwn_list(
722 			((state_change_cmd == CFGA_CMD_CONFIGURE) ?
723 			ADD_ENTRY : REMOVE_ENTRY),
724 			update_str, errstring);
725 	}
726 
727 	memset(&sense, 0, sizeof (sense));
728 	if ((ret = get_report_lun_data(apidt->xport_phys, apidt->dyncomp,
729 		&num_luns, &resp_buf, &sense, &l_errno)) != FPCFGA_OK) {
730 		/*
731 		 * Checking the sense key data as well as the additional
732 		 * sense key.  The SES Node is not required to repond
733 		 * to Report LUN.  In the case of Minnow, the SES node
734 		 * returns with KEY_ILLEGAL_REQUEST and the additional
735 		 * sense key of 0x20.  In this case we will blindly
736 		 * send the SCSI Inquiry call to lun 0
737 		 *
738 		 * if we get any other error we will set the inq_type
739 		 * appropriately
740 		 */
741 		if ((sense.es_key == KEY_ILLEGAL_REQUEST) &&
742 		    (sense.es_add_code == 0x20)) {
743 			lun = 0;
744 		} else {
745 			if (ret == FPCFGA_FCP_SEND_SCSI_DEV_NOT_TGT) {
746 				inq.inq_dtype = DTYPE_UNKNOWN;
747 			} else {
748 				/*
749 				 * Failed to get the LUN data for the device
750 				 * If we find that there is a lunlist for this
751 				 * device it could mean that there are dangling
752 				 * devinfo nodes. So, we will go ahead and try
753 				 * to unconfigure them.
754 				 */
755 				if ((apidt->lunlist == NULL) ||
756 				    (state_change_cmd == CFGA_CMD_CONFIGURE)) {
757 					S_FREE(update_str);
758 					status = getPortAttrsByWWN(handle,
759 					    *((HBA_WWN *)(pwwn)),
760 					    &discPortAttrs);
761 					if (status ==
762 					    HBA_STATUS_ERROR_ILLEGAL_WWN) {
763 						return (FPCFGA_APID_NOEXIST);
764 					} else {
765 						cfga_err(errstring, 0,
766 						    ERRARG_FC_REP_LUNS,
767 						    apidt->dyncomp, 0);
768 						return (FPCFGA_LIB_ERR);
769 					}
770 				} else {
771 					/* unconfig with lunlist not empty */
772 					no_config_attempt++;
773 				}
774 			}
775 		}
776 	}
777 	for (i = 0; i < num_luns; i++) {
778 		/*
779 		 * issue the inquiry to the first valid lun found
780 		 * in the lun_string
781 		 */
782 		lun_string = (uchar_t *)&(resp_buf->lun_string[i]);
783 		memcpy(&lun, lun_string, sizeof (lun));
784 
785 		memset(&sense, 0, sizeof (sense));
786 		status = HBA_ScsiInquiryV2(handle, portAttrs.PortWWN,
787 		    *(HBA_WWN *)(pwwn), lun, 0, 0, &inq, &inquirySize,
788 		    &scsiStatus, &sense, &senseSize);
789 		/*
790 		 * if Inquiry is returned correctly, check the
791 		 * peripheral qualifier for the lun.  if it is non-zero
792 		 * then try the SCSI Inquiry on the next lun
793 		 */
794 		if (status == HBA_STATUS_OK) {
795 			peri_qual = inq.inq_dtype & FP_PERI_QUAL_MASK;
796 			if (peri_qual == DPQ_POSSIBLE) {
797 				break;
798 			}
799 		}
800 	}
801 
802 	if (ret == FPCFGA_OK)
803 		S_FREE(resp_buf);
804 
805 	/*
806 	 * If there are no luns on this target, we will attempt to send
807 	 * the SCSI Inquiry to lun 0
808 	 */
809 	if (num_luns == 0) {
810 		lun = 0;
811 		status = HBA_ScsiInquiryV2(handle, portAttrs.PortWWN,
812 		    *(HBA_WWN *)(pwwn), lun, 0, 0, &inq, &inquirySize,
813 		    &scsiStatus, &sense, &senseSize);
814 	}
815 
816 	if (status != HBA_STATUS_OK) {
817 		if (status ==  HBA_STATUS_ERROR_NOT_A_TARGET) {
818 			inq.inq_dtype = DTYPE_UNKNOWN;
819 		} else if (status ==  HBA_STATUS_ERROR_ILLEGAL_WWN) {
820 			free(update_str);
821 			return (FPCFGA_APID_NOEXIST);
822 		} else {
823 			/*
824 			 * Failed to get the inq_dtype of device
825 			 * If we find that there is a lunlist for this
826 			 * device it could mean that there dangling
827 			 * devinfo nodes. So, we will go ahead and try
828 			 * to unconfigure them.  We'll just set the
829 			 * inq_dtype to some invalid value (0xFF)
830 			 */
831 			if ((apidt->lunlist == NULL) ||
832 			    (state_change_cmd == CFGA_CMD_CONFIGURE)) {
833 				cfga_err(errstring, 0,
834 				    ERRARG_FC_INQUIRY,
835 				    apidt->dyncomp, 0);
836 				free(update_str);
837 				return (FPCFGA_LIB_ERR);
838 			} else {
839 				/* unconfig with lunlist not empty */
840 				no_config_attempt++;
841 			}
842 		}
843 	}
844 	switch (state_change_cmd) {
845 	case CFGA_CMD_CONFIGURE:
846 	    if (portAttrs.PortType != HBA_PORTTYPE_NLPORT &&
847 		portAttrs.PortType != HBA_PORTTYPE_NPORT) {
848 		free(update_str);
849 		return (FPCFGA_OK);
850 	    }
851 
852 	    if (((inq.inq_dtype & DTYPE_MASK) == DTYPE_UNKNOWN) &&
853 		((flags & CFGA_FLAG_FORCE) == 0)) {
854 		/*
855 		 * We assume all DTYPE_UNKNOWNs are HBAs and we wont
856 		 * waste time trying to config them. If they are not
857 		 * HBAs, then there is something wrong since they should
858 		 * have had a valid dtype.
859 		 *
860 		 * However, if the force flag is set (cfgadm -f), we
861 		 * go ahead and try to configure.
862 		 *
863 		 * In this path, however, the force flag is not set.
864 		 */
865 		free(update_str);
866 		return (FPCFGA_OK);
867 	    }
868 
869 	    errno = 0;
870 		/*
871 		 * We'll issue the devctl_bus_dev_create() call even if the
872 		 * path exists in the devinfo tree. This is to take care of
873 		 * the situation where the device may be in a state other
874 		 * than the online and attached state.
875 		 */
876 	    if ((ret = do_devctl_dev_create(apidt, dev_path, MAXPATHLEN,
877 			inq.inq_dtype, errstring)) != FPCFGA_OK) {
878 		/*
879 		 * Could not configure device. To provide a more
880 		 * meaningful error message, first see if the supplied port
881 		 * WWN is there on the fabric. Otherwise print the error
882 		 * message using the information received from the driver
883 		 */
884 		status = getPortAttrsByWWN(handle, *((HBA_WWN *)(pwwn)),
885 		    &discPortAttrs);
886 		S_FREE(update_str);
887 		if (status == HBA_STATUS_ERROR_ILLEGAL_WWN) {
888 			return (FPCFGA_APID_NOEXIST);
889 		} else {
890 			return (FPCFGA_LIB_ERR);
891 		}
892 	    }
893 
894 	    if (((optflag & (FLAG_FORCE_UPDATE_REP|FLAG_NO_UPDATE_REP)) == 0) &&
895 		update_fabric_wwn_list(ADD_ENTRY, update_str, errstring)) {
896 		    cfga_err(errstring, 0, ERR_CONF_OK_UPD_REP, 0);
897 	    }
898 
899 	    S_FREE(update_str);
900 
901 	    if ((apidt->flags & FLAG_DISABLE_RCM) == 0) {
902 		/*
903 		 * There may be multiple LUNs associated with the
904 		 * WWN we created nodes for. So, we'll call
905 		 * apidt_create() again and let it build a list of
906 		 * all the LUNs for this WWN using the devinfo tree.
907 		 * We will then online all those devices in RCM
908 		 */
909 		    if ((t_apid = calloc(1, strlen(apidt->xport_phys) +
910 					strlen(DYN_SEP) +
911 					strlen(apidt->dyncomp) + 1)) == NULL) {
912 			    cfga_err(errstring, errno, ERR_MEM_ALLOC, 0);
913 			    return (FPCFGA_LIB_ERR);
914 		    }
915 		    sprintf(t_apid, "%s%s%s", apidt->xport_phys, DYN_SEP,
916 			apidt->dyncomp);
917 		    if ((ret = apidt_create(t_apid, &my_apidt,
918 					errstring)) != FPCFGA_OK) {
919 			    free(t_apid);
920 			    return (ret);
921 		    }
922 
923 		    my_apidt.flags = apidt->flags;
924 		    if ((ret = dev_rcm_online(&my_apidt, -1, flags,
925 					NULL)) != FPCFGA_OK) {
926 			    cfga_err(errstring, 0, ERRARG_RCM_ONLINE,
927 				apidt->lunlist->path, 0);
928 			    apidt_free(&my_apidt);
929 			    free(t_apid);
930 			    return (ret);
931 		    }
932 		    S_FREE(t_apid);
933 		    apidt_free(&my_apidt);
934 	    }
935 	    return (FPCFGA_OK);
936 
937 	case CFGA_CMD_UNCONFIGURE:
938 		if (portAttrs.PortType != HBA_PORTTYPE_NLPORT &&
939 		    portAttrs.PortType != HBA_PORTTYPE_NPORT) {
940 			free(update_str);
941 			return (FPCFGA_OPNOTSUPP);
942 		}
943 
944 		status = getPortAttrsByWWN(handle, *((HBA_WWN *)(pwwn)),
945 		    &discPortAttrs);
946 		if (apidt->lunlist == NULL) {
947 			/*
948 			 * But first, remove entry from the repository if it is
949 			 * there ... provided the force update flag is not set
950 			 * (in which case the update is already done) or if
951 			 * the no-update flag is not set.
952 			 */
953 			if ((optflag &
954 			(FLAG_FORCE_UPDATE_REP|FLAG_NO_UPDATE_REP)) == 0) {
955 				if (update_fabric_wwn_list(REMOVE_ENTRY,
956 						update_str, errstring)) {
957 					free(update_str);
958 					cfga_err(errstring, 0,
959 						ERR_UNCONF_OK_UPD_REP, 0);
960 					return
961 					(FPCFGA_UNCONF_OK_UPD_REP_FAILED);
962 				}
963 			}
964 			S_FREE(update_str);
965 			if (status == HBA_STATUS_ERROR_ILLEGAL_WWN) {
966 				return (FPCFGA_APID_NOEXIST);
967 			}
968 			return (FPCFGA_OK);
969 		}
970 		/*
971 		 * If there are multiple paths to the mpxio
972 		 * device, we will not check in RCM ONLY when there
973 		 * is atleast one other ONLINE/STANDBY path
974 		 */
975 		if (is_xport_phys_in_pathlist(apidt, errstring) !=
976 		    FPCFGA_OK) {
977 			free(update_str);
978 			return (FPCFGA_XPORT_NOT_IN_PHCI_LIST);
979 		}
980 
981 		/*
982 		 * dev_rcm_offline() updates errstring
983 		 */
984 		if ((ret = dev_rcm_offline(apidt, flags, errstring)) !=
985 		    FPCFGA_OK) {
986 			free(update_str);
987 			return (ret);
988 		}
989 		if ((ret = dev_unconf(apidt, errstring, &unconf_flag)) !=
990 		    FPCFGA_OK) {
991 			/* when inq failed don't attempt to reconfigure */
992 		    if (!no_config_attempt) {
993 			(void) do_devctl_dev_create(apidt, dev_path, MAXPATHLEN,
994 				inq.inq_dtype, NULL);
995 			(void) dev_rcm_online(apidt, -1, flags, NULL);
996 		    }
997 		    free(update_str);
998 		    return (ret);
999 		}
1000 		if ((ret = dev_rcm_remove(apidt, flags, errstring)) !=
1001 		    FPCFGA_OK) {
1002 			(void) do_devctl_dev_create(apidt, dev_path, MAXPATHLEN,
1003 				inq.inq_dtype, NULL);
1004 			(void) dev_rcm_online(apidt, -1, flags, NULL);
1005 			free(update_str);
1006 			return (ret);
1007 		}
1008 		/*
1009 		 * If we offlined a lun in RCM when there are multiple paths but
1010 		 * none of them are ONLINE/STANDBY, we have to online it back
1011 		 * in RCM now. This is a try best, will not fail for it.
1012 		 */
1013 		dev_rcm_online_nonoperationalpath(apidt, flags, NULL);
1014 
1015 		/* Update the repository if we havent already done it */
1016 		if ((optflag &
1017 			(FLAG_FORCE_UPDATE_REP|FLAG_NO_UPDATE_REP)) == 0) {
1018 			if (((optflag & FLAG_REMOVE_UNUSABLE_FCP_DEV) !=
1019 				    FLAG_REMOVE_UNUSABLE_FCP_DEV) ||
1020 			    (((optflag & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
1021 				FLAG_REMOVE_UNUSABLE_FCP_DEV) &&
1022 			    (unconf_flag == ALL_APID_LUNS_UNUSABLE))) {
1023 				if (update_fabric_wwn_list(REMOVE_ENTRY,
1024 					    update_str, errstring)) {
1025 				    free(update_str);
1026 				    cfga_err(errstring, errno,
1027 					ERR_UNCONF_OK_UPD_REP, 0);
1028 				    return (FPCFGA_UNCONF_OK_UPD_REP_FAILED);
1029 				}
1030 			}
1031 		}
1032 		free(update_str);
1033 		return (FPCFGA_OK);
1034 
1035 	default:
1036 		free(update_str);
1037 		return (FPCFGA_OPNOTSUPP);
1038 	}
1039 }
1040 
1041 /*
1042  * This function copies a port_wwn got by reading the property on a device
1043  * node (from_ptr in the function below) on to an array (to_ptr) so that it is
1044  * readable.
1045  *
1046  * Caller responsible to allocate enough memory in "to_ptr"
1047  */
1048 static void
1049 copy_pwwn_data_to_str(char *to_ptr, const uchar_t *from_ptr)
1050 {
1051 	if ((to_ptr == NULL) || (from_ptr == NULL))
1052 		return;
1053 
1054 	(void) sprintf(to_ptr, "%1.2x%1.2x%1.2x%1.2x%1.2x%1.2x%1.2x%1.2x",
1055 	from_ptr[0], from_ptr[1], from_ptr[2], from_ptr[3],
1056 	from_ptr[4], from_ptr[5], from_ptr[6], from_ptr[7]);
1057 }
1058 
1059 static fpcfga_ret_t
1060 unconf_vhci_nodes(di_path_t pnode, di_node_t fp_node, char *xport_phys,
1061 	char *dyncomp, int unusable_flag,
1062 	int *num_devs, int *failure_count, char **errstring,
1063 	cfga_flags_t flags)
1064 {
1065 	int		iret1, iret2, *lunnump;
1066 	char		*ptr;		/* scratch pad */
1067 	char		*node_path, *vhci_path, *update_str;
1068 	char		port_wwn[WWN_SIZE*2+1], pathname[MAXPATHLEN];
1069 	uchar_t		*port_wwn_data = NULL;
1070 	di_node_t	client_node;
1071 
1072 	while (pnode != DI_PATH_NIL) {
1073 
1074 		(*num_devs)++;
1075 
1076 
1077 		if ((node_path = di_devfs_path(fp_node)) == NULL) {
1078 			cfga_err(errstring, 0, ERRARG_DEVINFO,
1079 							xport_phys, 0);
1080 			(*failure_count)++;
1081 			pnode = di_path_next_client(fp_node, pnode);
1082 			continue;
1083 		}
1084 
1085 		iret1 = di_path_prop_lookup_bytes(pnode, PORT_WWN_PROP,
1086 			&port_wwn_data);
1087 
1088 		iret2 = di_path_prop_lookup_ints(pnode, LUN_PROP, &lunnump);
1089 
1090 		if ((iret1 == -1) || (iret2 == -1)) {
1091 			cfga_err(errstring, 0, ERRARG_DI_GET_PROP,
1092 								node_path, 0);
1093 			di_devfs_path_free(node_path);
1094 			node_path = NULL;
1095 			(*failure_count)++;
1096 			pnode = di_path_next_client(fp_node, pnode);
1097 			continue;
1098 		}
1099 
1100 		copy_pwwn_data_to_str(port_wwn, port_wwn_data);
1101 
1102 		if ((client_node = di_path_client_node(pnode)) ==
1103 								DI_NODE_NIL) {
1104 			(*failure_count)++;
1105 			di_devfs_path_free(node_path);
1106 			node_path = NULL;
1107 			pnode = di_path_next_client(fp_node, pnode);
1108 			continue;
1109 		}
1110 
1111 		if ((vhci_path = di_devfs_path(client_node)) == NULL) {
1112 			(*failure_count)++;
1113 			di_devfs_path_free(node_path);
1114 			node_path = NULL;
1115 			pnode = di_path_next_client(fp_node, pnode);
1116 			continue;
1117 		}
1118 
1119 		if ((ptr = strrchr(vhci_path, '@')) != NULL) {
1120 			*ptr = '\0';
1121 		}
1122 
1123 		if ((ptr = strrchr(vhci_path, '/')) == NULL) {
1124 			(*failure_count)++;
1125 			di_devfs_path_free(node_path);
1126 			node_path = NULL;
1127 			pnode = di_path_next_client(fp_node, pnode);
1128 			continue;
1129 		}
1130 
1131 		sprintf(pathname, "%s%s/%s@w%s,%x", DEVICES_DIR, node_path,
1132 					++ptr, port_wwn, *lunnump);
1133 
1134 		di_devfs_path_free(node_path);
1135 		di_devfs_path_free(vhci_path);
1136 		node_path = vhci_path = NULL;
1137 
1138 		/*
1139 		 * Try to offline in RCM first and if that is successful,
1140 		 * unconfigure the LUN. If offlining in RCM fails, then
1141 		 * update the failure_count which gets passed back to caller
1142 		 *
1143 		 * Here we got to check if unusable_flag is set or not.
1144 		 * If set, then unconfigure only those luns which are in
1145 		 * node_state DI_PATH_STATE_OFFLINE. If not set, unconfigure
1146 		 * all luns.
1147 		 */
1148 		if ((unusable_flag & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
1149 		    FLAG_REMOVE_UNUSABLE_FCP_DEV) {
1150 			if (pnode->path_state == DI_PATH_STATE_OFFLINE) {
1151 				if (fp_rcm_offline(pathname, errstring,
1152 				    flags) != 0) {
1153 					(*failure_count)++;
1154 					pnode = di_path_next_client(fp_node,
1155 					    pnode);
1156 					continue;
1157 				} else if (lun_unconf(pathname, *lunnump,
1158 				    xport_phys,dyncomp, errstring)
1159 				    != FPCFGA_OK) {
1160 					(void) fp_rcm_online(pathname,
1161 					    NULL, flags);
1162 					(*failure_count)++;
1163 					pnode = di_path_next_client(fp_node,
1164 					    pnode);
1165 					continue;
1166 				} else if (fp_rcm_remove(pathname, errstring,
1167 				    flags) != 0) {
1168 					/*
1169 					 * Bring everything back online
1170 					 * in rcm and continue
1171 					 */
1172 					(void) fp_rcm_online(pathname,
1173 					    NULL, flags);
1174 					(*failure_count)++;
1175 					pnode = di_path_next_client(fp_node,
1176 					    pnode);
1177 					continue;
1178 				}
1179 			} else {
1180 				pnode = di_path_next(fp_node, pnode);
1181 				continue;
1182 			}
1183 		} else {
1184 			if (fp_rcm_offline(pathname, errstring, flags) != 0) {
1185 				(*failure_count)++;
1186 				pnode = di_path_next_client(fp_node, pnode);
1187 				continue;
1188 			} else if (lun_unconf(pathname, *lunnump, xport_phys,
1189 			    dyncomp, errstring) != FPCFGA_OK) {
1190 				(void) fp_rcm_online(pathname, NULL, flags);
1191 				(*failure_count)++;
1192 				pnode = di_path_next_client(fp_node, pnode);
1193 				continue;
1194 			} else if (fp_rcm_remove(pathname, errstring,
1195 			    flags) != 0) {
1196 				/*
1197 				 * Bring everything back online
1198 				 * in rcm and continue
1199 				 */
1200 				(void) fp_rcm_online(pathname, NULL, flags);
1201 				(*failure_count)++;
1202 				pnode = di_path_next_client(fp_node, pnode);
1203 				continue;
1204 			}
1205 		}
1206 
1207 		/* Update the repository only on a successful unconfigure */
1208 		if ((update_str = calloc(1, strlen(xport_phys) +
1209 					strlen(DYN_SEP) +
1210 					strlen(port_wwn) + 1)) == NULL) {
1211 			cfga_err(errstring, errno, ERR_UNCONF_OK_UPD_REP, 0);
1212 			(*failure_count)++;
1213 			pnode = di_path_next_client(fp_node, pnode);
1214 			continue;
1215 		}
1216 
1217 		/* Init the string to be removed from repository */
1218 		sprintf(update_str, "%s%s%s", xport_phys, DYN_SEP, port_wwn);
1219 
1220 		if (update_fabric_wwn_list(REMOVE_ENTRY, update_str,
1221 								errstring)) {
1222 			S_FREE(update_str);
1223 			cfga_err(errstring, errno,
1224 					ERR_UNCONF_OK_UPD_REP, 0);
1225 			(*failure_count)++;
1226 			/* Cleanup and continue from here just for clarity */
1227 			pnode = di_path_next_client(fp_node, pnode);
1228 			continue;
1229 		}
1230 
1231 		S_FREE(update_str);
1232 		pnode = di_path_next_client(fp_node, pnode);
1233 	}
1234 
1235 	return (FPCFGA_OK);
1236 }
1237 
1238 static fpcfga_ret_t
1239 unconf_non_vhci_nodes(di_node_t dnode, char *xport_phys, char *dyncomp,
1240 	int unusable_flag, int *num_devs, int *failure_count,
1241 	char **errstring, cfga_flags_t flags)
1242 {
1243 	int	ret1, ret2, *lunnump;
1244 	char	pathname[MAXPATHLEN];
1245 	char	*node_path, *update_str;
1246 	char	port_wwn[WWN_SIZE*2+1];
1247 	uchar_t	*port_wwn_data = NULL;
1248 
1249 	while (dnode != DI_NODE_NIL) {
1250 
1251 		(*num_devs)++;
1252 
1253 		/* Get the physical path for this node */
1254 		if ((node_path = di_devfs_path(dnode)) == NULL) {
1255 			/*
1256 			 * We don't try to offline in RCM here because we
1257 			 * don't know the path to offline. Just continue to
1258 			 * the next node.
1259 			 */
1260 			cfga_err(errstring, 0, ERRARG_DEVINFO, xport_phys, 0);
1261 			(*failure_count)++;
1262 			dnode = di_sibling_node(dnode);
1263 			continue;
1264 		}
1265 
1266 		/* Now get the LUN # of this device thru the property */
1267 		ret1 = di_prop_lookup_ints(DDI_DEV_T_ANY, dnode,
1268 							LUN_PROP, &lunnump);
1269 
1270 		/* Next get the port WWN of the device */
1271 		ret2 = di_prop_lookup_bytes(DDI_DEV_T_ANY, dnode,
1272 						PORT_WWN_PROP, &port_wwn_data);
1273 
1274 		/* A failure in any of the above is not good */
1275 		if ((ret1 == -1) || (ret2 == -1)) {
1276 			/*
1277 			 * We don't try to offline in RCM here because we
1278 			 * don't know the path to offline. Just continue to
1279 			 * the next node.
1280 			 */
1281 			cfga_err(errstring, 0,
1282 					ERRARG_DI_GET_PROP, node_path, 0);
1283 			di_devfs_path_free(node_path);
1284 			node_path = NULL;
1285 			(*failure_count)++;
1286 			dnode = di_sibling_node(dnode);
1287 			continue;
1288 		}
1289 
1290 		/* Prepend the "/devices" prefix to the path and copy it */
1291 		sprintf(pathname, "%s%s", DEVICES_DIR, node_path);
1292 		di_devfs_path_free(node_path);
1293 		node_path = NULL;
1294 
1295 		copy_pwwn_data_to_str(port_wwn, port_wwn_data);
1296 
1297 		if (strstr(pathname, "@w") == NULL) {
1298 			/*
1299 			 * If the driver is detached, some part of the path
1300 			 * may be missing and so we'll manually construct it
1301 			 */
1302 			sprintf(&pathname[strlen(pathname)], "@w%s,%x",
1303 							port_wwn, *lunnump);
1304 		}
1305 
1306 		/*
1307 		 * Try to offline in RCM first and if that is successful,
1308 		 * unconfigure the LUN. If offlining in RCM fails, then
1309 		 * update the failure count
1310 		 *
1311 		 * Here we got to check if unusable_flag is set or not.
1312 		 * If set, then unconfigure only those luns which are in
1313 		 * node_state DI_DEVICE_OFFLINE or DI_DEVICE_DOWN.
1314 		 * If not set, unconfigure all luns.
1315 		 */
1316 		if ((unusable_flag & FLAG_REMOVE_UNUSABLE_FCP_DEV) ==
1317 		    FLAG_REMOVE_UNUSABLE_FCP_DEV) {
1318 			if ((dnode->node_state == DI_DEVICE_OFFLINE) ||
1319 			    (dnode->node_state == DI_DEVICE_DOWN)) {
1320 				if (fp_rcm_offline(pathname, errstring,
1321 				    flags) != 0) {
1322 					(*failure_count)++;
1323 					dnode = di_sibling_node(dnode);
1324 					continue;
1325 				} else if (lun_unconf(pathname, *lunnump,
1326 				    xport_phys,dyncomp, errstring)
1327 				    != FPCFGA_OK) {
1328 					(void) fp_rcm_online(pathname,
1329 					    NULL, flags);
1330 					(*failure_count)++;
1331 					dnode = di_sibling_node(dnode);
1332 					continue;
1333 				} else if (fp_rcm_remove(pathname, errstring,
1334 				    flags) != 0) {
1335 					/*
1336 					 * Bring everything back online
1337 					 * in rcm and continue
1338 					 */
1339 					(void) fp_rcm_online(pathname,
1340 					    NULL, flags);
1341 					(*failure_count)++;
1342 					dnode = di_sibling_node(dnode);
1343 					continue;
1344 				}
1345 			} else {
1346 				dnode = di_sibling_node(dnode);
1347 				continue;
1348 			}
1349 		} else {
1350 			if (fp_rcm_offline(pathname, errstring, flags) != 0) {
1351 				(*failure_count)++;
1352 				dnode = di_sibling_node(dnode);
1353 				continue;
1354 			} else if (lun_unconf(pathname, *lunnump, xport_phys,
1355 			    dyncomp, errstring) != FPCFGA_OK) {
1356 				(void) fp_rcm_online(pathname, NULL, flags);
1357 				(*failure_count)++;
1358 				dnode = di_sibling_node(dnode);
1359 				continue;
1360 			} else if (fp_rcm_remove(pathname, errstring,
1361 			    flags) != 0) {
1362 				/*
1363 				 * Bring everything back online
1364 				 * in rcm and continue
1365 				 */
1366 				(void) fp_rcm_online(pathname, NULL, flags);
1367 				(*failure_count)++;
1368 				dnode = di_sibling_node(dnode);
1369 				continue;
1370 			}
1371 		}
1372 
1373 		/* Update the repository only on a successful unconfigure */
1374 		if ((update_str = calloc(1, strlen(xport_phys) +
1375 					strlen(DYN_SEP) +
1376 					strlen(port_wwn) + 1)) == NULL) {
1377 			cfga_err(errstring, errno, ERR_UNCONF_OK_UPD_REP, 0);
1378 			(*failure_count)++;
1379 			dnode = di_sibling_node(dnode);
1380 			continue;
1381 		}
1382 
1383 		/* Init the string to be removed from repository */
1384 		sprintf(update_str, "%s%s%s", xport_phys, DYN_SEP, port_wwn);
1385 
1386 		if (update_fabric_wwn_list(REMOVE_ENTRY, update_str,
1387 								errstring)) {
1388 			S_FREE(update_str);
1389 			cfga_err(errstring, errno, ERR_UNCONF_OK_UPD_REP, 0);
1390 			(*failure_count)++;
1391 			dnode = di_sibling_node(dnode);
1392 			continue;
1393 		}
1394 
1395 		S_FREE(update_str);
1396 		dnode = di_sibling_node(dnode);
1397 	}
1398 
1399 	return (FPCFGA_OK);
1400 }
1401 
1402 /*
1403  * INPUT:
1404  * apidt - Pointer to apid_t structure with data filled in
1405  * flags - Flags for special handling
1406  *
1407  * OUTPUT:
1408  * errstring - Applicable only on a failure from plugin
1409  * num_devs  - Incremented per lun
1410  * failure_count - Incremented on any failed operation on lun
1411  *
1412  * RETURNS:
1413  * non-FPCFGA_OK on any validation check error. If this value is returned, no
1414  *             devices were handled.  Consequently num_devs and failure_count
1415  *             will not be incremented.
1416  * FPCFGA_OK This return value doesn't mean that all devices were successfully
1417  *             unconfigured, you have to check failure_count.
1418  */
1419 static fpcfga_ret_t
1420 unconf_any_devinfo_nodes(apid_t *apidt, cfga_flags_t flags, char **errstring,
1421 				int *num_devs, int *failure_count)
1422 {
1423 	char		*node_path = NULL;
1424 	char		pathname[MAXPATHLEN], *ptr;	/* scratch pad */
1425 	di_node_t	root_node, direct_node, fp_node;
1426 	di_path_t	path_node = DI_PATH_NIL;
1427 
1428 	/*
1429 	 * apidt->xport_phys is something like :
1430 	 * /devices/pci@.../SUNW,qlc@../fp@0,0:fc
1431 	 * Make sure we copy both the devinfo and pathinfo nodes
1432 	 */
1433 	(void) strlcpy(pathname, apidt->xport_phys, MAXPATHLEN);
1434 
1435 	/* Now get rid of the ':' at the end */
1436 	if ((ptr = strstr(pathname, MINOR_SEP)) != NULL)
1437 		*ptr = '\0';
1438 
1439 	if (strncmp(pathname, DEVICES_DIR, strlen(DEVICES_DIR))) {
1440 		cfga_err(errstring, 0, ERRARG_INVALID_PATH, pathname, 0);
1441 		return (FPCFGA_INVALID_PATH);
1442 	}
1443 
1444 	if ((root_node = di_init("/", DINFOCPYALL | DINFOPATH)) ==
1445 								DI_NODE_NIL) {
1446 		cfga_err(errstring, errno, ERRARG_DEVINFO,
1447 							apidt->xport_phys, 0);
1448 		return (FPCFGA_LIB_ERR);
1449 	}
1450 
1451 	if ((fp_node = di_drv_first_node("fp", root_node)) == DI_NODE_NIL) {
1452 		cfga_err(errstring, errno, ERRARG_DEVINFO,
1453 							apidt->xport_phys, 0);
1454 		di_fini(root_node);
1455 		return (FPCFGA_LIB_ERR);
1456 	}
1457 
1458 	/*
1459 	 * Search all the fp nodes to see if any match the one we are trying
1460 	 * to unconfigure
1461 	 */
1462 
1463 	/* Skip the "/devices" prefix */
1464 	ptr = pathname + strlen(DEVICES_DIR);
1465 
1466 	while (fp_node != DI_NODE_NIL) {
1467 		node_path = di_devfs_path(fp_node);
1468 		if (strcmp(node_path, ptr) == 0) {
1469 			/* Found the fp node. 'pathname' has the full path */
1470 			di_devfs_path_free(node_path);
1471 			node_path = NULL;
1472 			break;
1473 		}
1474 		fp_node = di_drv_next_node(fp_node);
1475 		di_devfs_path_free(node_path);
1476 	}
1477 
1478 	if (fp_node == DI_NODE_NIL) {
1479 		cfga_err(errstring, 0, ERRARG_NOT_IN_DEVINFO,
1480 							apidt->xport_phys, 0);
1481 		di_fini(root_node);
1482 		return (FPCFGA_LIB_ERR);
1483 	}
1484 
1485 	direct_node = di_child_node(fp_node);
1486 	path_node = di_path_next_client(fp_node, path_node);
1487 
1488 	if ((direct_node == DI_NODE_NIL) && (path_node == DI_PATH_NIL)) {
1489 		/* No devinfo or pathinfo nodes. Great ! Just return success */
1490 		di_fini(root_node);
1491 		return (FPCFGA_OK);
1492 	}
1493 
1494 	/* First unconfigure any non-MPXIO nodes */
1495 	unconf_non_vhci_nodes(direct_node, apidt->xport_phys, apidt->dyncomp,
1496 	    apidt->flags, num_devs, failure_count, errstring, flags);
1497 
1498 	/*
1499 	 * Now we will traverse any path info nodes that are there
1500 	 *
1501 	 * Only MPXIO devices have pathinfo nodes
1502 	 */
1503 	unconf_vhci_nodes(path_node, fp_node, apidt->xport_phys, apidt->dyncomp,
1504 	    apidt->flags, num_devs, failure_count, errstring, flags);
1505 
1506 	di_fini(root_node);
1507 
1508 	/*
1509 	 * We don't want to check the return value of unconf_non_vhci_nodes()
1510 	 * and unconf_vhci_nodes().  But instead, we are interested only in
1511 	 * consistently incrementing num_devs and failure_count so that we can
1512 	 * compare them.
1513 	 */
1514 	return (FPCFGA_OK);
1515 }
1516 
1517 /*
1518  * This function handles configuring/unconfiguring all the devices w.r.t
1519  * the FCA port specified by apidt.
1520  *
1521  * In the unconfigure case, it first unconfigures all the devices that are
1522  * seen through the given port at that moment and then unconfigures all the
1523  * devices that still (somehow) have devinfo nodes on the system for that FCA
1524  * port.
1525  *
1526  * INPUT:
1527  * cmd - CFGA_CMD_CONFIGURE or CFGA_CMD_UNCONFIGURE
1528  * apidt - Pointer to apid_t structure with data filled in
1529  * flags - Flags for special handling
1530  *
1531  * OUTPUT:
1532  * errstring - Applicable only on a failure from plugin
1533  *
1534  * RETURNS:
1535  * FPCFGA_OK on success
1536  * non-FPCFGA_OK otherwise
1537  */
1538 static fpcfga_ret_t
1539 handle_devs(cfga_cmd_t cmd, apid_t *apidt, cfga_flags_t flags,
1540 	char **errstring, HBA_HANDLE handle, int portIndex,
1541 	HBA_PORTATTRIBUTES portAttrs)
1542 {
1543 	int		num_devs = 0, dev_cs_failed = 0;
1544 	char		port_wwn[WWN_S_LEN];
1545 	la_wwn_t	pwwn;
1546 	apid_t		my_apidt = {NULL};
1547 	char		*my_apid;
1548 	HBA_PORTATTRIBUTES	discPortAttrs;
1549 	int			discIndex;
1550 	fpcfga_ret_t		rval = FPCFGA_OK;
1551 
1552 	if ((my_apid = calloc(
1553 		1, strlen(apidt->xport_phys) + strlen(DYN_SEP) +
1554 		(2 * FC_WWN_SIZE) + 1)) == NULL) {
1555 		cfga_err(errstring, errno, ERR_MEM_ALLOC, 0);
1556 		return (FPCFGA_LIB_ERR);
1557 	}
1558 
1559 	num_devs = portAttrs.NumberofDiscoveredPorts;
1560 	for (discIndex = 0; discIndex < portAttrs.NumberofDiscoveredPorts;
1561 		discIndex++) {
1562 	    if (getDiscPortAttrs(handle, portIndex,
1563 		discIndex, &discPortAttrs)) {
1564 		dev_cs_failed++;
1565 		/* Move on to the next target */
1566 		continue;
1567 	    }
1568 	    (void) sprintf(port_wwn, "%016llx",
1569 		wwnConversion(discPortAttrs.PortWWN.wwn));
1570 		/*
1571 		 * Construct a fake apid string similar to the one the
1572 		 * plugin gets from the framework and have apidt_create()
1573 		 * fill in the apid_t structure.
1574 		 */
1575 	    strcpy(my_apid, apidt->xport_phys);
1576 	    strcat(my_apid, DYN_SEP);
1577 	    strcat(my_apid, port_wwn);
1578 	    if (apidt_create(my_apid, &my_apidt, errstring) != FPCFGA_OK) {
1579 		dev_cs_failed++;
1580 		continue;
1581 	    }
1582 	    my_apidt.flags = apidt->flags;
1583 
1584 	    memcpy(&pwwn, &(discPortAttrs.PortWWN), sizeof (la_wwn_t));
1585 	    if (dev_change_state(cmd, &my_apidt, &pwwn,
1586 		flags, errstring, handle, portAttrs) != FPCFGA_OK) {
1587 		dev_cs_failed++;
1588 	    }
1589 	    apidt_free(&my_apidt);
1590 	}
1591 
1592 	S_FREE(my_apid);
1593 
1594 	/*
1595 	 * We have now handled all the devices that are currently visible
1596 	 * through the given FCA port. But, it is possible that there are
1597 	 * some devinfo nodes hanging around. For the unconfigure operation,
1598 	 * this has to be looked into too.
1599 	 */
1600 	if (cmd == CFGA_CMD_UNCONFIGURE) {
1601 		/* dev_cs_failed will be updated to indicate any failures */
1602 		rval = unconf_any_devinfo_nodes(apidt, flags, errstring,
1603 		    &num_devs, &dev_cs_failed);
1604 	}
1605 
1606 	if (rval == FPCFGA_OK) {
1607 		if (dev_cs_failed == 0)
1608 			return (FPCFGA_OK);
1609 
1610 		/*
1611 		 * For the discovered ports, num_devs is counted on target
1612 		 * basis, but for invisible targets, num_devs is counted on
1613 		 * lun basis.
1614 		 *
1615 		 * But if dev_cs_failed and num_devs are incremented
1616 		 * consistently, comparation of these two counters is still
1617 		 * meaningful.
1618 		 */
1619 		if (dev_cs_failed == num_devs) {
1620 			/* Failed on all devices seen through this FCA port */
1621 			cfga_err(errstring, 0,
1622 			((cmd == CFGA_CMD_CONFIGURE) ?
1623 				ERR_FCA_CONFIGURE : ERR_FCA_UNCONFIGURE), 0);
1624 			return (FPCFGA_LIB_ERR);
1625 		} else {
1626 			/* Failed only on some of the devices */
1627 			cfga_err(errstring, 0, ERR_PARTIAL_SUCCESS, 0);
1628 			return (FPCFGA_LIB_ERR);
1629 		}
1630 	} else {
1631 		if (dev_cs_failed == num_devs) {
1632 			/* Failed on all devices seen through this FCA port */
1633 			cfga_err(errstring, 0,
1634 			((cmd == CFGA_CMD_CONFIGURE) ?
1635 				ERR_FCA_CONFIGURE : ERR_FCA_UNCONFIGURE), 0);
1636 			return (FPCFGA_LIB_ERR);
1637 		} else {
1638 			/* Failed only on some of the devices */
1639 			cfga_err(errstring, 0, ERR_PARTIAL_SUCCESS, 0);
1640 			return (FPCFGA_LIB_ERR);
1641 		}
1642 	}
1643 
1644 	/*
1645 	 * Should never get here
1646 	 */
1647 }
1648 
1649 fpcfga_ret_t
1650 fca_change_state(cfga_cmd_t state_change_cmd, apid_t *apidt,
1651 		cfga_flags_t flags, char **errstring)
1652 {
1653 	fpcfga_ret_t	ret;
1654 	HBA_HANDLE	handle;
1655 	HBA_PORTATTRIBUTES	portAttrs;
1656 	int			portIndex;
1657 
1658 	if ((ret = findMatchingAdapterPort(apidt->xport_phys, &handle,
1659 	    &portIndex, &portAttrs, errstring)) != FPCFGA_OK) {
1660 		return (ret);
1661 	}
1662 
1663 	/*
1664 	 * Bail out if not fabric/public loop
1665 	 */
1666 	switch (state_change_cmd) {
1667 	case CFGA_CMD_CONFIGURE:
1668 	    if (portAttrs.PortType != HBA_PORTTYPE_NLPORT &&
1669 		portAttrs.PortType != HBA_PORTTYPE_NPORT) {
1670 			HBA_CloseAdapter(handle);
1671 			HBA_FreeLibrary();
1672 			return (FPCFGA_OK);
1673 	    }
1674 	    break;
1675 
1676 	case CFGA_CMD_UNCONFIGURE:
1677 	    if (portAttrs.PortType != HBA_PORTTYPE_NLPORT &&
1678 		portAttrs.PortType != HBA_PORTTYPE_NPORT) {
1679 		HBA_CloseAdapter(handle);
1680 		HBA_FreeLibrary();
1681 		return (FPCFGA_OPNOTSUPP);
1682 	    }
1683 	    break;
1684 	default:
1685 		HBA_CloseAdapter(handle);
1686 		HBA_FreeLibrary();
1687 		return (FPCFGA_LIB_ERR);
1688 	}
1689 	ret = (handle_devs(state_change_cmd, apidt, flags, errstring,
1690 	    handle, portIndex, portAttrs));
1691 	HBA_CloseAdapter(handle);
1692 	HBA_FreeLibrary();
1693 	return (ret);
1694 }
1695