/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License (the "License").
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright 2009 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
/*
* These functions are used to encode SAS SMP address data into
* Solaris devid / guid values.
*/
#ifndef _KERNEL
#include <stdio.h>
#endif /* _KERNEL */
#include <sys/inttypes.h>
#include <sys/types.h>
#include <sys/stropts.h>
#include <sys/debug.h>
#include <sys/isa_defs.h>
#include <sys/dditypes.h>
#include <sys/ddi_impldefs.h>
#include <sys/scsi/scsi.h>
#include <sys/scsi/generic/smp_frames.h>
#ifndef _KERNEL
#include <sys/libdevid.h>
#endif /* !_KERNEL */
#include "devid_impl.h"
/*
* Typically the wwnstr makes a good devid, however in some cases the wwnstr
* comes form the location of a FRU in the chassis instead of from the identity
* of the FRU. The table below provides vid/pid information for such cases.
* These vidpid strings are matched against smp_report_manufacturer_info_resp
* data. When a match occurs the srmir_vs_52 field, if non-zero, is used
* to form the devid.
*/
char *vidpid_devid_from_srmir_vs_52[] = {
/* " 111111" */
/* "012345670123456789012345" */
/* "|-VID--||-----PID------|" */
"SUN GENESIS",
NULL
};
/*
* Function: ddi_/devid_smp_encode
*
* Description: This routine finds and encodes a unique devid given the
* SAS address of an SMP node.
*
* Arguments: version - id encode algorithm version
* driver_name - binding driver name (if ! known use NULL)
* wwnstr - smp SAS address in wwnstr (unit-address) form.
* srmir_buf - REPORT MANUFACTURER INFORMATION response.
* srmir_len - amount of srmir_buf data.
* devid - id returned
*
* Return Code: DEVID_SUCCESS - success
* DEVID_FAILURE - failure
*/
int
#ifdef _KERNEL
ddi_devid_smp_encode(
#else /* ! _KERNEL */
devid_smp_encode(
#endif /* _KERNEL */
int version, /* IN */
char *driver_name, /* IN */
char *wwnstr, /* IN */
uchar_t *srmir_buf, /* IN */
size_t srmir_len, /* IN */
ddi_devid_t *devid) /* OUT */
{
uint64_t wwn;
ushort_t raw_id_type;
ushort_t raw_id_len;
impl_devid_t *i_devid;
int i_devid_len;
int i;
smp_response_frame_t *srs;
smp_report_manufacturer_info_resp_t *srmir;
char **vidpid;
uint8_t *vsp;
uint64_t s;
char sbuf[16 + 1];
int vlen, plen, slen;
int driver_name_len = 0;
DEVID_ASSERT(devid != NULL);
*devid = NULL;
/* verify valid version */
if (version > DEVID_SMP_ENCODE_VERSION_LATEST)
return (DEVID_FAILURE);
if (wwnstr == NULL)
return (DEVID_FAILURE);
/* convert wwnstr to binary */
if (scsi_wwnstr_to_wwn(wwnstr, &wwn) != DDI_SUCCESS)
return (DEVID_FAILURE);
if (srmir_buf &&
(srmir_len >= ((sizeof (*srs) - sizeof (srs->srf_data)) +
sizeof (*srmir)))) {
srs = (smp_response_frame_t *)srmir_buf;
srmir = (smp_report_manufacturer_info_resp_t *)srs->srf_data;
for (vidpid = vidpid_devid_from_srmir_vs_52; *vidpid; vidpid++)
if (strncmp(srmir->srmir_vendor_identification,
*vidpid, strlen(*vidpid)) == 0)
break;
/* no vid/pid match, use wwn for devid */
if (*vidpid == NULL)
goto usewwn;
/* extract the special vendor-specific 'devid serial number' */
vsp = &srmir->srmir_vs_52[0];
s = ((uint64_t)vsp[0] << 56) |
((uint64_t)vsp[1] << 48) |
((uint64_t)vsp[2] << 40) |
((uint64_t)vsp[3] << 32) |
((uint64_t)vsp[4] << 24) |
((uint64_t)vsp[5] << 16) |
((uint64_t)vsp[6] << 8) |
((uint64_t)vsp[7]);
/* discount zero value */
if (s == 0)
goto usewwn;
/* compute length (with trailing spaces removed) */
vlen = scsi_ascii_inquiry_len(
srmir->srmir_vendor_identification,
sizeof (srmir->srmir_vendor_identification));
plen = scsi_ascii_inquiry_len(
srmir->srmir_product_identification,
sizeof (srmir->srmir_product_identification));
slen = snprintf(sbuf, sizeof (sbuf), "%016" PRIx64, s);
if ((vlen <= 0) || (plen <= 0) || ((slen + 1) != sizeof (sbuf)))
goto usewwn;
/* this is most like a devid formed from inquiry data */
raw_id_type = DEVID_SCSI_SERIAL;
raw_id_len = vlen + 1 + plen + 1 + slen;
i_devid_len = sizeof (*i_devid) +
raw_id_len - sizeof (i_devid->did_id);
if ((i_devid = DEVID_MALLOC(i_devid_len)) == NULL)
return (DEVID_FAILURE);
bzero(i_devid, i_devid_len);
/* copy the vid to the beginning */
bcopy(&srmir->srmir_vendor_identification,
&i_devid->did_id[0], vlen);
i_devid->did_id[vlen] = '.';
/* copy the pid after the "vid." */
bcopy(&srmir->srmir_product_identification,
&i_devid->did_id[vlen + 1], plen);
i_devid->did_id[vlen + 1 + plen] = '.';
/* place the 'devid serial number' buffer the "vid.pid." */
bcopy(sbuf, &i_devid->did_id[vlen + 1 + plen + 1], slen);
} else {
usewwn: raw_id_type = DEVID_SCSI3_WWN;
raw_id_len = sizeof (wwn);
i_devid_len = sizeof (*i_devid) +
raw_id_len - sizeof (i_devid->did_id);
if ((i_devid = DEVID_MALLOC(i_devid_len)) == NULL)
return (DEVID_FAILURE);
bzero(i_devid, i_devid_len);
/* binary devid stores wwn bytes in big-endian order */
for (i = 0; i < sizeof (wwn); i++)
i_devid->did_id[i] =
(wwn >> ((sizeof (wwn) * 8) -
((i + 1) * 8))) & 0xFF;
}
i_devid->did_magic_hi = DEVID_MAGIC_MSB;
i_devid->did_magic_lo = DEVID_MAGIC_LSB;
i_devid->did_rev_hi = DEVID_REV_MSB;
i_devid->did_rev_lo = DEVID_REV_LSB;
DEVID_FORMTYPE(i_devid, raw_id_type);
DEVID_FORMLEN(i_devid, raw_id_len);
/* fill in driver name hint */
bzero(i_devid->did_driver, DEVID_HINT_SIZE);
if (driver_name != NULL) {
driver_name_len = strlen(driver_name);
if (driver_name_len > DEVID_HINT_SIZE) {
/* pick up last four characters of driver name */
driver_name += driver_name_len - DEVID_HINT_SIZE;
driver_name_len = DEVID_HINT_SIZE;
}
bcopy(driver_name, i_devid->did_driver, driver_name_len);
}
/* return device id */
*devid = (ddi_devid_t)i_devid;
return (DEVID_SUCCESS);
}