|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Copyright (c) 2002-2013 Mellanox Technologies LTD. All rights reserved.
|
|
Packit Service |
54dbc3 |
* Copyright (c) 2004-2009 Voltaire, Inc. All rights reserved.
|
|
Packit Service |
54dbc3 |
* Copyright (c) 1996-2003 Intel Corporation. All rights reserved.
|
|
Packit Service |
54dbc3 |
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
|
Packit Service |
54dbc3 |
*
|
|
Packit Service |
54dbc3 |
* This software is available to you under a choice of one of two
|
|
Packit Service |
54dbc3 |
* licenses. You may choose to be licensed under the terms of the GNU
|
|
Packit Service |
54dbc3 |
* General Public License (GPL) Version 2, available from the file
|
|
Packit Service |
54dbc3 |
* COPYING in the main directory of this source tree, or the
|
|
Packit Service |
54dbc3 |
* OpenIB.org BSD license below:
|
|
Packit Service |
54dbc3 |
*
|
|
Packit Service |
54dbc3 |
* Redistribution and use in source and binary forms, with or
|
|
Packit Service |
54dbc3 |
* without modification, are permitted provided that the following
|
|
Packit Service |
54dbc3 |
* conditions are met:
|
|
Packit Service |
54dbc3 |
*
|
|
Packit Service |
54dbc3 |
* - Redistributions of source code must retain the above
|
|
Packit Service |
54dbc3 |
* copyright notice, this list of conditions and the following
|
|
Packit Service |
54dbc3 |
* disclaimer.
|
|
Packit Service |
54dbc3 |
*
|
|
Packit Service |
54dbc3 |
* - Redistributions in binary form must reproduce the above
|
|
Packit Service |
54dbc3 |
* copyright notice, this list of conditions and the following
|
|
Packit Service |
54dbc3 |
* disclaimer in the documentation and/or other materials
|
|
Packit Service |
54dbc3 |
* provided with the distribution.
|
|
Packit Service |
54dbc3 |
*
|
|
Packit Service |
54dbc3 |
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
Packit Service |
54dbc3 |
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
Packit Service |
54dbc3 |
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
Packit Service |
54dbc3 |
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
Packit Service |
54dbc3 |
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
Packit Service |
54dbc3 |
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
Packit Service |
54dbc3 |
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
Packit Service |
54dbc3 |
* SOFTWARE.
|
|
Packit Service |
54dbc3 |
*
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Abstract:
|
|
Packit Service |
54dbc3 |
* Implementation of osm_sm_state_mgr_t.
|
|
Packit Service |
54dbc3 |
* This file implements the SM State Manager object.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
#if HAVE_CONFIG_H
|
|
Packit Service |
54dbc3 |
# include <config.h>
|
|
Packit Service |
54dbc3 |
#endif /* HAVE_CONFIG_H */
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
#include <string.h>
|
|
Packit Service |
54dbc3 |
#include <time.h>
|
|
Packit Service |
54dbc3 |
#include <iba/ib_types.h>
|
|
Packit Service |
54dbc3 |
#include <complib/cl_passivelock.h>
|
|
Packit Service |
54dbc3 |
#include <complib/cl_debug.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_file_ids.h>
|
|
Packit Service |
54dbc3 |
#define FILE_ID OSM_FILE_SM_STATE_MGR_C
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_sm.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_madw.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_switch.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_log.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_subnet.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_helper.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_msgdef.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_node.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_port.h>
|
|
Packit Service |
54dbc3 |
#include <vendor/osm_vendor_api.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_helper.h>
|
|
Packit Service |
54dbc3 |
#include <opensm/osm_opensm.h>
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
void osm_report_sm_state(osm_sm_t * sm)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
char buf[64];
|
|
Packit Service |
54dbc3 |
const char *state_str = osm_get_sm_mgr_state_str(sm->p_subn->sm_state);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
osm_log_v2(sm->p_log, OSM_LOG_SYS, FILE_ID, "Entering %s state\n", state_str);
|
|
Packit Service |
54dbc3 |
snprintf(buf, sizeof(buf), "ENTERING SM %s STATE", state_str);
|
|
Packit Service |
54dbc3 |
OSM_LOG_MSG_BOX(sm->p_log, OSM_LOG_VERBOSE, buf);
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
static boolean_t sm_state_mgr_send_master_sm_info_req(osm_sm_t * sm, uint8_t sm_state)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
osm_madw_context_t context;
|
|
Packit Service |
54dbc3 |
const osm_port_t *p_port;
|
|
Packit Service |
54dbc3 |
ib_api_status_t status;
|
|
Packit Service |
54dbc3 |
osm_dr_path_t dr_path;
|
|
Packit Service |
54dbc3 |
ib_net64_t guid;
|
|
Packit Service |
54dbc3 |
boolean_t sent_req = FALSE;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_ENTER(sm->p_log);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
memset(&context, 0, sizeof(context));
|
|
Packit Service |
54dbc3 |
if (sm_state == IB_SMINFO_STATE_STANDBY) {
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* We are in STANDBY state - this means we need to poll the
|
|
Packit Service |
54dbc3 |
* master SM (according to master_guid).
|
|
Packit Service |
54dbc3 |
* Send a query of SubnGet(SMInfo) to the subn
|
|
Packit Service |
54dbc3 |
* master_sm_base_lid object.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
guid = sm->master_sm_guid;
|
|
Packit Service |
54dbc3 |
} else {
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* We are not in STANDBY - this means we are in MASTER state -
|
|
Packit Service |
54dbc3 |
* so we need to poll the SM that is saved in polling_sm_guid
|
|
Packit Service |
54dbc3 |
* under sm.
|
|
Packit Service |
54dbc3 |
* Send a query of SubnGet(SMInfo) to that SM.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
guid = sm->polling_sm_guid;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/* Verify that SM is not polling itself */
|
|
Packit Service |
54dbc3 |
if (guid == sm->p_subn->sm_port_guid) {
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
|
|
Packit Service |
54dbc3 |
"OpenSM doesn't poll itself\n");
|
|
Packit Service |
54dbc3 |
goto Exit;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
p_port = osm_get_port_by_guid(sm->p_subn, guid);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
if (p_port == NULL) {
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3203: "
|
|
Packit Service |
54dbc3 |
"No port object for GUID 0x%016" PRIx64 "\n",
|
|
Packit Service |
54dbc3 |
cl_ntoh64(guid));
|
|
Packit Service |
54dbc3 |
goto Exit;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
context.smi_context.port_guid = guid;
|
|
Packit Service |
54dbc3 |
context.smi_context.set_method = FALSE;
|
|
Packit Service |
54dbc3 |
memcpy(&dr_path, osm_physp_get_dr_path_ptr(p_port->p_physp), sizeof(osm_dr_path_t));
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
status = osm_req_get(sm, &dr_path,
|
|
Packit Service |
54dbc3 |
IB_MAD_ATTR_SM_INFO, 0, FALSE,
|
|
Packit Service |
54dbc3 |
ib_port_info_get_m_key(&p_port->p_physp->port_info),
|
|
Packit Service |
54dbc3 |
0, CL_DISP_MSGID_NONE, &context);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
if (status != IB_SUCCESS)
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3204: "
|
|
Packit Service |
54dbc3 |
"Failure requesting SMInfo (%s)\n",
|
|
Packit Service |
54dbc3 |
ib_get_err_str(status));
|
|
Packit Service |
54dbc3 |
else
|
|
Packit Service |
54dbc3 |
sent_req = TRUE;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
Exit:
|
|
Packit Service |
54dbc3 |
OSM_LOG_EXIT(sm->p_log);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
return (sent_req);
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
static void sm_state_mgr_start_polling(osm_sm_t * sm)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
uint32_t timeout;
|
|
Packit Service |
54dbc3 |
cl_status_t cl_status;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_ENTER(sm->p_log);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Init the retry_number back to zero - need to restart counting
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->retry_number = 0;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Send a SubnGet(SMInfo) query to the current (or new) master found.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
CL_PLOCK_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
timeout = sm->p_subn->opt.sminfo_polling_timeout;
|
|
Packit Service |
54dbc3 |
sm_state_mgr_send_master_sm_info_req(sm, sm->p_subn->sm_state);
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Start a timer that will wake up every sminfo_polling_timeout milliseconds.
|
|
Packit Service |
54dbc3 |
* The callback of the timer will send a SubnGet(SMInfo) to the Master SM
|
|
Packit Service |
54dbc3 |
* and restart the timer
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
cl_status = cl_timer_start(&sm->polling_timer, timeout);
|
|
Packit Service |
54dbc3 |
if (cl_status != CL_SUCCESS)
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3210: "
|
|
Packit Service |
54dbc3 |
"Failed to start polling timer\n");
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_EXIT(sm->p_log);
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
void osm_sm_state_mgr_polling_callback(IN void *context)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
osm_sm_t *sm = context;
|
|
Packit Service |
54dbc3 |
uint32_t timeout;
|
|
Packit Service |
54dbc3 |
cl_status_t cl_status;
|
|
Packit Service |
54dbc3 |
uint8_t sm_state;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_ENTER(sm->p_log);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
cl_spinlock_acquire(&sm->state_lock);
|
|
Packit Service |
54dbc3 |
sm_state = sm->p_subn->sm_state;
|
|
Packit Service |
54dbc3 |
cl_spinlock_release(&sm->state_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
CL_PLOCK_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
timeout = sm->p_subn->opt.sminfo_polling_timeout;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* We can be here in one of two cases:
|
|
Packit Service |
54dbc3 |
* 1. We are a STANDBY sm polling on the master SM.
|
|
Packit Service |
54dbc3 |
* 2. We are a MASTER sm, waiting for a handover from a remote master sm.
|
|
Packit Service |
54dbc3 |
* If we are not in one of these cases - don't need to restart the poller.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
if (!((sm_state == IB_SMINFO_STATE_MASTER &&
|
|
Packit Service |
54dbc3 |
sm->polling_sm_guid != 0) ||
|
|
Packit Service |
54dbc3 |
sm_state == IB_SMINFO_STATE_STANDBY)) {
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
goto Exit;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* If we are a STANDBY sm and the osm_exit_flag is set, then let's
|
|
Packit Service |
54dbc3 |
* signal the subnet_up. This is relevant for the case of running only
|
|
Packit Service |
54dbc3 |
* once. In that case - the program is stuck until this signal is
|
|
Packit Service |
54dbc3 |
* received. In other cases - it is not relevant whether or not the
|
|
Packit Service |
54dbc3 |
* signal is on - since we are currently in exit flow
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
if (sm_state == IB_SMINFO_STATE_STANDBY && osm_exit_flag) {
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
|
|
Packit Service |
54dbc3 |
"Signalling subnet_up_event\n");
|
|
Packit Service |
54dbc3 |
cl_event_signal(&sm->subnet_up_event);
|
|
Packit Service |
54dbc3 |
goto Exit;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* If retry number reached the max_retry_number in the subnet opt - call
|
|
Packit Service |
54dbc3 |
* osm_sm_state_mgr_process with signal OSM_SM_SIGNAL_POLLING_TIMEOUT
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_VERBOSE, "SM State %d (%s), Retry number:%d\n",
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state, osm_get_sm_mgr_state_str(sm->p_subn->sm_state),
|
|
Packit Service |
54dbc3 |
sm->retry_number);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
if (sm->retry_number > sm->p_subn->opt.polling_retry_number) {
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_DEBUG,
|
|
Packit Service |
54dbc3 |
"Reached polling_retry_number value in retry_number. "
|
|
Packit Service |
54dbc3 |
"Go to DISCOVERY state\n");
|
|
Packit Service |
54dbc3 |
osm_sm_state_mgr_process(sm, OSM_SM_SIGNAL_POLLING_TIMEOUT);
|
|
Packit Service |
54dbc3 |
goto Exit;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/* Send a SubnGet(SMInfo) request to the remote sm (depends on our state) */
|
|
Packit Service |
54dbc3 |
if (sm_state_mgr_send_master_sm_info_req(sm, sm_state)) {
|
|
Packit Service |
54dbc3 |
/* Request sent, increment the retry number */
|
|
Packit Service |
54dbc3 |
sm->retry_number++;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/* restart the timer */
|
|
Packit Service |
54dbc3 |
cl_status = cl_timer_start(&sm->polling_timer, timeout);
|
|
Packit Service |
54dbc3 |
if (cl_status != CL_SUCCESS)
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3211: "
|
|
Packit Service |
54dbc3 |
"Failed to restart polling timer\n");
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
Exit:
|
|
Packit Service |
54dbc3 |
OSM_LOG_EXIT(sm->p_log);
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
static void sm_state_mgr_signal_error(osm_sm_t * sm, IN osm_sm_signal_t signal)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3207: "
|
|
Packit Service |
54dbc3 |
"Invalid signal %s in state %s\n",
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_signal_str(signal),
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_state_str(sm->p_subn->sm_state));
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
void osm_sm_state_mgr_signal_master_is_alive(osm_sm_t * sm)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
OSM_LOG_ENTER(sm->p_log);
|
|
Packit Service |
54dbc3 |
sm->retry_number = 0;
|
|
Packit Service |
54dbc3 |
OSM_LOG_EXIT(sm->p_log);
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
ib_api_status_t osm_sm_state_mgr_process(osm_sm_t * sm,
|
|
Packit Service |
54dbc3 |
IN osm_sm_signal_t signal)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
ib_api_status_t status = IB_SUCCESS;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
CL_ASSERT(sm);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_ENTER(sm->p_log);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* The state lock prevents many race conditions from screwing
|
|
Packit Service |
54dbc3 |
* up the state transition process.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
cl_spinlock_acquire(&sm->state_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_DEBUG,
|
|
Packit Service |
54dbc3 |
"Received signal %s in state %s\n",
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_signal_str(signal),
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_state_str(sm->p_subn->sm_state));
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
switch (sm->p_subn->sm_state) {
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_DISCOVERING:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISCOVERY_COMPLETED:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Update the state of the SM to MASTER
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
/* Turn on the first_time_master_sweep flag */
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_MASTER;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Make sure to set the subnet master_sm_base_lid
|
|
Packit Service |
54dbc3 |
* to the sm_base_lid value
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
CL_PLOCK_EXCL_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
sm->p_subn->first_time_master_sweep = TRUE;
|
|
Packit Service |
54dbc3 |
sm->p_subn->master_sm_base_lid =
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_base_lid;
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_MASTER_OR_HIGHER_SM_DETECTED:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Finished all discovery actions - move to STANDBY
|
|
Packit Service |
54dbc3 |
* start the polling
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_STANDBY;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Since another SM is doing the LFT config - we should not
|
|
Packit Service |
54dbc3 |
* ignore the results of it
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
CL_PLOCK_EXCL_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
sm->p_subn->ignore_existing_lfts = FALSE;
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
sm_state_mgr_start_polling(sm);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Signal for a new sweep. We need to discover the other SM.
|
|
Packit Service |
54dbc3 |
* If we already discovered this SM, and got the
|
|
Packit Service |
54dbc3 |
* HANDOVER - this means the remote SM is of lower priority.
|
|
Packit Service |
54dbc3 |
* In this case we will stop polling it (since it is a lower
|
|
Packit Service |
54dbc3 |
* priority SM in STANDBY state).
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
osm_sm_signal(sm, OSM_SIGNAL_SWEEP);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_STANDBY:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_POLLING_TIMEOUT:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISCOVER:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* case 1: Polling timeout occured - this means that the Master SM
|
|
Packit Service |
54dbc3 |
* is no longer alive.
|
|
Packit Service |
54dbc3 |
* case 2: Got a signal to move to DISCOVERING
|
|
Packit Service |
54dbc3 |
* Move to DISCOVERING state and start sweeping
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_DISCOVERING;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
CL_PLOCK_EXCL_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
sm->p_subn->coming_out_of_standby = TRUE;
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
osm_sm_signal(sm, OSM_SIGNAL_SWEEP);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISABLE:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Update the state to NOT_ACTIVE
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_NOTACTIVE;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Update the state to MASTER, and start sweeping
|
|
Packit Service |
54dbc3 |
* OPTIONAL: send ACKNOWLEDGE
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
/* Turn on the force_first_time_master_sweep flag */
|
|
Packit Service |
54dbc3 |
/* We want full reconfiguration to occur on the first */
|
|
Packit Service |
54dbc3 |
/* master sweep of this SM */
|
|
Packit Service |
54dbc3 |
CL_PLOCK_EXCL_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Make sure to set the subnet master_sm_base_lid
|
|
Packit Service |
54dbc3 |
* to the sm_base_lid value
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->p_subn->master_sm_base_lid =
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_base_lid;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
sm->p_subn->force_first_time_master_sweep = TRUE;
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_MASTER;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
osm_sm_signal(sm, OSM_SIGNAL_SWEEP);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_ACKNOWLEDGE:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Do nothing - already moved to STANDBY
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_NOTACTIVE:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_STANDBY:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Update the state to STANDBY
|
|
Packit Service |
54dbc3 |
* start the polling
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_STANDBY;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
sm_state_mgr_start_polling(sm);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_MASTER:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_POLLING_TIMEOUT:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* We received a polling timeout - this means that we
|
|
Packit Service |
54dbc3 |
* waited for a remote master sm to send us a handover,
|
|
Packit Service |
54dbc3 |
* but didn't get it, and didn't get a response from
|
|
Packit Service |
54dbc3 |
* that remote sm.
|
|
Packit Service |
54dbc3 |
* We want to force a heavy sweep - hopefully this
|
|
Packit Service |
54dbc3 |
* occurred because the remote sm died, and we'll find
|
|
Packit Service |
54dbc3 |
* this out and configure the subnet after a heavy sweep.
|
|
Packit Service |
54dbc3 |
* We also want to clear the polling_sm_guid - since
|
|
Packit Service |
54dbc3 |
* we are done polling on that remote sm - we are
|
|
Packit Service |
54dbc3 |
* sweeping again.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* If we received a handover in a master state - then we
|
|
Packit Service |
54dbc3 |
* want to force a heavy sweep. This means that either
|
|
Packit Service |
54dbc3 |
* we are in a sweep currently - in this case - no
|
|
Packit Service |
54dbc3 |
* change, or we are in idle state - since we
|
|
Packit Service |
54dbc3 |
* recognized a master SM before - so we want to make a
|
|
Packit Service |
54dbc3 |
* heavy sweep and reconfigure the new subnet.
|
|
Packit Service |
54dbc3 |
* We also want to clear the polling_sm_guid - since
|
|
Packit Service |
54dbc3 |
* we are done polling on that remote sm - we got a
|
|
Packit Service |
54dbc3 |
* handover from it.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
|
|
Packit Service |
54dbc3 |
"Forcing heavy sweep. Received signal %s\n",
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_signal_str(signal));
|
|
Packit Service |
54dbc3 |
CL_PLOCK_EXCL_ACQUIRE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
sm->polling_sm_guid = 0;
|
|
Packit Service |
54dbc3 |
sm->p_subn->force_first_time_master_sweep = TRUE;
|
|
Packit Service |
54dbc3 |
CL_PLOCK_RELEASE(sm->p_lock);
|
|
Packit Service |
54dbc3 |
osm_sm_signal(sm, OSM_SIGNAL_SWEEP);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER_SENT:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* Just sent a HANDOVER signal - move to STANDBY
|
|
Packit Service |
54dbc3 |
* start the polling
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_STANDBY;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
sm_state_mgr_start_polling(sm);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_WAIT_FOR_HANDOVER:
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* We found a remote master SM, and we are waiting for
|
|
Packit Service |
54dbc3 |
* it to handover the mastership to us. Need to start
|
|
Packit Service |
54dbc3 |
* polling that SM, to make sure it is alive, if it
|
|
Packit Service |
54dbc3 |
* isn't - then we should move back to discovering,
|
|
Packit Service |
54dbc3 |
* since something must have happened to it.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
sm_state_mgr_start_polling(sm);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISCOVER:
|
|
Packit Service |
54dbc3 |
sm->p_subn->sm_state = IB_SMINFO_STATE_DISCOVERING;
|
|
Packit Service |
54dbc3 |
osm_report_sm_state(sm);
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3208: "
|
|
Packit Service |
54dbc3 |
"Invalid state %s\n",
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_state_str(sm->p_subn->sm_state));
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
cl_spinlock_release(&sm->state_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_EXIT(sm->p_log);
|
|
Packit Service |
54dbc3 |
return status;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
ib_api_status_t osm_sm_state_mgr_check_legality(osm_sm_t * sm,
|
|
Packit Service |
54dbc3 |
IN osm_sm_signal_t signal)
|
|
Packit Service |
54dbc3 |
{
|
|
Packit Service |
54dbc3 |
ib_api_status_t status = IB_SUCCESS;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
CL_ASSERT(sm);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_ENTER(sm->p_log);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
/*
|
|
Packit Service |
54dbc3 |
* The state lock prevents many race conditions from screwing
|
|
Packit Service |
54dbc3 |
* up the state transition process.
|
|
Packit Service |
54dbc3 |
*/
|
|
Packit Service |
54dbc3 |
cl_spinlock_acquire(&sm->state_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_DEBUG, "Received signal %s in state %s\n",
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_signal_str(signal),
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_state_str(sm->p_subn->sm_state));
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
switch (sm->p_subn->sm_state) {
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_DISCOVERING:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISCOVERY_COMPLETED:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_MASTER_OR_HIGHER_SM_DETECTED:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER:
|
|
Packit Service |
54dbc3 |
status = IB_SUCCESS;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_STANDBY:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_POLLING_TIMEOUT:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISCOVER:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_DISABLE:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_ACKNOWLEDGE:
|
|
Packit Service |
54dbc3 |
status = IB_SUCCESS;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_NOTACTIVE:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_STANDBY:
|
|
Packit Service |
54dbc3 |
status = IB_SUCCESS;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
case IB_SMINFO_STATE_MASTER:
|
|
Packit Service |
54dbc3 |
switch (signal) {
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER:
|
|
Packit Service |
54dbc3 |
case OSM_SM_SIGNAL_HANDOVER_SENT:
|
|
Packit Service |
54dbc3 |
status = IB_SUCCESS;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
sm_state_mgr_signal_error(sm, signal);
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
break;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
default:
|
|
Packit Service |
54dbc3 |
OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3209: "
|
|
Packit Service |
54dbc3 |
"Invalid state %s\n",
|
|
Packit Service |
54dbc3 |
osm_get_sm_mgr_state_str(sm->p_subn->sm_state));
|
|
Packit Service |
54dbc3 |
status = IB_INVALID_PARAMETER;
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
}
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
cl_spinlock_release(&sm->state_lock);
|
|
Packit Service |
54dbc3 |
|
|
Packit Service |
54dbc3 |
OSM_LOG_EXIT(sm->p_log);
|
|
Packit Service |
54dbc3 |
return status;
|
|
Packit Service |
54dbc3 |
}
|