1*fcf3ce44SJohn Forte /* 2*fcf3ce44SJohn Forte * CDDL HEADER START 3*fcf3ce44SJohn Forte * 4*fcf3ce44SJohn Forte * The contents of this file are subject to the terms of the 5*fcf3ce44SJohn Forte * Common Development and Distribution License (the "License"). 6*fcf3ce44SJohn Forte * You may not use this file except in compliance with the License. 7*fcf3ce44SJohn Forte * 8*fcf3ce44SJohn Forte * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9*fcf3ce44SJohn Forte * or http://www.opensolaris.org/os/licensing. 10*fcf3ce44SJohn Forte * See the License for the specific language governing permissions 11*fcf3ce44SJohn Forte * and limitations under the License. 12*fcf3ce44SJohn Forte * 13*fcf3ce44SJohn Forte * When distributing Covered Code, include this CDDL HEADER in each 14*fcf3ce44SJohn Forte * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15*fcf3ce44SJohn Forte * If applicable, add the following below this CDDL HEADER, with the 16*fcf3ce44SJohn Forte * fields enclosed by brackets "[]" replaced with your own identifying 17*fcf3ce44SJohn Forte * information: Portions Copyright [yyyy] [name of copyright owner] 18*fcf3ce44SJohn Forte * 19*fcf3ce44SJohn Forte * CDDL HEADER END 20*fcf3ce44SJohn Forte */ 21*fcf3ce44SJohn Forte /* 22*fcf3ce44SJohn Forte * Copyright 2008 Sun Microsystems, Inc. All rights reserved. 23*fcf3ce44SJohn Forte * Use is subject to license terms. 24*fcf3ce44SJohn Forte */ 25*fcf3ce44SJohn Forte 26*fcf3ce44SJohn Forte 27*fcf3ce44SJohn Forte 28*fcf3ce44SJohn Forte #include "HBAPort.h" 29*fcf3ce44SJohn Forte #include "Exceptions.h" 30*fcf3ce44SJohn Forte #include "Trace.h" 31*fcf3ce44SJohn Forte #include <iostream> 32*fcf3ce44SJohn Forte #include <iomanip> 33*fcf3ce44SJohn Forte #include <cerrno> 34*fcf3ce44SJohn Forte #include <cstring> 35*fcf3ce44SJohn Forte #include <sys/types.h> 36*fcf3ce44SJohn Forte #include <sys/mkdev.h> 37*fcf3ce44SJohn Forte #include <sys/stat.h> 38*fcf3ce44SJohn Forte #include <fcntl.h> 39*fcf3ce44SJohn Forte #include <unistd.h> 40*fcf3ce44SJohn Forte #include <stropts.h> 41*fcf3ce44SJohn Forte #include <dirent.h> 42*fcf3ce44SJohn Forte #include <libdevinfo.h> 43*fcf3ce44SJohn Forte 44*fcf3ce44SJohn Forte using namespace std; 45*fcf3ce44SJohn Forte 46*fcf3ce44SJohn Forte /** 47*fcf3ce44SJohn Forte * Standard definition for general topology lookup (See T11 FC-FS) 48*fcf3ce44SJohn Forte */ 49*fcf3ce44SJohn Forte const int HBAPort::RNID_GENERAL_TOPOLOGY_DATA_FORMAT = 0xDF; 50*fcf3ce44SJohn Forte const uint8_t HBAPort::HBA_NPIV_PORT_MAX = UCHAR_MAX; 51*fcf3ce44SJohn Forte 52*fcf3ce44SJohn Forte /** 53*fcf3ce44SJohn Forte * @memo Construct a new deafult HBA Port 54*fcf3ce44SJohn Forte */ 55*fcf3ce44SJohn Forte HBAPort::HBAPort() { 56*fcf3ce44SJohn Forte } 57*fcf3ce44SJohn Forte 58*fcf3ce44SJohn Forte /** 59*fcf3ce44SJohn Forte * @memo Compare two HBA ports for equality 60*fcf3ce44SJohn Forte * @return TRUE if both ports are the same 61*fcf3ce44SJohn Forte * @return FALSE if the ports are different 62*fcf3ce44SJohn Forte * 63*fcf3ce44SJohn Forte * @doc Comparison is based on Node WWN, Port WWN and path 64*fcf3ce44SJohn Forte */ 65*fcf3ce44SJohn Forte bool HBAPort::operator==(HBAPort &comp) { 66*fcf3ce44SJohn Forte return (this->getPortWWN() == comp.getPortWWN() && 67*fcf3ce44SJohn Forte this->getNodeWWN() == comp.getNodeWWN() && 68*fcf3ce44SJohn Forte this->getPath() == comp.getPath()); 69*fcf3ce44SJohn Forte } 70*fcf3ce44SJohn Forte 71*fcf3ce44SJohn Forte /** 72*fcf3ce44SJohn Forte * @memo Validate that the port is still present in the system 73*fcf3ce44SJohn Forte * @exception UnavailableException if the port is not present 74*fcf3ce44SJohn Forte * 75*fcf3ce44SJohn Forte * @doc If the port is still present on the system, the routine 76*fcf3ce44SJohn Forte * will return normally. If the port is not present 77*fcf3ce44SJohn Forte * an exception will be thrown. 78*fcf3ce44SJohn Forte */ 79*fcf3ce44SJohn Forte void HBAPort::validatePresent() { 80*fcf3ce44SJohn Forte Trace log("HBAPort::validatePresent"); 81*fcf3ce44SJohn Forte string path = getPath(); 82*fcf3ce44SJohn Forte struct stat sbuf; 83*fcf3ce44SJohn Forte if (stat(path.c_str(), &sbuf) == -1) { 84*fcf3ce44SJohn Forte if (errno == ENOENT) { 85*fcf3ce44SJohn Forte throw UnavailableException(); 86*fcf3ce44SJohn Forte } else { 87*fcf3ce44SJohn Forte log.debug("Unable to stat %s: %s", path.c_str(), 88*fcf3ce44SJohn Forte strerror(errno)); 89*fcf3ce44SJohn Forte throw InternalError(); 90*fcf3ce44SJohn Forte } 91*fcf3ce44SJohn Forte } 92*fcf3ce44SJohn Forte } 93*fcf3ce44SJohn Forte 94*fcf3ce44SJohn Forte 95*fcf3ce44SJohn Forte /* 96*fcf3ce44SJohn Forte * structure for di_devlink_walk 97*fcf3ce44SJohn Forte */ 98*fcf3ce44SJohn Forte typedef struct walk_devlink { 99*fcf3ce44SJohn Forte char *path; 100*fcf3ce44SJohn Forte size_t len; 101*fcf3ce44SJohn Forte char **linkpp; 102*fcf3ce44SJohn Forte } walk_devlink_t; 103*fcf3ce44SJohn Forte 104*fcf3ce44SJohn Forte /** 105*fcf3ce44SJohn Forte * @memo callback funtion for di_devlink_walk 106*fcf3ce44SJohn Forte * @postcondition Find matching /dev link for the given path argument. 107*fcf3ce44SJohn Forte * @param devlink element and callback function argument. 108*fcf3ce44SJohn Forte * 109*fcf3ce44SJohn Forte * @doc The input path is expected to not have "/devices". 110*fcf3ce44SJohn Forte */ 111*fcf3ce44SJohn Forte extern "C" int 112*fcf3ce44SJohn Forte get_devlink(di_devlink_t devlink, void *arg) { 113*fcf3ce44SJohn Forte Trace log("get_devlink"); 114*fcf3ce44SJohn Forte walk_devlink_t *warg = (walk_devlink_t *)arg; 115*fcf3ce44SJohn Forte 116*fcf3ce44SJohn Forte /* 117*fcf3ce44SJohn Forte * When path is specified, it doesn't have minor 118*fcf3ce44SJohn Forte * name. Therefore, the ../.. prefixes needs to be stripped. 119*fcf3ce44SJohn Forte */ 120*fcf3ce44SJohn Forte if (warg->path) { 121*fcf3ce44SJohn Forte // di_devlink_content contains /devices 122*fcf3ce44SJohn Forte char *content = (char *)di_devlink_content(devlink); 123*fcf3ce44SJohn Forte char *start = strstr(content, "/devices"); 124*fcf3ce44SJohn Forte 125*fcf3ce44SJohn Forte if (start == NULL || 126*fcf3ce44SJohn Forte strncmp(start, warg->path, warg->len) != 0 || 127*fcf3ce44SJohn Forte // make it sure the device path has minor name 128*fcf3ce44SJohn Forte start[warg->len] != ':') 129*fcf3ce44SJohn Forte return (DI_WALK_CONTINUE); 130*fcf3ce44SJohn Forte } 131*fcf3ce44SJohn Forte 132*fcf3ce44SJohn Forte *(warg->linkpp) = strdup(di_devlink_path(devlink)); 133*fcf3ce44SJohn Forte return (DI_WALK_TERMINATE); 134*fcf3ce44SJohn Forte } 135*fcf3ce44SJohn Forte 136*fcf3ce44SJohn Forte /** 137*fcf3ce44SJohn Forte * @memo Convert /devices paths to /dev sym-link paths. 138*fcf3ce44SJohn Forte * @postcondition The mapping buffer OSDeviceName paths will be 139*fcf3ce44SJohn Forte * converted to short names. 140*fcf3ce44SJohn Forte * @param mappings The target mappings data to convert to 141*fcf3ce44SJohn Forte * short names 142*fcf3ce44SJohn Forte * 143*fcf3ce44SJohn Forte * @doc If no link 144*fcf3ce44SJohn Forte * is found, the long path is left as is. 145*fcf3ce44SJohn Forte * Note: The NumberOfEntries field MUST not be greater than the size 146*fcf3ce44SJohn Forte * of the array passed in. 147*fcf3ce44SJohn Forte */ 148*fcf3ce44SJohn Forte void HBAPort::convertToShortNames(PHBA_FCPTARGETMAPPINGV2 mappings) { 149*fcf3ce44SJohn Forte Trace log("HBAPort::convertToShortNames"); 150*fcf3ce44SJohn Forte di_devlink_handle_t hdl; 151*fcf3ce44SJohn Forte walk_devlink_t warg; 152*fcf3ce44SJohn Forte char *minor_path, *devlinkp; 153*fcf3ce44SJohn Forte 154*fcf3ce44SJohn Forte if ((hdl = di_devlink_init(NULL, 0)) == NULL) { 155*fcf3ce44SJohn Forte log.internalError("di_devlink_init failed. Errno:%d", errno); 156*fcf3ce44SJohn Forte // no need to check further, just return here. 157*fcf3ce44SJohn Forte return; 158*fcf3ce44SJohn Forte } 159*fcf3ce44SJohn Forte 160*fcf3ce44SJohn Forte for (int j = 0; j < mappings->NumberOfEntries; j++) { 161*fcf3ce44SJohn Forte if (strchr(mappings->entry[j].ScsiId.OSDeviceName, ':')) { 162*fcf3ce44SJohn Forte // search link for minor node 163*fcf3ce44SJohn Forte minor_path = mappings->entry[j].ScsiId.OSDeviceName; 164*fcf3ce44SJohn Forte if (strstr(minor_path, "/devices") != NULL) { 165*fcf3ce44SJohn Forte minor_path = mappings->entry[j].ScsiId.OSDeviceName + 166*fcf3ce44SJohn Forte strlen("/devices"); 167*fcf3ce44SJohn Forte } else { 168*fcf3ce44SJohn Forte minor_path = mappings->entry[j].ScsiId.OSDeviceName; 169*fcf3ce44SJohn Forte } 170*fcf3ce44SJohn Forte warg.path = NULL; 171*fcf3ce44SJohn Forte } else { 172*fcf3ce44SJohn Forte minor_path = NULL; 173*fcf3ce44SJohn Forte if (strstr(mappings->entry[j].ScsiId.OSDeviceName, 174*fcf3ce44SJohn Forte "/devices") != NULL) { 175*fcf3ce44SJohn Forte warg.len = strlen (mappings->entry[j].ScsiId.OSDeviceName) - 176*fcf3ce44SJohn Forte strlen ("/devices"); 177*fcf3ce44SJohn Forte warg.path = mappings->entry[j].ScsiId.OSDeviceName + 178*fcf3ce44SJohn Forte strlen ("/devices"); 179*fcf3ce44SJohn Forte } else { 180*fcf3ce44SJohn Forte warg.len = strlen(mappings->entry[j].ScsiId.OSDeviceName); 181*fcf3ce44SJohn Forte warg.path = mappings->entry[j].ScsiId.OSDeviceName; 182*fcf3ce44SJohn Forte } 183*fcf3ce44SJohn Forte } 184*fcf3ce44SJohn Forte 185*fcf3ce44SJohn Forte devlinkp = NULL; 186*fcf3ce44SJohn Forte warg.linkpp = &devlinkp; 187*fcf3ce44SJohn Forte (void) di_devlink_walk(hdl, NULL, minor_path, DI_PRIMARY_LINK, 188*fcf3ce44SJohn Forte (void *)&warg, get_devlink); 189*fcf3ce44SJohn Forte 190*fcf3ce44SJohn Forte if (devlinkp != NULL) { 191*fcf3ce44SJohn Forte snprintf(mappings->entry[j].ScsiId.OSDeviceName, 192*fcf3ce44SJohn Forte sizeof (mappings->entry[j].ScsiId.OSDeviceName), 193*fcf3ce44SJohn Forte "%s", devlinkp); 194*fcf3ce44SJohn Forte free(devlinkp); 195*fcf3ce44SJohn Forte } // else leave OSDeviceName alone. 196*fcf3ce44SJohn Forte 197*fcf3ce44SJohn Forte } 198*fcf3ce44SJohn Forte 199*fcf3ce44SJohn Forte di_devlink_fini(&hdl); 200*fcf3ce44SJohn Forte 201*fcf3ce44SJohn Forte } 202*fcf3ce44SJohn Forte 203*fcf3ce44SJohn Forte /* 204*fcf3ce44SJohn Forte * Finds controller path for a give device path. 205*fcf3ce44SJohn Forte * 206*fcf3ce44SJohn Forte * Return vale: controller path. 207*fcf3ce44SJohn Forte */ 208*fcf3ce44SJohn Forte string HBAPort::lookupControllerPath(string path) { 209*fcf3ce44SJohn Forte Trace log("lookupControllerPath"); 210*fcf3ce44SJohn Forte DIR *dp; 211*fcf3ce44SJohn Forte char buf[MAXPATHLEN]; 212*fcf3ce44SJohn Forte char node[MAXPATHLEN]; 213*fcf3ce44SJohn Forte struct dirent **dirpp, *dirp; 214*fcf3ce44SJohn Forte const char dir[] = "/dev/cfg"; 215*fcf3ce44SJohn Forte ssize_t count; 216*fcf3ce44SJohn Forte uchar_t *dir_buf = new uchar_t[sizeof (struct dirent) + MAXPATHLEN]; 217*fcf3ce44SJohn Forte 218*fcf3ce44SJohn Forte if ((dp = opendir(dir)) == NULL) { 219*fcf3ce44SJohn Forte string tmp = "Unable to open "; 220*fcf3ce44SJohn Forte tmp += dir; 221*fcf3ce44SJohn Forte tmp += "to find controller number."; 222*fcf3ce44SJohn Forte delete (dir_buf); 223*fcf3ce44SJohn Forte throw IOError(tmp); 224*fcf3ce44SJohn Forte } 225*fcf3ce44SJohn Forte 226*fcf3ce44SJohn Forte dirp = (struct dirent *) dir_buf; 227*fcf3ce44SJohn Forte dirpp = &dirp; 228*fcf3ce44SJohn Forte while ((readdir_r(dp, dirp, dirpp)) == 0 && dirp != NULL) { 229*fcf3ce44SJohn Forte if (strcmp(dirp->d_name, ".") == 0 || 230*fcf3ce44SJohn Forte strcmp(dirp->d_name, "..") == 0) { 231*fcf3ce44SJohn Forte continue; 232*fcf3ce44SJohn Forte } 233*fcf3ce44SJohn Forte sprintf(node, "%s/%s", dir, dirp->d_name); 234*fcf3ce44SJohn Forte if ((count = readlink(node,buf,sizeof(buf)))) { 235*fcf3ce44SJohn Forte buf[count] = '\0'; 236*fcf3ce44SJohn Forte if (strstr(buf, path.c_str())) { 237*fcf3ce44SJohn Forte string cfg_path = dir; 238*fcf3ce44SJohn Forte cfg_path += "/"; 239*fcf3ce44SJohn Forte cfg_path += dirp->d_name; 240*fcf3ce44SJohn Forte closedir(dp); 241*fcf3ce44SJohn Forte delete (dir_buf); 242*fcf3ce44SJohn Forte return (cfg_path); 243*fcf3ce44SJohn Forte } 244*fcf3ce44SJohn Forte } 245*fcf3ce44SJohn Forte } 246*fcf3ce44SJohn Forte 247*fcf3ce44SJohn Forte closedir(dp); 248*fcf3ce44SJohn Forte delete (dir_buf); 249*fcf3ce44SJohn Forte throw InternalError("Unable to find controller path"); 250*fcf3ce44SJohn Forte } 251*fcf3ce44SJohn Forte 252*fcf3ce44SJohn Forte void HBAPort::addPort(HBANPIVPort *port) { 253*fcf3ce44SJohn Forte Trace log("HBAPort::addPort"); 254*fcf3ce44SJohn Forte lock(); 255*fcf3ce44SJohn Forte // support hba with up to UCHAR_MAX number of ports. 256*fcf3ce44SJohn Forte if (npivportsByIndex.size() + 1 > HBA_NPIV_PORT_MAX) { 257*fcf3ce44SJohn Forte unlock(); 258*fcf3ce44SJohn Forte throw InternalError("HBA NPIV Port count exceeds max number of ports"); 259*fcf3ce44SJohn Forte } 260*fcf3ce44SJohn Forte 261*fcf3ce44SJohn Forte try { 262*fcf3ce44SJohn Forte npivportsByWWN[port->getPortWWN()] = port; 263*fcf3ce44SJohn Forte npivportsByIndex.insert(npivportsByIndex.end(), port); 264*fcf3ce44SJohn Forte unlock(); 265*fcf3ce44SJohn Forte } catch (...) { 266*fcf3ce44SJohn Forte unlock(); 267*fcf3ce44SJohn Forte throw; 268*fcf3ce44SJohn Forte } 269*fcf3ce44SJohn Forte } 270*fcf3ce44SJohn Forte 271*fcf3ce44SJohn Forte HBANPIVPort* HBAPort::getPort(uint64_t wwn) { 272*fcf3ce44SJohn Forte Trace log("HBAPort::getPort"); 273*fcf3ce44SJohn Forte HBANPIVPort *port = NULL; 274*fcf3ce44SJohn Forte 275*fcf3ce44SJohn Forte lock(); 276*fcf3ce44SJohn Forte try { 277*fcf3ce44SJohn Forte if (npivportsByWWN.find(wwn) == npivportsByWWN.end()) { 278*fcf3ce44SJohn Forte throw IllegalWWNException(); 279*fcf3ce44SJohn Forte } 280*fcf3ce44SJohn Forte port = npivportsByWWN[wwn]; 281*fcf3ce44SJohn Forte unlock(); 282*fcf3ce44SJohn Forte return (port); 283*fcf3ce44SJohn Forte } catch (...) { 284*fcf3ce44SJohn Forte unlock(); 285*fcf3ce44SJohn Forte throw; 286*fcf3ce44SJohn Forte } 287*fcf3ce44SJohn Forte } 288*fcf3ce44SJohn Forte 289*fcf3ce44SJohn Forte HBANPIVPort* HBAPort::getPortByIndex(int index) { 290*fcf3ce44SJohn Forte Trace log("HBAPort::getPortByIndex"); 291*fcf3ce44SJohn Forte lock(); 292*fcf3ce44SJohn Forte try { 293*fcf3ce44SJohn Forte if (index >= npivportsByIndex.size() || index < 0) { 294*fcf3ce44SJohn Forte throw IllegalIndexException(); 295*fcf3ce44SJohn Forte } 296*fcf3ce44SJohn Forte HBANPIVPort *tmp = npivportsByIndex[index]; 297*fcf3ce44SJohn Forte unlock(); 298*fcf3ce44SJohn Forte return (tmp); 299*fcf3ce44SJohn Forte } catch (...) { 300*fcf3ce44SJohn Forte unlock(); 301*fcf3ce44SJohn Forte throw; 302*fcf3ce44SJohn Forte } 303*fcf3ce44SJohn Forte } 304*fcf3ce44SJohn Forte 305