xref: /titanic_51/usr/src/lib/sun_sas/common/Sun_sasSendSMPPassThru.c (revision 9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bb)
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 #include <sys/scsi/impl/usmp.h>
29*9e86db79SHyon Kim 
30*9e86db79SHyon Kim /*
31*9e86db79SHyon Kim  * Pass usmp_cmd into ioctl
32*9e86db79SHyon Kim  */
33*9e86db79SHyon Kim static HBA_STATUS
34*9e86db79SHyon Kim SendSMPPassThru(const char *devpath, void *reqframe, HBA_UINT32 *reqsize,
35*9e86db79SHyon Kim     void *rspframe, HBA_UINT32 *rspsize) {
36*9e86db79SHyon Kim 	const char		ROUTINE[] = "SendSMPPassThru";
37*9e86db79SHyon Kim 	int			fd;
38*9e86db79SHyon Kim 	usmp_cmd_t		ucmd_buf;
39*9e86db79SHyon Kim 	HBA_STATUS		ret;
40*9e86db79SHyon Kim 
41*9e86db79SHyon Kim 	bzero(&ucmd_buf, sizeof (ucmd_buf));
42*9e86db79SHyon Kim 
43*9e86db79SHyon Kim 	ucmd_buf.usmp_req = (caddr_t)reqframe;
44*9e86db79SHyon Kim 	ucmd_buf.usmp_rsp = (caddr_t)rspframe;
45*9e86db79SHyon Kim 	ucmd_buf.usmp_reqsize = (size_t)(*reqsize);
46*9e86db79SHyon Kim 	ucmd_buf.usmp_rspsize = (size_t)(*rspsize);
47*9e86db79SHyon Kim 	ucmd_buf.usmp_timeout = SMP_DEFAULT_TIMEOUT;
48*9e86db79SHyon Kim 
49*9e86db79SHyon Kim 	/*
50*9e86db79SHyon Kim 	 * open smp device
51*9e86db79SHyon Kim 	 */
52*9e86db79SHyon Kim 
53*9e86db79SHyon Kim 	if ((fd = open(devpath, O_RDONLY | O_NONBLOCK)) == -1) {
54*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE,
55*9e86db79SHyon Kim 		    "open devpath %s failed due to %s",
56*9e86db79SHyon Kim 		    devpath, strerror(errno));
57*9e86db79SHyon Kim 		return (HBA_STATUS_ERROR);
58*9e86db79SHyon Kim 	}
59*9e86db79SHyon Kim 
60*9e86db79SHyon Kim 	/*
61*9e86db79SHyon Kim 	 * send usmp command
62*9e86db79SHyon Kim 	 */
63*9e86db79SHyon Kim 	if (ioctl(fd, USMPFUNC, &ucmd_buf) == -1) {
64*9e86db79SHyon Kim 		if ((errno == ETIME) || (errno == ETIMEDOUT) ||
65*9e86db79SHyon Kim 		    (errno == EAGAIN)) {
66*9e86db79SHyon Kim 		    ret = HBA_STATUS_ERROR_TRY_AGAIN;
67*9e86db79SHyon Kim 		} else if (errno == EBUSY) {
68*9e86db79SHyon Kim 		    ret = HBA_STATUS_ERROR_BUSY;
69*9e86db79SHyon Kim 		} else {
70*9e86db79SHyon Kim 		    ret = HBA_STATUS_ERROR;
71*9e86db79SHyon Kim 		}
72*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "ioctl:USMPFUNC failed due to %s",
73*9e86db79SHyon Kim 		    strerror(errno));
74*9e86db79SHyon Kim 		(void) close(fd);
75*9e86db79SHyon Kim 		return (ret);
76*9e86db79SHyon Kim 	}
77*9e86db79SHyon Kim 
78*9e86db79SHyon Kim 	(void) close(fd);
79*9e86db79SHyon Kim 	return (HBA_STATUS_OK);
80*9e86db79SHyon Kim }
81*9e86db79SHyon Kim 
82*9e86db79SHyon Kim /*
83*9e86db79SHyon Kim  * Send a USMP command to a remote SMP node
84*9e86db79SHyon Kim  */
85*9e86db79SHyon Kim HBA_STATUS
86*9e86db79SHyon Kim Sun_sasSendSMPPassThru(HBA_HANDLE handle, HBA_WWN hbaPortWWN,
87*9e86db79SHyon Kim     HBA_WWN destPortWWN, HBA_WWN domainPortWWN, void *pReqBuffer,
88*9e86db79SHyon Kim     HBA_UINT32 ReqBufferSize, void *pRspBuffer, HBA_UINT32 *pRspBufferSize)
89*9e86db79SHyon Kim {
90*9e86db79SHyon Kim 	const char		ROUTINE[] = "Sun_sasSendSMPPassThru";
91*9e86db79SHyon Kim 	HBA_STATUS		status;
92*9e86db79SHyon Kim 	struct sun_sas_hba	*hba_ptr;
93*9e86db79SHyon Kim 	int			domainPortFound = 0;
94*9e86db79SHyon Kim 	int			chkDomainPort = 0;
95*9e86db79SHyon Kim 	int			hbaPortFound = 0;
96*9e86db79SHyon Kim 	struct sun_sas_port	*hba_port_ptr, *hba_disco_port;
97*9e86db79SHyon Kim 	hrtime_t		start, end;
98*9e86db79SHyon Kim 	double			duration;
99*9e86db79SHyon Kim 
100*9e86db79SHyon Kim 	start = gethrtime();
101*9e86db79SHyon Kim 	/* Validate the arguments */
102*9e86db79SHyon Kim 	if (pRspBuffer == NULL) {
103*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "NULL response buffer");
104*9e86db79SHyon Kim 		return (HBA_STATUS_ERROR_ARG);
105*9e86db79SHyon Kim 	}
106*9e86db79SHyon Kim 	if (pReqBuffer == NULL) {
107*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "NULL sense buffer");
108*9e86db79SHyon Kim 		return (HBA_STATUS_ERROR_ARG);
109*9e86db79SHyon Kim 	}
110*9e86db79SHyon Kim 	if (pRspBufferSize == NULL) {
111*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "NULL response size");
112*9e86db79SHyon Kim 		return (HBA_STATUS_ERROR_ARG);
113*9e86db79SHyon Kim 	}
114*9e86db79SHyon Kim 
115*9e86db79SHyon Kim 	lock(&all_hbas_lock);
116*9e86db79SHyon Kim 	if ((hba_ptr = Retrieve_Sun_sasHandle(handle)) == NULL) {
117*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "Invalid handle %08lx", handle);
118*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
119*9e86db79SHyon Kim 		return (HBA_STATUS_ERROR_INVALID_HANDLE);
120*9e86db79SHyon Kim 	}
121*9e86db79SHyon Kim 
122*9e86db79SHyon Kim 	/* Check for stale data */
123*9e86db79SHyon Kim 	status = verifyAdapter(hba_ptr);
124*9e86db79SHyon Kim 	if (status != HBA_STATUS_OK) {
125*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "Verify adapter failed");
126*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
127*9e86db79SHyon Kim 		return (status);
128*9e86db79SHyon Kim 	}
129*9e86db79SHyon Kim 
130*9e86db79SHyon Kim 	/*
131*9e86db79SHyon Kim 	 * We are not checking to see if our data is stale.
132*9e86db79SHyon Kim 	 * By verifying this information here, we will take a big performance
133*9e86db79SHyon Kim 	 * hit.  This check will be done later only if the Inquiry ioctl fails
134*9e86db79SHyon Kim 	 */
135*9e86db79SHyon Kim 
136*9e86db79SHyon Kim 	if (hba_ptr->device_path == NULL) {
137*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE,
138*9e86db79SHyon Kim 		    "HBA handle had NULL device path.\
139*9e86db79SHyon Kim 		    Unable to send SCSI cmd");
140*9e86db79SHyon Kim 		unlock(&all_hbas_lock);
141*9e86db79SHyon Kim 		return (HBA_STATUS_ERROR);
142*9e86db79SHyon Kim 	}
143*9e86db79SHyon Kim 
144*9e86db79SHyon Kim 	if (wwnConversion(domainPortWWN.wwn))
145*9e86db79SHyon Kim 		chkDomainPort = 1;
146*9e86db79SHyon Kim 
147*9e86db79SHyon Kim 	/* Determine which port to use */
148*9e86db79SHyon Kim 	for (hba_port_ptr = hba_ptr->first_port;
149*9e86db79SHyon Kim 	    hba_port_ptr != NULL;
150*9e86db79SHyon Kim 	    hba_port_ptr = hba_port_ptr->next) {
151*9e86db79SHyon Kim 
152*9e86db79SHyon Kim 		if (hbaPortFound == 0) {
153*9e86db79SHyon Kim 			if (wwnConversion(hba_port_ptr->port_attributes.
154*9e86db79SHyon Kim 			    PortSpecificAttribute.SASPort->LocalSASAddress.wwn)
155*9e86db79SHyon Kim 			    != wwnConversion(hbaPortWWN.wwn)) {
156*9e86db79SHyon Kim 				/*
157*9e86db79SHyon Kim 				 * Since all the ports under the same HBA have
158*9e86db79SHyon Kim 				 * the same LocalSASAddress, we should break
159*9e86db79SHyon Kim 				 * the loop once we find it dosn't match.
160*9e86db79SHyon Kim 				 */
161*9e86db79SHyon Kim 				break;
162*9e86db79SHyon Kim 			} else {
163*9e86db79SHyon Kim 				hbaPortFound = 1;
164*9e86db79SHyon Kim 			}
165*9e86db79SHyon Kim 		}
166*9e86db79SHyon Kim 
167*9e86db79SHyon Kim 		if (chkDomainPort != 0) {
168*9e86db79SHyon Kim 			if (hba_port_ptr->first_phy != NULL &&
169*9e86db79SHyon Kim 			    wwnConversion(hba_port_ptr->first_phy->
170*9e86db79SHyon Kim 			    phy.domainPortWWN.wwn) ==
171*9e86db79SHyon Kim 			    wwnConversion(domainPortWWN.wwn)) {
172*9e86db79SHyon Kim 				domainPortFound = 1;
173*9e86db79SHyon Kim 			}
174*9e86db79SHyon Kim 			if (!(domainPortFound)) {
175*9e86db79SHyon Kim 				continue;
176*9e86db79SHyon Kim 			}
177*9e86db79SHyon Kim 		}
178*9e86db79SHyon Kim 
179*9e86db79SHyon Kim 		for (hba_disco_port = hba_port_ptr->first_attached_port;
180*9e86db79SHyon Kim 		    hba_disco_port != NULL;
181*9e86db79SHyon Kim 		    hba_disco_port = hba_disco_port->next) {
182*9e86db79SHyon Kim 
183*9e86db79SHyon Kim 			/*
184*9e86db79SHyon Kim 			 * If discoveredPort is not given targetPort, just skip
185*9e86db79SHyon Kim 			 */
186*9e86db79SHyon Kim 			if (wwnConversion(hba_disco_port->port_attributes.\
187*9e86db79SHyon Kim 			    PortSpecificAttribute.SASPort->LocalSASAddress.wwn)
188*9e86db79SHyon Kim 			    != wwnConversion(destPortWWN.wwn)) {
189*9e86db79SHyon Kim 				/* Does not match */
190*9e86db79SHyon Kim 				continue;
191*9e86db79SHyon Kim 			}
192*9e86db79SHyon Kim 
193*9e86db79SHyon Kim 			/*
194*9e86db79SHyon Kim 			 * If matching targetPort does not support SMP protocal
195*9e86db79SHyon Kim 			 * return error.
196*9e86db79SHyon Kim 			 * comment it out for testing only
197*9e86db79SHyon Kim 			 */
198*9e86db79SHyon Kim 			if ((hba_disco_port->port_attributes.\
199*9e86db79SHyon Kim 			    PortSpecificAttribute.SASPort->PortProtocol &
200*9e86db79SHyon Kim 			    HBA_SASPORTPROTOCOL_SMP) == 0) {
201*9e86db79SHyon Kim 				log(LOG_DEBUG, ROUTINE, "Input WWN %01611x\
202*9e86db79SHyon Kim 				    does not support SMP protocol",
203*9e86db79SHyon Kim 				    wwnConversion(hbaPortWWN.wwn));
204*9e86db79SHyon Kim 				unlock(&all_hbas_lock);
205*9e86db79SHyon Kim 				return (HBA_STATUS_ERROR_INVALID_PROTOCOL_TYPE);
206*9e86db79SHyon Kim 			}
207*9e86db79SHyon Kim 
208*9e86db79SHyon Kim 			/*
209*9e86db79SHyon Kim 			 * SMP target port doesn't have any scsi info.
210*9e86db79SHyon Kim 			 *   - like /dev/rdsk/cxtxdxsx
211*9e86db79SHyon Kim 			 * So we use OSDeviceName from port attributes.
212*9e86db79SHyon Kim 			 *   - like /dev/smp/expd[0-9]
213*9e86db79SHyon Kim 			 */
214*9e86db79SHyon Kim 			status = SendSMPPassThru(
215*9e86db79SHyon Kim 			    hba_disco_port->port_attributes.OSDeviceName,
216*9e86db79SHyon Kim 			    pReqBuffer, &ReqBufferSize,
217*9e86db79SHyon Kim 			    pRspBuffer, pRspBufferSize);
218*9e86db79SHyon Kim 
219*9e86db79SHyon Kim 			unlock(&all_hbas_lock);
220*9e86db79SHyon Kim 			end = gethrtime();
221*9e86db79SHyon Kim 			duration = end - start;
222*9e86db79SHyon Kim 			duration /= HR_SECOND;
223*9e86db79SHyon Kim 			log(LOG_DEBUG, ROUTINE, "Took total\
224*9e86db79SHyon Kim 			    of %.4f seconds", duration);
225*9e86db79SHyon Kim 			return (status);
226*9e86db79SHyon Kim 		}
227*9e86db79SHyon Kim 		if (chkDomainPort) {
228*9e86db79SHyon Kim 			unlock(&all_hbas_lock);
229*9e86db79SHyon Kim 			log(LOG_DEBUG, ROUTINE, "Unable to locate"
230*9e86db79SHyon Kim 			    "requested SMP target port %16llx",
231*9e86db79SHyon Kim 			    wwnConversion(destPortWWN.wwn));
232*9e86db79SHyon Kim 			return (HBA_STATUS_ERROR_ILLEGAL_WWN);
233*9e86db79SHyon Kim 		}
234*9e86db79SHyon Kim 	}
235*9e86db79SHyon Kim 	unlock(&all_hbas_lock);
236*9e86db79SHyon Kim 	if (hbaPortFound == 0) {
237*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE,
238*9e86db79SHyon Kim 		    "Unable to locate requested Port WWN %016llx on "
239*9e86db79SHyon Kim 		    "handle %08lx", wwnConversion(hbaPortWWN.wwn), handle);
240*9e86db79SHyon Kim 	} else if (chkDomainPort && !domainPortFound) {
241*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "Unable to locate requested"
242*9e86db79SHyon Kim 		    " domainPortWWN %016llx on handle %08lx",
243*9e86db79SHyon Kim 		    wwnConversion(domainPortWWN.wwn), handle);
244*9e86db79SHyon Kim 	} else {
245*9e86db79SHyon Kim 		log(LOG_DEBUG, ROUTINE, "Unable to locate"
246*9e86db79SHyon Kim 		    "requested SMP target port %16llx",
247*9e86db79SHyon Kim 		    wwnConversion(destPortWWN.wwn));
248*9e86db79SHyon Kim 	}
249*9e86db79SHyon Kim 	return (HBA_STATUS_ERROR_ILLEGAL_WWN);
250*9e86db79SHyon Kim }
251