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