ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim/*
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * CDDL HEADER START
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim *
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * The contents of this file are subject to the terms of the
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * Common Development and Distribution License (the "License").
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * You may not use this file except in compliance with the License.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim *
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * or http://www.opensolaris.org/os/licensing.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * See the License for the specific language governing permissions
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * and limitations under the License.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim *
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * When distributing Covered Code, include this CDDL HEADER in each
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * If applicable, add the following below this CDDL HEADER, with the
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * fields enclosed by brackets "[]" replaced with your own identifying
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * information: Portions Copyright [yyyy] [name of copyright owner]
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim *
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * CDDL HEADER END
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim */
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim/*
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * Copyright (c) 2010, Oracle and/or its affiliates. All rights reserved.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim */
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <sys/scsi/impl/spc3_types.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <stddef.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <string.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <strings.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <libnvpair.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/libses.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/libses_plugin.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/plugins/ses/framework/ses2.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/plugins/ses/framework/ses2_impl.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/plugins/ses/framework/libses.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/plugins/ses/vendor/sun.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim#include <scsi/plugins/ses/vendor/sun_impl.h>
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim/*ARGSUSED*/
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kimstatic int
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kimenc_parse_feature_block(ses_plugin_t *sp, ses_node_t *np)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim{
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim sun_feature_block_impl_t *sfbip;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim nvlist_t *encprops;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim uint8_t *vsp;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim uint_t vsp_len;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim uint_t cid_off, cid_len;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim uint16_t revision;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim uint64_t chunk;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim int nverr;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim encprops = ses_node_props(np);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (nvlist_lookup_byte_array(encprops, SES_EN_PROP_VS,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim &vsp, &vsp_len) != 0 ||
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim vsp_len < offsetof(sun_feature_block_impl_t, _reserved2))
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim return (0);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim sfbip = (sun_feature_block_impl_t *)vsp;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (strncmp((char *)sfbip->sfbi_spms_header, "SPMS", 4) != 0 ||
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim sfbip->sfbi_spms_major_ver != 1)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim return (0);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim revision = SCSI_READ16(&sfbip->sfbi_spms_revision);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim /*
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * The offset read from the Sun Feature Block needs to be adjusted
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * so that the difference in the sizes of the Enclosure
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * Descriptor and the INQUIRY data format is accounted for.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim */
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim cid_len = sfbip->sfbi_chassis_id_len;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (sfbip->sfbi_chassis_id_off >= 96 && cid_len >= 4) {
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim cid_off = sfbip->sfbi_chassis_id_off -
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim (sizeof (ses2_ed_impl_t) - 1);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim cid_off += offsetof(ses2_ed_impl_t, st_priv[0]) -
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim offsetof(spc3_inquiry_data_t, id_vs_36[0]);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (cid_off + cid_len <= vsp_len) {
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim SES_NV_ADD(fixed_string, nverr, encprops,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim LIBSES_EN_PROP_CSN, (char *)(vsp + cid_off),
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim cid_len);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim }
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim }
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (revision >= 104) {
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim SES_NV_ADD(boolean_value, nverr, encprops,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim LIBSES_EN_PROP_INTERNAL, sfbip->sfbi_int);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim }
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (revision >= 105) {
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (sfbip->sfbi_fw_upload_max_chunk_sz == 0)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim chunk = 512;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim else if (sfbip->sfbi_fw_upload_max_chunk_sz == 0x7f)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim chunk = 65536;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim else
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim chunk = 512 * sfbip->sfbi_fw_upload_max_chunk_sz;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim SES_NV_ADD(uint64, nverr, encprops,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim LIBSES_EN_PROP_FIRMWARE_CHUNK_SIZE, chunk);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim }
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim /*
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * If this is a subchassis, it will have a subchassis index field
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * with a value other than 0. See SPMS-1r111 4.1.3.1. If not, we
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * will see 0 and will not create the subchassis member at all; note
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * that this is backward-compatible with pre-111 implementations that
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * treated this as a reserved field. No such implementation contains
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim * a subchassis.
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim */
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if (sfbip->sfbi_spms_revision >= 111 &&
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim sfbip->sfbi_subchassis_index != 0) {
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim SES_NV_ADD(uint64, nverr, encprops,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim LIBSES_EN_PROP_SUBCHASSIS_ID,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim sfbip->sfbi_subchassis_index - 1);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim }
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim return (0);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim}
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kimint
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kimsun_fill_enclosure_node(ses_plugin_t *sp, ses_node_t *np)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim{
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim ses_snap_t *snap = ses_node_snapshot(np);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim nvlist_t *props = ses_node_props(np);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim sun_fru_descr_impl_t *sfdi;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim int err;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim size_t len;
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if ((err = enc_parse_feature_block(sp, np)) != 0)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim return (err);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if ((sfdi = ses_plugin_page_lookup(sp, snap,
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim SUN_DIAGPAGE_FRUID, np, &len)) != NULL) {
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim if ((err = sun_fruid_parse_common(sfdi, props)) != 0)
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim return (err);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim }
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim return (0);
ac88567a7a5bb7f01cf22cf366bc9d6203e24d7aHyon Kim}