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
mad_send(ib_rpc_t * rpc,ib_portid_t * dport,ib_rmpp_hdr_t * rmpp,void * data)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
mad_send_via(ib_rpc_t * rpc,ib_portid_t * dport,ib_rmpp_hdr_t * rmpp,void * data,struct ibmad_port * srcport)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
mad_respond(void * umad,ib_portid_t * portid,uint32_t rstatus)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
mad_respond_via(void * umad,ib_portid_t * portid,uint32_t rstatus,struct ibmad_port * srcport)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
mad_receive(void * umad,int timeout)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
mad_receive_via(void * umad,int timeout,struct ibmad_port * srcport)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
mad_alloc(void)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
mad_free(void * umad)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