1*d6b92ffaSHans Petter Selasky /* 2*d6b92ffaSHans Petter Selasky * Copyright (c) 2004-2009 Voltaire Inc. All rights reserved. 3*d6b92ffaSHans Petter Selasky * 4*d6b92ffaSHans Petter Selasky * This software is available to you under a choice of one of two 5*d6b92ffaSHans Petter Selasky * licenses. You may choose to be licensed under the terms of the GNU 6*d6b92ffaSHans Petter Selasky * General Public License (GPL) Version 2, available from the file 7*d6b92ffaSHans Petter Selasky * COPYING in the main directory of this source tree, or the 8*d6b92ffaSHans Petter Selasky * OpenIB.org BSD license below: 9*d6b92ffaSHans Petter Selasky * 10*d6b92ffaSHans Petter Selasky * Redistribution and use in source and binary forms, with or 11*d6b92ffaSHans Petter Selasky * without modification, are permitted provided that the following 12*d6b92ffaSHans Petter Selasky * conditions are met: 13*d6b92ffaSHans Petter Selasky * 14*d6b92ffaSHans Petter Selasky * - Redistributions of source code must retain the above 15*d6b92ffaSHans Petter Selasky * copyright notice, this list of conditions and the following 16*d6b92ffaSHans Petter Selasky * disclaimer. 17*d6b92ffaSHans Petter Selasky * 18*d6b92ffaSHans Petter Selasky * - Redistributions in binary form must reproduce the above 19*d6b92ffaSHans Petter Selasky * copyright notice, this list of conditions and the following 20*d6b92ffaSHans Petter Selasky * disclaimer in the documentation and/or other materials 21*d6b92ffaSHans Petter Selasky * provided with the distribution. 22*d6b92ffaSHans Petter Selasky * 23*d6b92ffaSHans Petter Selasky * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 24*d6b92ffaSHans Petter Selasky * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 25*d6b92ffaSHans Petter Selasky * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 26*d6b92ffaSHans Petter Selasky * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS 27*d6b92ffaSHans Petter Selasky * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN 28*d6b92ffaSHans Petter Selasky * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 29*d6b92ffaSHans Petter Selasky * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 30*d6b92ffaSHans Petter Selasky * SOFTWARE. 31*d6b92ffaSHans Petter Selasky * 32*d6b92ffaSHans Petter Selasky */ 33*d6b92ffaSHans Petter Selasky 34*d6b92ffaSHans Petter Selasky #if HAVE_CONFIG_H 35*d6b92ffaSHans Petter Selasky # include <config.h> 36*d6b92ffaSHans Petter Selasky #endif /* HAVE_CONFIG_H */ 37*d6b92ffaSHans Petter Selasky 38*d6b92ffaSHans Petter Selasky #include <stdio.h> 39*d6b92ffaSHans Petter Selasky #include <stdlib.h> 40*d6b92ffaSHans Petter Selasky #include <string.h> 41*d6b92ffaSHans Petter Selasky #include <errno.h> 42*d6b92ffaSHans Petter Selasky 43*d6b92ffaSHans Petter Selasky #include <infiniband/umad.h> 44*d6b92ffaSHans Petter Selasky #include <infiniband/mad.h> 45*d6b92ffaSHans Petter Selasky 46*d6b92ffaSHans Petter Selasky #include "mad_internal.h" 47*d6b92ffaSHans Petter Selasky 48*d6b92ffaSHans Petter Selasky #undef DEBUG 49*d6b92ffaSHans Petter Selasky #define DEBUG if (ibdebug) IBWARN 50*d6b92ffaSHans Petter Selasky 51*d6b92ffaSHans Petter Selasky int mad_send(ib_rpc_t * rpc, ib_portid_t * dport, ib_rmpp_hdr_t * rmpp, 52*d6b92ffaSHans Petter Selasky void *data) 53*d6b92ffaSHans Petter Selasky { 54*d6b92ffaSHans Petter Selasky return mad_send_via(rpc, dport, rmpp, data, ibmp); 55*d6b92ffaSHans Petter Selasky } 56*d6b92ffaSHans Petter Selasky 57*d6b92ffaSHans Petter Selasky int mad_send_via(ib_rpc_t * rpc, ib_portid_t * dport, ib_rmpp_hdr_t * rmpp, 58*d6b92ffaSHans Petter Selasky void *data, struct ibmad_port *srcport) 59*d6b92ffaSHans Petter Selasky { 60*d6b92ffaSHans Petter Selasky uint8_t pktbuf[1024]; 61*d6b92ffaSHans Petter Selasky void *umad = pktbuf; 62*d6b92ffaSHans Petter Selasky 63*d6b92ffaSHans Petter Selasky memset(pktbuf, 0, umad_size() + IB_MAD_SIZE); 64*d6b92ffaSHans Petter Selasky 65*d6b92ffaSHans Petter Selasky DEBUG("rmpp %p data %p", rmpp, data); 66*d6b92ffaSHans Petter Selasky 67*d6b92ffaSHans Petter Selasky if (mad_build_pkt(umad, rpc, dport, rmpp, data) < 0) 68*d6b92ffaSHans Petter Selasky return -1; 69*d6b92ffaSHans Petter Selasky 70*d6b92ffaSHans Petter Selasky if (ibdebug) { 71*d6b92ffaSHans Petter Selasky IBWARN("data offs %d sz %d", rpc->dataoffs, rpc->datasz); 72*d6b92ffaSHans Petter Selasky xdump(stderr, "mad send data\n", 73*d6b92ffaSHans Petter Selasky (char *)umad_get_mad(umad) + rpc->dataoffs, rpc->datasz); 74*d6b92ffaSHans Petter Selasky } 75*d6b92ffaSHans Petter Selasky 76*d6b92ffaSHans Petter Selasky if (umad_send(srcport->port_id, srcport->class_agents[rpc->mgtclass & 0xff], 77*d6b92ffaSHans Petter Selasky umad, IB_MAD_SIZE, mad_get_timeout(srcport, rpc->timeout), 78*d6b92ffaSHans Petter Selasky 0) < 0) { 79*d6b92ffaSHans Petter Selasky IBWARN("send failed; %s", strerror(errno)); 80*d6b92ffaSHans Petter Selasky return -1; 81*d6b92ffaSHans Petter Selasky } 82*d6b92ffaSHans Petter Selasky 83*d6b92ffaSHans Petter Selasky return 0; 84*d6b92ffaSHans Petter Selasky } 85*d6b92ffaSHans Petter Selasky 86*d6b92ffaSHans Petter Selasky int mad_respond(void *umad, ib_portid_t * portid, uint32_t rstatus) 87*d6b92ffaSHans Petter Selasky { 88*d6b92ffaSHans Petter Selasky return mad_respond_via(umad, portid, rstatus, ibmp); 89*d6b92ffaSHans Petter Selasky } 90*d6b92ffaSHans Petter Selasky 91*d6b92ffaSHans Petter Selasky int mad_respond_via(void *umad, ib_portid_t * portid, uint32_t rstatus, 92*d6b92ffaSHans Petter Selasky struct ibmad_port *srcport) 93*d6b92ffaSHans Petter Selasky { 94*d6b92ffaSHans Petter Selasky uint8_t *mad = umad_get_mad(umad); 95*d6b92ffaSHans Petter Selasky ib_mad_addr_t *mad_addr; 96*d6b92ffaSHans Petter Selasky ib_rpc_t rpc = { 0 }; 97*d6b92ffaSHans Petter Selasky ib_portid_t rport; 98*d6b92ffaSHans Petter Selasky int is_smi; 99*d6b92ffaSHans Petter Selasky 100*d6b92ffaSHans Petter Selasky if (!portid) { 101*d6b92ffaSHans Petter Selasky if (!(mad_addr = umad_get_mad_addr(umad))) { 102*d6b92ffaSHans Petter Selasky errno = EINVAL; 103*d6b92ffaSHans Petter Selasky return -1; 104*d6b92ffaSHans Petter Selasky } 105*d6b92ffaSHans Petter Selasky 106*d6b92ffaSHans Petter Selasky memset(&rport, 0, sizeof(rport)); 107*d6b92ffaSHans Petter Selasky 108*d6b92ffaSHans Petter Selasky rport.lid = ntohs(mad_addr->lid); 109*d6b92ffaSHans Petter Selasky rport.qp = ntohl(mad_addr->qpn); 110*d6b92ffaSHans Petter Selasky rport.qkey = ntohl(mad_addr->qkey); 111*d6b92ffaSHans Petter Selasky rport.sl = mad_addr->sl; 112*d6b92ffaSHans Petter Selasky 113*d6b92ffaSHans Petter Selasky if (mad_addr->grh_present) { 114*d6b92ffaSHans Petter Selasky rport.grh_present = 1; 115*d6b92ffaSHans Petter Selasky memcpy(&rport.gid, &mad_addr->gid, sizeof(rport.gid)); 116*d6b92ffaSHans Petter Selasky } 117*d6b92ffaSHans Petter Selasky 118*d6b92ffaSHans Petter Selasky portid = &rport; 119*d6b92ffaSHans Petter Selasky } 120*d6b92ffaSHans Petter Selasky 121*d6b92ffaSHans Petter Selasky DEBUG("dest %s", portid2str(portid)); 122*d6b92ffaSHans Petter Selasky 123*d6b92ffaSHans Petter Selasky rpc.mgtclass = mad_get_field(mad, 0, IB_MAD_MGMTCLASS_F); 124*d6b92ffaSHans Petter Selasky 125*d6b92ffaSHans Petter Selasky rpc.method = mad_get_field(mad, 0, IB_MAD_METHOD_F); 126*d6b92ffaSHans Petter Selasky if (rpc.method == IB_MAD_METHOD_SET) 127*d6b92ffaSHans Petter Selasky rpc.method = IB_MAD_METHOD_GET; 128*d6b92ffaSHans Petter Selasky if (rpc.method != IB_MAD_METHOD_SEND) 129*d6b92ffaSHans Petter Selasky rpc.method |= IB_MAD_RESPONSE; 130*d6b92ffaSHans Petter Selasky 131*d6b92ffaSHans Petter Selasky rpc.attr.id = mad_get_field(mad, 0, IB_MAD_ATTRID_F); 132*d6b92ffaSHans Petter Selasky rpc.attr.mod = mad_get_field(mad, 0, IB_MAD_ATTRMOD_F); 133*d6b92ffaSHans Petter Selasky if (rpc.mgtclass == IB_SA_CLASS) 134*d6b92ffaSHans Petter Selasky rpc.recsz = mad_get_field(mad, 0, IB_SA_ATTROFFS_F); 135*d6b92ffaSHans Petter Selasky if (mad_is_vendor_range2(rpc.mgtclass)) 136*d6b92ffaSHans Petter Selasky rpc.oui = mad_get_field(mad, 0, IB_VEND2_OUI_F); 137*d6b92ffaSHans Petter Selasky 138*d6b92ffaSHans Petter Selasky rpc.trid = mad_get_field64(mad, 0, IB_MAD_TRID_F); 139*d6b92ffaSHans Petter Selasky rpc.rstatus = rstatus; 140*d6b92ffaSHans Petter Selasky 141*d6b92ffaSHans Petter Selasky /* cleared by default: timeout, datasz, dataoffs, mkey, mask */ 142*d6b92ffaSHans Petter Selasky 143*d6b92ffaSHans Petter Selasky is_smi = rpc.mgtclass == IB_SMI_CLASS || 144*d6b92ffaSHans Petter Selasky rpc.mgtclass == IB_SMI_DIRECT_CLASS; 145*d6b92ffaSHans Petter Selasky 146*d6b92ffaSHans Petter Selasky if (is_smi) 147*d6b92ffaSHans Petter Selasky portid->qp = 0; 148*d6b92ffaSHans Petter Selasky else if (!portid->qp) 149*d6b92ffaSHans Petter Selasky portid->qp = 1; 150*d6b92ffaSHans Petter Selasky 151*d6b92ffaSHans Petter Selasky if (!portid->qkey && portid->qp == 1) 152*d6b92ffaSHans Petter Selasky portid->qkey = IB_DEFAULT_QP1_QKEY; 153*d6b92ffaSHans Petter Selasky 154*d6b92ffaSHans Petter Selasky DEBUG 155*d6b92ffaSHans Petter Selasky ("qp 0x%x class 0x%x method %d attr 0x%x mod 0x%x datasz %d off %d qkey %x", 156*d6b92ffaSHans Petter Selasky portid->qp, rpc.mgtclass, rpc.method, rpc.attr.id, rpc.attr.mod, 157*d6b92ffaSHans Petter Selasky rpc.datasz, rpc.dataoffs, portid->qkey); 158*d6b92ffaSHans Petter Selasky 159*d6b92ffaSHans Petter Selasky if (mad_build_pkt(umad, &rpc, portid, 0, 0) < 0) 160*d6b92ffaSHans Petter Selasky return -1; 161*d6b92ffaSHans Petter Selasky 162*d6b92ffaSHans Petter Selasky if (ibdebug > 1) 163*d6b92ffaSHans Petter Selasky xdump(stderr, "mad respond pkt\n", mad, IB_MAD_SIZE); 164*d6b92ffaSHans Petter Selasky 165*d6b92ffaSHans Petter Selasky if (umad_send 166*d6b92ffaSHans Petter Selasky (srcport->port_id, srcport->class_agents[rpc.mgtclass], umad, 167*d6b92ffaSHans Petter Selasky IB_MAD_SIZE, mad_get_timeout(srcport, rpc.timeout), 0) < 0) { 168*d6b92ffaSHans Petter Selasky DEBUG("send failed; %s", strerror(errno)); 169*d6b92ffaSHans Petter Selasky return -1; 170*d6b92ffaSHans Petter Selasky } 171*d6b92ffaSHans Petter Selasky 172*d6b92ffaSHans Petter Selasky return 0; 173*d6b92ffaSHans Petter Selasky } 174*d6b92ffaSHans Petter Selasky 175*d6b92ffaSHans Petter Selasky void *mad_receive(void *umad, int timeout) 176*d6b92ffaSHans Petter Selasky { 177*d6b92ffaSHans Petter Selasky return mad_receive_via(umad, timeout, ibmp); 178*d6b92ffaSHans Petter Selasky } 179*d6b92ffaSHans Petter Selasky 180*d6b92ffaSHans Petter Selasky void *mad_receive_via(void *umad, int timeout, struct ibmad_port *srcport) 181*d6b92ffaSHans Petter Selasky { 182*d6b92ffaSHans Petter Selasky void *mad = umad ? umad : umad_alloc(1, umad_size() + IB_MAD_SIZE); 183*d6b92ffaSHans Petter Selasky int agent; 184*d6b92ffaSHans Petter Selasky int length = IB_MAD_SIZE; 185*d6b92ffaSHans Petter Selasky 186*d6b92ffaSHans Petter Selasky if ((agent = umad_recv(srcport->port_id, mad, &length, 187*d6b92ffaSHans Petter Selasky mad_get_timeout(srcport, timeout))) < 0) { 188*d6b92ffaSHans Petter Selasky if (!umad) 189*d6b92ffaSHans Petter Selasky umad_free(mad); 190*d6b92ffaSHans Petter Selasky DEBUG("recv failed: %s", strerror(errno)); 191*d6b92ffaSHans Petter Selasky return 0; 192*d6b92ffaSHans Petter Selasky } 193*d6b92ffaSHans Petter Selasky 194*d6b92ffaSHans Petter Selasky return mad; 195*d6b92ffaSHans Petter Selasky } 196*d6b92ffaSHans Petter Selasky 197*d6b92ffaSHans Petter Selasky void *mad_alloc(void) 198*d6b92ffaSHans Petter Selasky { 199*d6b92ffaSHans Petter Selasky return umad_alloc(1, umad_size() + IB_MAD_SIZE); 200*d6b92ffaSHans Petter Selasky } 201*d6b92ffaSHans Petter Selasky 202*d6b92ffaSHans Petter Selasky void mad_free(void *umad) 203*d6b92ffaSHans Petter Selasky { 204*d6b92ffaSHans Petter Selasky umad_free(umad); 205*d6b92ffaSHans Petter Selasky } 206