Blame src/ibsendtrap.c

Packit db064d
/*
Packit db064d
 * Copyright (c) 2008 Lawrence Livermore National Security
Packit db064d
 * Copyright (c) 2008-2009 Voltaire Inc.  All rights reserved.
Packit db064d
 * Copyright (c) 2009 HNR Consulting.  All rights reserved.
Packit db064d
 * Copyright (c) 2011 Mellanox Technologies LTD.  All rights reserved.
Packit db064d
 *
Packit db064d
 * Produced at Lawrence Livermore National Laboratory.
Packit db064d
 * Written by Ira Weiny <weiny2@llnl.gov>.
Packit db064d
 *
Packit db064d
 * This software is available to you under a choice of one of two
Packit db064d
 * licenses.  You may choose to be licensed under the terms of the GNU
Packit db064d
 * General Public License (GPL) Version 2, available from the file
Packit db064d
 * COPYING in the main directory of this source tree, or the
Packit db064d
 * OpenIB.org BSD license below:
Packit db064d
 *
Packit db064d
 *     Redistribution and use in source and binary forms, with or
Packit db064d
 *     without modification, are permitted provided that the following
Packit db064d
 *     conditions are met:
Packit db064d
 *
Packit db064d
 *      - Redistributions of source code must retain the above
Packit db064d
 *        copyright notice, this list of conditions and the following
Packit db064d
 *        disclaimer.
Packit db064d
 *
Packit db064d
 *      - Redistributions in binary form must reproduce the above
Packit db064d
 *        copyright notice, this list of conditions and the following
Packit db064d
 *        disclaimer in the documentation and/or other materials
Packit db064d
 *        provided with the distribution.
Packit db064d
 *
Packit db064d
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
Packit db064d
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
Packit db064d
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
Packit db064d
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
Packit db064d
 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
Packit db064d
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
Packit db064d
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
Packit db064d
 * SOFTWARE.
Packit db064d
 *
Packit db064d
 */
Packit db064d
Packit db064d
#include <stdlib.h>
Packit db064d
#include <stdio.h>
Packit db064d
#include <unistd.h>
Packit db064d
#include <string.h>
Packit db064d
Packit db064d
#define _GNU_SOURCE
Packit db064d
Packit db064d
#include <infiniband/mad.h>
Packit db064d
#include <iba/ib_types.h>
Packit db064d
Packit db064d
#include "ibdiag_common.h"
Packit db064d
Packit db064d
static struct ibmad_port *srcport;
Packit db064d
/* for local link integrity */
Packit db064d
static int error_port = 1;
Packit db064d
Packit db064d
static uint16_t get_node_type(ib_portid_t * port)
Packit db064d
{
Packit db064d
	uint16_t node_type = IB_NODE_TYPE_CA;
Packit db064d
	uint8_t data[IB_SMP_DATA_SIZE] = { 0 };
Packit db064d
Packit db064d
	if (smp_query_via(data, port, IB_ATTR_NODE_INFO, 0, 0, srcport))
Packit db064d
		node_type = (uint16_t) mad_get_field(data, 0, IB_NODE_TYPE_F);
Packit db064d
	return node_type;
Packit db064d
}
Packit db064d
Packit db064d
static uint32_t get_cap_mask(ib_portid_t * port)
Packit db064d
{
Packit db064d
	uint8_t data[IB_SMP_DATA_SIZE] = { 0 };
Packit db064d
	uint32_t cap_mask = 0;
Packit db064d
Packit db064d
	if (smp_query_via(data, port, IB_ATTR_PORT_INFO, 0, 0, srcport))
Packit db064d
		cap_mask = (uint32_t) mad_get_field(data, 0, IB_PORT_CAPMASK_F);
Packit db064d
	return cap_mask;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap145(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	n->generic_type = 0x80 | IB_NOTICE_TYPE_INFO;
Packit db064d
	n->g_or_v.generic.prod_type_lsb = htobe16(get_node_type(port));
Packit db064d
	n->g_or_v.generic.trap_num = htobe16(145);
Packit db064d
	n->issuer_lid = htobe16((uint16_t) port->lid);
Packit db064d
	n->data_details.ntc_145.new_sys_guid = htobe64(0x1234567812345678ULL);
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap144_local(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	n->generic_type = 0x80 | IB_NOTICE_TYPE_INFO;
Packit db064d
	n->g_or_v.generic.prod_type_lsb = htobe16(get_node_type(port));
Packit db064d
	n->g_or_v.generic.trap_num = htobe16(144);
Packit db064d
	n->issuer_lid = htobe16((uint16_t) port->lid);
Packit db064d
	n->data_details.ntc_144.lid = n->issuer_lid;
Packit db064d
	n->data_details.ntc_144.new_cap_mask = htobe32(get_cap_mask(port));
Packit db064d
	n->data_details.ntc_144.local_changes =
Packit db064d
	    TRAP_144_MASK_OTHER_LOCAL_CHANGES;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap144_nodedesc(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	build_trap144_local(n, port);
Packit db064d
	n->data_details.ntc_144.change_flgs =
Packit db064d
	    TRAP_144_MASK_NODE_DESCRIPTION_CHANGE;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap144_linkspeed(ib_mad_notice_attr_t * n,
Packit db064d
				    ib_portid_t * port)
Packit db064d
{
Packit db064d
	build_trap144_local(n, port);
Packit db064d
	n->data_details.ntc_144.change_flgs =
Packit db064d
	    TRAP_144_MASK_LINK_SPEED_ENABLE_CHANGE;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap129(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	n->generic_type = 0x80 | IB_NOTICE_TYPE_URGENT;
Packit db064d
	n->g_or_v.generic.prod_type_lsb = htobe16(get_node_type(port));
Packit db064d
	n->g_or_v.generic.trap_num = htobe16(129);
Packit db064d
	n->issuer_lid = htobe16((uint16_t) port->lid);
Packit db064d
	n->data_details.ntc_129_131.lid = n->issuer_lid;
Packit db064d
	n->data_details.ntc_129_131.pad = 0;
Packit db064d
	n->data_details.ntc_129_131.port_num = (uint8_t) error_port;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap256_local(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	n->generic_type = 0x80 | IB_NOTICE_TYPE_SECURITY;
Packit db064d
	n->g_or_v.generic.prod_type_lsb = htobe16(get_node_type(port));
Packit db064d
	n->g_or_v.generic.trap_num = htobe16(256);
Packit db064d
	n->issuer_lid = htobe16((uint16_t) port->lid);
Packit db064d
	n->data_details.ntc_256.lid = n->issuer_lid;
Packit db064d
	n->data_details.ntc_256.dr_slid = htobe16(0xffff);
Packit db064d
	n->data_details.ntc_256.method = 1;
Packit db064d
	n->data_details.ntc_256.attr_id = htobe16(0x15);
Packit db064d
	n->data_details.ntc_256.attr_mod = htobe32(0x12);
Packit db064d
	n->data_details.ntc_256.mkey = htobe64(0x1234567812345678ULL);
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap256_lid(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	build_trap256_local(n, port);
Packit db064d
	n->data_details.ntc_256.dr_trunc_hop = 0;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap256_dr(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	build_trap256_local(n, port);
Packit db064d
	n->data_details.ntc_256.dr_trunc_hop = 0x80 | 0x4;
Packit db064d
	n->data_details.ntc_256.dr_rtn_path[0] = 5;
Packit db064d
	n->data_details.ntc_256.dr_rtn_path[1] = 6;
Packit db064d
	n->data_details.ntc_256.dr_rtn_path[2] = 7;
Packit db064d
	n->data_details.ntc_256.dr_rtn_path[3] = 8;
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap257_258(ib_mad_notice_attr_t * n, ib_portid_t * port,
Packit db064d
			      uint16_t trap_num)
Packit db064d
{
Packit db064d
	n->generic_type = 0x80 | IB_NOTICE_TYPE_SECURITY;
Packit db064d
	n->g_or_v.generic.prod_type_lsb = htobe16(get_node_type(port));
Packit db064d
	n->g_or_v.generic.trap_num = htobe16(trap_num);
Packit db064d
	n->issuer_lid = htobe16((uint16_t) port->lid);
Packit db064d
	n->data_details.ntc_257_258.lid1 = htobe16(1);
Packit db064d
	n->data_details.ntc_257_258.lid2 = htobe16(2);
Packit db064d
	n->data_details.ntc_257_258.key = htobe32(0x12345678);
Packit db064d
	n->data_details.ntc_257_258.qp1 = htobe32(0x010101);
Packit db064d
	n->data_details.ntc_257_258.qp2 = htobe32(0x020202);
Packit db064d
	n->data_details.ntc_257_258.gid1.unicast.prefix = htobe64(0xf8c0000000000001ULL);
Packit db064d
	n->data_details.ntc_257_258.gid1.unicast.interface_id = htobe64(0x1111222233334444ULL);
Packit db064d
	n->data_details.ntc_257_258.gid2.unicast.prefix = htobe64(0xf8c0000000000001ULL);
Packit db064d
	n->data_details.ntc_257_258.gid2.unicast.interface_id = htobe64(0x5678567812341234ULL);
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap257(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	build_trap257_258(n, port, 257);
Packit db064d
}
Packit db064d
Packit db064d
static void build_trap258(ib_mad_notice_attr_t * n, ib_portid_t * port)
Packit db064d
{
Packit db064d
	build_trap257_258(n, port, 258);
Packit db064d
}
Packit db064d
Packit db064d
static int send_trap(void (*build) (ib_mad_notice_attr_t *, ib_portid_t *))
Packit db064d
{
Packit db064d
	ib_portid_t sm_port;
Packit db064d
	ib_portid_t selfportid;
Packit db064d
	int selfport;
Packit db064d
	ib_rpc_t trap_rpc;
Packit db064d
	ib_mad_notice_attr_t notice;
Packit db064d
Packit db064d
	if (resolve_self(ibd_ca, ibd_ca_port, &selfportid, &selfport, NULL))
Packit db064d
		IBEXIT("can't resolve self");
Packit db064d
Packit db064d
	if (resolve_sm_portid(ibd_ca, ibd_ca_port, &sm_port))
Packit db064d
		IBEXIT("can't resolve SM destination port");
Packit db064d
Packit db064d
	memset(&trap_rpc, 0, sizeof(trap_rpc));
Packit db064d
	trap_rpc.mgtclass = IB_SMI_CLASS;
Packit db064d
	trap_rpc.method = IB_MAD_METHOD_TRAP;
Packit db064d
	trap_rpc.trid = mad_trid();
Packit db064d
	trap_rpc.attr.id = NOTICE;
Packit db064d
	trap_rpc.datasz = IB_SMP_DATA_SIZE;
Packit db064d
	trap_rpc.dataoffs = IB_SMP_DATA_OFFS;
Packit db064d
Packit db064d
	memset(&notice, 0, sizeof(notice));
Packit db064d
	build(&notice, &selfportid);
Packit db064d
Packit db064d
	return mad_send_via(&trap_rpc, &sm_port, NULL, &notice, srcport);
Packit db064d
}
Packit db064d
Packit db064d
typedef struct _trap_def {
Packit db064d
	const char *trap_name;
Packit db064d
	void (*build_func) (ib_mad_notice_attr_t *, ib_portid_t *);
Packit db064d
} trap_def_t;
Packit db064d
Packit db064d
static const trap_def_t traps[] = {
Packit db064d
	{"node_desc_change", build_trap144_nodedesc},
Packit db064d
	{"link_speed_enabled_change", build_trap144_linkspeed},
Packit db064d
	{"local_link_integrity", build_trap129},
Packit db064d
	{"sys_image_guid_change", build_trap145},
Packit db064d
	{"mkey_lid", build_trap256_lid},
Packit db064d
	{"mkey_dr", build_trap256_dr},
Packit db064d
	{"pkey", build_trap257},
Packit db064d
	{"qkey", build_trap258},
Packit db064d
	{NULL, NULL}
Packit db064d
};
Packit db064d
Packit db064d
static int process_send_trap(const char *trap_name)
Packit db064d
{
Packit db064d
	int i;
Packit db064d
Packit db064d
	for (i = 0; traps[i].trap_name; i++)
Packit db064d
		if (strcmp(traps[i].trap_name, trap_name) == 0)
Packit db064d
			return send_trap(traps[i].build_func);
Packit db064d
	ibdiag_show_usage();
Packit db064d
	return 1;
Packit db064d
}
Packit db064d
Packit db064d
int main(int argc, char **argv)
Packit db064d
{
Packit db064d
	char usage_args[1024];
Packit db064d
	int mgmt_classes[2] = { IB_SMI_CLASS, IB_SMI_DIRECT_CLASS };
Packit db064d
	const char *trap_name = NULL;
Packit db064d
	int i, n, rc;
Packit db064d
Packit db064d
	n = sprintf(usage_args, "[<trap_name>] [<error_port>]\n"
Packit db064d
		    "\nArgument <trap_name> can be one of the following:\n");
Packit db064d
	for (i = 0; traps[i].trap_name; i++) {
Packit db064d
		n += snprintf(usage_args + n, sizeof(usage_args) - n,
Packit db064d
			      "  %s\n", traps[i].trap_name);
Packit db064d
		if (n >= sizeof(usage_args))
Packit db064d
			exit(-1);
Packit db064d
	}
Packit db064d
	snprintf(usage_args + n, sizeof(usage_args) - n,
Packit db064d
		 "\n  default behavior is to send \"%s\"", traps[0].trap_name);
Packit db064d
Packit db064d
	ibdiag_process_opts(argc, argv, NULL, "DGKL", NULL, NULL,
Packit db064d
			    usage_args, NULL);
Packit db064d
Packit db064d
	argc -= optind;
Packit db064d
	argv += optind;
Packit db064d
Packit db064d
	trap_name = argv[0] ? argv[0] : traps[0].trap_name;
Packit db064d
Packit db064d
	if (argc > 1)
Packit db064d
		error_port = atoi(argv[1]);
Packit db064d
Packit db064d
	madrpc_show_errors(1);
Packit db064d
Packit db064d
	srcport = mad_rpc_open_port(ibd_ca, ibd_ca_port, mgmt_classes, 2);
Packit db064d
	if (!srcport)
Packit db064d
		IBEXIT("Failed to open '%s' port '%d'", ibd_ca, ibd_ca_port);
Packit db064d
Packit db064d
	smp_mkey_set(srcport, ibd_mkey);
Packit db064d
Packit db064d
	rc = process_send_trap(trap_name);
Packit db064d
	mad_rpc_close_port(srcport);
Packit db064d
	return rc;
Packit db064d
}