xref: /freebsd/contrib/ofed/libibnetdisc/query_smp.c (revision 87181516ef48be852d5e5fee53c6e0dbfc62f21e)
1*d6b92ffaSHans Petter Selasky /*
2*d6b92ffaSHans Petter Selasky  * Copyright (c) 2010 Lawrence Livermore National Laboratory
3*d6b92ffaSHans Petter Selasky  * Copyright (c) 2011 Mellanox Technologies LTD.  All rights reserved.
4*d6b92ffaSHans Petter Selasky  *
5*d6b92ffaSHans Petter Selasky  * This software is available to you under a choice of one of two
6*d6b92ffaSHans Petter Selasky  * licenses.  You may choose to be licensed under the terms of the GNU
7*d6b92ffaSHans Petter Selasky  * General Public License (GPL) Version 2, available from the file
8*d6b92ffaSHans Petter Selasky  * COPYING in the main directory of this source tree, or the
9*d6b92ffaSHans Petter Selasky  * OpenIB.org BSD license below:
10*d6b92ffaSHans Petter Selasky  *
11*d6b92ffaSHans Petter Selasky  *     Redistribution and use in source and binary forms, with or
12*d6b92ffaSHans Petter Selasky  *     without modification, are permitted provided that the following
13*d6b92ffaSHans Petter Selasky  *     conditions are met:
14*d6b92ffaSHans Petter Selasky  *
15*d6b92ffaSHans Petter Selasky  *      - Redistributions of source code must retain the above
16*d6b92ffaSHans Petter Selasky  *        copyright notice, this list of conditions and the following
17*d6b92ffaSHans Petter Selasky  *        disclaimer.
18*d6b92ffaSHans Petter Selasky  *
19*d6b92ffaSHans Petter Selasky  *      - Redistributions in binary form must reproduce the above
20*d6b92ffaSHans Petter Selasky  *        copyright notice, this list of conditions and the following
21*d6b92ffaSHans Petter Selasky  *        disclaimer in the documentation and/or other materials
22*d6b92ffaSHans Petter Selasky  *        provided with the distribution.
23*d6b92ffaSHans Petter Selasky  *
24*d6b92ffaSHans Petter Selasky  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
25*d6b92ffaSHans Petter Selasky  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
26*d6b92ffaSHans Petter Selasky  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
27*d6b92ffaSHans Petter Selasky  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
28*d6b92ffaSHans Petter Selasky  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
29*d6b92ffaSHans Petter Selasky  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
30*d6b92ffaSHans Petter Selasky  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
31*d6b92ffaSHans Petter Selasky  * SOFTWARE.
32*d6b92ffaSHans Petter Selasky  *
33*d6b92ffaSHans Petter Selasky  */
34*d6b92ffaSHans Petter Selasky 
35*d6b92ffaSHans Petter Selasky #if HAVE_CONFIG_H
36*d6b92ffaSHans Petter Selasky #  include <config.h>
37*d6b92ffaSHans Petter Selasky #endif				/* HAVE_CONFIG_H */
38*d6b92ffaSHans Petter Selasky 
39*d6b92ffaSHans Petter Selasky #include <errno.h>
40*d6b92ffaSHans Petter Selasky #include <infiniband/ibnetdisc.h>
41*d6b92ffaSHans Petter Selasky #include <infiniband/umad.h>
42*d6b92ffaSHans Petter Selasky #include "internal.h"
43*d6b92ffaSHans Petter Selasky 
44*d6b92ffaSHans Petter Selasky extern int mlnx_ext_port_info_err(smp_engine_t * engine, ibnd_smp_t * smp,
45*d6b92ffaSHans Petter Selasky 				  uint8_t * mad, void *cb_data);
46*d6b92ffaSHans Petter Selasky 
queue_smp(smp_engine_t * engine,ibnd_smp_t * smp)47*d6b92ffaSHans Petter Selasky static void queue_smp(smp_engine_t * engine, ibnd_smp_t * smp)
48*d6b92ffaSHans Petter Selasky {
49*d6b92ffaSHans Petter Selasky 	smp->qnext = NULL;
50*d6b92ffaSHans Petter Selasky 	if (!engine->smp_queue_head) {
51*d6b92ffaSHans Petter Selasky 		engine->smp_queue_head = smp;
52*d6b92ffaSHans Petter Selasky 		engine->smp_queue_tail = smp;
53*d6b92ffaSHans Petter Selasky 	} else {
54*d6b92ffaSHans Petter Selasky 		engine->smp_queue_tail->qnext = smp;
55*d6b92ffaSHans Petter Selasky 		engine->smp_queue_tail = smp;
56*d6b92ffaSHans Petter Selasky 	}
57*d6b92ffaSHans Petter Selasky }
58*d6b92ffaSHans Petter Selasky 
get_smp(smp_engine_t * engine)59*d6b92ffaSHans Petter Selasky static ibnd_smp_t *get_smp(smp_engine_t * engine)
60*d6b92ffaSHans Petter Selasky {
61*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *head = engine->smp_queue_head;
62*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *tail = engine->smp_queue_tail;
63*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *rc = head;
64*d6b92ffaSHans Petter Selasky 	if (head) {
65*d6b92ffaSHans Petter Selasky 		if (tail == head)
66*d6b92ffaSHans Petter Selasky 			engine->smp_queue_tail = NULL;
67*d6b92ffaSHans Petter Selasky 		engine->smp_queue_head = head->qnext;
68*d6b92ffaSHans Petter Selasky 	}
69*d6b92ffaSHans Petter Selasky 	return rc;
70*d6b92ffaSHans Petter Selasky }
71*d6b92ffaSHans Petter Selasky 
send_smp(ibnd_smp_t * smp,smp_engine_t * engine)72*d6b92ffaSHans Petter Selasky static int send_smp(ibnd_smp_t * smp, smp_engine_t * engine)
73*d6b92ffaSHans Petter Selasky {
74*d6b92ffaSHans Petter Selasky 	int rc = 0;
75*d6b92ffaSHans Petter Selasky 	uint8_t umad[1024];
76*d6b92ffaSHans Petter Selasky 	ib_rpc_t *rpc = &smp->rpc;
77*d6b92ffaSHans Petter Selasky 	int agent = 0;
78*d6b92ffaSHans Petter Selasky 
79*d6b92ffaSHans Petter Selasky 	memset(umad, 0, umad_size() + IB_MAD_SIZE);
80*d6b92ffaSHans Petter Selasky 
81*d6b92ffaSHans Petter Selasky 	if (rpc->mgtclass == IB_SMI_CLASS) {
82*d6b92ffaSHans Petter Selasky 		agent = engine->smi_agent;
83*d6b92ffaSHans Petter Selasky 	} else if (rpc->mgtclass == IB_SMI_DIRECT_CLASS) {
84*d6b92ffaSHans Petter Selasky 		agent = engine->smi_dir_agent;
85*d6b92ffaSHans Petter Selasky 	} else {
86*d6b92ffaSHans Petter Selasky 		IBND_ERROR("Invalid class for RPC\n");
87*d6b92ffaSHans Petter Selasky 		return (-EIO);
88*d6b92ffaSHans Petter Selasky 	}
89*d6b92ffaSHans Petter Selasky 
90*d6b92ffaSHans Petter Selasky 	if ((rc = mad_build_pkt(umad, &smp->rpc, &smp->path, NULL, NULL))
91*d6b92ffaSHans Petter Selasky 	    < 0) {
92*d6b92ffaSHans Petter Selasky 		IBND_ERROR("mad_build_pkt failed; %d\n", rc);
93*d6b92ffaSHans Petter Selasky 		return rc;
94*d6b92ffaSHans Petter Selasky 	}
95*d6b92ffaSHans Petter Selasky 
96*d6b92ffaSHans Petter Selasky 	if ((rc = umad_send(engine->umad_fd, agent, umad, IB_MAD_SIZE,
97*d6b92ffaSHans Petter Selasky 			    engine->cfg->timeout_ms, engine->cfg->retries)) < 0) {
98*d6b92ffaSHans Petter Selasky 		IBND_ERROR("send failed; %d\n", rc);
99*d6b92ffaSHans Petter Selasky 		return rc;
100*d6b92ffaSHans Petter Selasky 	}
101*d6b92ffaSHans Petter Selasky 
102*d6b92ffaSHans Petter Selasky 	return 0;
103*d6b92ffaSHans Petter Selasky }
104*d6b92ffaSHans Petter Selasky 
process_smp_queue(smp_engine_t * engine)105*d6b92ffaSHans Petter Selasky static int process_smp_queue(smp_engine_t * engine)
106*d6b92ffaSHans Petter Selasky {
107*d6b92ffaSHans Petter Selasky 	int rc = 0;
108*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *smp;
109*d6b92ffaSHans Petter Selasky 	while (cl_qmap_count(&engine->smps_on_wire)
110*d6b92ffaSHans Petter Selasky 	       < engine->cfg->max_smps) {
111*d6b92ffaSHans Petter Selasky 		smp = get_smp(engine);
112*d6b92ffaSHans Petter Selasky 		if (!smp)
113*d6b92ffaSHans Petter Selasky 			return 0;
114*d6b92ffaSHans Petter Selasky 
115*d6b92ffaSHans Petter Selasky 		if ((rc = send_smp(smp, engine)) != 0) {
116*d6b92ffaSHans Petter Selasky 			free(smp);
117*d6b92ffaSHans Petter Selasky 			return rc;
118*d6b92ffaSHans Petter Selasky 		}
119*d6b92ffaSHans Petter Selasky 		cl_qmap_insert(&engine->smps_on_wire, (uint32_t) smp->rpc.trid,
120*d6b92ffaSHans Petter Selasky 			       (cl_map_item_t *) smp);
121*d6b92ffaSHans Petter Selasky 		engine->total_smps++;
122*d6b92ffaSHans Petter Selasky 	}
123*d6b92ffaSHans Petter Selasky 	return 0;
124*d6b92ffaSHans Petter Selasky }
125*d6b92ffaSHans Petter Selasky 
issue_smp(smp_engine_t * engine,ib_portid_t * portid,unsigned attrid,unsigned mod,smp_comp_cb_t cb,void * cb_data)126*d6b92ffaSHans Petter Selasky int issue_smp(smp_engine_t * engine, ib_portid_t * portid,
127*d6b92ffaSHans Petter Selasky 	      unsigned attrid, unsigned mod, smp_comp_cb_t cb, void *cb_data)
128*d6b92ffaSHans Petter Selasky {
129*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *smp = calloc(1, sizeof *smp);
130*d6b92ffaSHans Petter Selasky 	if (!smp) {
131*d6b92ffaSHans Petter Selasky 		IBND_ERROR("OOM\n");
132*d6b92ffaSHans Petter Selasky 		return -ENOMEM;
133*d6b92ffaSHans Petter Selasky 	}
134*d6b92ffaSHans Petter Selasky 
135*d6b92ffaSHans Petter Selasky 	smp->cb = cb;
136*d6b92ffaSHans Petter Selasky 	smp->cb_data = cb_data;
137*d6b92ffaSHans Petter Selasky 	smp->path = *portid;
138*d6b92ffaSHans Petter Selasky 	smp->rpc.method = IB_MAD_METHOD_GET;
139*d6b92ffaSHans Petter Selasky 	smp->rpc.attr.id = attrid;
140*d6b92ffaSHans Petter Selasky 	smp->rpc.attr.mod = mod;
141*d6b92ffaSHans Petter Selasky 	smp->rpc.timeout = engine->cfg->timeout_ms;
142*d6b92ffaSHans Petter Selasky 	smp->rpc.datasz = IB_SMP_DATA_SIZE;
143*d6b92ffaSHans Petter Selasky 	smp->rpc.dataoffs = IB_SMP_DATA_OFFS;
144*d6b92ffaSHans Petter Selasky 	smp->rpc.trid = mad_trid();
145*d6b92ffaSHans Petter Selasky 	smp->rpc.mkey = engine->cfg->mkey;
146*d6b92ffaSHans Petter Selasky 
147*d6b92ffaSHans Petter Selasky 	if (portid->lid <= 0 || portid->drpath.drslid == 0xffff ||
148*d6b92ffaSHans Petter Selasky 	    portid->drpath.drdlid == 0xffff)
149*d6b92ffaSHans Petter Selasky 		smp->rpc.mgtclass = IB_SMI_DIRECT_CLASS;	/* direct SMI */
150*d6b92ffaSHans Petter Selasky 	else
151*d6b92ffaSHans Petter Selasky 		smp->rpc.mgtclass = IB_SMI_CLASS;	/* Lid routed SMI */
152*d6b92ffaSHans Petter Selasky 
153*d6b92ffaSHans Petter Selasky 	portid->sl = 0;
154*d6b92ffaSHans Petter Selasky 	portid->qp = 0;
155*d6b92ffaSHans Petter Selasky 
156*d6b92ffaSHans Petter Selasky 	queue_smp(engine, smp);
157*d6b92ffaSHans Petter Selasky 	return process_smp_queue(engine);
158*d6b92ffaSHans Petter Selasky }
159*d6b92ffaSHans Petter Selasky 
process_one_recv(smp_engine_t * engine)160*d6b92ffaSHans Petter Selasky static int process_one_recv(smp_engine_t * engine)
161*d6b92ffaSHans Petter Selasky {
162*d6b92ffaSHans Petter Selasky 	int rc = 0;
163*d6b92ffaSHans Petter Selasky 	int status = 0;
164*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *smp;
165*d6b92ffaSHans Petter Selasky 	uint8_t *mad;
166*d6b92ffaSHans Petter Selasky 	uint32_t trid;
167*d6b92ffaSHans Petter Selasky 	uint8_t umad[sizeof(struct ib_user_mad) + IB_MAD_SIZE];
168*d6b92ffaSHans Petter Selasky 	int length = umad_size() + IB_MAD_SIZE;
169*d6b92ffaSHans Petter Selasky 
170*d6b92ffaSHans Petter Selasky 	memset(umad, 0, sizeof(umad));
171*d6b92ffaSHans Petter Selasky 
172*d6b92ffaSHans Petter Selasky 	/* wait for the next message */
173*d6b92ffaSHans Petter Selasky 	if ((rc = umad_recv(engine->umad_fd, umad, &length,
174*d6b92ffaSHans Petter Selasky 			    -1)) < 0) {
175*d6b92ffaSHans Petter Selasky 		IBND_ERROR("umad_recv failed: %d\n", rc);
176*d6b92ffaSHans Petter Selasky 		return -1;
177*d6b92ffaSHans Petter Selasky 	}
178*d6b92ffaSHans Petter Selasky 
179*d6b92ffaSHans Petter Selasky 	mad = umad_get_mad(umad);
180*d6b92ffaSHans Petter Selasky 	trid = (uint32_t) mad_get_field64(mad, 0, IB_MAD_TRID_F);
181*d6b92ffaSHans Petter Selasky 
182*d6b92ffaSHans Petter Selasky 	smp = (ibnd_smp_t *) cl_qmap_remove(&engine->smps_on_wire, trid);
183*d6b92ffaSHans Petter Selasky 	if ((cl_map_item_t *) smp == cl_qmap_end(&engine->smps_on_wire)) {
184*d6b92ffaSHans Petter Selasky 		IBND_ERROR("Failed to find matching smp for trid (%x)\n", trid);
185*d6b92ffaSHans Petter Selasky 		return -1;
186*d6b92ffaSHans Petter Selasky 	}
187*d6b92ffaSHans Petter Selasky 
188*d6b92ffaSHans Petter Selasky 	rc = process_smp_queue(engine);
189*d6b92ffaSHans Petter Selasky 	if (rc)
190*d6b92ffaSHans Petter Selasky 		goto error;
191*d6b92ffaSHans Petter Selasky 
192*d6b92ffaSHans Petter Selasky 	if ((status = umad_status(umad))) {
193*d6b92ffaSHans Petter Selasky 		IBND_ERROR("umad (%s Attr 0x%x:%u) bad status %d; %s\n",
194*d6b92ffaSHans Petter Selasky 			   portid2str(&smp->path), smp->rpc.attr.id,
195*d6b92ffaSHans Petter Selasky 			   smp->rpc.attr.mod, status, strerror(status));
196*d6b92ffaSHans Petter Selasky 		if (smp->rpc.attr.id == IB_ATTR_MLNX_EXT_PORT_INFO)
197*d6b92ffaSHans Petter Selasky 			rc = mlnx_ext_port_info_err(engine, smp, mad,
198*d6b92ffaSHans Petter Selasky 						    smp->cb_data);
199*d6b92ffaSHans Petter Selasky 	} else if ((status = mad_get_field(mad, 0, IB_DRSMP_STATUS_F))) {
200*d6b92ffaSHans Petter Selasky 		IBND_ERROR("mad (%s Attr 0x%x:%u) bad status 0x%x\n",
201*d6b92ffaSHans Petter Selasky 			   portid2str(&smp->path), smp->rpc.attr.id,
202*d6b92ffaSHans Petter Selasky 			   smp->rpc.attr.mod, status);
203*d6b92ffaSHans Petter Selasky 		if (smp->rpc.attr.id == IB_ATTR_MLNX_EXT_PORT_INFO)
204*d6b92ffaSHans Petter Selasky 			rc = mlnx_ext_port_info_err(engine, smp, mad,
205*d6b92ffaSHans Petter Selasky 						    smp->cb_data);
206*d6b92ffaSHans Petter Selasky 	} else
207*d6b92ffaSHans Petter Selasky 		rc = smp->cb(engine, smp, mad, smp->cb_data);
208*d6b92ffaSHans Petter Selasky 
209*d6b92ffaSHans Petter Selasky error:
210*d6b92ffaSHans Petter Selasky 	free(smp);
211*d6b92ffaSHans Petter Selasky 	return rc;
212*d6b92ffaSHans Petter Selasky }
213*d6b92ffaSHans Petter Selasky 
smp_engine_init(smp_engine_t * engine,char * ca_name,int ca_port,void * user_data,ibnd_config_t * cfg)214*d6b92ffaSHans Petter Selasky int smp_engine_init(smp_engine_t * engine, char * ca_name, int ca_port,
215*d6b92ffaSHans Petter Selasky 		    void *user_data, ibnd_config_t *cfg)
216*d6b92ffaSHans Petter Selasky {
217*d6b92ffaSHans Petter Selasky 	memset(engine, 0, sizeof(*engine));
218*d6b92ffaSHans Petter Selasky 
219*d6b92ffaSHans Petter Selasky 	if (umad_init() < 0) {
220*d6b92ffaSHans Petter Selasky 		IBND_ERROR("umad_init failed\n");
221*d6b92ffaSHans Petter Selasky 		return -EIO;
222*d6b92ffaSHans Petter Selasky 	}
223*d6b92ffaSHans Petter Selasky 
224*d6b92ffaSHans Petter Selasky 	engine->umad_fd = umad_open_port(ca_name, ca_port);
225*d6b92ffaSHans Petter Selasky 	if (engine->umad_fd < 0) {
226*d6b92ffaSHans Petter Selasky 		IBND_ERROR("can't open UMAD port (%s:%d)\n", ca_name, ca_port);
227*d6b92ffaSHans Petter Selasky 		return -EIO;
228*d6b92ffaSHans Petter Selasky 	}
229*d6b92ffaSHans Petter Selasky 
230*d6b92ffaSHans Petter Selasky 	if ((engine->smi_agent = umad_register(engine->umad_fd,
231*d6b92ffaSHans Petter Selasky 	     IB_SMI_CLASS, 1, 0, 0)) < 0) {
232*d6b92ffaSHans Petter Selasky 		IBND_ERROR("Failed to register SMI agent on (%s:%d)\n",
233*d6b92ffaSHans Petter Selasky 			   ca_name, ca_port);
234*d6b92ffaSHans Petter Selasky 		goto eio_close;
235*d6b92ffaSHans Petter Selasky 	}
236*d6b92ffaSHans Petter Selasky 
237*d6b92ffaSHans Petter Selasky 	if ((engine->smi_dir_agent = umad_register(engine->umad_fd,
238*d6b92ffaSHans Petter Selasky 	     IB_SMI_DIRECT_CLASS, 1, 0, 0)) < 0) {
239*d6b92ffaSHans Petter Selasky 		IBND_ERROR("Failed to register SMI_DIRECT agent on (%s:%d)\n",
240*d6b92ffaSHans Petter Selasky 			   ca_name, ca_port);
241*d6b92ffaSHans Petter Selasky 		goto eio_close;
242*d6b92ffaSHans Petter Selasky 	}
243*d6b92ffaSHans Petter Selasky 
244*d6b92ffaSHans Petter Selasky 	engine->user_data = user_data;
245*d6b92ffaSHans Petter Selasky 	cl_qmap_init(&engine->smps_on_wire);
246*d6b92ffaSHans Petter Selasky 	engine->cfg = cfg;
247*d6b92ffaSHans Petter Selasky 	return (0);
248*d6b92ffaSHans Petter Selasky 
249*d6b92ffaSHans Petter Selasky eio_close:
250*d6b92ffaSHans Petter Selasky 	umad_close_port(engine->umad_fd);
251*d6b92ffaSHans Petter Selasky 	return (-EIO);
252*d6b92ffaSHans Petter Selasky }
253*d6b92ffaSHans Petter Selasky 
smp_engine_destroy(smp_engine_t * engine)254*d6b92ffaSHans Petter Selasky void smp_engine_destroy(smp_engine_t * engine)
255*d6b92ffaSHans Petter Selasky {
256*d6b92ffaSHans Petter Selasky 	cl_map_item_t *item;
257*d6b92ffaSHans Petter Selasky 	ibnd_smp_t *smp;
258*d6b92ffaSHans Petter Selasky 
259*d6b92ffaSHans Petter Selasky 	/* remove queued smps */
260*d6b92ffaSHans Petter Selasky 	smp = get_smp(engine);
261*d6b92ffaSHans Petter Selasky 	if (smp)
262*d6b92ffaSHans Petter Selasky 		IBND_ERROR("outstanding SMP's\n");
263*d6b92ffaSHans Petter Selasky 	for ( /* */ ; smp; smp = get_smp(engine))
264*d6b92ffaSHans Petter Selasky 		free(smp);
265*d6b92ffaSHans Petter Selasky 
266*d6b92ffaSHans Petter Selasky 	/* remove smps from the wire queue */
267*d6b92ffaSHans Petter Selasky 	item = cl_qmap_head(&engine->smps_on_wire);
268*d6b92ffaSHans Petter Selasky 	if (item != cl_qmap_end(&engine->smps_on_wire))
269*d6b92ffaSHans Petter Selasky 		IBND_ERROR("outstanding SMP's on wire\n");
270*d6b92ffaSHans Petter Selasky 	for ( /* */ ; item != cl_qmap_end(&engine->smps_on_wire);
271*d6b92ffaSHans Petter Selasky 	     item = cl_qmap_head(&engine->smps_on_wire)) {
272*d6b92ffaSHans Petter Selasky 		cl_qmap_remove_item(&engine->smps_on_wire, item);
273*d6b92ffaSHans Petter Selasky 		free(item);
274*d6b92ffaSHans Petter Selasky 	}
275*d6b92ffaSHans Petter Selasky 
276*d6b92ffaSHans Petter Selasky 	umad_close_port(engine->umad_fd);
277*d6b92ffaSHans Petter Selasky }
278*d6b92ffaSHans Petter Selasky 
process_mads(smp_engine_t * engine)279*d6b92ffaSHans Petter Selasky int process_mads(smp_engine_t * engine)
280*d6b92ffaSHans Petter Selasky {
281*d6b92ffaSHans Petter Selasky 	int rc;
282*d6b92ffaSHans Petter Selasky 	while (!cl_is_qmap_empty(&engine->smps_on_wire))
283*d6b92ffaSHans Petter Selasky 		if ((rc = process_one_recv(engine)) != 0)
284*d6b92ffaSHans Petter Selasky 			return rc;
285*d6b92ffaSHans Petter Selasky 	return 0;
286*d6b92ffaSHans Petter Selasky }
287