xref: /illumos-gate/usr/src/uts/common/io/ib/mgt/ibdma/ibdma.c (revision 2e837a72011f54762249b6612c2a64f171efcd43)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright (c) 2009, 2010, Oracle and/or its affiliates. All rights reserved.
24  */
25 
26 /*
27  * Infiniband Device Management Agent for IB storage.
28  */
29 
30 #include <sys/conf.h>
31 #include <sys/file.h>
32 #include <sys/ddi.h>
33 #include <sys/sunddi.h>
34 #include <sys/modctl.h>
35 #include <sys/priv.h>
36 #include <sys/sysmacros.h>
37 
38 #include <sys/ib/ibtl/ibti.h>		/* IB public interfaces */
39 
40 #include <sys/ib/mgt/ibdma/ibdma.h>
41 #include <sys/ib/mgt/ibdma/ibdma_impl.h>
42 
43 /*
44  * NOTE: The IB Device Management Agent function, like other IB
45  * managers and agents is best implemented as a kernel misc.
46  * module.
47  * Eventually we could modify IBT_DM_AGENT so that we don't need to
48  * open each HCA to receive asynchronous events.
49  */
50 
51 #define	IBDMA_NAME_VERSION	"IB Device Management Agent"
52 
53 extern struct mod_ops mod_miscops;
54 
55 static void ibdma_ibt_async_handler(void *clnt, ibt_hca_hdl_t hdl,
56 	ibt_async_code_t code, ibt_async_event_t *event);
57 
58 static void ibdma_mad_recv_cb(ibmf_handle_t ibmf_hdl,
59 	ibmf_msg_t *msgp, void *args);
60 static void ibdma_create_resp_mad(ibmf_msg_t *msgp);
61 
62 /*
63  * Misc. kernel module for now.
64  */
65 static struct modlmisc modlmisc = {
66 	&mod_miscops,
67 	IBDMA_NAME_VERSION
68 };
69 
70 static struct modlinkage modlinkage = {
71 	MODREV_1, (void *)&modlmisc, NULL
72 };
73 
74 static ibt_clnt_modinfo_t ibdma_ibt_modinfo = {
75 	IBTI_V_CURR,
76 	IBT_DM_AGENT,
77 	ibdma_ibt_async_handler,
78 	NULL,
79 	"ibdma"
80 };
81 
82 /*
83  * Module global state allocated at init().
84  */
85 static ibdma_mod_state_t	*ibdma = NULL;
86 
87 /*
88  * Init/Fini handlers and IBTL HCA management prototypes.
89  */
90 static int ibdma_init();
91 static int ibdma_fini();
92 static int ibdma_ibt_init();
93 static void ibdma_ibt_fini();
94 static ibdma_hca_t *ibdma_hca_init(ib_guid_t guid);
95 static void ibdma_hca_fini(ibdma_hca_t *hca);
96 static ibdma_hca_t *ibdma_find_hca(ib_guid_t guid);
97 
98 /*
99  * DevMgmt Agent MAD attribute handlers prototypes.
100  */
101 static void ibdma_get_class_portinfo(ibmf_msg_t *msg);
102 static void ibdma_get_io_unitinfo(ibdma_hca_t *hca, ibmf_msg_t *msg);
103 static void ibdma_get_ioc_profile(ibdma_hca_t *hca, ibmf_msg_t *msg);
104 static void ibdma_get_ioc_services(ibdma_hca_t *hca, ibmf_msg_t *msg);
105 
106 /*
107  * _init()
108  */
109 int
110 _init(void)
111 {
112 	int status;
113 
114 	ASSERT(ibdma == NULL);
115 
116 	ibdma = kmem_zalloc(sizeof (*ibdma), KM_SLEEP);
117 	ASSERT(ibdma != NULL);
118 
119 	status = ibdma_init();
120 	if (status != DDI_SUCCESS) {
121 		kmem_free(ibdma, sizeof (*ibdma));
122 		ibdma = NULL;
123 		return (status);
124 	}
125 
126 	status = mod_install(&modlinkage);
127 	if (status != DDI_SUCCESS) {
128 		cmn_err(CE_NOTE, "_init, mod_install error (%d)", status);
129 		(void) ibdma_fini();
130 		kmem_free(ibdma, sizeof (*ibdma));
131 		ibdma = NULL;
132 	}
133 	return (status);
134 }
135 
136 /*
137  * _info()
138  */
139 int
140 _info(struct modinfo *modinfop)
141 {
142 	return (mod_info(&modlinkage, modinfop));
143 }
144 
145 /*
146  * _fini()
147  */
148 int
149 _fini(void)
150 {
151 	int		status;
152 	int		slot;
153 	ibdma_hca_t	*hca;
154 
155 	status = mod_remove(&modlinkage);
156 	if (status != DDI_SUCCESS) {
157 		cmn_err(CE_NOTE, "_fini, mod_remove error (%d)", status);
158 		return (status);
159 	}
160 
161 	/*
162 	 * Sanity check to see if anyone is not cleaning
163 	 * up appropriately.
164 	 */
165 	mutex_enter(&ibdma->ms_hca_list_lock);
166 	hca = list_head(&ibdma->ms_hca_list);
167 	while (hca != NULL) {
168 		for (slot = 0; slot < IBDMA_MAX_IOC; slot++) {
169 			if (hca->ih_ioc[slot].ii_inuse) {
170 				cmn_err(CE_NOTE, "_fini, IOC %d still attached"
171 				    " for (0x%0llx)", slot+1,
172 				    (u_longlong_t)hca->ih_iou_guid);
173 			}
174 		}
175 		hca = list_next(&ibdma->ms_hca_list, hca);
176 	}
177 	mutex_exit(&ibdma->ms_hca_list_lock);
178 
179 	(void) ibdma_fini();
180 	kmem_free(ibdma, sizeof (*ibdma));
181 	return (status);
182 }
183 
184 /*
185  * ibdma_init()
186  *
187  * Initialize I/O Unit structure, generate initial HCA list and register
188  * it port with the IBMF.
189  */
190 static int
191 ibdma_init()
192 {
193 	int		status;
194 
195 	/*
196 	 * Global lock and I/O Unit initialization.
197 	 */
198 	mutex_init(&ibdma->ms_hca_list_lock, NULL, MUTEX_DRIVER, NULL);
199 
200 	/*
201 	 * Discover IB hardware and setup for device management agent
202 	 * support.
203 	 */
204 	status = ibdma_ibt_init();
205 	if (status != DDI_SUCCESS) {
206 		cmn_err(CE_NOTE, "ibdma_init, ibt_attach failed (%d)",
207 		    status);
208 		mutex_destroy(&ibdma->ms_hca_list_lock);
209 		return (status);
210 	}
211 
212 	return (status);
213 }
214 
215 /*
216  * ibdma_fini()
217  *
218  * Release resource if we are no longer in use.
219  */
220 static int
221 ibdma_fini()
222 {
223 	ibdma_ibt_fini();
224 	mutex_destroy(&ibdma->ms_hca_list_lock);
225 	return (DDI_SUCCESS);
226 }
227 
228 /*
229  * ibdma_ibt_async_handler()
230  */
231 /* ARGSUSED */
232 static void
233 ibdma_ibt_async_handler(void *clnt, ibt_hca_hdl_t hdl,
234 	ibt_async_code_t code, ibt_async_event_t *event)
235 {
236 	ibdma_hca_t	*hca;
237 
238 	switch (code) {
239 
240 	case IBT_EVENT_PORT_UP:
241 	case IBT_ERROR_PORT_DOWN:
242 	case IBT_PORT_CHANGE_EVENT:
243 	case IBT_CLNT_REREG_EVENT:
244 		break;
245 
246 	case IBT_HCA_ATTACH_EVENT:
247 		mutex_enter(&ibdma->ms_hca_list_lock);
248 		hca = ibdma_hca_init(event->ev_hca_guid);
249 		if (hca != NULL) {
250 			list_insert_tail(&ibdma->ms_hca_list, hca);
251 			cmn_err(CE_NOTE, "hca ibt hdl (%p)",
252 			    (void *)hca->ih_ibt_hdl);
253 			ibdma->ms_num_hcas++;
254 		}
255 		mutex_exit(&ibdma->ms_hca_list_lock);
256 		break;
257 
258 	case IBT_HCA_DETACH_EVENT:
259 		mutex_enter(&ibdma->ms_hca_list_lock);
260 		hca = ibdma_find_hca(event->ev_hca_guid);
261 		if (hca != NULL) {
262 			list_remove(&ibdma->ms_hca_list, hca);
263 			cmn_err(CE_NOTE, "removing hca (%p) (0x%llx)",
264 			    (void *)hca, hca ?
265 			    (u_longlong_t)hca->ih_iou_guid : 0x0ll);
266 			ibdma_hca_fini(hca);
267 		}
268 		mutex_exit(&ibdma->ms_hca_list_lock);
269 		break;
270 
271 	default:
272 #ifdef	DEBUG
273 		cmn_err(CE_NOTE, "ibt_async_handler, unhandled event(%d)",
274 		    code);
275 #endif
276 		break;
277 	}
278 
279 }
280 
281 /*
282  * ibdma_ibt_init()
283  */
284 static int
285 ibdma_ibt_init()
286 {
287 	int		status;
288 	int		hca_cnt;
289 	int		hca_ndx;
290 	ib_guid_t	*guid;
291 	ibdma_hca_t	*hca;
292 
293 	/*
294 	 * Attach to IBTF and get HCA list.
295 	 */
296 	status = ibt_attach(&ibdma_ibt_modinfo, NULL,
297 	    ibdma, &ibdma->ms_ibt_hdl);
298 	if (status != DDI_SUCCESS) {
299 		cmn_err(CE_NOTE, "ibt_init, ibt_attach failed (%d)",
300 		    status);
301 		return (status);
302 	}
303 
304 	list_create(&ibdma->ms_hca_list, sizeof (ibdma_hca_t),
305 	    offsetof(ibdma_hca_t, ih_node));
306 
307 	hca_cnt = ibt_get_hca_list(&guid);
308 	if (hca_cnt < 1) {
309 #ifdef	DEBUG_IBDMA
310 		cmn_err(CE_NOTE, "ibt_init, no HCA(s) found");
311 #endif
312 		/* not an error if no HCAs, but nothing more to do here */
313 		return (DDI_SUCCESS);
314 	}
315 
316 	mutex_enter(&ibdma->ms_hca_list_lock);
317 
318 	for (hca_ndx = 0; hca_ndx < hca_cnt; hca_ndx++) {
319 #ifdef	DEBUG_IBDMA
320 		cmn_err(CE_NOTE, "adding hca GUID(0x%llx)",
321 		    (u_longlong_t)guid[hca_ndx]);
322 #endif
323 
324 		hca = ibdma_hca_init(guid[hca_ndx]);
325 		if (hca == NULL) {
326 			cmn_err(CE_NOTE, "ibt_init, hca_init GUID(0x%llx)"
327 			    " failed", (u_longlong_t)guid[hca_ndx]);
328 			continue;
329 		}
330 		list_insert_tail(&ibdma->ms_hca_list, hca);
331 		ibdma->ms_num_hcas++;
332 	}
333 
334 	mutex_exit(&ibdma->ms_hca_list_lock);
335 
336 	ibt_free_hca_list(guid, hca_cnt);
337 #ifdef	DEBUG_IBDMA
338 	cmn_err(CE_NOTE, "Added %d HCA(s)",
339 	    ibdma->ms_num_hcas);
340 #endif
341 	return (DDI_SUCCESS);
342 }
343 
344 /*
345  * ibdma_ibt_fini()
346  */
347 static void
348 ibdma_ibt_fini()
349 {
350 	ibdma_hca_t		*hca;
351 	ibdma_hca_t		*next;
352 
353 	mutex_enter(&ibdma->ms_hca_list_lock);
354 	hca = list_head(&ibdma->ms_hca_list);
355 	while (hca != NULL) {
356 		next = list_next(&ibdma->ms_hca_list, hca);
357 		list_remove(&ibdma->ms_hca_list, hca);
358 #ifdef	DEBUG_IBDMA
359 		cmn_err(CE_NOTE, "removing hca (%p) (0x%llx)",
360 		    (void *)hca, hca ?
361 		    (u_longlong_t)hca->ih_iou_guid : 0x0ll);
362 		cmn_err(CE_NOTE, "hca ibt hdl (%p)",
363 		    (void *)hca->ih_ibt_hdl);
364 #endif
365 		ibdma_hca_fini(hca);
366 		hca = next;
367 	}
368 	list_destroy(&ibdma->ms_hca_list);
369 
370 	(void) ibt_detach(ibdma->ms_ibt_hdl);
371 	ibdma->ms_ibt_hdl   = NULL;
372 	ibdma->ms_num_hcas  = 0;
373 	mutex_exit(&ibdma->ms_hca_list_lock);
374 }
375 
376 /*
377  * ibdma_find_hca()
378  */
379 static ibdma_hca_t *
380 ibdma_find_hca(ib_guid_t guid)
381 {
382 	ibdma_hca_t	*hca;
383 
384 	ASSERT(mutex_owned(&ibdma->ms_hca_list_lock));
385 
386 	hca = list_head(&ibdma->ms_hca_list);
387 	while (hca != NULL) {
388 		if (hca->ih_iou_guid == guid) {
389 			break;
390 		}
391 		hca = list_next(&ibdma->ms_hca_list, hca);
392 	}
393 	return (hca);
394 }
395 
396 /*
397  * ibdma_hca_init()
398  */
399 static ibdma_hca_t *
400 ibdma_hca_init(ib_guid_t guid)
401 {
402 	ibt_status_t		status;
403 	ibdma_hca_t		*hca;
404 	ibdma_port_t		*port;
405 	ibt_hca_attr_t		hca_attr;
406 	int			ndx;
407 
408 	ASSERT(mutex_owned(&ibdma->ms_hca_list_lock));
409 
410 	status = ibt_query_hca_byguid(guid, &hca_attr);
411 	if (status != IBT_SUCCESS) {
412 		cmn_err(CE_NOTE, "hca_init HCA query error (%d)",
413 		    status);
414 		return (NULL);
415 	}
416 
417 	if (ibdma_find_hca(guid) != NULL) {
418 #ifdef	DEBUG_IBDMA
419 		cmn_err(CE_NOTE, "hca_init HCA already exists");
420 #endif
421 		return (NULL);
422 	}
423 
424 	hca = kmem_zalloc(sizeof (ibdma_hca_t) +
425 	    (hca_attr.hca_nports-1)*sizeof (ibdma_port_t), KM_SLEEP);
426 	ASSERT(hca != NULL);
427 
428 	hca->ih_nports   = hca_attr.hca_nports;
429 
430 	rw_init(&hca->ih_iou_rwlock, NULL, RW_DRIVER, NULL);
431 	rw_enter(&hca->ih_iou_rwlock, RW_WRITER);
432 	hca->ih_iou_guid		= guid;
433 	hca->ih_iou.iou_changeid	= h2b16(1);
434 	hca->ih_iou.iou_num_ctrl_slots	= IBDMA_MAX_IOC;
435 	hca->ih_iou.iou_flag		= IB_DM_IOU_OPTIONROM_ABSENT;
436 
437 	list_create(&hca->ih_hdl_list, sizeof (ibdma_hdl_impl_t),
438 	    offsetof(ibdma_hdl_impl_t, ih_node));
439 	rw_exit(&hca->ih_iou_rwlock);
440 
441 	/*
442 	 * It would be better to not open, but IBTL is setup to only allow
443 	 * certain managers to get async call backs if not open.
444 	 */
445 	status = ibt_open_hca(ibdma->ms_ibt_hdl, guid, &hca->ih_ibt_hdl);
446 	if (status != IBT_SUCCESS) {
447 		cmn_err(CE_NOTE, "hca_init() IBT open failed (%d)",
448 		    status);
449 
450 		list_destroy(&hca->ih_hdl_list);
451 		rw_destroy(&hca->ih_iou_rwlock);
452 		kmem_free(hca, sizeof (ibdma_hca_t) +
453 		    (hca_attr.hca_nports-1)*sizeof (ibdma_port_t));
454 		return (NULL);
455 	}
456 
457 	/*
458 	 * Register with the IB Management Framework and setup MAD call-back.
459 	 */
460 	for (ndx = 0; ndx < hca->ih_nports; ndx++) {
461 		port = &hca->ih_port[ndx];
462 		port->ip_hcap = hca;
463 		port->ip_ibmf_reg.ir_ci_guid	= hca->ih_iou_guid;
464 		port->ip_ibmf_reg.ir_port_num	= ndx + 1;
465 		port->ip_ibmf_reg.ir_client_class = DEV_MGT_AGENT;
466 
467 		status = ibmf_register(&port->ip_ibmf_reg, IBMF_VERSION,
468 		    0, NULL, NULL, &port->ip_ibmf_hdl, &port->ip_ibmf_caps);
469 		if (status != IBMF_SUCCESS) {
470 			cmn_err(CE_NOTE, "hca_init, IBMF register failed (%d)",
471 			    status);
472 			port->ip_ibmf_hdl = NULL;
473 			ibdma_hca_fini(hca);
474 			return (NULL);
475 		}
476 
477 		status = ibmf_setup_async_cb(port->ip_ibmf_hdl,
478 		    IBMF_QP_HANDLE_DEFAULT, ibdma_mad_recv_cb, port, 0);
479 		if (status != IBMF_SUCCESS) {
480 			cmn_err(CE_NOTE, "hca_init, IBMF cb setup failed (%d)",
481 			    status);
482 			ibdma_hca_fini(hca);
483 			return (NULL);
484 		}
485 
486 		status = ibt_modify_port_byguid(hca->ih_iou_guid,
487 		    ndx+1, IBT_PORT_SET_DEVMGT, 0);
488 		if (status != IBT_SUCCESS) {
489 			cmn_err(CE_NOTE, "hca_init, IBT modify port caps"
490 			    " error (%d)", status);
491 			ibdma_hca_fini(hca);
492 			return (NULL);
493 		}
494 	}
495 	return (hca);
496 }
497 
498 /*
499  * ibdma_hca_fini()
500  */
501 static void
502 ibdma_hca_fini(ibdma_hca_t *hca)
503 {
504 	int			status;
505 	int			ndx;
506 	ibdma_port_t		*port;
507 	ibdma_hdl_impl_t	*hdl;
508 	ibdma_hdl_impl_t	*hdl_next;
509 
510 	ASSERT(mutex_owned(&ibdma->ms_hca_list_lock));
511 	ASSERT(hca != NULL);
512 
513 	rw_enter(&hca->ih_iou_rwlock, RW_WRITER);
514 
515 	/*
516 	 * All handles should have been de-registered, but release
517 	 * any that are outstanding.
518 	 */
519 	hdl = list_head(&hca->ih_hdl_list);
520 	while (hdl != NULL) {
521 		hdl_next = list_next(&hca->ih_hdl_list, hdl);
522 		list_remove(&hca->ih_hdl_list, hdl);
523 		cmn_err(CE_NOTE, "hca_fini, unexpected ibdma user handle"
524 		    " exists");
525 		kmem_free(hdl, sizeof (*hdl));
526 		hdl = hdl_next;
527 	}
528 	list_destroy(&hca->ih_hdl_list);
529 
530 	/*
531 	 * Un-register with the IBMF.
532 	 */
533 	for (ndx = 0; ndx < hca->ih_nports; ndx++) {
534 		port = &hca->ih_port[ndx];
535 		port->ip_hcap = NULL;
536 
537 		status = ibt_modify_port_byguid(hca->ih_iou_guid,
538 		    ndx+1, IBT_PORT_RESET_DEVMGT, 0);
539 		if (status != IBT_SUCCESS)
540 			cmn_err(CE_NOTE, "hca_fini, IBT modify port caps"
541 			    " error (%d)", status);
542 
543 		if (port->ip_ibmf_hdl == NULL)
544 			continue;
545 
546 		status = ibmf_tear_down_async_cb(port->ip_ibmf_hdl,
547 		    IBMF_QP_HANDLE_DEFAULT, 0);
548 		if (status != IBMF_SUCCESS)
549 			cmn_err(CE_NOTE, "hca_fini, IBMF tear down cb"
550 			    " error (%d)", status);
551 
552 		status = ibmf_unregister(&port->ip_ibmf_hdl, 0);
553 		if (status != IBMF_SUCCESS)
554 			cmn_err(CE_NOTE, "hca_fini, IBMF un-register"
555 			    " error (%d)", status);
556 		port->ip_ibmf_hdl = NULL;
557 	}
558 
559 	status = ibt_close_hca(hca->ih_ibt_hdl);
560 	if (status != IBT_SUCCESS)
561 		cmn_err(CE_NOTE, "hca_fini close error (%d)", status);
562 
563 	rw_exit(&hca->ih_iou_rwlock);
564 	rw_destroy(&hca->ih_iou_rwlock);
565 	kmem_free(hca, sizeof (ibdma_hca_t) +
566 	    (hca->ih_nports-1) * sizeof (ibdma_port_t));
567 }
568 
569 /* DM IBMF MAD handlers */
570 /*
571  * ibdma_create_resp_mad()
572  */
573 static void
574 ibdma_create_resp_mad(ibmf_msg_t *msgp)
575 {
576 	/*
577 	 * Allocate send buffer fix up hdr for response.
578 	 */
579 	msgp->im_msgbufs_send.im_bufs_mad_hdr =
580 	    kmem_zalloc(IBDMA_MAD_SIZE, KM_SLEEP);
581 
582 	msgp->im_msgbufs_send.im_bufs_cl_hdr = (uchar_t *)
583 	    msgp->im_msgbufs_send.im_bufs_mad_hdr + sizeof (ib_mad_hdr_t);
584 	msgp->im_msgbufs_send.im_bufs_cl_hdr_len = IBDMA_DM_MAD_HDR_SIZE;
585 	msgp->im_msgbufs_send.im_bufs_cl_data =
586 	    ((char *)msgp->im_msgbufs_send.im_bufs_cl_hdr +
587 	    IBDMA_DM_MAD_HDR_SIZE);
588 	msgp->im_msgbufs_send.im_bufs_cl_data_len =
589 	    IBDMA_MAD_SIZE - sizeof (ib_mad_hdr_t) - IBDMA_DM_MAD_HDR_SIZE;
590 	(void) memcpy(msgp->im_msgbufs_send.im_bufs_mad_hdr,
591 	    msgp->im_msgbufs_recv.im_bufs_mad_hdr, IBDMA_MAD_SIZE);
592 
593 	/*
594 	 * We may want to support a GRH since this is a GMP; not
595 	 * required for current SRP device manager platforms.
596 	 */
597 #if 0
598 	if (msgp->im_msg_flags & IBMF_MSG_FLAGS_GLOBAL_ADDRESS) {
599 		ib_gid_t	temp = msgp->im_global_addr.ig_recver_gid;
600 
601 		msgp->im_global_addr.ig_recver_gid =
602 		    msgp->im_global_addr.ig_sender_gid;
603 		msgp->im_global_addr.ig_sender_gid = temp;
604 	}
605 #endif
606 }
607 
608 /*
609  * ibdma_mad_send_cb()
610  */
611 /* ARGSUSED */
612 static void
613 ibdma_mad_send_cb(ibmf_handle_t ibmf_hdl, ibmf_msg_t *msgp, void *arg)
614 {
615 	/*
616 	 * Just free the buffers and release the message.
617 	 */
618 	if (msgp->im_msgbufs_send.im_bufs_mad_hdr != NULL) {
619 		kmem_free(msgp->im_msgbufs_send.im_bufs_mad_hdr,
620 		    IBDMA_MAD_SIZE);
621 		msgp->im_msgbufs_send.im_bufs_mad_hdr = NULL;
622 	}
623 	if (ibmf_free_msg(ibmf_hdl, &msgp) != IBMF_SUCCESS) {
624 		cmn_err(CE_NOTE, "mad_send_cb, IBMF message free error");
625 	}
626 }
627 
628 /*
629  * ibdma_mad_recv_cb()
630  */
631 static void
632 ibdma_mad_recv_cb(ibmf_handle_t ibmf_hdl, ibmf_msg_t *msgp, void *args)
633 {
634 	int		status;
635 	ib_mad_hdr_t	*in_mad;
636 	ib_mad_hdr_t	*out_mad;
637 	ibdma_port_t	*port = args;
638 
639 	ASSERT(msgp != NULL);
640 	ASSERT(port != NULL);
641 
642 	if (msgp->im_msg_status != IBMF_SUCCESS) {
643 		cmn_err(CE_NOTE, "mad_recv_cb, bad MAD receive status (%d)",
644 		    msgp->im_msg_status);
645 		goto drop;
646 	}
647 
648 	in_mad = msgp->im_msgbufs_recv.im_bufs_mad_hdr;
649 
650 	if (in_mad->MgmtClass != MAD_MGMT_CLASS_DEV_MGT) {
651 #ifdef	DEBUG_IBDMA
652 		cmn_err(CE_NOTE, "mad_recv_cb, MAD not of Dev Mgmt Class");
653 #endif
654 		goto drop;
655 	}
656 
657 	ibdma_create_resp_mad(msgp);
658 	out_mad = msgp->im_msgbufs_send.im_bufs_mad_hdr;
659 
660 	out_mad->R_Method = IB_DM_DEVMGT_METHOD_GET_RESP;
661 	out_mad->Status   = 0;
662 
663 	if (in_mad->R_Method == MAD_METHOD_SET) {
664 #ifdef	DEBUG_IBDMA
665 		cmn_err(CE_NOTE, "mad_recv_cb, no attributes supported"
666 		    " for set");
667 #endif
668 		out_mad->Status = MAD_STATUS_UNSUPP_METHOD_ATTR;
669 		goto send_resp;
670 	}
671 
672 	if (in_mad->R_Method != MAD_METHOD_GET) {
673 #ifdef	DEBUG_IBDMA
674 		cmn_err(CE_NOTE, "mad_recv_cb, no attributes supported"
675 		    " for set");
676 #endif
677 		out_mad->Status = MAD_STATUS_UNSUPP_METHOD;
678 		goto send_resp;
679 	}
680 
681 	/*
682 	 * Process a GET method.
683 	 */
684 	switch (b2h16(in_mad->AttributeID)) {
685 
686 	case IB_DM_ATTR_CLASSPORTINFO:
687 		ibdma_get_class_portinfo(msgp);
688 		break;
689 
690 	case IB_DM_ATTR_IO_UNITINFO:
691 		ibdma_get_io_unitinfo(port->ip_hcap, msgp);
692 		break;
693 
694 	case IB_DM_ATTR_IOC_CTRL_PROFILE:
695 		ibdma_get_ioc_profile(port->ip_hcap, msgp);
696 		break;
697 
698 	case IB_DM_ATTR_SERVICE_ENTRIES:
699 		ibdma_get_ioc_services(port->ip_hcap, msgp);
700 		break;
701 
702 	default:
703 		out_mad->Status = MAD_STATUS_UNSUPP_METHOD_ATTR;
704 		break;
705 	}
706 
707 send_resp:
708 	status = ibmf_msg_transport(ibmf_hdl, IBMF_QP_HANDLE_DEFAULT,
709 	    msgp, NULL, ibdma_mad_send_cb, NULL, 0);
710 	if (status != IBMF_SUCCESS) {
711 		cmn_err(CE_NOTE, "mad_recv_cb, send error (%d)", status);
712 		ibdma_mad_send_cb(ibmf_hdl, msgp, NULL);
713 	}
714 	return;
715 
716 drop:
717 	status = ibmf_free_msg(ibmf_hdl, &msgp);
718 	if (status != IBMF_SUCCESS) {
719 		cmn_err(CE_NOTE, "mad_recv_cb, error dropping (%d)",
720 		    status);
721 	}
722 }
723 
724 /*
725  * ibdma_get_class_portinfo()
726  */
727 static void
728 ibdma_get_class_portinfo(ibmf_msg_t *msg)
729 {
730 	ib_mad_classportinfo_t	*cpip;
731 
732 	cpip = (ib_mad_classportinfo_t *)msg->im_msgbufs_send.im_bufs_cl_data;
733 	bzero(cpip, sizeof (*cpip));
734 	cpip->BaseVersion   = MAD_CLASS_BASE_VERS_1;
735 	cpip->ClassVersion  = IB_DM_CLASS_VERSION_1;
736 	cpip->RespTimeValue = h2b32(IBDMA_DM_RESP_TIME);
737 }
738 
739 /*
740  * ibdma_get_io_unitinfo()
741  */
742 static void
743 ibdma_get_io_unitinfo(ibdma_hca_t *hca, ibmf_msg_t *msg)
744 {
745 	ib_dm_io_unitinfo_t	*uip;
746 
747 	uip = (ib_dm_io_unitinfo_t *)msg->im_msgbufs_send.im_bufs_cl_data;
748 	rw_enter(&hca->ih_iou_rwlock, RW_READER);
749 	bcopy(&hca->ih_iou, uip, sizeof (ib_dm_io_unitinfo_t));
750 	rw_exit(&hca->ih_iou_rwlock);
751 }
752 
753 /*
754  * ibdma_get_ioc_profile()
755  */
756 static void
757 ibdma_get_ioc_profile(ibdma_hca_t *hca, ibmf_msg_t *msg)
758 {
759 	ib_dm_ioc_ctrl_profile_t	*iocp;
760 	uint32_t			slot;
761 
762 	ASSERT(msg != NULL);
763 
764 	slot = b2h32(msg->im_msgbufs_recv.im_bufs_mad_hdr->AttributeModifier);
765 	iocp = (ib_dm_ioc_ctrl_profile_t *)
766 	    msg->im_msgbufs_send.im_bufs_cl_data;
767 	if (slot == 0 || slot > IBDMA_MAX_IOC) {
768 		msg->im_msgbufs_send.im_bufs_mad_hdr->Status =
769 		    MAD_STATUS_INVALID_FIELD;
770 		return;
771 	}
772 
773 	slot--;
774 	rw_enter(&hca->ih_iou_rwlock, RW_READER);
775 	if (ibdma_get_ioc_state(hca, slot) == IBDMA_IOC_PRESENT) {
776 		bcopy(&hca->ih_ioc[slot].ii_profile, iocp,
777 		    sizeof (ib_dm_ioc_ctrl_profile_t));
778 	} else {
779 		msg->im_msgbufs_send.im_bufs_mad_hdr->Status =
780 		    IB_DM_DEVMGT_MAD_STAT_NORESP;
781 	}
782 	rw_exit(&hca->ih_iou_rwlock);
783 }
784 
785 /*
786  * ibdma_get_ioc_services()
787  */
788 static void
789 ibdma_get_ioc_services(ibdma_hca_t *hca, ibmf_msg_t *msg)
790 {
791 	ib_dm_srv_t	*to_svcp;
792 	ib_dm_srv_t	*from_svcp;
793 	uint32_t	slot;
794 	uint8_t		hi;
795 	uint8_t		low;
796 
797 	ASSERT(msg != NULL);
798 
799 	slot = b2h32(msg->im_msgbufs_recv.im_bufs_mad_hdr->AttributeModifier);
800 	hi   = (slot >> 8) & 0x00FF;
801 	low  = slot  & 0x00FF;
802 	slot = (slot >> 16) & 0x0FFFF;
803 	if (slot == 0 || slot > IBDMA_MAX_IOC) {
804 		msg->im_msgbufs_send.im_bufs_mad_hdr->Status =
805 		    MAD_STATUS_INVALID_FIELD;
806 		return;
807 	}
808 
809 	slot--;
810 
811 	rw_enter(&hca->ih_iou_rwlock, RW_READER);
812 	if (ibdma_get_ioc_state(hca, slot) != IBDMA_IOC_PRESENT) {
813 		msg->im_msgbufs_send.im_bufs_mad_hdr->Status =
814 		    IB_DM_DEVMGT_MAD_STAT_NORESP;
815 		rw_exit(&hca->ih_iou_rwlock);
816 		return;
817 	}
818 
819 	if ((low > hi) || (hi - low > 4)) {
820 		msg->im_msgbufs_send.im_bufs_mad_hdr->Status =
821 		    MAD_STATUS_INVALID_FIELD;
822 		rw_exit(&hca->ih_iou_rwlock);
823 		return;
824 	}
825 
826 	if (hi > hca->ih_ioc[slot].ii_profile.ioc_service_entries) {
827 		msg->im_msgbufs_send.im_bufs_mad_hdr->Status =
828 		    MAD_STATUS_INVALID_FIELD;
829 		rw_exit(&hca->ih_iou_rwlock);
830 		return;
831 	}
832 
833 	to_svcp = (ib_dm_srv_t *)msg->im_msgbufs_send.im_bufs_cl_data;
834 	from_svcp = hca->ih_ioc[slot].ii_srvcs + low;
835 	bcopy(from_svcp, to_svcp, sizeof (ib_dm_srv_t) * (hi - low + 1));
836 	rw_exit(&hca->ih_iou_rwlock);
837 }
838 
839 
840 /*
841  * Client API internal helpers
842  */
843 
844 /*
845  * ibdma_hdl_to_ioc()
846  */
847 ibdma_hdl_impl_t *
848 ibdma_get_hdl_impl(ibdma_hdl_t hdl)
849 {
850 	ibdma_hca_t		*hca;
851 	ibdma_hdl_impl_t	*hdl_tmp = hdl;
852 	ibdma_hdl_impl_t	*hdl_impl;
853 
854 	ASSERT(mutex_owned(&ibdma->ms_hca_list_lock));
855 
856 	if (hdl_tmp == NULL) {
857 		cmn_err(CE_NOTE, "get_hdl_impl, NULL handle");
858 		return (NULL);
859 	}
860 
861 	hca = ibdma_find_hca(hdl_tmp->ih_iou_guid);
862 	if (hca == NULL) {
863 		cmn_err(CE_NOTE, "get_hdl_impl, invalid handle, bad IOU");
864 		return (NULL);
865 	}
866 
867 	hdl_impl = list_head(&hca->ih_hdl_list);
868 	while (hdl_impl != NULL) {
869 		if (hdl_impl == hdl_tmp) {
870 			break;
871 		}
872 		hdl_impl = list_next(&hca->ih_hdl_list, hdl_impl);
873 	}
874 	return (hdl_impl);
875 }
876 
877 /*
878  * ibdma_set_ioc_state()
879  *
880  * slot should be 0 based (not DM 1 based slot).
881  *
882  * I/O Unit write lock should be held outside of this function.
883  */
884 static void
885 ibdma_set_ioc_state(ibdma_hca_t *hca, int slot, ibdma_ioc_state_t state)
886 {
887 	uint8_t		cur;
888 	uint16_t	id;
889 
890 	cur = hca->ih_iou.iou_ctrl_list[slot >> 1];
891 	if (slot & 1) {
892 		cur = (cur & 0xF0) | state;
893 	} else {
894 		cur = (cur & 0x0F) | (state << 4);
895 	}
896 	hca->ih_iou.iou_ctrl_list[slot >> 1] = cur;
897 	id = b2h16(hca->ih_iou.iou_changeid);
898 	id++;
899 	hca->ih_iou.iou_changeid = h2b16(id);
900 #ifdef	DEBUG_IBDMA
901 	cmn_err(CE_NOTE, "set_ioc_state, slot offset(%d), value(%d)",
902 	    slot, hca->ih_iou.iou_ctrl_list[slot >> 1]);
903 #endif
904 }
905 
906 /*
907  * ibdma_get_ioc_state()
908  *
909  * slot should be 0 based (not DM 1 based slot).
910  *
911  * I/O Unit read lock should be held outside of this function.
912  */
913 static ibdma_ioc_state_t
914 ibdma_get_ioc_state(ibdma_hca_t *hca, int slot)
915 {
916 	uint8_t		cur;
917 
918 	if (slot >= IBDMA_MAX_IOC)
919 		return (0xFF);
920 
921 	cur = hca->ih_iou.iou_ctrl_list[slot >> 1];
922 	cur = slot & 1 ?  cur & 0x0F : cur >> 4;
923 	return (cur);
924 }
925 
926 /* CLIENT API Implementation */
927 /*
928  * ibdma_ioc_register()
929  *
930  */
931 ibdma_hdl_t
932 ibdma_ioc_register(ib_guid_t iou_guid, ib_dm_ioc_ctrl_profile_t *profile,
933 	ib_dm_srv_t *services)
934 {
935 	int			free_slot = -1;
936 	int			svc_entries;
937 	int			slot;
938 	ibdma_hca_t		*hca;
939 	ibdma_hdl_impl_t	*hdl;
940 
941 	if (profile == NULL || services == NULL) {
942 		cmn_err(CE_NOTE, "ioc_register, bad parameter");
943 		return (NULL);
944 	}
945 
946 	svc_entries = profile->ioc_service_entries;
947 	if (svc_entries == 0) {
948 		cmn_err(CE_NOTE, "ioc_register, bad profile no service");
949 		return (NULL);
950 	}
951 
952 	/*
953 	 * Find the associated I/O Unit.
954 	 */
955 	mutex_enter(&ibdma->ms_hca_list_lock);
956 	hca = ibdma_find_hca(iou_guid);
957 	if (hca == NULL) {
958 		mutex_exit(&ibdma->ms_hca_list_lock);
959 		cmn_err(CE_NOTE, "ioc_register, bad I/O Unit GUID (0x%llx)",
960 		    (u_longlong_t)iou_guid);
961 		return (NULL);
962 	}
963 
964 	rw_enter(&hca->ih_iou_rwlock, RW_WRITER);
965 	for (slot = 0; slot < IBDMA_MAX_IOC; slot++) {
966 		if (hca->ih_ioc[slot].ii_inuse == 0) {
967 			if (free_slot == -1) {
968 				free_slot = slot;
969 			}
970 			continue;
971 		}
972 
973 		if (profile->ioc_guid ==
974 		    hca->ih_ioc[slot].ii_profile.ioc_guid) {
975 			rw_exit(&hca->ih_iou_rwlock);
976 			mutex_exit(&ibdma->ms_hca_list_lock);
977 #ifdef	DEBUG_IBDMA
978 			cmn_err(CE_NOTE, "ioc_register, IOC previously"
979 			    " registered");
980 #endif
981 			return (NULL);
982 		}
983 	}
984 
985 	if (free_slot < 0) {
986 		rw_exit(&hca->ih_iou_rwlock);
987 		cmn_err(CE_NOTE, "ioc_register, error - I/O Unit full");
988 		return (NULL);
989 	}
990 #ifdef	DEBUG_IBDMA
991 	cmn_err(CE_NOTE, "ibdma_ioc_register, assigned to 0 based slot (%d)",
992 	    free_slot);
993 #endif
994 
995 	hca->ih_ioc[free_slot].ii_inuse = 1;
996 	hca->ih_ioc[free_slot].ii_slot  = free_slot;
997 	hca->ih_ioc[free_slot].ii_hcap  = hca;
998 
999 	/*
1000 	 * Allocate local copy of profile and services.
1001 	 */
1002 	hca->ih_ioc[free_slot].ii_srvcs =
1003 	    kmem_zalloc(sizeof (ib_dm_srv_t) * svc_entries, KM_SLEEP);
1004 	bcopy(profile, &hca->ih_ioc[free_slot].ii_profile,
1005 	    sizeof (ib_dm_ioc_ctrl_profile_t));
1006 	bcopy(services, hca->ih_ioc[free_slot].ii_srvcs,
1007 	    sizeof (ib_dm_srv_t) * svc_entries);
1008 
1009 	/*
1010 	 * Update the profile copy with the I/O controller slot assigned.
1011 	 * The slot occupies the lower 8 biths of the vendor ID/slot 32bit
1012 	 * field.
1013 	 */
1014 	profile->ioc_vendorid |= h2b32(free_slot);
1015 
1016 	ibdma_set_ioc_state(hca, free_slot, IBDMA_IOC_PRESENT);
1017 
1018 	hdl = kmem_alloc(sizeof (*hdl), KM_SLEEP);
1019 	hdl->ih_iou_guid = hca->ih_iou_guid;
1020 	hdl->ih_ioc_ndx = (uint8_t)free_slot;
1021 	list_insert_tail(&hca->ih_hdl_list, hdl);
1022 
1023 	rw_exit(&hca->ih_iou_rwlock);
1024 	mutex_exit(&ibdma->ms_hca_list_lock);
1025 
1026 	return ((ibdma_hdl_t)hdl);
1027 }
1028 
1029 /*
1030  * ibdma_ioc_unregister()
1031  *
1032  */
1033 ibdma_status_t
1034 ibdma_ioc_unregister(ibdma_hdl_t hdl)
1035 {
1036 	ibdma_ioc_t		*ioc;
1037 	ibdma_hca_t		*hca;
1038 	int			slot;
1039 	ibdma_hdl_impl_t	*hdl_tmp = hdl;
1040 	ibdma_hdl_impl_t	*hdl_impl;
1041 
1042 	if (hdl == NULL) {
1043 		cmn_err(CE_NOTE, "ioc_unregister, NULL handle");
1044 		return (IBDMA_BAD_PARAM);
1045 	}
1046 
1047 	mutex_enter(&ibdma->ms_hca_list_lock);
1048 	hca = ibdma_find_hca(hdl_tmp->ih_iou_guid);
1049 	if (hca == NULL) {
1050 		cmn_err(CE_NOTE, "ioc_unregsiter, invalid handle, IOU"
1051 		    " not found");
1052 		mutex_exit(&ibdma->ms_hca_list_lock);
1053 		return (IBDMA_BAD_PARAM);
1054 	}
1055 
1056 	hdl_impl = list_head(&hca->ih_hdl_list);
1057 	while (hdl_impl != NULL) {
1058 		if (hdl_impl == hdl_tmp) {
1059 			break;
1060 		}
1061 		hdl_impl = list_next(&hca->ih_hdl_list, hdl_impl);
1062 	}
1063 
1064 	if (hdl_impl == NULL) {
1065 		cmn_err(CE_NOTE, "ioc_unregsiter, invalid handle, not found");
1066 		mutex_exit(&ibdma->ms_hca_list_lock);
1067 		return (IBDMA_BAD_PARAM);
1068 	}
1069 
1070 	list_remove(&hca->ih_hdl_list, hdl_impl);
1071 
1072 	if (hdl_impl->ih_ioc_ndx >= IBDMA_MAX_IOC) {
1073 		cmn_err(CE_NOTE, "ioc_unregister, corrupted handle");
1074 		kmem_free(hdl_impl, sizeof (*hdl_impl));
1075 		mutex_exit(&ibdma->ms_hca_list_lock);
1076 		return (IBDMA_BAD_PARAM);
1077 	}
1078 	ioc = &hca->ih_ioc[hdl_impl->ih_ioc_ndx];
1079 	kmem_free(hdl_impl, sizeof (*hdl_impl));
1080 
1081 	if (ioc->ii_slot > IBDMA_MAX_IOC) {
1082 		cmn_err(CE_NOTE, "ioc_unregister, IOC corrupted, bad"
1083 		    " slot in IOC");
1084 		mutex_exit(&ibdma->ms_hca_list_lock);
1085 		return (IBDMA_BAD_PARAM);
1086 	}
1087 
1088 	rw_enter(&ioc->ii_hcap->ih_iou_rwlock, RW_WRITER);
1089 	if (ioc->ii_inuse == 0) {
1090 		rw_exit(&ioc->ii_hcap->ih_iou_rwlock);
1091 		mutex_exit(&ibdma->ms_hca_list_lock);
1092 		cmn_err(CE_NOTE, "ioc_unregister, slot not in use (%d)",
1093 		    ioc->ii_slot+1);
1094 		return (IBDMA_BAD_PARAM);
1095 	}
1096 
1097 	ASSERT(ioc->ii_srvcs != NULL);
1098 
1099 	slot = ioc->ii_slot;
1100 	hca  = ioc->ii_hcap;
1101 	kmem_free(ioc->ii_srvcs, sizeof (ib_dm_srv_t) *
1102 	    ioc->ii_profile.ioc_service_entries);
1103 	bzero(ioc, sizeof (ibdma_ioc_t));
1104 	ibdma_set_ioc_state(hca, slot, IBDMA_IOC_NOT_INSTALLED);
1105 
1106 	rw_exit(&hca->ih_iou_rwlock);
1107 	mutex_exit(&ibdma->ms_hca_list_lock);
1108 
1109 	return (IBDMA_SUCCESS);
1110 }
1111 
1112 /*
1113  * ibdma_ioc_update()
1114  *
1115  */
1116 ibdma_status_t
1117 ibdma_ioc_update(ibdma_hdl_t hdl, ib_dm_ioc_ctrl_profile_t *profile,
1118 	ib_dm_srv_t *services)
1119 {
1120 	ibdma_ioc_t		*ioc;
1121 	ibdma_hca_t		*hca;
1122 	ibdma_hdl_impl_t	*hdl_tmp = hdl;
1123 	ibdma_hdl_impl_t	*hdl_impl;
1124 
1125 	if (hdl == NULL) {
1126 		cmn_err(CE_NOTE, "ioc_update, NULL handle");
1127 		return (IBDMA_BAD_PARAM);
1128 	}
1129 
1130 	if (profile == NULL || services == NULL) {
1131 		cmn_err(CE_NOTE, "ioc_update, NULL parameter");
1132 		return (IBDMA_BAD_PARAM);
1133 	}
1134 
1135 	mutex_enter(&ibdma->ms_hca_list_lock);
1136 	hca = ibdma_find_hca(hdl_tmp->ih_iou_guid);
1137 	if (hca == NULL) {
1138 		cmn_err(CE_NOTE, "ioc_update, invalid handle, IOU not found");
1139 		mutex_exit(&ibdma->ms_hca_list_lock);
1140 		return (IBDMA_BAD_PARAM);
1141 	}
1142 
1143 	hdl_impl = list_head(&hca->ih_hdl_list);
1144 	while (hdl_impl != NULL) {
1145 		if (hdl_impl == hdl_tmp) {
1146 			break;
1147 		}
1148 		hdl_impl = list_next(&hca->ih_hdl_list, hdl_impl);
1149 	}
1150 
1151 	if (hdl_impl == NULL) {
1152 		cmn_err(CE_NOTE, "ioc_update, invalid handle, not found");
1153 		mutex_exit(&ibdma->ms_hca_list_lock);
1154 		return (IBDMA_BAD_PARAM);
1155 	}
1156 
1157 	if (hdl_impl->ih_ioc_ndx >= IBDMA_MAX_IOC) {
1158 		cmn_err(CE_NOTE, "ioc_update, corrupted handle");
1159 		mutex_exit(&ibdma->ms_hca_list_lock);
1160 		return (IBDMA_BAD_PARAM);
1161 	}
1162 	ioc = &hca->ih_ioc[hdl_impl->ih_ioc_ndx];
1163 
1164 	if (ioc->ii_slot >= IBDMA_MAX_IOC || ioc->ii_hcap == NULL) {
1165 		cmn_err(CE_NOTE, "ioc_update, bad handle (%p)",
1166 		    (void *)hdl);
1167 		mutex_exit(&ibdma->ms_hca_list_lock);
1168 		return (IBDMA_BAD_PARAM);
1169 	}
1170 
1171 	rw_enter(&ioc->ii_hcap->ih_iou_rwlock, RW_WRITER);
1172 	if (ioc->ii_inuse == 0) {
1173 		rw_exit(&ioc->ii_hcap->ih_iou_rwlock);
1174 		mutex_exit(&ibdma->ms_hca_list_lock);
1175 		cmn_err(CE_NOTE, "ioc_udate slot not in use (%d)",
1176 		    ioc->ii_slot+1);
1177 		return (IBDMA_BAD_PARAM);
1178 	}
1179 
1180 	ASSERT(ioc->ii_srvcs != NULL);
1181 
1182 	kmem_free(ioc->ii_srvcs, ioc->ii_profile.ioc_service_entries *
1183 	    sizeof (ib_dm_srv_t));
1184 	ioc->ii_srvcs = kmem_zalloc(profile->ioc_service_entries  *
1185 	    sizeof (ib_dm_srv_t), KM_SLEEP);
1186 
1187 	bcopy(profile, &ioc->ii_profile, sizeof (ib_dm_ioc_ctrl_profile_t));
1188 	bcopy(services, ioc->ii_srvcs, sizeof (ib_dm_srv_t) *
1189 	    profile->ioc_service_entries);
1190 	/*
1191 	 * Update the profile copy with the I/O controller slot assigned.
1192 	 * The slot occupies the lower 8 biths of the vendor ID/slot 32bit
1193 	 * field.
1194 	 */
1195 	profile->ioc_vendorid |= h2b32(ioc->ii_slot);
1196 	ibdma_set_ioc_state(ioc->ii_hcap, ioc->ii_slot, IBDMA_IOC_PRESENT);
1197 	rw_exit(&ioc->ii_hcap->ih_iou_rwlock);
1198 	mutex_exit(&ibdma->ms_hca_list_lock);
1199 
1200 	return (IBDMA_SUCCESS);
1201 }
1202