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 * Combine uscsi command ans send it out via ioctl
31*9e86db79SHyon Kim */
32*9e86db79SHyon Kim static HBA_STATUS
SendScsiInquiry(const char * devpath,HBA_UINT8 cdb1,HBA_UINT8 cdb2,void * responseBuffer,HBA_UINT32 * responseSize,HBA_UINT8 * scsiStatus,void * senseBuffer,HBA_UINT32 * senseSize)33*9e86db79SHyon Kim SendScsiInquiry(const char *devpath, HBA_UINT8 cdb1, HBA_UINT8 cdb2,
34*9e86db79SHyon Kim void *responseBuffer, HBA_UINT32 *responseSize, HBA_UINT8 *scsiStatus,
35*9e86db79SHyon Kim void *senseBuffer, HBA_UINT32 *senseSize)
36*9e86db79SHyon Kim {
37*9e86db79SHyon Kim HBA_UINT32 status;
38*9e86db79SHyon Kim struct uscsi_cmd ucmd_buf;
39*9e86db79SHyon Kim union scsi_cdb cdb;
40*9e86db79SHyon Kim
41*9e86db79SHyon Kim bzero(&cdb, sizeof (cdb));
42*9e86db79SHyon Kim bzero(&ucmd_buf, sizeof (ucmd_buf));
43*9e86db79SHyon Kim bzero(senseBuffer, *senseSize);
44*9e86db79SHyon Kim bzero(responseBuffer, *responseSize);
45*9e86db79SHyon Kim
46*9e86db79SHyon Kim cdb.scc_cmd = SCMD_INQUIRY;
47*9e86db79SHyon Kim cdb.g0_addr1 = cdb2;
48*9e86db79SHyon Kim cdb.g0_addr2 = cdb1;
49*9e86db79SHyon Kim cdb.g0_count0 = *responseSize;
50*9e86db79SHyon Kim
51*9e86db79SHyon Kim ucmd_buf.uscsi_cdb = (char *)&cdb;
52*9e86db79SHyon Kim ucmd_buf.uscsi_cdblen = CDB_GROUP0;
53*9e86db79SHyon Kim ucmd_buf.uscsi_bufaddr = (caddr_t)responseBuffer;
54*9e86db79SHyon Kim ucmd_buf.uscsi_buflen = *responseSize;
55*9e86db79SHyon Kim ucmd_buf.uscsi_rqbuf = (caddr_t)senseBuffer;
56*9e86db79SHyon Kim ucmd_buf.uscsi_rqlen = *senseSize;
57*9e86db79SHyon Kim ucmd_buf.uscsi_flags = USCSI_READ | USCSI_SILENT | USCSI_RQENABLE;
58*9e86db79SHyon Kim
59*9e86db79SHyon Kim status = send_uscsi_cmd(devpath, &ucmd_buf);
60*9e86db79SHyon Kim *scsiStatus = ucmd_buf.uscsi_status;
61*9e86db79SHyon Kim return (status);
62*9e86db79SHyon Kim }
63*9e86db79SHyon Kim
64*9e86db79SHyon Kim
65*9e86db79SHyon Kim /*
66*9e86db79SHyon Kim * Send a SCSI inquiry to a remote WWN
67*9e86db79SHyon Kim */
68*9e86db79SHyon Kim HBA_STATUS
Sun_sasScsiInquiry(HBA_HANDLE handle,HBA_WWN portWWN,HBA_WWN targetPortWWN,HBA_WWN domainPortWWN,SMHBA_SCSILUN smhbaLUN,HBA_UINT8 cdb1,HBA_UINT8 cdb2,void * responseBuffer,HBA_UINT32 * responseSize,HBA_UINT8 * scsiStatus,void * senseBuffer,HBA_UINT32 * senseSize)69*9e86db79SHyon Kim Sun_sasScsiInquiry(HBA_HANDLE handle, HBA_WWN portWWN, HBA_WWN targetPortWWN,
70*9e86db79SHyon Kim HBA_WWN domainPortWWN, SMHBA_SCSILUN smhbaLUN, HBA_UINT8 cdb1,
71*9e86db79SHyon Kim HBA_UINT8 cdb2, void *responseBuffer, HBA_UINT32 *responseSize,
72*9e86db79SHyon Kim HBA_UINT8 *scsiStatus, void *senseBuffer, HBA_UINT32 *senseSize)
73*9e86db79SHyon Kim {
74*9e86db79SHyon Kim const char ROUTINE[] = "Sun_sasScsiInquiry";
75*9e86db79SHyon Kim HBA_STATUS status;
76*9e86db79SHyon Kim int index = 0;
77*9e86db79SHyon Kim int domainPortFound = 0;
78*9e86db79SHyon Kim int hbaPortFound = 0;
79*9e86db79SHyon Kim int chkDomainPort = 0;
80*9e86db79SHyon Kim struct sun_sas_hba *hba_ptr = NULL;
81*9e86db79SHyon Kim struct sun_sas_port *hba_port_ptr, *hba_disco_port;
82*9e86db79SHyon Kim struct ScsiEntryList *mapping_ptr;
83*9e86db79SHyon Kim hrtime_t start, end;
84*9e86db79SHyon Kim double duration;
85*9e86db79SHyon Kim HBA_SCSILUN hba_lun;
86*9e86db79SHyon Kim
87*9e86db79SHyon Kim start = gethrtime();
88*9e86db79SHyon Kim /* Validate the arguments */
89*9e86db79SHyon Kim if (responseBuffer == NULL) {
90*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "NULL response buffer");
91*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG);
92*9e86db79SHyon Kim }
93*9e86db79SHyon Kim if (senseBuffer == NULL) {
94*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "NULL sense buffer");
95*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG);
96*9e86db79SHyon Kim }
97*9e86db79SHyon Kim if (responseSize == NULL) {
98*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "NULL response size");
99*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG);
100*9e86db79SHyon Kim }
101*9e86db79SHyon Kim if (senseSize == NULL) {
102*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "NULL sense size");
103*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG);
104*9e86db79SHyon Kim }
105*9e86db79SHyon Kim if (scsiStatus == NULL) {
106*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "NULL scsi status");
107*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG);
108*9e86db79SHyon Kim }
109*9e86db79SHyon Kim
110*9e86db79SHyon Kim lock(&all_hbas_lock);
111*9e86db79SHyon Kim index = RetrieveIndex(handle);
112*9e86db79SHyon Kim lock(&open_handles_lock);
113*9e86db79SHyon Kim if ((hba_ptr = RetrieveHandle(index)) == NULL) {
114*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Invalid handle %08lx", handle);
115*9e86db79SHyon Kim unlock(&open_handles_lock);
116*9e86db79SHyon Kim unlock(&all_hbas_lock);
117*9e86db79SHyon Kim return (HBA_STATUS_ERROR_INVALID_HANDLE);
118*9e86db79SHyon Kim }
119*9e86db79SHyon Kim
120*9e86db79SHyon Kim /* Check for stale data */
121*9e86db79SHyon Kim status = verifyAdapter(hba_ptr);
122*9e86db79SHyon Kim if (status != HBA_STATUS_OK) {
123*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Verify adapter failed");
124*9e86db79SHyon Kim unlock(&open_handles_lock);
125*9e86db79SHyon Kim unlock(&all_hbas_lock);
126*9e86db79SHyon Kim return (status);
127*9e86db79SHyon Kim }
128*9e86db79SHyon Kim
129*9e86db79SHyon Kim /*
130*9e86db79SHyon Kim * We are not checking to see if our data is stale.
131*9e86db79SHyon Kim * By verifying this information here, we will take a big performance
132*9e86db79SHyon Kim * hit. This check will be done later only if the Inquiry ioctl fails
133*9e86db79SHyon Kim */
134*9e86db79SHyon Kim if (hba_ptr->device_path == NULL) {
135*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE,
136*9e86db79SHyon Kim "HBA handle had NULL device path. \
137*9e86db79SHyon Kim Unable to send SCSI cmd");
138*9e86db79SHyon Kim unlock(&open_handles_lock);
139*9e86db79SHyon Kim unlock(&all_hbas_lock);
140*9e86db79SHyon Kim return (HBA_STATUS_ERROR);
141*9e86db79SHyon Kim }
142*9e86db79SHyon Kim
143*9e86db79SHyon Kim if (wwnConversion(domainPortWWN.wwn))
144*9e86db79SHyon Kim chkDomainPort = 1;
145*9e86db79SHyon Kim
146*9e86db79SHyon Kim /* Determine which port to use */
147*9e86db79SHyon Kim for (hba_port_ptr = hba_ptr->first_port;
148*9e86db79SHyon Kim hba_port_ptr != NULL;
149*9e86db79SHyon Kim hba_port_ptr = hba_port_ptr->next) {
150*9e86db79SHyon Kim
151*9e86db79SHyon Kim if (hbaPortFound == 0) {
152*9e86db79SHyon Kim if (wwnConversion(hba_port_ptr->port_attributes.
153*9e86db79SHyon Kim PortSpecificAttribute.SASPort->LocalSASAddress.wwn)
154*9e86db79SHyon Kim != wwnConversion(portWWN.wwn)) {
155*9e86db79SHyon Kim /*
156*9e86db79SHyon Kim * Since all the ports under the same HBA have
157*9e86db79SHyon Kim * the same LocalSASAddress, we should break
158*9e86db79SHyon Kim * the loop once we find it dosn't match.
159*9e86db79SHyon Kim */
160*9e86db79SHyon Kim break;
161*9e86db79SHyon Kim } else {
162*9e86db79SHyon Kim hbaPortFound = 1;
163*9e86db79SHyon Kim }
164*9e86db79SHyon Kim }
165*9e86db79SHyon Kim
166*9e86db79SHyon Kim if (chkDomainPort) {
167*9e86db79SHyon Kim if (hba_port_ptr->first_phy != NULL &&
168*9e86db79SHyon Kim wwnConversion(hba_port_ptr->first_phy->
169*9e86db79SHyon Kim phy.domainPortWWN.wwn) ==
170*9e86db79SHyon Kim wwnConversion(domainPortWWN.wwn)) {
171*9e86db79SHyon Kim domainPortFound = 1;
172*9e86db79SHyon Kim }
173*9e86db79SHyon Kim if (!(domainPortFound)) {
174*9e86db79SHyon Kim continue;
175*9e86db79SHyon Kim }
176*9e86db79SHyon Kim }
177*9e86db79SHyon Kim
178*9e86db79SHyon Kim for (hba_disco_port = hba_port_ptr->first_attached_port;
179*9e86db79SHyon Kim hba_disco_port != NULL;
180*9e86db79SHyon Kim hba_disco_port = hba_disco_port->next) {
181*9e86db79SHyon Kim
182*9e86db79SHyon Kim /*
183*9e86db79SHyon Kim * If discoveredPort is not given targetPort, just skip
184*9e86db79SHyon Kim */
185*9e86db79SHyon Kim if (wwnConversion(hba_disco_port->port_attributes.\
186*9e86db79SHyon Kim PortSpecificAttribute.SASPort->LocalSASAddress.wwn)
187*9e86db79SHyon Kim != wwnConversion(targetPortWWN.wwn)) {
188*9e86db79SHyon Kim /* Does not match */
189*9e86db79SHyon Kim continue;
190*9e86db79SHyon Kim }
191*9e86db79SHyon Kim
192*9e86db79SHyon Kim /*
193*9e86db79SHyon Kim * If discoveredPort is not a SAS/SATA port, it is not a
194*9e86db79SHyon Kim * target port
195*9e86db79SHyon Kim */
196*9e86db79SHyon Kim if ((hba_disco_port->port_attributes.PortType !=
197*9e86db79SHyon Kim HBA_PORTTYPE_SATADEVICE) &&
198*9e86db79SHyon Kim (hba_disco_port->port_attributes.PortType !=
199*9e86db79SHyon Kim HBA_PORTTYPE_SASDEVICE)) {
200*9e86db79SHyon Kim unlock(&open_handles_lock);
201*9e86db79SHyon Kim unlock(&all_hbas_lock);
202*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Target Port WWN "
203*9e86db79SHyon Kim "%016llx on handle %08lx is not a Target",
204*9e86db79SHyon Kim wwnConversion(targetPortWWN.wwn), handle);
205*9e86db79SHyon Kim return (HBA_STATUS_ERROR_NOT_A_TARGET);
206*9e86db79SHyon Kim }
207*9e86db79SHyon Kim
208*9e86db79SHyon Kim /*
209*9e86db79SHyon Kim * Iterating and matching is needed.
210*9e86db79SHyon Kim */
211*9e86db79SHyon Kim for (mapping_ptr = hba_disco_port->scsiInfo;
212*9e86db79SHyon Kim mapping_ptr != NULL;
213*9e86db79SHyon Kim mapping_ptr = mapping_ptr->next) {
214*9e86db79SHyon Kim
215*9e86db79SHyon Kim if (memcmp(
216*9e86db79SHyon Kim &mapping_ptr->entry.PortLun.TargetLun,
217*9e86db79SHyon Kim &smhbaLUN, sizeof (HBA_SCSILUN))
218*9e86db79SHyon Kim != 0) {
219*9e86db79SHyon Kim continue;
220*9e86db79SHyon Kim }
221*9e86db79SHyon Kim
222*9e86db79SHyon Kim status = SendScsiInquiry(
223*9e86db79SHyon Kim mapping_ptr->entry.ScsiId.OSDeviceName,
224*9e86db79SHyon Kim cdb1, cdb2,
225*9e86db79SHyon Kim responseBuffer, responseSize,
226*9e86db79SHyon Kim scsiStatus, senseBuffer,
227*9e86db79SHyon Kim senseSize);
228*9e86db79SHyon Kim
229*9e86db79SHyon Kim unlock(&open_handles_lock);
230*9e86db79SHyon Kim unlock(&all_hbas_lock);
231*9e86db79SHyon Kim end = gethrtime();
232*9e86db79SHyon Kim duration = end - start;
233*9e86db79SHyon Kim duration /= HR_SECOND;
234*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Took total\
235*9e86db79SHyon Kim of %.4f seconds", duration);
236*9e86db79SHyon Kim return (status);
237*9e86db79SHyon Kim }
238*9e86db79SHyon Kim unlock(&open_handles_lock);
239*9e86db79SHyon Kim unlock(&all_hbas_lock);
240*9e86db79SHyon Kim (void *) memcpy(&hba_lun, &smhbaLUN,
241*9e86db79SHyon Kim sizeof (HBA_SCSILUN));
242*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Unable to locate lun"
243*9e86db79SHyon Kim " %08lx for target %016llx on handle %08lx",
244*9e86db79SHyon Kim hba_lun, wwnConversion(targetPortWWN.wwn), handle);
245*9e86db79SHyon Kim return (HBA_STATUS_ERROR_INVALID_LUN);
246*9e86db79SHyon Kim }
247*9e86db79SHyon Kim if (chkDomainPort) {
248*9e86db79SHyon Kim unlock(&open_handles_lock);
249*9e86db79SHyon Kim unlock(&all_hbas_lock);
250*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Unable to locate requested "
251*9e86db79SHyon Kim "Port WWN %016llx on handle %08lx",
252*9e86db79SHyon Kim wwnConversion(targetPortWWN.wwn), handle);
253*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ILLEGAL_WWN);
254*9e86db79SHyon Kim }
255*9e86db79SHyon Kim }
256*9e86db79SHyon Kim
257*9e86db79SHyon Kim unlock(&open_handles_lock);
258*9e86db79SHyon Kim unlock(&all_hbas_lock);
259*9e86db79SHyon Kim if (hbaPortFound == 0) {
260*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE,
261*9e86db79SHyon Kim "Unable to locate requested Port WWN %016llx on "
262*9e86db79SHyon Kim "handle %08lx", wwnConversion(portWWN.wwn), handle);
263*9e86db79SHyon Kim } else if (chkDomainPort && !domainPortFound) {
264*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Unable to locate requested"
265*9e86db79SHyon Kim " domainPortWWN %016llx on handle %08lx",
266*9e86db79SHyon Kim wwnConversion(domainPortWWN.wwn), handle);
267*9e86db79SHyon Kim } else {
268*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Unable to locate requested "
269*9e86db79SHyon Kim "Port WWN %016llx on handle %08lx",
270*9e86db79SHyon Kim wwnConversion(targetPortWWN.wwn), handle);
271*9e86db79SHyon Kim }
272*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ILLEGAL_WWN);
273*9e86db79SHyon Kim }
274