Blame opensm/osm_trap_rcv.c

Packit Service 54dbc3
/*
Packit Service 54dbc3
 * Copyright (c) 2004-2009 Voltaire, Inc. All rights reserved.
Packit Service 54dbc3
 * Copyright (c) 2002-2007 Mellanox Technologies LTD. All rights reserved.
Packit Service 54dbc3
 * Copyright (c) 1996-2003 Intel Corporation. All rights reserved.
Packit Service 54dbc3
 * Copyright (c) 2009,2010 HNR Consulting. 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_trap_rcv_t.
Packit Service 54dbc3
 * This object represents the Trap Receiver object.
Packit Service 54dbc3
 * This object is part of the opensm family of objects.
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 <iba/ib_types.h>
Packit Service 54dbc3
#include <complib/cl_qmap.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_TRAP_RCV_C
Packit Service 54dbc3
#include <opensm/osm_madw.h>
Packit Service 54dbc3
#include <opensm/osm_log.h>
Packit Service 54dbc3
#include <opensm/osm_node.h>
Packit Service 54dbc3
#include <opensm/osm_helper.h>
Packit Service 54dbc3
#include <opensm/osm_subnet.h>
Packit Service 54dbc3
#include <opensm/osm_inform.h>
Packit Service 54dbc3
#include <opensm/osm_opensm.h>
Packit Service 54dbc3
Packit Service 54dbc3
extern void osm_req_get_node_desc(IN osm_sm_t * sm, osm_physp_t *p_physp);
Packit Service 54dbc3
Packit Service 54dbc3
/**********************************************************************
Packit Service 54dbc3
 *
Packit Service 54dbc3
 * TRAP HANDLING:
Packit Service 54dbc3
 *
Packit Service 54dbc3
 * Assuming traps can be caused by bad hardware we should provide
Packit Service 54dbc3
 * a mechanism for filtering their propagation into the actual logic
Packit Service 54dbc3
 * of OpenSM such that it is not overloaded by them.
Packit Service 54dbc3
 *
Packit Service 54dbc3
 * We will provide a trap filtering mechanism with "Aging" capability.
Packit Service 54dbc3
 * This mechanism will track incoming traps, clasify them by their
Packit Service 54dbc3
 * source and content and provide back their age.
Packit Service 54dbc3
 *
Packit Service 54dbc3
 * A timer running in the background will toggle a timer counter
Packit Service 54dbc3
 * that should be referenced by the aging algorithm.
Packit Service 54dbc3
 * To provide an efficient handling of aging, we also track all traps
Packit Service 54dbc3
 * in a sorted list by their aging.
Packit Service 54dbc3
 *
Packit Service 54dbc3
 * The generic Aging Tracker mechanism is implemented in the
Packit Service 54dbc3
 * cl_aging_tracker object.
Packit Service 54dbc3
 *
Packit Service 54dbc3
 **********************************************************************/
Packit Service 54dbc3
Packit Service 54dbc3
static osm_physp_t *get_physp_by_lid_and_num(IN osm_sm_t * sm,
Packit Service 54dbc3
					     IN ib_net16_t lid, IN uint8_t num)
Packit Service 54dbc3
{
Packit Service 54dbc3
	osm_port_t *p_port = osm_get_port_by_lid(sm->p_subn, lid);
Packit Service 54dbc3
	if (!p_port)
Packit Service 54dbc3
		return NULL;
Packit Service 54dbc3
Packit Service 54dbc3
	if (osm_node_get_num_physp(p_port->p_node) <= num)
Packit Service 54dbc3
		return NULL;
Packit Service 54dbc3
Packit Service 54dbc3
	return osm_node_get_physp_ptr(p_port->p_node, num);
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
static uint64_t aging_tracker_callback(IN uint64_t key, IN uint32_t num_regs,
Packit Service 54dbc3
				       IN void *context)
Packit Service 54dbc3
{
Packit Service 54dbc3
	osm_sm_t *sm = context;
Packit Service 54dbc3
	ib_net16_t lid;
Packit Service 54dbc3
	uint8_t port_num;
Packit Service 54dbc3
	osm_physp_t *p_physp;
Packit Service 54dbc3
Packit Service 54dbc3
	OSM_LOG_ENTER(sm->p_log);
Packit Service 54dbc3
Packit Service 54dbc3
	if (osm_exit_flag)
Packit Service 54dbc3
		/* We got an exit flag - do nothing */
Packit Service 54dbc3
		return 0;
Packit Service 54dbc3
Packit Service 54dbc3
	lid = (ib_net16_t) ((key & 0x0000FFFF00000000ULL) >> 32);
Packit Service 54dbc3
	port_num = (uint8_t) ((key & 0x00FF000000000000ULL) >> 48);
Packit Service 54dbc3
Packit Service 54dbc3
	CL_PLOCK_ACQUIRE(sm->p_lock);
Packit Service 54dbc3
Packit Service 54dbc3
	p_physp = get_physp_by_lid_and_num(sm, lid, port_num);
Packit Service 54dbc3
	if (!p_physp)
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
			"Cannot find port num:%u with lid:%u\n",
Packit Service 54dbc3
			port_num, cl_ntoh16(lid));
Packit Service 54dbc3
	/* make sure the physp is still valid */
Packit Service 54dbc3
	/* If the health port was false - set it to true */
Packit Service 54dbc3
	else if (!osm_physp_is_healthy(p_physp)) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
			"Clearing health bit of port num:%u with lid:%u\n",
Packit Service 54dbc3
			port_num, cl_ntoh16(lid));
Packit Service 54dbc3
Packit Service 54dbc3
		/* Clear its health bit */
Packit Service 54dbc3
		osm_physp_set_health(p_physp, TRUE);
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	CL_PLOCK_RELEASE(sm->p_lock);
Packit Service 54dbc3
	OSM_LOG_EXIT(sm->p_log);
Packit Service 54dbc3
Packit Service 54dbc3
	/* We want to remove the event from the tracker - so
Packit Service 54dbc3
	   need to return zero. */
Packit Service 54dbc3
	return 0;
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
/**********************************************************************
Packit Service 54dbc3
 * CRC calculation for notice identification
Packit Service 54dbc3
 **********************************************************************/
Packit Service 54dbc3
Packit Service 54dbc3
#define CRC32_POLYNOMIAL   0xEDB88320L
Packit Service 54dbc3
Packit Service 54dbc3
/* calculate the crc for a given buffer */
Packit Service 54dbc3
static uint32_t trap_calc_crc32(void *buffer, uint32_t count)
Packit Service 54dbc3
{
Packit Service 54dbc3
	uint32_t temp1, temp2;
Packit Service 54dbc3
	uint32_t crc = -1L;
Packit Service 54dbc3
	unsigned char *p = (unsigned char *)buffer;
Packit Service 54dbc3
	/* precalculated table for faster crc calculation */
Packit Service 54dbc3
	static uint32_t crc_table[256];
Packit Service 54dbc3
	static boolean_t first = TRUE;
Packit Service 54dbc3
	int i, j;
Packit Service 54dbc3
Packit Service 54dbc3
	/* if we need to initialize the lookup table */
Packit Service 54dbc3
	if (first) {
Packit Service 54dbc3
		/* calc the CRC table */
Packit Service 54dbc3
		for (i = 0; i <= 255; i++) {
Packit Service 54dbc3
			crc = i;
Packit Service 54dbc3
			for (j = 8; j > 0; j--)
Packit Service 54dbc3
				if (crc & 1)
Packit Service 54dbc3
					crc = (crc >> 1) ^ CRC32_POLYNOMIAL;
Packit Service 54dbc3
				else
Packit Service 54dbc3
					crc >>= 1;
Packit Service 54dbc3
			crc_table[i] = crc;
Packit Service 54dbc3
		}
Packit Service 54dbc3
		first = FALSE;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	crc = -1L;
Packit Service 54dbc3
	/* do the calculation */
Packit Service 54dbc3
	while (count-- != 0) {
Packit Service 54dbc3
		temp1 = (crc >> 8) & 0x00FFFFFFL;
Packit Service 54dbc3
		temp2 = crc_table[((int)crc ^ *p++) & 0xFF];
Packit Service 54dbc3
		crc = temp1 ^ temp2;
Packit Service 54dbc3
	}
Packit Service 54dbc3
	return crc;
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
/* The key is created in the following manner:
Packit Service 54dbc3
   port_num  lid   crc
Packit Service 54dbc3
   \______/ \___/ \___/
Packit Service 54dbc3
     16b     16b   32b
Packit Service 54dbc3
*/
Packit Service 54dbc3
static uint64_t trap_get_key(IN uint16_t lid, IN uint8_t port_num,
Packit Service 54dbc3
			     IN ib_mad_notice_attr_t * p_ntci)
Packit Service 54dbc3
{
Packit Service 54dbc3
	uint32_t crc = trap_calc_crc32(p_ntci, sizeof(ib_mad_notice_attr_t));
Packit Service 54dbc3
	return ((uint64_t) port_num << 48) | ((uint64_t) lid << 32) | crc;
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
static int print_num_received(IN uint32_t num_received)
Packit Service 54dbc3
{
Packit Service 54dbc3
	uint32_t i;
Packit Service 54dbc3
Packit Service 54dbc3
	/* Series is 10, 20, 50, 100, 200, 500, ... */
Packit Service 54dbc3
	i = num_received;
Packit Service 54dbc3
	while (i >= 10) {
Packit Service 54dbc3
		if (i % 10)
Packit Service 54dbc3
			break;
Packit Service 54dbc3
		i = i / 10;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	if (i == 1 || i == 2 || i == 5)
Packit Service 54dbc3
		return 1;
Packit Service 54dbc3
	else
Packit Service 54dbc3
		return 0;
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
static int disable_port(osm_sm_t *sm, osm_physp_t *p)
Packit Service 54dbc3
{
Packit Service 54dbc3
	uint8_t payload[IB_SMP_DATA_SIZE];
Packit Service 54dbc3
	osm_madw_context_t context;
Packit Service 54dbc3
	ib_port_info_t *pi = (ib_port_info_t *)payload;
Packit Service 54dbc3
	osm_physp_t *physp0;
Packit Service 54dbc3
	osm_port_t *p_port;
Packit Service 54dbc3
	ib_net64_t m_key;
Packit Service 54dbc3
	ib_api_status_t status;
Packit Service 54dbc3
Packit Service 54dbc3
	/* select the nearest port to master opensm */
Packit Service 54dbc3
	if (p->p_remote_physp &&
Packit Service 54dbc3
	    p->dr_path.hop_count > p->p_remote_physp->dr_path.hop_count)
Packit Service 54dbc3
		p = p->p_remote_physp;
Packit Service 54dbc3
Packit Service 54dbc3
	/* If trap 131, might want to disable peer port if available */
Packit Service 54dbc3
	/* but peer port has been observed not to respond to SM requests */
Packit Service 54dbc3
Packit Service 54dbc3
	memcpy(payload, &p->port_info, sizeof(ib_port_info_t));
Packit Service 54dbc3
Packit Service 54dbc3
	/* Set port to disabled/down */
Packit Service 54dbc3
	ib_port_info_set_port_state(pi, IB_LINK_DOWN);
Packit Service 54dbc3
	ib_port_info_set_port_phys_state(IB_PORT_PHYS_STATE_DISABLED, pi);
Packit Service 54dbc3
Packit Service 54dbc3
	/* Issue set of PortInfo */
Packit Service 54dbc3
	context.pi_context.node_guid = osm_node_get_node_guid(p->p_node);
Packit Service 54dbc3
	context.pi_context.port_guid = osm_physp_get_port_guid(p);
Packit Service 54dbc3
	context.pi_context.set_method = TRUE;
Packit Service 54dbc3
	context.pi_context.light_sweep = FALSE;
Packit Service 54dbc3
	context.pi_context.active_transition = FALSE;
Packit Service 54dbc3
	context.pi_context.client_rereg = FALSE;
Packit Service 54dbc3
	if (osm_node_get_type(p->p_node) == IB_NODE_TYPE_SWITCH &&
Packit Service 54dbc3
	    osm_physp_get_port_num(p) != 0) {
Packit Service 54dbc3
		physp0 = osm_node_get_physp_ptr(p->p_node, 0);
Packit Service 54dbc3
		m_key = ib_port_info_get_m_key(&physp0->port_info);
Packit Service 54dbc3
	} else
Packit Service 54dbc3
		m_key = ib_port_info_get_m_key(&p->port_info);
Packit Service 54dbc3
Packit Service 54dbc3
	if (osm_node_get_type(p->p_node) != IB_NODE_TYPE_SWITCH) {
Packit Service 54dbc3
		if (!pi->base_lid) {
Packit Service 54dbc3
			p_port = osm_get_port_by_guid(sm->p_subn,
Packit Service 54dbc3
						      osm_physp_get_port_guid(p));
Packit Service 54dbc3
			if (p_port)
Packit Service 54dbc3
				pi->base_lid = p_port->lid;
Packit Service 54dbc3
			else {
Packit Service 54dbc3
				OSM_LOG(sm->p_log, OSM_LOG_ERROR,
Packit Service 54dbc3
					"ERR 3804: Port 0x%" PRIx64
Packit Service 54dbc3
					" not found, port set failed\n",
Packit Service 54dbc3
					cl_ntoh64(osm_physp_get_port_guid(p)));
Packit Service 54dbc3
				status = IB_ERROR;
Packit Service 54dbc3
				goto EXIT;
Packit Service 54dbc3
			}
Packit Service 54dbc3
		}
Packit Service 54dbc3
		pi->master_sm_base_lid = sm->p_subn->sm_base_lid;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	status = osm_req_set(sm, osm_physp_get_dr_path_ptr(p),
Packit Service 54dbc3
			   payload, sizeof(payload), IB_MAD_ATTR_PORT_INFO,
Packit Service 54dbc3
			   cl_hton32(osm_physp_get_port_num(p)),
Packit Service 54dbc3
			   FALSE, m_key,
Packit Service 54dbc3
			   0, CL_DISP_MSGID_NONE, &context);
Packit Service 54dbc3
Packit Service 54dbc3
EXIT:
Packit Service 54dbc3
	return status;
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
static void log_trap_info(osm_log_t *p_log, ib_mad_notice_attr_t *p_ntci,
Packit Service 54dbc3
			  ib_net16_t source_lid, ib_net64_t trans_id)
Packit Service 54dbc3
{
Packit Service 54dbc3
	if (!OSM_LOG_IS_ACTIVE_V2(p_log, OSM_LOG_ERROR))
Packit Service 54dbc3
		return;
Packit Service 54dbc3
Packit Service 54dbc3
	if (ib_notice_is_generic(p_ntci)) {
Packit Service 54dbc3
		char str[32];
Packit Service 54dbc3
Packit Service 54dbc3
		if ((p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_LINK_INTEGRITY_THRESHOLD_TRAP)) ||
Packit Service 54dbc3
		    (p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_BUFFER_OVERRUN_THRESHOLD_TRAP)) ||
Packit Service 54dbc3
		    (p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_WATCHDOG_TIMER_EXPIRED_TRAP)))
Packit Service 54dbc3
			snprintf(str, sizeof(str), " Port %u",
Packit Service 54dbc3
				 p_ntci->data_details.ntc_129_131.port_num);
Packit Service 54dbc3
		else
Packit Service 54dbc3
			str[0] = '\0';
Packit Service 54dbc3
Packit Service 54dbc3
		OSM_LOG(p_log, OSM_LOG_ERROR,
Packit Service 54dbc3
			"Received Generic Notice type:%u "
Packit Service 54dbc3
			"num:%u (%s) Producer:%u (%s) "
Packit Service 54dbc3
			"from LID:%u%s TID:0x%016" PRIx64 "\n",
Packit Service 54dbc3
			ib_notice_get_type(p_ntci),
Packit Service 54dbc3
			cl_ntoh16(p_ntci->g_or_v.generic.trap_num),
Packit Service 54dbc3
			ib_get_trap_str(p_ntci->g_or_v.generic.trap_num),
Packit Service 54dbc3
			cl_ntoh32(ib_notice_get_prod_type(p_ntci)),
Packit Service 54dbc3
			ib_get_producer_type_str(ib_notice_get_prod_type(p_ntci)),
Packit Service 54dbc3
			cl_hton16(source_lid), str, cl_ntoh64(trans_id));
Packit Service 54dbc3
		if ((p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_BAD_PKEY_TRAP)) ||
Packit Service 54dbc3
		    (p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_BAD_QKEY_TRAP))) {
Packit Service 54dbc3
			OSM_LOG(p_log, OSM_LOG_ERROR,
Packit Service 54dbc3
				"Bad %s_Key:0x%x on SL:%d from "
Packit Service 54dbc3
				"LID1:%u QP1:0x%x to "
Packit Service 54dbc3
				"LID2:%u QP2:0x%x\n",
Packit Service 54dbc3
				(p_ntci->g_or_v.generic.trap_num == CL_HTON16(257)) ? "P" : "Q",
Packit Service 54dbc3
				cl_ntoh32(p_ntci->data_details.ntc_257_258.key),
Packit Service 54dbc3
				cl_ntoh32(p_ntci->data_details.ntc_257_258.qp1) >> 28,
Packit Service 54dbc3
				cl_ntoh16(p_ntci->data_details.ntc_257_258.lid1),
Packit Service 54dbc3
				cl_ntoh32(p_ntci->data_details.ntc_257_258.qp1) & 0xfff,
Packit Service 54dbc3
				cl_ntoh16(p_ntci->data_details.ntc_257_258.lid2),
Packit Service 54dbc3
				cl_ntoh32(p_ntci->data_details.ntc_257_258.qp2));
Packit Service 54dbc3
		}
Packit Service 54dbc3
	} else
Packit Service 54dbc3
		OSM_LOG(p_log, OSM_LOG_ERROR,
Packit Service 54dbc3
			"Received Vendor Notice type:%u vend:0x%06X "
Packit Service 54dbc3
			"dev:%u from LID:%u TID:0x%016" PRIx64 "\n",
Packit Service 54dbc3
			ib_notice_get_type(p_ntci),
Packit Service 54dbc3
			cl_ntoh32(ib_notice_get_vend_id(p_ntci)),
Packit Service 54dbc3
			cl_ntoh16(p_ntci->g_or_v.vend.dev_id),
Packit Service 54dbc3
			cl_ntoh16(source_lid), cl_ntoh64(trans_id));
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
static int shutup_noisy_port(osm_sm_t *sm, ib_net16_t lid, uint8_t port,
Packit Service 54dbc3
			     unsigned num)
Packit Service 54dbc3
{
Packit Service 54dbc3
	osm_physp_t *p = get_physp_by_lid_and_num(sm, lid, port);
Packit Service 54dbc3
	if (!p) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3805: "
Packit Service 54dbc3
			"Failed to find physical port by lid:%u num:%u\n",
Packit Service 54dbc3
			cl_ntoh16(lid), port);
Packit Service 54dbc3
		return -1;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/* When babbling port policy option is enabled and
Packit Service 54dbc3
	   Threshold for disabling a "babbling" port is exceeded */
Packit Service 54dbc3
	if (sm->p_subn->opt.babbling_port_policy && num >= 250) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
			"Disabling noisy physical port 0x%016" PRIx64
Packit Service 54dbc3
			": lid %u, num %u\n",
Packit Service 54dbc3
			cl_ntoh64(osm_physp_get_port_guid(p)),
Packit Service 54dbc3
			cl_ntoh16(lid), port);
Packit Service 54dbc3
		if (disable_port(sm, p))
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3811: "
Packit Service 54dbc3
				"Failed to disable noisy physical port 0x%016"
Packit Service 54dbc3
				PRIx64 ": lid %u, num %u\n",
Packit Service 54dbc3
				cl_ntoh64(osm_physp_get_port_guid(p)),
Packit Service 54dbc3
				cl_ntoh16(lid), port);
Packit Service 54dbc3
		else
Packit Service 54dbc3
			return 1;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/* check if the current state of the p_physp is healthy. If
Packit Service 54dbc3
	   it is - then this is a first change of state. Run a heavy sweep. */
Packit Service 54dbc3
	if (osm_physp_is_healthy(p)) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
			"Marking unhealthy physical port by lid:%u num:%u\n",
Packit Service 54dbc3
			cl_ntoh16(lid), port);
Packit Service 54dbc3
		osm_physp_set_health(p, FALSE);
Packit Service 54dbc3
		return 2;
Packit Service 54dbc3
	}
Packit Service 54dbc3
	return 0;
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
static void trap_rcv_process_request(IN osm_sm_t * sm,
Packit Service 54dbc3
				     IN const osm_madw_t * p_madw)
Packit Service 54dbc3
{
Packit Service 54dbc3
	uint8_t payload[sizeof(ib_mad_notice_attr_t)];
Packit Service 54dbc3
	ib_smp_t *p_smp;
Packit Service 54dbc3
	ib_mad_notice_attr_t *p_ntci = (ib_mad_notice_attr_t *) payload;
Packit Service 54dbc3
	ib_api_status_t status;
Packit Service 54dbc3
	osm_madw_t tmp_madw;	/* we need a copy to last after repress */
Packit Service 54dbc3
	uint64_t trap_key;
Packit Service 54dbc3
	uint32_t num_received;
Packit Service 54dbc3
	osm_physp_t *p_physp;
Packit Service 54dbc3
	osm_port_t *p_port;
Packit Service 54dbc3
	ib_net16_t source_lid = 0;
Packit Service 54dbc3
	boolean_t is_gsi = TRUE;
Packit Service 54dbc3
	uint8_t port_num = 0;
Packit Service 54dbc3
	boolean_t physp_change_trap = FALSE;
Packit Service 54dbc3
	uint64_t event_wheel_timeout = OSM_DEFAULT_TRAP_SUPPRESSION_TIMEOUT;
Packit Service 54dbc3
	boolean_t run_heavy_sweep = FALSE;
Packit Service 54dbc3
	char buf[1024];
Packit Service 54dbc3
	osm_dr_path_t *p_path;
Packit Service 54dbc3
	unsigned n;
Packit Service 54dbc3
Packit Service 54dbc3
	OSM_LOG_ENTER(sm->p_log);
Packit Service 54dbc3
Packit Service 54dbc3
	CL_ASSERT(p_madw);
Packit Service 54dbc3
Packit Service 54dbc3
	if (osm_exit_flag)
Packit Service 54dbc3
		/*
Packit Service 54dbc3
		   We got an exit flag - do nothing
Packit Service 54dbc3
		   Otherwise we start a sweep on the trap 144 caused by
Packit Service 54dbc3
		   cleaning up SM Cap bit...
Packit Service 54dbc3
		 */
Packit Service 54dbc3
		goto Exit2;
Packit Service 54dbc3
Packit Service 54dbc3
	/* update the is_gsi flag according to the mgmt_class field */
Packit Service 54dbc3
	if (p_madw->p_mad->mgmt_class == IB_MCLASS_SUBN_LID ||
Packit Service 54dbc3
	    p_madw->p_mad->mgmt_class == IB_MCLASS_SUBN_DIR)
Packit Service 54dbc3
		is_gsi = FALSE;
Packit Service 54dbc3
Packit Service 54dbc3
	/* No real need to grab the lock for this function. */
Packit Service 54dbc3
	memset(payload, 0, sizeof(payload));
Packit Service 54dbc3
	memset(&tmp_madw, 0, sizeof(tmp_madw));
Packit Service 54dbc3
Packit Service 54dbc3
	p_smp = osm_madw_get_smp_ptr(p_madw);
Packit Service 54dbc3
Packit Service 54dbc3
	if (p_smp->method != IB_MAD_METHOD_TRAP) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3801: "
Packit Service 54dbc3
			"Unsupported method 0x%X\n", p_smp->method);
Packit Service 54dbc3
		goto Exit2;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/*
Packit Service 54dbc3
	 * The NOTICE Attribute is part of the SMP CLASS attributes
Packit Service 54dbc3
	 * As such the actual attribute data resides inside the SMP
Packit Service 54dbc3
	 * payload.
Packit Service 54dbc3
	 */
Packit Service 54dbc3
Packit Service 54dbc3
	memcpy(payload, &p_smp->data, IB_SMP_DATA_SIZE);
Packit Service 54dbc3
	memcpy(&tmp_madw, p_madw, sizeof(tmp_madw));
Packit Service 54dbc3
Packit Service 54dbc3
	if (is_gsi == FALSE) {
Packit Service 54dbc3
		/* We are in smi flow */
Packit Service 54dbc3
		/*
Packit Service 54dbc3
		 * When we receive a TRAP with dlid = 0 - it means it
Packit Service 54dbc3
		 * came from our own node. So we need to fix it.
Packit Service 54dbc3
		 */
Packit Service 54dbc3
Packit Service 54dbc3
		if (p_madw->mad_addr.addr_type.smi.source_lid == 0) {
Packit Service 54dbc3
			/* Check if the sm_base_lid is 0. If yes - this means
Packit Service 54dbc3
			   that the local lid wasn't configured yet. Don't send
Packit Service 54dbc3
			   a response to the trap. */
Packit Service 54dbc3
			if (sm->p_subn->sm_base_lid == 0) {
Packit Service 54dbc3
				OSM_LOG(sm->p_log, OSM_LOG_DEBUG,
Packit Service 54dbc3
					"Received SLID=0 Trap with local LID=0. Ignoring MAD\n");
Packit Service 54dbc3
				goto Exit2;
Packit Service 54dbc3
			}
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_DEBUG,
Packit Service 54dbc3
				"Received SLID=0 Trap. Using local LID:%u instead\n",
Packit Service 54dbc3
				cl_ntoh16(sm->p_subn->sm_base_lid));
Packit Service 54dbc3
			tmp_madw.mad_addr.addr_type.smi.source_lid =
Packit Service 54dbc3
			    sm->p_subn->sm_base_lid;
Packit Service 54dbc3
		}
Packit Service 54dbc3
Packit Service 54dbc3
		source_lid = tmp_madw.mad_addr.addr_type.smi.source_lid;
Packit Service 54dbc3
Packit Service 54dbc3
		/* Print some info about the incoming Trap */
Packit Service 54dbc3
		log_trap_info(sm->p_log, p_ntci, source_lid, p_smp->trans_id);
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	osm_dump_notice_v2(sm->p_log, p_ntci, FILE_ID, OSM_LOG_VERBOSE);
Packit Service 54dbc3
	CL_PLOCK_ACQUIRE(sm->p_lock);
Packit Service 54dbc3
	p_physp = osm_get_physp_by_mad_addr(sm->p_log, sm->p_subn,
Packit Service 54dbc3
					    &tmp_madw.mad_addr);
Packit Service 54dbc3
	if (p_physp)
Packit Service 54dbc3
		p_smp->m_key = ib_port_info_get_m_key(&p_physp->port_info);
Packit Service 54dbc3
	else
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3809: "
Packit Service 54dbc3
			"Failed to find source physical port for trap\n");
Packit Service 54dbc3
Packit Service 54dbc3
	status = osm_resp_send(sm, &tmp_madw, 0, payload);
Packit Service 54dbc3
	if (status != IB_SUCCESS) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3802: "
Packit Service 54dbc3
			"Error sending response (%s)\n",
Packit Service 54dbc3
			ib_get_err_str(status));
Packit Service 54dbc3
		goto Exit;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/*
Packit Service 54dbc3
	 * We would like to filter out recurring Traps so we track them by
Packit Service 54dbc3
	 * their source lid and content. If the same trap was already
Packit Service 54dbc3
	 * received within the aging time window more than 10 times,
Packit Service 54dbc3
	 * we simply ignore it. This is done only if we are in smi mode
Packit Service 54dbc3
	 */
Packit Service 54dbc3
Packit Service 54dbc3
	if (is_gsi == FALSE) {
Packit Service 54dbc3
		if (ib_notice_is_generic(p_ntci) &&
Packit Service 54dbc3
		    (p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_LINK_INTEGRITY_THRESHOLD_TRAP) ||
Packit Service 54dbc3
		     p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_BUFFER_OVERRUN_THRESHOLD_TRAP) ||
Packit Service 54dbc3
		     p_ntci->g_or_v.generic.trap_num == CL_HTON16(SM_WATCHDOG_TIMER_EXPIRED_TRAP))) {
Packit Service 54dbc3
			/* If this is a trap 129, 130, or 131 - then this is a
Packit Service 54dbc3
			 * trap signaling a change on a physical port.
Packit Service 54dbc3
			 * Mark the physp_change_trap flag as TRUE.
Packit Service 54dbc3
			 */
Packit Service 54dbc3
			physp_change_trap = TRUE;
Packit Service 54dbc3
			/* The source_lid should be based on the source_lid from the trap */
Packit Service 54dbc3
			source_lid = p_ntci->data_details.ntc_129_131.lid;
Packit Service 54dbc3
			port_num = p_ntci->data_details.ntc_129_131.port_num;
Packit Service 54dbc3
		}
Packit Service 54dbc3
Packit Service 54dbc3
		/* try to find it in the aging tracker */
Packit Service 54dbc3
		trap_key = trap_get_key(source_lid, port_num, p_ntci);
Packit Service 54dbc3
		num_received = cl_event_wheel_num_regs(&sm->trap_aging_tracker,
Packit Service 54dbc3
						       trap_key);
Packit Service 54dbc3
Packit Service 54dbc3
		/* Now we know how many times it provided this trap */
Packit Service 54dbc3
		if (num_received >= 10) {
Packit Service 54dbc3
			if (print_num_received(num_received))
Packit Service 54dbc3
				OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
					"Received trap %u times consecutively\n",
Packit Service 54dbc3
					num_received);
Packit Service 54dbc3
			/*
Packit Service 54dbc3
			 * If the trap provides info about a bad port
Packit Service 54dbc3
			 * we mark it as unhealthy.
Packit Service 54dbc3
			 */
Packit Service 54dbc3
			if (physp_change_trap == TRUE) {
Packit Service 54dbc3
				int ret = shutup_noisy_port(sm, source_lid,
Packit Service 54dbc3
							    port_num,
Packit Service 54dbc3
							    num_received);
Packit Service 54dbc3
				if (ret == 1) /* port disabled */
Packit Service 54dbc3
					goto Exit;
Packit Service 54dbc3
				else if (ret == 2) /* unhealthy - run sweep */
Packit Service 54dbc3
					run_heavy_sweep = TRUE;
Packit Service 54dbc3
				/* in any case increase timeout interval */
Packit Service 54dbc3
				event_wheel_timeout =
Packit Service 54dbc3
				    OSM_DEFAULT_UNHEALTHY_TIMEOUT;
Packit Service 54dbc3
			}
Packit Service 54dbc3
		}
Packit Service 54dbc3
Packit Service 54dbc3
		/* restart the aging anyway */
Packit Service 54dbc3
		/* If physp_change_trap is TRUE - then use a callback to unset
Packit Service 54dbc3
		   the healthy bit. If not - no need to use a callback. */
Packit Service 54dbc3
		if (physp_change_trap == TRUE)
Packit Service 54dbc3
			cl_event_wheel_reg(&sm->trap_aging_tracker, trap_key,
Packit Service 54dbc3
					   cl_get_time_stamp() + event_wheel_timeout,
Packit Service 54dbc3
					   aging_tracker_callback, sm);
Packit Service 54dbc3
		else
Packit Service 54dbc3
			cl_event_wheel_reg(&sm->trap_aging_tracker, trap_key,
Packit Service 54dbc3
					   cl_get_time_stamp() + event_wheel_timeout,
Packit Service 54dbc3
					   NULL, NULL);
Packit Service 54dbc3
Packit Service 54dbc3
		/* If was already registered do nothing more */
Packit Service 54dbc3
		if (num_received >= 10 && run_heavy_sweep == FALSE) {
Packit Service 54dbc3
			if (print_num_received(num_received))
Packit Service 54dbc3
				OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
					"Ignoring noisy traps.\n");
Packit Service 54dbc3
			goto Exit;
Packit Service 54dbc3
		}
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/* Check for node description update. IB Spec v1.2.1 pg 823 */
Packit Service 54dbc3
	if (!ib_notice_is_generic(p_ntci))
Packit Service 54dbc3
		goto check_sweep;
Packit Service 54dbc3
	if (cl_ntoh16(p_ntci->g_or_v.generic.trap_num) == SM_LOCAL_CHANGES_TRAP &&
Packit Service 54dbc3
	    p_ntci->data_details.ntc_144.local_changes & TRAP_144_MASK_OTHER_LOCAL_CHANGES &&
Packit Service 54dbc3
	    p_ntci->data_details.ntc_144.change_flgs & TRAP_144_MASK_NODE_DESCRIPTION_CHANGE) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_INFO, "Trap 144 Node description update\n");
Packit Service 54dbc3
Packit Service 54dbc3
		if (p_physp) {
Packit Service 54dbc3
			osm_req_get_node_desc(sm, p_physp);
Packit Service 54dbc3
			if (!(p_ntci->data_details.ntc_144.change_flgs & ~TRAP_144_MASK_NODE_DESCRIPTION_CHANGE) &&
Packit Service 54dbc3
			    p_ntci->data_details.ntc_144.new_cap_mask == p_physp->port_info.capability_mask)
Packit Service 54dbc3
				goto check_report;
Packit Service 54dbc3
		} else
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_ERROR,
Packit Service 54dbc3
				"ERR 3812: No physical port found for "
Packit Service 54dbc3
				"trap 144: \"node description update\"\n");
Packit Service 54dbc3
		goto check_sweep;
Packit Service 54dbc3
	} else if (cl_ntoh16(p_ntci->g_or_v.generic.trap_num) == SM_SYS_IMG_GUID_CHANGED_TRAP) {
Packit Service 54dbc3
		if (p_physp) {
Packit Service 54dbc3
			CL_PLOCK_RELEASE(sm->p_lock);
Packit Service 54dbc3
			CL_PLOCK_EXCL_ACQUIRE(sm->p_lock);
Packit Service 54dbc3
			p_physp = osm_get_physp_by_mad_addr(sm->p_log,
Packit Service 54dbc3
							    sm->p_subn,
Packit Service 54dbc3
							    &tmp_madw.mad_addr);
Packit Service 54dbc3
			if (p_physp) {
Packit Service 54dbc3
				/* this assumes that trap 145 content is not broken? */
Packit Service 54dbc3
				p_physp->p_node->node_info.sys_guid =
Packit Service 54dbc3
					p_ntci->data_details.ntc_145.new_sys_guid;
Packit Service 54dbc3
			}
Packit Service 54dbc3
			CL_PLOCK_RELEASE(sm->p_lock);
Packit Service 54dbc3
			CL_PLOCK_ACQUIRE(sm->p_lock);
Packit Service 54dbc3
		} else
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_ERROR,
Packit Service 54dbc3
				"ERR 3813: No physical port found for "
Packit Service 54dbc3
				"trap 145: \"SystemImageGUID update\"\n");
Packit Service 54dbc3
		goto check_report;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
check_sweep:
Packit Service 54dbc3
	if (osm_log_is_active_v2(sm->p_log, OSM_LOG_INFO, FILE_ID)) {
Packit Service 54dbc3
		if (ib_notice_is_generic(p_ntci) &&
Packit Service 54dbc3
		    cl_ntoh16(p_ntci->g_or_v.generic.trap_num) == SM_LINK_STATE_CHANGED_TRAP) {
Packit Service 54dbc3
			p_path = (p_physp) ?
Packit Service 54dbc3
			    osm_physp_get_dr_path_ptr(p_physp) : NULL;
Packit Service 54dbc3
			if (p_path) {
Packit Service 54dbc3
				n = sprintf(buf, "SM class trap %u: ",
Packit Service 54dbc3
					    cl_ntoh16(p_ntci->g_or_v.generic.trap_num));
Packit Service 54dbc3
				n += snprintf(buf + n, sizeof(buf) - n,
Packit Service 54dbc3
					      "Directed Path Dump of %u hop path: "
Packit Service 54dbc3
					      "Path = ", p_path->hop_count);
Packit Service 54dbc3
Packit Service 54dbc3
				osm_dump_dr_path_as_buf(sizeof(buf) - n, p_path,
Packit Service 54dbc3
							buf + n);
Packit Service 54dbc3
Packit Service 54dbc3
				osm_log_v2(sm->p_log, OSM_LOG_INFO, FILE_ID,
Packit Service 54dbc3
					   "%s\n", buf);
Packit Service 54dbc3
			}
Packit Service 54dbc3
		}
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/* do a sweep if we received a trap */
Packit Service 54dbc3
	if (sm->p_subn->opt.sweep_on_trap) {
Packit Service 54dbc3
		/* if this is trap number 128 or run_heavy_sweep is TRUE -
Packit Service 54dbc3
		   update the force_heavy_sweep flag of the subnet.
Packit Service 54dbc3
		   Sweep also on traps 144 - these traps signal a change of
Packit Service 54dbc3
		   certain port capabilities.
Packit Service 54dbc3
		   TODO: In the future this can be changed to just getting
Packit Service 54dbc3
		   PortInfo on this port instead of sweeping the entire subnet. */
Packit Service 54dbc3
		if (ib_notice_is_generic(p_ntci) &&
Packit Service 54dbc3
		    (cl_ntoh16(p_ntci->g_or_v.generic.trap_num) == SM_LINK_STATE_CHANGED_TRAP ||
Packit Service 54dbc3
		     cl_ntoh16(p_ntci->g_or_v.generic.trap_num) == SM_LOCAL_CHANGES_TRAP ||
Packit Service 54dbc3
		     run_heavy_sweep)) {
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
				"Forcing heavy sweep. Received trap:%u\n",
Packit Service 54dbc3
				cl_ntoh16(p_ntci->g_or_v.generic.trap_num));
Packit Service 54dbc3
Packit Service 54dbc3
			sm->p_subn->force_heavy_sweep = TRUE;
Packit Service 54dbc3
		}
Packit Service 54dbc3
		osm_sm_signal(sm, OSM_SIGNAL_SWEEP);
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/* If we reached here due to trap 129/130/131 - do not need to do
Packit Service 54dbc3
	   the notice report. Just goto exit. We know this is the case
Packit Service 54dbc3
	   if physp_change_trap is TRUE. */
Packit Service 54dbc3
	if (physp_change_trap == TRUE)
Packit Service 54dbc3
		goto Exit;
Packit Service 54dbc3
Packit Service 54dbc3
check_report:
Packit Service 54dbc3
	/* We are going to report the notice - so need to fix the IssuerGID
Packit Service 54dbc3
	   accordingly. See IBA 1.2 p.739 or IBA 1.1 p.653 for details. */
Packit Service 54dbc3
	if (is_gsi) {
Packit Service 54dbc3
		if (!tmp_madw.mad_addr.addr_type.gsi.global_route) {
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3806: "
Packit Service 54dbc3
				"Received gsi trap with global_route FALSE. "
Packit Service 54dbc3
				"Cannot update issuer_gid!\n");
Packit Service 54dbc3
			goto Exit;
Packit Service 54dbc3
		}
Packit Service 54dbc3
		memcpy(&p_ntci->issuer_gid,
Packit Service 54dbc3
		       &tmp_madw.mad_addr.addr_type.gsi.grh_info.src_gid,
Packit Service 54dbc3
		       sizeof(ib_gid_t));
Packit Service 54dbc3
	} else {
Packit Service 54dbc3
		/* Need to use the IssuerLID */
Packit Service 54dbc3
		p_port = osm_get_port_by_lid(sm->p_subn, source_lid);
Packit Service 54dbc3
		if (!p_port) {
Packit Service 54dbc3
			OSM_LOG(sm->p_log, OSM_LOG_VERBOSE,
Packit Service 54dbc3
				"Cannot find port corresponding to lid:%u\n",
Packit Service 54dbc3
				cl_ntoh16(source_lid));
Packit Service 54dbc3
Packit Service 54dbc3
			goto Exit;
Packit Service 54dbc3
		}
Packit Service 54dbc3
Packit Service 54dbc3
		p_ntci->issuer_gid.unicast.prefix =
Packit Service 54dbc3
		    sm->p_subn->opt.subnet_prefix;
Packit Service 54dbc3
		p_ntci->issuer_gid.unicast.interface_id = p_port->guid;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
	/* we need a lock here as the InformInfo DB must be stable */
Packit Service 54dbc3
	status = osm_report_notice(sm->p_log, sm->p_subn, p_ntci);
Packit Service 54dbc3
	if (status != IB_SUCCESS) {
Packit Service 54dbc3
		OSM_LOG(sm->p_log, OSM_LOG_ERROR, "ERR 3803: "
Packit Service 54dbc3
			"Error sending trap reports (%s)\n",
Packit Service 54dbc3
			ib_get_err_str(status));
Packit Service 54dbc3
		goto Exit;
Packit Service 54dbc3
	}
Packit Service 54dbc3
Packit Service 54dbc3
Exit:
Packit Service 54dbc3
	CL_PLOCK_RELEASE(sm->p_lock);
Packit Service 54dbc3
Exit2:
Packit Service 54dbc3
	OSM_LOG_EXIT(sm->p_log);
Packit Service 54dbc3
}
Packit Service 54dbc3
Packit Service 54dbc3
void osm_trap_rcv_process(IN void *context, IN void *data)
Packit Service 54dbc3
{
Packit Service 54dbc3
	osm_sm_t *sm = context;
Packit Service 54dbc3
	osm_madw_t *p_madw = data;
Packit Service 54dbc3
	ib_smp_t __attribute__((unused)) *p_smp;
Packit Service 54dbc3
Packit Service 54dbc3
	OSM_LOG_ENTER(sm->p_log);
Packit Service 54dbc3
Packit Service 54dbc3
	CL_ASSERT(p_madw);
Packit Service 54dbc3
Packit Service 54dbc3
	p_smp = osm_madw_get_smp_ptr(p_madw);
Packit Service 54dbc3
Packit Service 54dbc3
	/* Only Trap requests get here */
Packit Service 54dbc3
	CL_ASSERT(!ib_smp_is_response(p_smp));
Packit Service 54dbc3
	trap_rcv_process_request(sm, p_madw);
Packit Service 54dbc3
Packit Service 54dbc3
	OSM_LOG_EXIT(sm->p_log);
Packit Service 54dbc3
}