9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim/*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * CDDL HEADER START
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * The contents of this file are subject to the terms of the
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Common Development and Distribution License (the "License").
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * You may not use this file except in compliance with the License.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * or http://www.opensolaris.org/os/licensing.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * See the License for the specific language governing permissions
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * and limitations under the License.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * When distributing Covered Code, include this CDDL HEADER in each
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * If applicable, add the following below this CDDL HEADER, with the
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * fields enclosed by brackets "[]" replaced with your own identifying
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * information: Portions Copyright [yyyy] [name of copyright owner]
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * CDDL HEADER END
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim/*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Use is subject to license terms.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim#include <sun_sas.h>
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim/*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Discover an HBA node with matching path.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * The di_node_t argument should be the root of the device tree.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * This routine assumes the locks have been taken
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kimstatic int
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kimfind_matching_hba(di_node_t node, void *arg)
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim{
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim int *propData, rval;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim walkarg_t *wa = (walkarg_t *)arg;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim char *devpath, fulldevpath[MAXPATHLEN];
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /* Skip stub(instance -1) nodes */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if (IS_STUB_NODE(node)) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return (DI_WALK_CONTINUE);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim rval = di_prop_lookup_ints(DDI_DEV_T_ANY, node,
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim "sm-hba-supported", &propData);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if (rval < 0) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return (DI_WALK_CONTINUE);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim } else {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if ((devpath = di_devfs_path(node)) == NULL) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /* still continue to see if there is matching one. */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return (DI_WALK_CONTINUE);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim (void) snprintf(fulldevpath, MAXPATHLEN, "%s%s", DEVICES_DIR,
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim devpath);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if ((strstr(fulldevpath, wa->devpath)) != NULL) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *wa->flag = B_TRUE;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /* Found a node. No need to walk any more. */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim di_devfs_path_free(devpath);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return (DI_WALK_TERMINATE);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim di_devfs_path_free(devpath);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return (DI_WALK_CONTINUE);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim}
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim/*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Refreshes information about an HBA
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Note: This routine holds the locks in write mode
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * during most of the processing, and as such, will cause
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * all other threads to block on entry into the library
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * until the refresh is complete. An optimization would be
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * to put fine-grain locking in for the open_handle structures.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kimvoid
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon KimSun_sasRefreshAdapterConfiguration()
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim{
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim const char ROUTINE[] =
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim "Sun_sasRefreshAdapterConfiguration";
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim struct sun_sas_hba *hba_ptr;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim di_node_t root;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim hrtime_t start;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim hrtime_t end;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim double duration;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim walkarg_t wa;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * We're going to be tweaking a lot of global lists, so write
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * lock them.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim lock(&all_hbas_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim lock(&open_handles_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim start = gethrtime();
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /* Grab device tree */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if ((root = di_init("/", DINFOCACHE)) == DI_NODE_NIL) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim log(LOG_DEBUG, ROUTINE,
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim "Unable to load device tree for reason \"%s\"",
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim strerror(errno));
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim unlock(&open_handles_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim unlock(&all_hbas_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim end = gethrtime();
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim duration = end - start;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim duration /= HR_SECOND;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim log(LOG_DEBUG, ROUTINE, "Device tree init took "
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim "%.6f seconds", duration);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim for (hba_ptr = global_hba_head; hba_ptr != NULL;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim hba_ptr = hba_ptr->next) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim wa.devpath = hba_ptr->device_path;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim wa.flag = (boolean_t *)calloc(1, sizeof (boolean_t));
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim *wa.flag = B_FALSE;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if (di_walk_node(root, DI_WALK_SIBFIRST, &wa,
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim find_matching_hba) != 0) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim log(LOG_DEBUG, ROUTINE, "di_walk_node failed.");
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim unlock(&open_handles_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim unlock(&all_hbas_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim S_FREE(wa.flag);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim di_fini(root);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim return;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if (*wa.flag == B_FALSE) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Keep the HBA as it is including open handles
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * per the SM-HBA standards. Just mark it as invalid.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim log(LOG_DEBUG, ROUTINE, "No matching HBA found. %s",
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim hba_ptr->device_path);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim hba_ptr->invalid = B_TRUE;
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim S_FREE(wa.flag);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim /*
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Now we marked missing HBA(s). Redisoover hbas to refresh
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * or add any new adapter to the hba list.
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim * Simply call devtree_get_all_hbas().
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim */
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim if (devtree_get_all_hbas(root) != HBA_STATUS_OK) {
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim log(LOG_DEBUG, ROUTINE, "devtree_get_all_hbas failed.");
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim }
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim di_fini(root);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim unlock(&open_handles_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim unlock(&all_hbas_lock);
9e86db79b7d1bbc5f2f04e99954cbd5eae0e22bbHyon Kim}