1*9e86db79SHyon Kim /*
2*9e86db79SHyon Kim  * CDDL HEADER START
3*9e86db79SHyon Kim  *
4*9e86db79SHyon Kim  * The contents of this file are subject to the terms of the
5*9e86db79SHyon Kim  * Common Development and Distribution License (the "License").
6*9e86db79SHyon Kim  * You may not use this file except in compliance with the License.
7*9e86db79SHyon Kim  *
8*9e86db79SHyon Kim  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*9e86db79SHyon Kim  * or http://www.opensolaris.org/os/licensing.
10*9e86db79SHyon Kim  * See the License for the specific language governing permissions
11*9e86db79SHyon Kim  * and limitations under the License.
12*9e86db79SHyon Kim  *
13*9e86db79SHyon Kim  * When distributing Covered Code, include this CDDL HEADER in each
14*9e86db79SHyon Kim  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*9e86db79SHyon Kim  * If applicable, add the following below this CDDL HEADER, with the
16*9e86db79SHyon Kim  * fields enclosed by brackets "[]" replaced with your own identifying
17*9e86db79SHyon Kim  * information: Portions Copyright [yyyy] [name of copyright owner]
18*9e86db79SHyon Kim  *
19*9e86db79SHyon Kim  * CDDL HEADER END
20*9e86db79SHyon Kim  */
21*9e86db79SHyon Kim 
22*9e86db79SHyon Kim /*
23*9e86db79SHyon Kim  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
24*9e86db79SHyon Kim  * Use is subject to license terms.
25*9e86db79SHyon Kim  */
26*9e86db79SHyon Kim 
27*9e86db79SHyon Kim #include    "sun_sas.h"
28*9e86db79SHyon Kim 
29*9e86db79SHyon Kim /*
30*9e86db79SHyon Kim  * Discover an HBA node with  mtaching path.
31*9e86db79SHyon Kim  * The di_node_t argument should be the root of the device tree.
32*9e86db79SHyon Kim  * This routine assumes the locks have been taken
33*9e86db79SHyon Kim  */
34*9e86db79SHyon Kim static int
match_smhba_sas_hba(di_node_t node,void * arg)35*9e86db79SHyon Kim match_smhba_sas_hba(di_node_t node, void *arg)
36*9e86db79SHyon Kim {
37*9e86db79SHyon Kim 	int *propData, rval;
38*9e86db79SHyon Kim 	walkarg_t *wa = (walkarg_t *)arg;
39*9e86db79SHyon Kim 	char	*devpath, fulldevpath[MAXPATHLEN];
40*9e86db79SHyon Kim 
41*9e86db79SHyon Kim 	/* Skip stub(instance -1) nodes */
42*9e86db79SHyon Kim 	if (IS_STUB_NODE(node)) {
43*9e86db79SHyon Kim 		return (DI_WALK_CONTINUE);
44*9e86db79SHyon Kim 	}
45*9e86db79SHyon Kim 
46*9e86db79SHyon Kim 	rval = di_prop_lookup_ints(DDI_DEV_T_ANY, node,
47*9e86db79SHyon Kim 	    "sm-hba-supported", &propData);
48*9e86db79SHyon Kim 	if (rval < 0) {
49*9e86db79SHyon Kim 		return (DI_WALK_CONTINUE);
50*9e86db79SHyon Kim 	} else {
51*9e86db79SHyon Kim 		if ((devpath = di_devfs_path(node)) == NULL) {
52*9e86db79SHyon Kim 			/* still continue to see if there is matching one. */
53*9e86db79SHyon Kim 			return (DI_WALK_CONTINUE);
54*9e86db79SHyon Kim 		}
55*9e86db79SHyon Kim 		(void) snprintf(fulldevpath, MAXPATHLEN, "%s%s", DEVICES_DIR,
56*9e86db79SHyon Kim 		    devpath);
57*9e86db79SHyon Kim 
58*9e86db79SHyon Kim 		if ((strstr(fulldevpath, wa->devpath)) != NULL) {
59*9e86db79SHyon Kim 			/* add the hba to the hba list */
60*9e86db79SHyon Kim 			if (devtree_get_one_hba(node) ==
61*9e86db79SHyon Kim 			    HBA_STATUS_OK) {
62*9e86db79SHyon Kim 				/* succeed to refresh the adapater. */
63*9e86db79SHyon Kim 				*wa->flag = B_TRUE;
64*9e86db79SHyon Kim 			}
65*9e86db79SHyon Kim 			/* Found a node. No need to walk any more. */
66*9e86db79SHyon Kim 			di_devfs_path_free(devpath);
67*9e86db79SHyon Kim 			return (DI_WALK_TERMINATE);
68*9e86db79SHyon Kim 		}
69*9e86db79SHyon Kim 		di_devfs_path_free(devpath);
70*9e86db79SHyon Kim 	}
71*9e86db79SHyon Kim 
72*9e86db79SHyon Kim 	return (DI_WALK_CONTINUE);
73*9e86db79SHyon Kim }
74*9e86db79SHyon Kim 
75*9e86db79SHyon Kim /*
76*9e86db79SHyon Kim  * Refreshes information about an HBA
77*9e86db79SHyon Kim  *
78*9e86db79SHyon Kim  * Note: This routine holds the locks in write mode
79*9e86db79SHyon Kim  *       during most of the processing, and as such, will cause
80*9e86db79SHyon Kim  *	 all other threads to block on entry into the library
81*9e86db79SHyon Kim  *	 until the refresh is complete.  An optimization would be
82*9e86db79SHyon Kim  *       to put fine-grain locking in for the open_handle structures.
83*9e86db79SHyon Kim  */
84*9e86db79SHyon Kim void
Sun_sasRefreshInformation(HBA_HANDLE handle)85*9e86db79SHyon Kim Sun_sasRefreshInformation(HBA_HANDLE handle)
86*9e86db79SHyon Kim {
87*9e86db79SHyon Kim 	const char		    ROUTINE[] = "Sun_sasRefreshInformation";
88*9e86db79SHyon Kim 	struct sun_sas_hba	    *hba_ptr;
89*9e86db79SHyon Kim 	struct open_handle	    *oHandle;
90*9e86db79SHyon Kim 	di_node_t		    root;
91*9e86db79SHyon Kim 	hrtime_t		    start;
92*9e86db79SHyon Kim 	hrtime_t		    end;
93*9e86db79SHyon Kim 	double			    duration;
94*9e86db79SHyon Kim 	walkarg_t		    wa;
95*9e86db79SHyon Kim 
96*9e86db79SHyon Kim 	/* take a lock for hbas and handles during rerfresh. */
97*9e86db79SHyon Kim 	lock(&all_hbas_lock);
98*9e86db79SHyon Kim 	lock(&open_handles_lock);
99*9e86db79SHyon Kim 
100*9e86db79SHyon Kim 	oHandle = RetrieveOpenHandle(handle);
101*9e86db79SHyon Kim 	if (oHandle == NULL) {
102*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "Invalid handle %08lx", handle);
103*9e86db79SHyon Kim 		unlock(&open_handles_lock);
104*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
105*9e86db79SHyon Kim 		return;
106*9e86db79SHyon Kim 	}
107*9e86db79SHyon Kim 
108*9e86db79SHyon Kim 	/* now we know the associated hba exists in the global list. */
109*9e86db79SHyon Kim 	start = gethrtime();
110*9e86db79SHyon Kim 	/* Grab device tree */
111*9e86db79SHyon Kim 	if ((root = di_init("/", DINFOCACHE)) == DI_NODE_NIL) {
112*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE,
113*9e86db79SHyon Kim 		    "Unable to load device tree for reason \"%s\"",
114*9e86db79SHyon Kim 		    strerror(errno));
115*9e86db79SHyon Kim 		unlock(&open_handles_lock);
116*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
117*9e86db79SHyon Kim 		return;
118*9e86db79SHyon Kim 	}
119*9e86db79SHyon Kim 
120*9e86db79SHyon Kim 	end = gethrtime();
121*9e86db79SHyon Kim 	duration = end - start;
122*9e86db79SHyon Kim 	duration /= HR_SECOND;
123*9e86db79SHyon Kim 	log(LOG_DEBUG, ROUTINE, "Device tree init took "
124*9e86db79SHyon Kim 	    "%.6f seconds", duration);
125*9e86db79SHyon Kim 
126*9e86db79SHyon Kim 	hba_ptr = RetrieveHandle(oHandle->adapterIndex);
127*9e86db79SHyon Kim 	wa.devpath = hba_ptr->device_path;
128*9e86db79SHyon Kim 	wa.flag = (boolean_t *)calloc(1, sizeof (boolean_t));
129*9e86db79SHyon Kim 	*wa.flag = B_FALSE;
130*9e86db79SHyon Kim 
131*9e86db79SHyon Kim 	/* found the matching hba node and refresh hba ports and targets. */
132*9e86db79SHyon Kim 	if (di_walk_node(root, DI_WALK_SIBFIRST, &wa,
133*9e86db79SHyon Kim 	    match_smhba_sas_hba) != 0) {
134*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "di_walk_node failed.");
135*9e86db79SHyon Kim 		unlock(&open_handles_lock);
136*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
137*9e86db79SHyon Kim 		S_FREE(wa.flag);
138*9e86db79SHyon Kim 		di_fini(root);
139*9e86db79SHyon Kim 		return;
140*9e86db79SHyon Kim 	}
141*9e86db79SHyon Kim 
142*9e86db79SHyon Kim 	if (*wa.flag != B_TRUE) {
143*9e86db79SHyon Kim 		/* no matching HBA. */
144*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "No matching HBA found.");
145*9e86db79SHyon Kim 		unlock(&open_handles_lock);
146*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
147*9e86db79SHyon Kim 		S_FREE(wa.flag);
148*9e86db79SHyon Kim 		di_fini(root);
149*9e86db79SHyon Kim 		return;
150*9e86db79SHyon Kim 	}
151*9e86db79SHyon Kim 
152*9e86db79SHyon Kim 	S_FREE(wa.flag);
153*9e86db79SHyon Kim 
154*9e86db79SHyon Kim 	di_fini(root);
155*9e86db79SHyon Kim 
156*9e86db79SHyon Kim 	/* All done, release the locks */
157*9e86db79SHyon Kim 	unlock(&open_handles_lock);
158*9e86db79SHyon Kim 	unlock(&all_hbas_lock);
159*9e86db79SHyon Kim }
160