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 * Retrieves the mapping between targets and OS SCSI information 31*9e86db79SHyon Kim */ 32*9e86db79SHyon Kim HBA_STATUS 33*9e86db79SHyon Kim Sun_sasGetTargetMapping(HBA_HANDLE handle, HBA_WWN hbaPortWWN, 34*9e86db79SHyon Kim HBA_WWN domainPortWWN, SMHBA_TARGETMAPPING *mapping) 35*9e86db79SHyon Kim { 36*9e86db79SHyon Kim const char ROUTINE[] = "Sun_sasGetTargetMapping"; 37*9e86db79SHyon Kim int i, index; 38*9e86db79SHyon Kim int hbaPortFound = 0; 39*9e86db79SHyon Kim int domainPortFound = 0; 40*9e86db79SHyon Kim uint_t total_entries = 0; 41*9e86db79SHyon Kim struct sun_sas_hba *hba_ptr; 42*9e86db79SHyon Kim struct sun_sas_port *hba_port_ptr, *hba_disco_port; 43*9e86db79SHyon Kim struct ScsiEntryList *mapping_ptr; 44*9e86db79SHyon Kim 45*9e86db79SHyon Kim if (mapping == NULL) { 46*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "NULL mapping buffer"); 47*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG); 48*9e86db79SHyon Kim } 49*9e86db79SHyon Kim 50*9e86db79SHyon Kim lock(&all_hbas_lock); 51*9e86db79SHyon Kim index = RetrieveIndex(handle); 52*9e86db79SHyon Kim lock(&open_handles_lock); 53*9e86db79SHyon Kim hba_ptr = RetrieveHandle(index); 54*9e86db79SHyon Kim if (hba_ptr == NULL) { 55*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Invalid handle %08lx.", handle); 56*9e86db79SHyon Kim /* on error, need to set NumberOfEntries to 0 */ 57*9e86db79SHyon Kim mapping->NumberOfEntries = 0; 58*9e86db79SHyon Kim unlock(&open_handles_lock); 59*9e86db79SHyon Kim unlock(&all_hbas_lock); 60*9e86db79SHyon Kim return (HBA_STATUS_ERROR_INVALID_HANDLE); 61*9e86db79SHyon Kim } 62*9e86db79SHyon Kim 63*9e86db79SHyon Kim /* 64*9e86db79SHyon Kim * We should indicate an error if no domainPortWWN passed in. 65*9e86db79SHyon Kim */ 66*9e86db79SHyon Kim if (wwnConversion(domainPortWWN.wwn) == 0) { 67*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "domainPortWWN must be provided"); 68*9e86db79SHyon Kim mapping->NumberOfEntries = 0; 69*9e86db79SHyon Kim unlock(&open_handles_lock); 70*9e86db79SHyon Kim unlock(&all_hbas_lock); 71*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ARG); 72*9e86db79SHyon Kim } 73*9e86db79SHyon Kim /* 74*9e86db79SHyon Kim * walk through the list of ports for this hba and count up the number 75*9e86db79SHyon Kim * of discovered ports on each hba port 76*9e86db79SHyon Kim */ 77*9e86db79SHyon Kim i = 0; 78*9e86db79SHyon Kim for (hba_port_ptr = hba_ptr->first_port; hba_port_ptr != NULL; 79*9e86db79SHyon Kim hba_port_ptr = hba_port_ptr->next) { 80*9e86db79SHyon Kim if (hbaPortFound == 0) { 81*9e86db79SHyon Kim if (wwnConversion(hba_port_ptr->port_attributes. 82*9e86db79SHyon Kim PortSpecificAttribute.SASPort->LocalSASAddress.wwn) 83*9e86db79SHyon Kim != wwnConversion(hbaPortWWN.wwn)) { 84*9e86db79SHyon Kim /* 85*9e86db79SHyon Kim * Since all the ports under the same HBA have 86*9e86db79SHyon Kim * the same LocalSASAddress, we should break 87*9e86db79SHyon Kim * the loop once we find it dosn't match. 88*9e86db79SHyon Kim */ 89*9e86db79SHyon Kim break; 90*9e86db79SHyon Kim } else { 91*9e86db79SHyon Kim hbaPortFound = 1; 92*9e86db79SHyon Kim } 93*9e86db79SHyon Kim } 94*9e86db79SHyon Kim 95*9e86db79SHyon Kim /* 96*9e86db79SHyon Kim * Check whether the domainPortWWN matches. 97*9e86db79SHyon Kim */ 98*9e86db79SHyon Kim if ((validateDomainAddress(hba_port_ptr, domainPortWWN)) 99*9e86db79SHyon Kim != HBA_STATUS_OK) { 100*9e86db79SHyon Kim continue; 101*9e86db79SHyon Kim } 102*9e86db79SHyon Kim domainPortFound = 1; 103*9e86db79SHyon Kim 104*9e86db79SHyon Kim for (hba_disco_port = hba_port_ptr->first_attached_port; 105*9e86db79SHyon Kim hba_disco_port != NULL; 106*9e86db79SHyon Kim hba_disco_port = hba_disco_port->next) { 107*9e86db79SHyon Kim for (mapping_ptr = hba_disco_port->scsiInfo; 108*9e86db79SHyon Kim mapping_ptr != NULL; 109*9e86db79SHyon Kim mapping_ptr = mapping_ptr->next) { 110*9e86db79SHyon Kim /* 111*9e86db79SHyon Kim * Add the information as much as mapping 112*9e86db79SHyon Kim * can hold. 113*9e86db79SHyon Kim */ 114*9e86db79SHyon Kim if (wwnConversion(domainPortWWN.wwn) != 115*9e86db79SHyon Kim wwnConversion(mapping_ptr->entry. 116*9e86db79SHyon Kim PortLun.domainPortWWN.wwn)) { 117*9e86db79SHyon Kim continue; 118*9e86db79SHyon Kim } 119*9e86db79SHyon Kim 120*9e86db79SHyon Kim if (total_entries < mapping->NumberOfEntries) { 121*9e86db79SHyon Kim (void) memcpy(&mapping->entry[i].ScsiId, 122*9e86db79SHyon Kim &mapping_ptr->entry.ScsiId, 123*9e86db79SHyon Kim sizeof (SMHBA_SCSIID)); 124*9e86db79SHyon Kim (void) memcpy(&mapping->entry[i]. 125*9e86db79SHyon Kim PortLun, &mapping_ptr->entry. 126*9e86db79SHyon Kim PortLun, sizeof (SMHBA_PORTLUN)); 127*9e86db79SHyon Kim (void) memcpy(&mapping->entry[i].LUID, 128*9e86db79SHyon Kim &mapping_ptr->entry.LUID, 129*9e86db79SHyon Kim sizeof (SMHBA_LUID)); 130*9e86db79SHyon Kim i++; 131*9e86db79SHyon Kim } 132*9e86db79SHyon Kim total_entries++; 133*9e86db79SHyon Kim } 134*9e86db79SHyon Kim } 135*9e86db79SHyon Kim } 136*9e86db79SHyon Kim 137*9e86db79SHyon Kim /* 138*9e86db79SHyon Kim * check to make sure user has passed in an acceptable PortWWN for 139*9e86db79SHyon Kim * the given handle 140*9e86db79SHyon Kim */ 141*9e86db79SHyon Kim if (hbaPortFound == 0) { 142*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "Unable to locate requested " 143*9e86db79SHyon Kim "HBA Port WWN %016llx on handle %08lx", 144*9e86db79SHyon Kim wwnConversion(hbaPortWWN.wwn), handle); 145*9e86db79SHyon Kim unlock(&open_handles_lock); 146*9e86db79SHyon Kim unlock(&all_hbas_lock); 147*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ILLEGAL_WWN); 148*9e86db79SHyon Kim } 149*9e86db79SHyon Kim 150*9e86db79SHyon Kim if (domainPortFound == 0) { 151*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, "No matching domain " 152*9e86db79SHyon Kim "port %016llx for port %016llx on handle " 153*9e86db79SHyon Kim "%08lx", wwnConversion(domainPortWWN.wwn), 154*9e86db79SHyon Kim wwnConversion(hbaPortWWN.wwn), handle); 155*9e86db79SHyon Kim unlock(&open_handles_lock); 156*9e86db79SHyon Kim unlock(&all_hbas_lock); 157*9e86db79SHyon Kim return (HBA_STATUS_ERROR_ILLEGAL_WWN); 158*9e86db79SHyon Kim } 159*9e86db79SHyon Kim 160*9e86db79SHyon Kim if (total_entries > mapping->NumberOfEntries) { 161*9e86db79SHyon Kim log(LOG_DEBUG, ROUTINE, 162*9e86db79SHyon Kim "total entries: %d: mapping->NumberofEntries: %d.", 163*9e86db79SHyon Kim total_entries, mapping->NumberOfEntries); 164*9e86db79SHyon Kim mapping->NumberOfEntries = total_entries; 165*9e86db79SHyon Kim unlock(&open_handles_lock); 166*9e86db79SHyon Kim unlock(&all_hbas_lock); 167*9e86db79SHyon Kim return (HBA_STATUS_ERROR_MORE_DATA); 168*9e86db79SHyon Kim } 169*9e86db79SHyon Kim 170*9e86db79SHyon Kim mapping->NumberOfEntries = total_entries; 171*9e86db79SHyon Kim 172*9e86db79SHyon Kim /* convert devpath to dev link */ 173*9e86db79SHyon Kim convertDevpathToDevlink(mapping); 174*9e86db79SHyon Kim 175*9e86db79SHyon Kim unlock(&open_handles_lock); 176*9e86db79SHyon Kim unlock(&all_hbas_lock); 177*9e86db79SHyon Kim 178*9e86db79SHyon Kim return (HBA_STATUS_OK); 179*9e86db79SHyon Kim } 180