xref: /illumos-gate/usr/src/lib/smhba/common/SMHBAAPILIB.c (revision e8921a52c53ee69f7b65f054d9b2e886139daa59)
1 /*
2  * ************************************************************************
3  * Description
4  *	HBAAPILIB.c - Implements a sample common (wrapper) HBA API library
5  *
6  * License:
7  *	The contents of this file are subject to the SNIA Public License
8  *	Version 1.0 (the "License"); you may not use this file except in
9  *	compliance with the License. You may obtain a copy of the License at
10  *
11  *	/http://www.snia.org/English/Resources/Code/OpenSource.html
12  *
13  *	Software distributed under the License is distributed on an "AS IS"
14  *	basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
15  *	the License for the specific language governing rights and limitations
16  *	under the License.
17  *
18  * The Original Code is  SNIA HBA API Wrapper Library
19  *
20  * The Initial Developer of the Original Code is:
21  *	Benjamin F. Kuo, Troika Networks, Inc. (benk@troikanetworks.com)
22  *
23  * Contributor(s):
24  *	Tuan Lam, QLogic Corp. (t_lam@qlc.com)
25  *	Dan Willie, Emulex Corp. (Dan.Willie@emulex.com)
26  *	Dixon Hutchinson, Legato Systems, Inc. (dhutchin@legato.com)
27  *	David Dillard, VERITAS Software Corp. (david.dillard@veritas.com)
28  *
29  * ************************************************************************
30  *
31  * Adding on SM-HBA support
32  *
33  * The implementation includes Three different categories functions to support
34  * both HBAAPI and SM-HBA through the same library.
35  *
36  * SM-HBA unique interface:
37  *	1. CHECKLIBRARYANDVERSION(SMHBA) : match SMHBA VSL
38  *	   Or checking specifically if version is SMHBA beforehand.
39  *	2. resolved to ftable.smhbafunctiontable.{interface}
40  * HBAAPIV2 unique functions
41  *	1. CHECKLIBRARYANDVERSION(HBAAPIV2) : validate and match HBAAPI V2 VSL.
42  *	   Or checking specifically if version is HBAAPIV2 beforehand.
43  *	2. resolved to ftable.functiontable.{interface}
44  * Common interface between SM-HBA and HBAAPIV2.
45  *	1. CHECKLIBRARY() : to validate the VSL.
46  *	2. FUNCCOMMON macro to map the appropriate entry point table
47  *	    (union ftable).
48  *	3. If the interface is not supported by HBAAPI(Version 1)
49  *	   the funtiion ptr will be set to NULL.
50  * Common interface between HBAAPI and HBAAPIV2.
51  *	1. Check if version is not SMHBA).
52  *	2. ftable.functiontalbe.(interface)
53  *
54  * ************************************************************************
55  */
56 /*
57  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
58  * Use is subject to license terms.
59  */
60 
61 #ifdef WIN32
62 #include <windows.h>
63 #include <string.h>
64 /*
65  * Next define forces entry points in the dll to be exported
66  * See hbaapi.h to see what it does.
67  */
68 #define	HBAAPI_EXPORTS
69 #else
70 #include <dlfcn.h>
71 #include <strings.h>
72 #endif
73 #include <stdio.h>
74 #include <time.h>
75 #include "smhbaapi.h"
76 #include "vendorsmhbaapi.h"
77 #include <stdlib.h>
78 #ifdef USESYSLOG
79 #include <syslog.h>
80 #endif
81 #ifdef SOLARIS
82 #include <link.h>
83 #include <limits.h>
84 static int	*handle;
85 static Link_map *map, *mp;
86 #endif
87 
88 /*
89  * LIBRARY_NUM is a shortcut to figure out which library we need to call.
90  *  The top 16 bits of handle are the library index
91  */
92 #define	LIBRARY_NUM(handle)	((handle)>>16)
93 
94 /*
95  * VENDOR_HANDLE turns a global library handle into a vendor specific handle,
96  * with all upper 16 bits set to 0
97  */
98 #define	VENDOR_HANDLE(handle)	((handle)&0xFFFF)
99 
100 #define	HBA_HANDLE_FROM_LOCAL(library, vendor) \
101 				(((library)<<16) | ((vendor)&0x0000FFFF))
102 
103 int _hbaapi_debuglevel = 0;
104 #define	DEBUG(L, STR, A1, A2, A3)
105 
106 #if defined(USESYSLOG) && defined(USELOGFILE)
107 FILE *_hbaapi_debug_fd = NULL;
108 int _hbaapi_sysloginit = 0;
109 #undef DEBUG
110 #ifdef WIN32
111 #define	DEBUG(L, STR, A1, A2, A3)\
112     if ((L) <= _hbaapi_debuglevel) {\
113 	if (_hbaapi_sysloginit == 0) {\
114 	    openlog("HBAAPI", LOG_PID|LOG_ODELAY, LOG_USER);\
115 	    _hbaapi_sysloginit = 1;\
116 	}\
117 	syslog(LOG_INFO, (STR), (A1), (A2), (A3));\
118 	if (_hbaapi_debug_fd == NULL) {\
119 	    char _logFile[MAX_PATH]; \
120 	    GetTempPath(MAX_PATH, _logFile); \
121 	    strcat(_logFile, "HBAAPI.log"); \
122 	    _hbaapi_debug_fd = fopen(_logFile, "a");\
123 	}\
124 	if (_hbaapi_debug_fd != NULL) {\
125 	    fprintf(_hbaapi_debug_fd, #STR "\n", (A1), (A2), (A3));\
126 	}\
127 	}
128 #else /* WIN32 */
129 #define	DEBUG(L, STR, A1, A2, A3)\
130 	if ((L) <= _hbaapi_debuglevel) {\
131 	if (_hbaapi_sysloginit == 0) {\
132 	    openlog("HBAAPI", LOG_PID|LOG_ODELAY, LOG_USER);\
133 	    _hbaapi_sysloginit = 1;\
134 	}\
135 	syslog(LOG_INFO, (STR), (A1), (A2), (A3));\
136 	if (_hbaapi_debug_fd == NULL) {\
137 	    _hbaapi_debug_fd = fopen("/tmp/HBAAPI.log", "a");\
138 	}\
139 	if (_hbaapi_debug_fd != NULL) {\
140 	    fprintf(_hbaapi_debug_fd, #STR  "\n", (A1), (A2), (A3));\
141 	}\
142 	}
143 #endif /* WIN32 */
144 
145 #else /* Not both USESYSLOG and USELOGFILE */
146 #if defined(USESYSLOG)
147 int _hbaapi_sysloginit = 0;
148 #undef DEBUG
149 #define	DEBUG(L, STR, A1, A2, A3) \
150     if ((L) <= _hbaapi_debuglevel) {\
151 	if (_hbaapi_sysloginit == 0) {\
152 	    openlog("HBAAPI", LOG_PID|LOG_ODELAY, LOG_USER);\
153 	    _hbaapi_sysloginit = 1;\
154 	}\
155 	syslog(LOG_DEBUG, (STR), (A1), (A2), (A3));\
156 	}
157 #endif /* USESYSLOG */
158 #if defined(USELOGFILE)
159 FILE *_hbaapi_debug_fd = NULL;
160 #undef DEBUG
161 #ifdef WIN32
162 #define	DEBUG(L, STR, A1, A2, A3) \
163     if ((L) <= _hbaapi_debuglevel) {\
164 	if (_hbaapi_debug_fd == NULL) {\
165 	    char _logFile[MAX_PATH]; \
166 	    GetTempPath(MAX_PATH, _logFile); \
167 	    strcat(_logFile, "HBAAPI.log"); \
168 	    _hbaapi_debug_fd = fopen(_logFile, "a");\
169 	}\
170 	}
171 #else /* WIN32 */
172 #define	DEBUG(L, STR, A1, A2, A3) \
173     if ((L) <= _hbaapi_debuglevel) {\
174 	if (_hbaapi_debug_fd == NULL) {\
175 	    _hbaapi_debug_fd = fopen("/tmp/HBAAPI.log", "a");\
176 	}\
177 	if (_hbaapi_debug_fd != NULL) { \
178 	    fprintf(_hbaapi_debug_fd, #STR "\n", (A1), (A2), (A3));\
179 	}\
180 	}
181 #endif /* WIN32 */
182 #endif /* USELOGFILE */
183 #endif /* Not both USELOGFILE and USESYSLOG */
184 
185 #ifdef POSIX_THREADS
186 #include <pthread.h>
187 /*
188  * When multiple mutex's are grabed, they must be always be grabbed in
189  * the same order, or deadlock can result.  There are three levels
190  * of mutex's involved in this API.  If LL_mutex is grabbed, always grap
191  * it first.  If AL_mutex is grabbed, it may not be grabbed before
192  * LL_mutex.  If grabbed in a multi grab sequence, the mutex's protecting
193  * the callback lists must always be grabbed last and release before calling
194  * a vendor specific library function that might invoke a callback function
195  * on the same thread.
196  */
197 #define	GRAB_MUTEX(M)			grab_mutex(M)
198 #define	RELEASE_MUTEX(M)		release_mutex(M)
199 #define	RELEASE_MUTEX_RETURN(M, RET)	release_mutex(M); return (RET)
200 #elif defined(WIN32)
201 #define	GRAB_MUTEX(m)			EnterCriticalSection(m)
202 #define	RELEASE_MUTEX(m)		LeaveCriticalSection(m)
203 #define	RELEASE_MUTEX_RETURN(m, RET)	LeaveCriticalSection(m); return (RET)
204 #else
205 #define	GRAB_MUTEX(M)
206 #define	RELEASE_MUTEX(M)
207 #define	RELEASE_MUTEX_RETURN(M, RET)	return (RET)
208 #endif
209 
210 /*
211  * Vendor library information
212  */
213 typedef enum {
214     HBA_LIBRARY_UNKNOWN,
215     HBA_LIBRARY_LOADED,
216     HBA_LIBRARY_NOT_LOADED
217 } HBA_LIBRARY_STATUS;
218 
219 typedef enum {
220     UNKNOWN = 1,
221     SMHBA,
222     HBAAPIV2,
223     HBAAPI
224 } LIBRARY_VERSION;
225 
226 typedef struct hba_library_info {
227     struct hba_library_info
228 			*next;
229 #ifdef WIN32
230     HINSTANCE		hLibrary;		/* Handle to a loaded DLL */
231 #else
232     char		*LibraryName;
233     void*		hLibrary;		/* Handle to a loaded DLL */
234 #endif
235     char		*LibraryPath;
236     LIBRARY_VERSION	version;		/* resolve union */
237     HBA_UINT32		numOfAdapters;
238     union {
239 	SMHBA_ENTRYPOINTS   smhbafunctionTable;	/* smhba function pointers */
240 	HBA_ENTRYPOINTSV2   functionTable;	/* hba api function pointers */
241 	} ftable;
242     HBA_LIBRARY_STATUS	status;			/* info on this library */
243     HBA_UINT32		index;
244 } HBA_LIBRARY_INFO, *PHBA_LIBRARY_INFO;
245 
246 #define	ARE_WE_INITED() \
247 	if (_hbaapi_librarylist == NULL) { \
248 		return (HBA_STATUS_ERROR_NOT_LOADED); \
249 	}
250 HBA_LIBRARY_INFO *_hbaapi_librarylist = NULL;
251 HBA_UINT32 _hbaapi_total_library_count = 0;
252 #ifdef POSIX_THREADS
253 pthread_mutex_t _hbaapi_LL_mutex = PTHREAD_MUTEX_INITIALIZER;
254 #elif defined(WIN32)
255 CRITICAL_SECTION _hbaapi_LL_mutex;
256 #endif
257 
258 /*
259  * Macro to use the right function table between smhba and hbaapi.
260  */
261 #define	FUNCTABLE(lib_infop) \
262 	((lib_infop->version == SMHBA) ? \
263 	lib_infop->ftable.smhbafunctionTable : \
264 	lib_infop->ftable.functionTable);
265 
266 /*
267  * Macro to use the right function ptr between smhba and hbaapi function table.
268  * Should be used for an interface common to SM-HBA and HBAAPIV2.
269  */
270 #define	FUNCCOMMON(lib_infop, func) \
271 	((lib_infop->version == SMHBA) ? \
272 	lib_infop->ftable.smhbafunctionTable.func : \
273 	lib_infop->ftable.functionTable.func)
274 
275 /*
276  * Macro to use the hbaapi function ptr.
277  * Should be used for an interface applicable only HBAAPIV2.
278  */
279 #define	FUNCHBAAPIV2(lib_infop, func) \
280 	lib_infop->ftable.functionTable.func
281 
282 /*
283  * Macro to use the hbaapi function ptr.
284  * Should be used for an interface applicable only HBAAPIV2.
285  */
286 #define	FUNCSMHBA(lib_infop, func) \
287 	lib_infop->ftable.smhbafunctionTable.func
288 
289 /*
290  * Individual adapter (hba) information
291  */
292 typedef struct hba_adapter_info {
293     struct hba_adapter_info
294 			*next;
295     HBA_STATUS		GNstatus;	/* status from GetAdapterNameFunc */
296     char		*name;
297     HBA_WWN		nodeWWN;
298     HBA_LIBRARY_INFO	*library;
299     HBA_UINT32		index;
300 } HBA_ADAPTER_INFO;
301 
302 HBA_ADAPTER_INFO *_hbaapi_adapterlist = NULL;
303 HBA_UINT32 _hbaapi_total_adapter_count = 0;
304 #ifdef POSIX_THREADS
305 pthread_mutex_t _hbaapi_AL_mutex = PTHREAD_MUTEX_INITIALIZER;
306 #elif defined(WIN32)
307 CRITICAL_SECTION _hbaapi_AL_mutex;
308 #endif
309 
310 /*
311  * Call back registration
312  */
313 typedef struct hba_vendorcallback_elem {
314     struct hba_vendorcallback_elem
315 				*next;
316     HBA_CALLBACKHANDLE		vendorcbhandle;
317     HBA_LIBRARY_INFO		*lib_info;
318 } HBA_VENDORCALLBACK_ELEM;
319 
320 /*
321  * Each instance of HBA_ADAPTERCALLBACK_ELEM represents a call to one of
322  * "register" functions that apply to a particular adapter.
323  * HBA_ALLADAPTERSCALLBACK_ELEM is used just for HBA_RegisterForAdapterAddEvents
324  */
325 typedef struct hba_adaptercallback_elem {
326     struct hba_adaptercallback_elem
327 			*next;
328     HBA_LIBRARY_INFO	*lib_info;
329     void		*userdata;
330     HBA_CALLBACKHANDLE	vendorcbhandle;
331     void		(*callback)();
332 } HBA_ADAPTERCALLBACK_ELEM;
333 
334 typedef struct hba_alladapterscallback_elem {
335     struct hba_alladapterscallback_elem
336 				*next;
337     void			*userdata;
338     HBA_VENDORCALLBACK_ELEM	*vendorhandlelist;
339     void			(*callback)();
340 } HBA_ALLADAPTERSCALLBACK_ELEM;
341 
342 HBA_ALLADAPTERSCALLBACK_ELEM *_hbaapi_adapteraddevents_callback_list = NULL;
343 HBA_ADAPTERCALLBACK_ELEM *_hbaapi_adapterevents_callback_list = NULL;
344 HBA_ADAPTERCALLBACK_ELEM *_hbaapi_adapterportevents_callback_list = NULL;
345 HBA_ADAPTERCALLBACK_ELEM *_hbaapi_adapterportstatevents_callback_list = NULL;
346 HBA_ADAPTERCALLBACK_ELEM *_hbaapi_targetevents_callback_list = NULL;
347 HBA_ADAPTERCALLBACK_ELEM *_hbaapi_linkevents_callback_list = NULL;
348 
349 HBA_ALLADAPTERSCALLBACK_ELEM *_smhba_adapteraddevents_callback_list = NULL;
350 HBA_ADAPTERCALLBACK_ELEM *_smhba_adapterevents_callback_list = NULL;
351 HBA_ADAPTERCALLBACK_ELEM *_smhba_adapterportevents_callback_list = NULL;
352 HBA_ADAPTERCALLBACK_ELEM *_smhba_adapterportstatevents_callback_list = NULL;
353 HBA_ADAPTERCALLBACK_ELEM *_smhba_adapterphystatevents_callback_list = NULL;
354 HBA_ADAPTERCALLBACK_ELEM *_smhba_targetevents_callback_list = NULL;
355 
356 #ifdef POSIX_THREADS
357 /* mutex's to protect each list */
358 pthread_mutex_t _hbaapi_AAE_mutex = PTHREAD_MUTEX_INITIALIZER;
359 pthread_mutex_t _hbaapi_AE_mutex = PTHREAD_MUTEX_INITIALIZER;
360 pthread_mutex_t _hbaapi_APE_mutex = PTHREAD_MUTEX_INITIALIZER;
361 pthread_mutex_t _hbaapi_APSE_mutex = PTHREAD_MUTEX_INITIALIZER;
362 pthread_mutex_t _hbaapi_TE_mutex = PTHREAD_MUTEX_INITIALIZER;
363 pthread_mutex_t _hbaapi_LE_mutex = PTHREAD_MUTEX_INITIALIZER;
364 pthread_mutex_t _smhba_AAE_mutex = PTHREAD_MUTEX_INITIALIZER;
365 pthread_mutex_t _smhba_AE_mutex = PTHREAD_MUTEX_INITIALIZER;
366 pthread_mutex_t _smhba_APE_mutex = PTHREAD_MUTEX_INITIALIZER;
367 pthread_mutex_t _smhba_APSE_mutex = PTHREAD_MUTEX_INITIALIZER;
368 pthread_mutex_t _smhba_APHYSE_mutex = PTHREAD_MUTEX_INITIALIZER;
369 pthread_mutex_t _smhba_TE_mutex = PTHREAD_MUTEX_INITIALIZER;
370 pthread_mutex_t _smhba_LE_mutex = PTHREAD_MUTEX_INITIALIZER;
371 #elif defined(WIN32)
372 CRITICAL_SECTION _hbaapi_AAE_mutex;
373 CRITICAL_SECTION _hbaapi_AE_mutex;
374 CRITICAL_SECTION _hbaapi_APE_mutex;
375 CRITICAL_SECTION _hbaapi_APSE_mutex;
376 CRITICAL_SECTION _hbaapi_TE_mutex;
377 CRITICAL_SECTION _smhba_AAE_mutex;
378 CRITICAL_SECTION _smhba_AE_mutex;
379 CRITICAL_SECTION _smhba_APE_mutex;
380 CRITICAL_SECTION _smhba_APSE_mutex;
381 CRITICAL_SECTION _smhba_APHYSE_mutex;
382 CRITICAL_SECTION _smhba_TE_mutex;
383 CRITICAL_SECTION _hbaapi_LE_mutex;
384 #endif
385 
386 HBA_ADAPTERCALLBACK_ELEM **cb_lists_array[] = {
387 	&_hbaapi_adapterevents_callback_list,
388 	&_hbaapi_adapterportevents_callback_list,
389 	&_hbaapi_adapterportstatevents_callback_list,
390 	&_hbaapi_targetevents_callback_list,
391 	&_hbaapi_linkevents_callback_list,
392 	&_smhba_adapterevents_callback_list,
393 	&_smhba_adapterportevents_callback_list,
394 	&_smhba_adapterportstatevents_callback_list,
395 	&_smhba_adapterphystatevents_callback_list,
396 	&_smhba_targetevents_callback_list,
397 	NULL};
398 
399 /*
400  * Common library internal. Mutex handling
401  */
402 #ifdef POSIX_THREADS
403 static void
404 grab_mutex(pthread_mutex_t *mp) {
405 /* LINTED E_FUNC_SET_NOT_USED */
406     int ret;
407     if ((ret = pthread_mutex_lock(mp)) != 0) {
408 	perror("pthread_mutex_lock - HBAAPI:");
409 	DEBUG(1, "pthread_mutex_lock returned %d", ret, 0, 0);
410 	}
411 }
412 
413 static void
414 release_mutex(pthread_mutex_t *mp) {
415 /* LINTED E_FUNC_SET_NOT_USED */
416     int ret;
417     if ((ret = pthread_mutex_unlock(mp)) != 0) {
418 	perror("pthread_mutex_unlock - HBAAPI:");
419 	DEBUG(1, "pthread_mutex_unlock returned %d", ret, 0, 0);
420 	}
421 }
422 #endif
423 
424 /*
425  * Common library internal. Check library and return vendorhandle
426  */
427 static HBA_STATUS
428 HBA_CheckLibrary(HBA_HANDLE handle,
429     HBA_LIBRARY_INFO **lib_infopp,
430     HBA_HANDLE *vendorhandle) {
431 
432     HBA_UINT32		libraryIndex;
433     HBA_LIBRARY_INFO	*lib_infop;
434 
435     if (_hbaapi_librarylist == NULL) {
436 	return (HBA_STATUS_ERROR);
437 	}
438     libraryIndex = LIBRARY_NUM(handle);
439 
440     GRAB_MUTEX(&_hbaapi_LL_mutex);
441     for (lib_infop = _hbaapi_librarylist;
442 	lib_infop != NULL;
443 	lib_infop = lib_infop->next) {
444 	if (lib_infop->index == libraryIndex) {
445 	    if (lib_infop->status != HBA_LIBRARY_LOADED) {
446 		return (HBA_STATUS_ERROR);
447 	    }
448 	    *lib_infopp = lib_infop;
449 	    *vendorhandle = VENDOR_HANDLE(handle);
450 	    /* caller will release the mutex */
451 	    return (HBA_STATUS_OK);
452 	}
453 	}
454     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INVALID_HANDLE);
455 }
456 #define	CHECKLIBRARY() \
457 	status = HBA_CheckLibrary(handle, &lib_infop, &vendorHandle);\
458 	if (status != HBA_STATUS_OK) { \
459 	    return (status); \
460 	}
461 
462 #define	CHECKLIBRARYANDVERSION(ver) \
463 	status = HBA_CheckLibrary(handle, &lib_infop, &vendorHandle); \
464 	if (status != HBA_STATUS_OK) { \
465 	    return (status); \
466 	} else { \
467 	    if (ver != lib_infop->version) { \
468 		RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, \
469 		    HBA_STATUS_ERROR_INCOMPATIBLE); \
470 	    } \
471 	}
472 
473 /*
474  * freevendorhandlelist is called with _hbaapi_LL_mutex already held
475  */
476 static void
477 freevendorhandlelist(HBA_VENDORCALLBACK_ELEM *vhlist) {
478     HBA_VENDORCALLBACK_ELEM	*vhlp;
479     HBA_VENDORCALLBACK_ELEM	*vnext;
480     HBARemoveCallbackFunc	registeredfunc;
481 
482     for (vhlp = vhlist; vhlp != NULL; vhlp = vnext) {
483 	vnext = vhlp->next;
484 	registeredfunc =
485 	    FUNCCOMMON(vhlp->lib_info, RemoveCallbackHandler);
486 	if (registeredfunc == NULL) {
487 	    continue;
488 	}
489 	(registeredfunc)(vhlp->vendorcbhandle);
490 	free(vhlp);
491 	}
492 }
493 
494 static
495 HBA_STATUS
496 local_remove_callback(HBA_CALLBACKHANDLE cbhandle) {
497     HBA_ADAPTERCALLBACK_ELEM		***listp;
498     HBA_ADAPTERCALLBACK_ELEM		**lastp;
499     HBA_ALLADAPTERSCALLBACK_ELEM	**lap;
500     HBA_ALLADAPTERSCALLBACK_ELEM	*allcbp;
501     HBA_ADAPTERCALLBACK_ELEM		*cbp;
502     HBARemoveCallbackFunc		registeredfunc;
503     HBA_VENDORCALLBACK_ELEM		*vhlp;
504     HBA_VENDORCALLBACK_ELEM		*vnext;
505     int					found;
506     HBA_STATUS			status = HBA_STATUS_ERROR_INVALID_HANDLE;
507 
508 
509 	/* search through the simple lists first */
510     GRAB_MUTEX(&_hbaapi_AAE_mutex);
511     GRAB_MUTEX(&_hbaapi_AE_mutex);
512     GRAB_MUTEX(&_hbaapi_APE_mutex);
513     GRAB_MUTEX(&_hbaapi_APSE_mutex);
514     GRAB_MUTEX(&_hbaapi_TE_mutex);
515     GRAB_MUTEX(&_hbaapi_LE_mutex);
516     GRAB_MUTEX(&_smhba_AAE_mutex);
517     GRAB_MUTEX(&_smhba_AE_mutex);
518     GRAB_MUTEX(&_smhba_APE_mutex);
519     GRAB_MUTEX(&_smhba_APSE_mutex);
520     GRAB_MUTEX(&_smhba_TE_mutex);
521     for (listp = cb_lists_array, found = 0;
522 	    (found == 0 && *listp != NULL); listp++) {
523 	lastp = *listp;
524 	for (cbp = **listp; cbp != NULL; cbp = cbp->next) {
525 	    if (cbhandle != (HBA_CALLBACKHANDLE)cbp) {
526 		lastp = &(cbp->next);
527 		continue;
528 	    }
529 	    found = 1;
530 	    registeredfunc =
531 		FUNCCOMMON(cbp->lib_info, RemoveCallbackHandler);
532 	    if (registeredfunc == NULL) {
533 		break;
534 	    }
535 	    (registeredfunc)(cbp->vendorcbhandle);
536 	    *lastp = cbp->next;
537 	    free(cbp);
538 	    break;
539 	}
540 	}
541     RELEASE_MUTEX(&_hbaapi_LE_mutex);
542     RELEASE_MUTEX(&_hbaapi_TE_mutex);
543     RELEASE_MUTEX(&_hbaapi_APSE_mutex);
544     RELEASE_MUTEX(&_hbaapi_APE_mutex);
545     RELEASE_MUTEX(&_hbaapi_AE_mutex);
546     RELEASE_MUTEX(&_hbaapi_AAE_mutex);
547     RELEASE_MUTEX(&_smhba_AAE_mutex);
548     RELEASE_MUTEX(&_smhba_AE_mutex);
549     RELEASE_MUTEX(&_smhba_APE_mutex);
550     RELEASE_MUTEX(&_smhba_APSE_mutex);
551     RELEASE_MUTEX(&_smhba_TE_mutex);
552 
553     if (found != 0) {
554 	if (registeredfunc == NULL) {
555 	    return (HBA_STATUS_ERROR_NOT_SUPPORTED);
556 	}
557 	return (HBA_STATUS_OK);
558 	}
559 
560     GRAB_MUTEX(&_hbaapi_AAE_mutex);
561 	/*
562 	 * if it wasnt in the simple lists,
563 	 * look in the list for adapteraddevents
564 	 */
565     lap = &_hbaapi_adapteraddevents_callback_list;
566     for (allcbp = _hbaapi_adapteraddevents_callback_list;
567 	    allcbp != NULL;
568 	    allcbp = allcbp->next) {
569 	if (cbhandle != (HBA_CALLBACKHANDLE)allcbp) {
570 	    lap = &allcbp->next;
571 	    continue;
572 	}
573 	for (vhlp = allcbp->vendorhandlelist; vhlp != NULL; vhlp = vnext) {
574 	    vnext = vhlp->next;
575 	    /* should be HBAAPIV2 VSL to get to here */
576 	    registeredfunc =
577 		    vhlp->lib_info->ftable.functionTable.RemoveCallbackHandler;
578 	    if (registeredfunc == NULL) {
579 		continue;
580 	    }
581 	    (registeredfunc)(vhlp->vendorcbhandle);
582 	    free(vhlp);
583 	}
584 	*lap = allcbp->next;
585 	free(allcbp);
586 	status = HBA_STATUS_OK;
587 	break;
588 	}
589     RELEASE_MUTEX(&_hbaapi_AAE_mutex);
590 
591 	/* now search smhba adapteradd events. */
592     GRAB_MUTEX(&_smhba_AAE_mutex);
593     lap = &_smhba_adapteraddevents_callback_list;
594     for (allcbp = _smhba_adapteraddevents_callback_list;
595 	allcbp != NULL;
596 	allcbp = allcbp->next) {
597 	if (cbhandle != (HBA_CALLBACKHANDLE)allcbp) {
598 	    lap = &allcbp->next;
599 	    continue;
600 	}
601 	for (vhlp = allcbp->vendorhandlelist; vhlp != NULL; vhlp = vnext) {
602 	    vnext = vhlp->next;
603 	    /* should be SMHBA VSL to get to here */
604 	    registeredfunc =
605 		    vhlp->lib_info->
606 			ftable.smhbafunctionTable.RemoveCallbackHandler;
607 	    if (registeredfunc == NULL) {
608 		continue;
609 	    }
610 	    (registeredfunc)(vhlp->vendorcbhandle);
611 	    free(vhlp);
612 	}
613 	*lap = allcbp->next;
614 	free(allcbp);
615 	status = HBA_STATUS_OK;
616 	break;
617 	}
618     RELEASE_MUTEX(&_smhba_AAE_mutex);
619 
620     return (status);
621 }
622 
623 /* LINTED E_STATIC_UE_STATIC_UNUSED */
624 static char wwn_str1[17];
625 /* LINTED E_STATIC_UE_STATIC_UNUSED */
626 static char wwn_str2[17];
627 /* LINTED E_STATIC_UE_STATIC_UNUSED */
628 static char wwn_str3[17];
629 #define	WWN2STR1(wwn) WWN2str(wwn_str1, (wwn))
630 #define	WWN2STR2(wwn) WWN2str(wwn_str2, (wwn))
631 #define	WWN2STR3(wwn) WWN2str(wwn_str3, (wwn))
632 static char *
633 /* LINTED E_STATIC_UE_STATIC_UNUSED */
634 WWN2str(char *buf, HBA_WWN *wwn) {
635     int j;
636     unsigned char *pc = (unsigned char *)&(wwn->wwn[0]);
637     buf[0] = '\0';
638     for (j = 0; j < 16; j += 2) {
639 		(void) sprintf(&buf[j], "%02X", (int)*pc++);
640 	}
641     return (buf);
642 }
643 
644 #ifdef WIN32
645 BOOL APIENTRY
646 DllMain(HANDLE hModule,
647     DWORD  ul_reason_for_call,
648     LPVOID lpReserved)
649 {
650 	switch (ul_reason_for_call) {
651 	case DLL_PROCESS_ATTACH:
652 		break;
653 	case DLL_PROCESS_DETACH:
654 		break;
655 	case DLL_THREAD_ATTACH:
656 	case DLL_THREAD_DETACH:
657 		break;
658 	}
659 	return (TRUE);
660 }
661 #endif
662 
663 /*
664  * Read in the config file and load all the specified vendor specific
665  * libraries and perform the function registration exercise
666  */
667 HBA_STATUS
668 HBA_LoadLibrary()
669 {
670 	HBARegisterLibraryFunc RegisterFunc;
671 	HBARegisterLibraryV2Func RegisterV2Func;
672 	SMHBARegisterLibraryFunc RegisterSMHBAFunc;
673 	HBALoadLibraryFunc	LoadLibraryFunc;
674 	HBAGetVersionFunc	GetVersionFunc;
675 #ifdef	POSIX_THREADS
676 	int			ret;
677 #endif
678 	HBA_STATUS		status;
679 	HBA_UINT32		libversion;
680 
681 	/* Open configuration file from known location */
682 #ifdef WIN32
683 	LONG		lStatus;
684 	HKEY		hkSniaHba, hkVendorLib;
685 	FILETIME		ftLastWriteTime;
686 	TCHAR		cSubKeyName[256];
687 	DWORD		i, dwSize, dwType;
688 	BYTE		byFileName[MAX_PATH];
689 	HBA_LIBRARY_INFO	*lib_infop;
690 
691 	if (_hbaapi_librarylist != NULL) {
692 		/* this is an app programming error */
693 		return (HBA_STATUS_ERROR);
694 	}
695 
696 	lStatus = RegOpenKeyEx(HKEY_LOCAL_MACHINE, "SOFTWARE\\SNIA\\HBA",
697 	    0, KEY_READ, &hkSniaHba);
698 	if (lStatus != ERROR_SUCCESS) {
699 		/* ???Opportunity to send error msg, configuration error */
700 		return (HBA_STATUS_ERROR);
701 	}
702 	/*
703 	 * Enumerate all the subkeys. These have the form:
704 	 * HKLM\Software\SNIA\HBA\<Vendor id> - note that we don't care
705 	 * what the vendor id is
706 	 */
707 	for (i = 0; ; i++) {
708 		dwSize = 255;	/* how big the buffer is */
709 		lStatus = RegEnumKeyEx(hkSniaHba, i,
710 		    (char *)&cSubKeyName, &dwSize, NULL,
711 		    NULL, NULL, &ftLastWriteTime);
712 	if (lStatus == ERROR_NO_MORE_ITEMS) {
713 		break;	/* we're done */
714 	} else if (lStatus == ERROR_MORE_DATA) { /* buffer not big enough */
715 		/* do whatever */
716 		;
717 	}
718 	/* Now open the subkey that pertains to this vendor's library */
719 	lStatus = RegOpenKeyEx(hkSniaHba, cSubKeyName, 0, KEY_READ,
720 	    &hkVendorLib);
721 	if (lStatus != ERROR_SUCCESS) {
722 		RegCloseKey(hkSniaHba);
723 	    /* ???Opportunity to send error msg, installation error */
724 		return (HBA_STATUS_ERROR);
725 		/*
726 		 * you may want to return something
727 		 * else or keep trying
728 		 */
729 	}
730 	/*
731 	 * The name of the library is contained in a REG_SZ Value
732 	 * keyed to "LibraryFile"
733 	 */
734 	dwSize = MAX_PATH;
735 	lStatus = RegQueryValueEx(hkVendorLib, "LibraryFile", NULL, &dwType,
736 	    byFileName, &dwSize);
737 	if (lStatus != ERROR_SUCCESS) {
738 		RegCloseKey(hkVendorLib);
739 	    /* ???Opportunity to send error msg, installation error */
740 		continue;
741 	}
742 	lib_infop = (HBA_LIBRARY_INFO *)calloc(1, sizeof (HBA_LIBRARY_INFO));
743 	if (lib_infop == NULL) {
744 	    /* what is the right thing to do in MS land??? */
745 		RegCloseKey(hkVendorLib);
746 		/* ???Opportunity to send error msg, installation error */
747 		return (HBA_STATUS_ERROR);
748 	}
749 	lib_infop->status = HBA_LIBRARY_NOT_LOADED;
750 	lib_infop->next = _hbaapi_librarylist;
751 	lib_infop->index = _hbaapi_total_library_count;
752 	_hbaapi_total_library_count++;
753 	_hbaapi_librarylist = lib_infop;
754 
755 	/* Now I can try to load the library */
756 	lib_infop->hLibrary = LoadLibrary(byFileName);
757 	if (lib_infop->hLibrary == NULL) {
758 	    /* printf("unable to load library %s\n", librarypath); */
759 	    /* ???Opportunity to send error msg, installation error */
760 		goto dud_library;
761 	}
762 	lib_infop->LibraryPath = strdup(byFileName);
763 	DEBUG(1, "HBAAPI loading: %s\n", byFileName, 0, 0);
764 
765 	RegisterSMHBAFunc = (SMHBARegisterLibraryFunc)
766 	    GetProcAddress(lib_infop->hLibrary, "SMHBA_RegisterLibrary");
767 	if (RegisterSMHBAFunc != NULL) {
768 		status = ((RegisterSMHBAFunc)(SMHBA_ENTRYPOINTS *)
769 		    (&lib_infop->ftable.smhbafunctionTable));
770 		if (status != HBA_STATUS_OK) {
771 			/* library not loaded */
772 			/* ???Opportunity to send error msg, library error? */
773 			goto dud_library;
774 		} else {
775 			lib_infop->version = SMHBA;
776 		}
777 	} else {
778 	    /* Call the registration function to get the list of pointers */
779 		RegisterV2Func = (HBARegisterLibraryV2Func)GetProcAddress(
780 		    lib_infop->hLibrary, "HBA_RegisterLibraryV2");
781 		if (RegisterV2Func != NULL) {
782 		/*
783 		 * Load the function pointers directly into
784 		 * the table of functions
785 		 */
786 		status = ((RegisterV2Func)
787 		    (HBA_ENTRYPOINTSV2 *)(&lib_infop->ftable.functionTable));
788 		if (status != HBA_STATUS_OK) {
789 		    /* library not loaded */
790 		    /* ???Opportunity to send error msg, library error? */
791 			goto dud_library;
792 		} else {
793 			lib_infop->version = HBAAPIV2;
794 		}
795 		} else {
796 		/* Maybe the vendor library is only Rev1 */
797 		RegisterFunc = (HBARegisterLibraryFunc)
798 		    GetProcAddress(lib_infop->hLibrary, "HBA_RegisterLibrary");
799 		if (RegisterFunc == NULL) {
800 		    /* ???Opportunity to send error msg, library error? */
801 			goto dud_library;
802 		}
803 		/*
804 		 * Load the function points directly into
805 		 * the Rev 2 table of functions
806 		 */
807 		status = ((RegisterFunc)(
808 		    (HBA_ENTRYPOINTS *)(&lib_infop->ftable.functionTable)));
809 		if (status != HBA_STATUS_OK) {
810 		    /* library not loaded */
811 		    /* ???Opportunity to send error msg, library error? */
812 			goto dud_library;
813 		} else {
814 			lib_infop->version = HBAAPI;
815 		}
816 		}
817 	}
818 
819 	/* successfully loaded library */
820 	/*
821 	 * SM-HBA and HBAAPI has a seperate handler for GetVersion but
822 	 * they have the same function signature so use the same variable here.
823 	 */
824 	GetVersionFunc = FUNCCOMMON(lib_infop, GetVersionHandler);
825 	if (GetVersionFunc != NULL) {
826 		if (lib_infop->version == SMHBA) {
827 		/* Check the version of this library before loading */
828 		libversion = ((GetVersionFunc)());
829 #ifdef NOTDEF /* save for a later time... when it matters */
830 		if (libversion < SMHBA_LIBVERSION) {
831 			goto dud_library;
832 		}
833 #endif
834 		} else {
835 		/* Check the version of this library before loading */
836 	    /* Actually... This wrapper is compatible with version 1 */
837 		libversion = ((GetVersionFunc)());
838 #ifdef NOTDEF /* save for a later time... when it matters */
839 		if (libversion < HBA_LIBVERSION) {
840 			goto dud_library;
841 		}
842 #endif
843 		}
844 	} else {
845 	    /* ???Opportunity to send error msg, library error? */
846 		goto dud_library;
847 	}
848 
849 	LoadLibraryFunc = FUNCCOMMON(lib_infop, LoadLibraryHandler);
850 	if (LoadLibraryFunc == NULL) {
851 	    /* Hmmm, dont we need to flag this in a realy big way??? */
852 	    /* How about messages to the system event logger ??? */
853 	    /* ???Opportunity to send error msg, library error? */
854 		goto dud_library;
855 	}
856 	/* Initialize this library */
857 	status = ((LoadLibraryFunc)());
858 	if (status != HBA_STATUS_OK) {
859 	    /* ???Opportunity to send error msg, library error? */
860 		continue;
861 	}
862 	/* successfully loaded library */
863 	lib_infop->status = HBA_LIBRARY_LOADED;
864 
865 	dud_library: /* its also just the end of the loop */
866 	RegCloseKey(hkVendorLib);
867 	}
868 	RegCloseKey(hkSniaHba);
869 
870 #else /* Unix as opposed to Win32 */
871 	FILE		*hbaconf;
872 	char		fullline[512];		/* line read from HBA.conf */
873 	char		*libraryname;		/* Read in from file HBA.conf */
874 	char		*librarypath;		/* Read in from file HBA.conf */
875 	char		hbaConfFilePath[256];
876 	char		*charPtr;
877 	HBA_LIBRARY_INFO	*lib_infop;
878 
879 	GRAB_MUTEX(&_hbaapi_LL_mutex);
880 	if (_hbaapi_librarylist != NULL) {
881 		(void) fprintf(stderr,
882 		    "HBA_LoadLibrary: previously unfreed "
883 		    "libraries exist, call HBA_FreeLibrary().\n");
884 		RELEASE_MUTEX(&_hbaapi_LL_mutex);
885 		return (HBA_STATUS_ERROR);
886 	}
887 
888 	(void) strcpy(hbaConfFilePath, "/etc/smhba.conf");
889 
890 	if ((hbaconf = fopen(hbaConfFilePath, "r")) == NULL) {
891 		(void) printf("Cannot open %s\n", hbaConfFilePath);
892 		RELEASE_MUTEX(&_hbaapi_LL_mutex);
893 		return (HBA_STATUS_ERROR);
894 	}
895 
896 	/* Read in each line and load library */
897 	while ((hbaconf != NULL) &&
898 	    (fgets(fullline, sizeof (fullline), hbaconf))) {
899 		/* Skip the comments... */
900 		if ((fullline[0] == '#') || (fullline[0] == '\n')) {
901 			continue;
902 		}
903 
904 	/* grab first 'thing' in line (if its there) */
905 	if ((libraryname = strtok(fullline, " \t\n")) != NULL) {
906 		if (strlen(libraryname) >= 64) {
907 			(void) fprintf(stderr,
908 			    "Library name(%s) in %s is > 64 characters\n",
909 			    libraryname, hbaConfFilePath);
910 		}
911 	}
912 	/* grab second 'thing' in line (if its there) */
913 	if ((librarypath = strtok(NULL, " \t\n")) != NULL) {
914 		if (strlen(librarypath) >= 256) {
915 		(void) fprintf(stderr,
916 		    "Library path(%s) in %s is > 256 characters\n",
917 		    librarypath, hbaConfFilePath);
918 		}
919 	}
920 
921 	/* there should be no more 'things' in the line */
922 	if ((charPtr = strtok(NULL, " \n\t")) != NULL) {
923 		(void) fprintf(stderr, "Extraneous characters (\"%s\") in %s\n",
924 		    charPtr, hbaConfFilePath);
925 	}
926 
927 	/* Continue to the next line if library name or path is invalid */
928 	if (libraryname == NULL ||
929 	    strlen(libraryname) == 0 ||
930 	    librarypath == NULL ||
931 	    (strlen(librarypath) == 0)) {
932 		continue;
933 	}
934 
935 	/*
936 	 * Special case....
937 	 * Look for loglevel
938 	 */
939 	if (strcmp(libraryname, "debuglevel") == 0) {
940 		_hbaapi_debuglevel = strtol(librarypath, NULL, 10);
941 	    /* error handling does the right thing automagically */
942 		continue;
943 	}
944 
945 	lib_infop = (HBA_LIBRARY_INFO *)calloc(1, sizeof (HBA_LIBRARY_INFO));
946 	if (lib_infop == NULL) {
947 		(void) fprintf(stderr, "HBA_LoadLibrary: out of memeory\n");
948 		RELEASE_MUTEX(&_hbaapi_LL_mutex);
949 		return (HBA_STATUS_ERROR);
950 	}
951 	lib_infop->status = HBA_LIBRARY_NOT_LOADED;
952 	lib_infop->LibraryName = strdup(libraryname);
953 	lib_infop->LibraryPath = strdup(librarypath);
954 	lib_infop->numOfAdapters = 0;
955 	lib_infop->version = UNKNOWN;
956 	lib_infop->index = _hbaapi_total_library_count;
957 	_hbaapi_total_library_count++;
958 	lib_infop->next = _hbaapi_librarylist;
959 	_hbaapi_librarylist = lib_infop;
960 
961 	/* Load the DLL now */
962 	if ((lib_infop->hLibrary = dlopen(librarypath, RTLD_LAZY)) == NULL) {
963 	    /* printf("unable to load library %s\n", librarypath); */
964 		continue;
965 	}
966 	/* Call the registration function to get the list of pointers */
967 	RegisterSMHBAFunc = (SMHBARegisterLibraryFunc)
968 	    dlsym(lib_infop->hLibrary, "SMHBA_RegisterLibrary");
969 	if (RegisterSMHBAFunc != NULL) {
970 		/*
971 		 * Load the function points directly into
972 		 * the table of functions
973 		 */
974 		status = ((RegisterSMHBAFunc)
975 		    (&lib_infop->ftable.smhbafunctionTable));
976 		if (status != HBA_STATUS_OK) {
977 			/* library not loaded */
978 			continue;
979 		} else {
980 			lib_infop->version = SMHBA;
981 		}
982 	} else {
983 		RegisterV2Func = (HBARegisterLibraryV2Func)
984 		    dlsym(lib_infop->hLibrary, "HBA_RegisterLibraryV2");
985 		if (RegisterV2Func != NULL) {
986 		/*
987 		 * Load the function points directly into
988 		 * the table of functions
989 		 */
990 		status = ((RegisterV2Func)((HBA_ENTRYPOINTSV2 *)
991 		    (&lib_infop->ftable.functionTable)));
992 		if (status != HBA_STATUS_OK) {
993 		    /* library not loaded */
994 			continue;
995 		} else {
996 			lib_infop->version = HBAAPIV2;
997 		}
998 		} else {
999 		/* Maybe the vendor library is only Rev1 */
1000 		RegisterFunc = (HBARegisterLibraryFunc)
1001 		    dlsym(lib_infop->hLibrary, "HBA_RegisterLibrary");
1002 		if (RegisterFunc == NULL) {
1003 		    /* This function is required */
1004 			(void) fprintf(stderr,
1005 			    "HBA_LoadLibrary: vendor specific RegisterLibrary "
1006 			    "function not found.  lib: %s\n", librarypath);
1007 			DEBUG(1, "HBA_LoadLibrary: vendor specific "
1008 			    "RegisterLibrary function not found.  lib: %s\n",
1009 			    librarypath, 0, 0);
1010 			continue;
1011 		}
1012 		/*
1013 		 * Load the function points directly into
1014 		 * the table of functions
1015 		 */
1016 		status = ((RegisterFunc)
1017 		    ((HBA_ENTRYPOINTS *)(&lib_infop->ftable.functionTable)));
1018 		if (status != HBA_STATUS_OK) {
1019 		    /* library not loaded */
1020 			(void) fprintf(stderr,
1021 			    "HBA_LoadLibrary: vendor specific RegisterLibrary "
1022 			    "function encountered an error.  lib: %s\n",
1023 			    librarypath);
1024 			DEBUG(1,
1025 			    "HBA_LoadLibrary: vendor specific RegisterLibrary "
1026 			    "function encountered an error. lib: %s\n",
1027 			    librarypath, 0, 0);
1028 			continue;
1029 		} else {
1030 			lib_infop->version = HBAAPI;
1031 		}
1032 		}
1033 	}
1034 
1035 	/* successfully loaded library */
1036 	/*
1037 	 * SM-HBA and HBAAPI has a seperate handler for GetVersion but
1038 	 * they have the same function signature so use the same variable here.
1039 	 */
1040 	if ((GetVersionFunc = FUNCCOMMON(lib_infop, GetVersionHandler))
1041 	    == NULL) {
1042 		continue;
1043 	}
1044 	if (lib_infop->version == SMHBA) {
1045 		libversion = ((GetVersionFunc)());
1046 		if (libversion < SMHBA_LIBVERSION) {
1047 			(void) printf("Library version mismatch."
1048 			    "Got %d expected %d.\n",
1049 			    libversion, SMHBA_LIBVERSION);
1050 			continue;
1051 		}
1052 	} else {
1053 		libversion = ((GetVersionFunc)());
1054 	    /* Check the version of this library before loading */
1055 	    /* Actually... This wrapper is compatible with version 1 */
1056 		if (libversion < HBA_LIBVERSION) {
1057 			(void) printf("Library version mismatch."
1058 			    "Got %d expected %d.\n",
1059 			    libversion, HBA_LIBVERSION);
1060 			continue;
1061 		}
1062 	}
1063 
1064 	DEBUG(1, "%s libversion = %d", librarypath, libversion, 0);
1065 	LoadLibraryFunc = FUNCCOMMON(lib_infop, LoadLibraryHandler);
1066 	if (LoadLibraryFunc == NULL) {
1067 	    /* this function is required */
1068 		(void) fprintf(stderr,
1069 		    "HBA_LoadLibrary: vendor specific LoadLibrary "
1070 		    "function not found.  lib: %s\n", librarypath);
1071 		DEBUG(1, "HBA_LoadLibrary: vendor specific LoadLibrary "
1072 		    "function not found.  lib: %s\n", librarypath, 0, 0);
1073 		continue;
1074 	}
1075 	/* Initialize this library */
1076 	if ((status = ((LoadLibraryFunc)())) != HBA_STATUS_OK) {
1077 	    /* maybe this should be a printf so that we CANNOT miss it */
1078 		(void) fprintf(stderr,
1079 		    "HBA_LoadLibrary: Encounterd and error loading: %s",
1080 		    librarypath);
1081 		DEBUG(1, "Encounterd and error loading: %s", librarypath, 0, 0);
1082 		DEBUG(1, "  HBA_STATUS: %d", status, 0, 0);
1083 		continue;
1084 	}
1085 	/* successfully loaded library */
1086 	lib_infop->status = HBA_LIBRARY_LOADED;
1087 	}
1088 #endif /* WIN32 or UNIX */
1089 #ifdef POSIX_THREADS
1090 	/*
1091 	 * The _hbaapi_LL_mutex is already grabbed to proctect the caller of
1092 	 * HBA_FreeLibrary() during loading.
1093 	 * The mutexes are already initialized
1094 	 * with PTHREAD_MUTEX_INITIALIZER.  Do we need to init again?
1095 	 * Keeping the code from HBAAPI source...
1096 	 */
1097 	ret = pthread_mutex_init(&_hbaapi_AL_mutex, NULL);
1098 	if (ret == 0) {
1099 		ret = pthread_mutex_init(&_hbaapi_AAE_mutex, NULL);
1100 	}
1101 	if (ret == 0) {
1102 		ret = pthread_mutex_init(&_hbaapi_AE_mutex, NULL);
1103 	}
1104 	if (ret == 0) {
1105 		ret = pthread_mutex_init(&_hbaapi_APE_mutex, NULL);
1106 	}
1107 	if (ret == 0) {
1108 		ret = pthread_mutex_init(&_hbaapi_APSE_mutex, NULL);
1109 	}
1110 	if (ret == 0) {
1111 		ret = pthread_mutex_init(&_hbaapi_TE_mutex, NULL);
1112 	}
1113 	if (ret == 0) {
1114 		ret = pthread_mutex_init(&_smhba_AAE_mutex, NULL);
1115 	}
1116 	if (ret == 0) {
1117 		ret = pthread_mutex_init(&_smhba_AE_mutex, NULL);
1118 	}
1119 	if (ret == 0) {
1120 		ret = pthread_mutex_init(&_smhba_APE_mutex, NULL);
1121 	}
1122 	if (ret == 0) {
1123 		ret = pthread_mutex_init(&_smhba_APSE_mutex, NULL);
1124 	}
1125 	if (ret == 0) {
1126 		ret = pthread_mutex_init(&_smhba_TE_mutex, NULL);
1127 	}
1128 	if (ret == 0) {
1129 		ret = pthread_mutex_init(&_hbaapi_LE_mutex, NULL);
1130 	}
1131 	if (ret != 0) {
1132 		perror("pthread_mutex_init - HBA_LoadLibrary");
1133 		RELEASE_MUTEX(&_hbaapi_LL_mutex);
1134 		return (HBA_STATUS_ERROR);
1135 	}
1136 	RELEASE_MUTEX(&_hbaapi_LL_mutex);
1137 #elif defined(WIN32)
1138 	InitializeCriticalSection(&_hbaapi_LL_mutex);
1139 	InitializeCriticalSection(&_hbaapi_AL_mutex);
1140 	InitializeCriticalSection(&_hbaapi_AAE_mutex);
1141 	InitializeCriticalSection(&_hbaapi_AE_mutex);
1142 	InitializeCriticalSection(&_hbaapi_APE_mutex);
1143 	InitializeCriticalSection(&_hbaapi_APSE_mutex);
1144 	InitializeCriticalSection(&_hbaapi_TE_mutex);
1145 	InitializeCriticalSection(&_hbaapi_LE_mutex);
1146 	InitializeCriticalSection(&_smhba_AAE_mutex);
1147 	InitializeCriticalSection(&_smhba_AE_mutex);
1148 	InitializeCriticalSection(&_smhba_APE_mutex);
1149 	InitializeCriticalSection(&_smhba_APSE_mutex);
1150 	InitializeCriticalSection(&_smhba_TE_mutex);
1151 #endif
1152 
1153 	return (HBA_STATUS_OK);
1154 }
1155 
1156 HBA_STATUS
1157 HBA_FreeLibrary() {
1158     HBAFreeLibraryFunc	FreeLibraryFunc;
1159 /* LINTED E_FUNC_SET_NOT_USED */
1160     HBA_STATUS		status __unused;
1161     HBA_LIBRARY_INFO	*lib_infop;
1162     HBA_LIBRARY_INFO	*lib_next;
1163     HBA_ADAPTERCALLBACK_ELEM
1164 			***listp;
1165     HBA_ADAPTER_INFO	*adapt_infop;
1166     HBA_ADAPTER_INFO	*adapt_next;
1167 
1168     GRAB_MUTEX(&_hbaapi_LL_mutex);
1169     if (_hbaapi_librarylist == NULL) {
1170 	RELEASE_MUTEX(&_hbaapi_LL_mutex);
1171 	return (HBA_STATUS_ERROR_NOT_LOADED);
1172 	}
1173 
1174     GRAB_MUTEX(&_hbaapi_AL_mutex);
1175 
1176     DEBUG(1, "HBA_FreeLibrary()", 0, 0, 0);
1177     for (lib_infop = _hbaapi_librarylist; lib_infop != NULL;
1178 	    lib_infop = lib_next) {
1179 	lib_next = lib_infop->next;
1180 	if (lib_infop->status == HBA_LIBRARY_LOADED) {
1181 	    FreeLibraryFunc = FUNCCOMMON(lib_infop, FreeLibraryHandler);
1182 	    if (FreeLibraryFunc != NULL) {
1183 		/* Free this library */
1184 		status = ((FreeLibraryFunc)());
1185 		DEBUG(1, "HBA_FreeLibrary() Failed %d", status, 0, 0);
1186 	    }
1187 #ifdef WIN32
1188 	    FreeLibrary(lib_infop->hLibrary);	/* Unload DLL from memory */
1189 #else
1190 	    (void) dlclose(lib_infop->hLibrary); /* Unload DLL from memory */
1191 #endif
1192 	}
1193 #ifndef WIN32
1194 	free(lib_infop->LibraryName);
1195 #endif
1196 	free(lib_infop->LibraryPath);
1197 	free(lib_infop);
1198 
1199 	}
1200     _hbaapi_librarylist = NULL;
1201 	/*
1202 	 * OK, now all functions are disabled except for LoadLibrary,
1203 	 * Hope no other thread calls it before we have returned
1204 	 */
1205     _hbaapi_total_library_count = 0;
1206 
1207     for (adapt_infop = _hbaapi_adapterlist;
1208 	    adapt_infop != NULL;
1209 	    adapt_infop = adapt_next) {
1210 		adapt_next = adapt_infop->next;
1211 		free(adapt_infop->name);
1212 		free(adapt_infop);
1213 	}
1214     _hbaapi_adapterlist = NULL;
1215     _hbaapi_total_adapter_count = 0;
1216 
1217 	/*
1218 	 * Free up the callbacks, this is not the most efficient, but it works
1219 	 */
1220 	while ((volatile HBA_ADAPTERCALLBACK_ELEM *)
1221 	    _hbaapi_adapteraddevents_callback_list
1222 	    != NULL) {
1223 	(void) local_remove_callback((HBA_CALLBACKHANDLE)
1224 	    _hbaapi_adapteraddevents_callback_list);
1225 	}
1226 	while ((volatile HBA_ADAPTERCALLBACK_ELEM *)
1227 	    _smhba_adapteraddevents_callback_list
1228 	    != NULL) {
1229 	(void) local_remove_callback((HBA_CALLBACKHANDLE)
1230 	    _smhba_adapteraddevents_callback_list);
1231 	}
1232     for (listp = cb_lists_array; *listp != NULL; listp++) {
1233 	while ((volatile HBA_ADAPTERCALLBACK_ELEM ***)**listp != NULL) {
1234 	    (void) local_remove_callback((HBA_CALLBACKHANDLE)**listp);
1235 	}
1236 	}
1237 
1238     RELEASE_MUTEX(&_hbaapi_AL_mutex);
1239     RELEASE_MUTEX(&_hbaapi_LL_mutex);
1240 
1241 #ifdef USESYSLOG
1242     closelog();
1243 #endif
1244 #ifdef USELOGFILE
1245     if (_hbaapi_debug_fd != NULL) {
1246 	fclose(_hbaapi_debug_fd);
1247 	}
1248     _hbaapi_debug_fd = NULL;
1249 #endif
1250 #ifdef POSIX_THREADS
1251 	/* this will unlock them as well, but who cares */
1252 	(void) pthread_mutex_destroy(&_hbaapi_LE_mutex);
1253 	(void) pthread_mutex_destroy(&_hbaapi_TE_mutex);
1254 	(void) pthread_mutex_destroy(&_hbaapi_APSE_mutex);
1255 	(void) pthread_mutex_destroy(&_hbaapi_APE_mutex);
1256 	(void) pthread_mutex_destroy(&_hbaapi_AE_mutex);
1257 	(void) pthread_mutex_destroy(&_hbaapi_AAE_mutex);
1258 	(void) pthread_mutex_destroy(&_smhba_TE_mutex);
1259 	(void) pthread_mutex_destroy(&_smhba_APSE_mutex);
1260 	(void) pthread_mutex_destroy(&_smhba_APE_mutex);
1261 	(void) pthread_mutex_destroy(&_smhba_AE_mutex);
1262 	(void) pthread_mutex_destroy(&_smhba_AAE_mutex);
1263 	(void) pthread_mutex_destroy(&_hbaapi_AL_mutex);
1264 	(void) pthread_mutex_destroy(&_hbaapi_LL_mutex);
1265 #elif defined(WIN32)
1266     DeleteCriticalSection(&_hbaapi_LL_mutex);
1267     DeleteCriticalSection(&_hbaapi_AL_mutex);
1268     DeleteCriticalSection(&_hbaapi_AAE_mutex);
1269     DeleteCriticalSection(&_hbaapi_AE_mutex);
1270     DeleteCriticalSection(&_hbaapi_APE_mutex);
1271     DeleteCriticalSection(&_hbaapi_APSE_mutex);
1272     DeleteCriticalSection(&_hbaapi_TE_mutex);
1273     DeleteCriticalSection(&_hbaapi_LE_mutex);
1274     DeleteCriticalSection(&_smhba_TE_mutex);
1275     DeleteCriticalSection(&_smhba_APSE_mutex);
1276     DeleteCriticalSection(&_smhba_APE_mutex);
1277     DeleteCriticalSection(&_smhba_AE_mutex);
1278     DeleteCriticalSection(&_smhba_AAE_mutex);
1279 #endif
1280 
1281 	return (HBA_STATUS_OK);
1282 }
1283 
1284 /*
1285  * The API used to use fixed size tables as its primary data structure.
1286  * Indexing from 1 to N identified each adapters.  Now the adapters are
1287  * on a linked list.  There is a unique "index" foreach each adapter.
1288  * Adapters always keep their index, even if they are removed from the
1289  * hardware.  The only time the indexing is reset is on HBA_FreeLibrary
1290  */
1291 HBA_UINT32
1292 HBA_GetNumberOfAdapters()
1293 {
1294 	int j = 0;
1295 	HBA_LIBRARY_INFO	*lib_infop;
1296 	HBAGetNumberOfAdaptersFunc GetNumberOfAdaptersFunc;
1297 	HBAGetAdapterNameFunc GetAdapterNameFunc;
1298 	HBA_BOOLEAN		found_name;
1299 	HBA_ADAPTER_INFO	*adapt_infop;
1300 	HBA_STATUS		status;
1301 
1302 	char adaptername[256];
1303 	int num_adapters; /* local */
1304 
1305 	if (_hbaapi_librarylist == NULL) {
1306 		return (0);
1307 	}
1308 	GRAB_MUTEX(&_hbaapi_LL_mutex); /* pay attention to order */
1309 	GRAB_MUTEX(&_hbaapi_AL_mutex);
1310 
1311 	for (lib_infop = _hbaapi_librarylist;
1312 	    lib_infop != NULL;
1313 	    lib_infop = lib_infop->next) {
1314 
1315 	if (lib_infop->status != HBA_LIBRARY_LOADED) {
1316 		continue;
1317 	}
1318 
1319 	GetNumberOfAdaptersFunc =
1320 	    FUNCCOMMON(lib_infop, GetNumberOfAdaptersHandler);
1321 	if (GetNumberOfAdaptersFunc == NULL)  {
1322 		continue;
1323 	}
1324 	num_adapters = ((GetNumberOfAdaptersFunc)());
1325 #ifndef WIN32
1326 	DEBUG(1, "HBAAPI: num_adapters for %s = %d\n",
1327 	    lib_infop->LibraryName, num_adapters, 0);
1328 #else
1329 	DEBUG(1, "HBAAPI: num_adapters for %s = %d\n",
1330 	    lib_infop->LibraryPath, num_adapters, 0);
1331 #endif
1332 
1333 	/* Also get the names of all the adapters here and cache */
1334 	GetAdapterNameFunc = FUNCCOMMON(lib_infop, GetAdapterNameHandler);
1335 	if (GetAdapterNameFunc == NULL) {
1336 		continue;
1337 	}
1338 
1339 	for (j = 0; j < num_adapters; j++) {
1340 		found_name = 0;
1341 		status = (GetAdapterNameFunc)(j, (char *)&adaptername);
1342 		if (status == HBA_STATUS_OK) {
1343 		for (adapt_infop = _hbaapi_adapterlist;
1344 		    adapt_infop != NULL;
1345 		    adapt_infop = adapt_infop->next) {
1346 			/*
1347 			 * check for duplicates, really,
1348 			 * this may just be a second
1349 			 * call to this function
1350 			 * ??? how do we know when a name becomes stale?
1351 			 */
1352 			if (strcmp(adaptername, adapt_infop->name) == 0) {
1353 				/* already got this one */
1354 				found_name++;
1355 			break;
1356 			}
1357 		}
1358 		if (found_name != 0) {
1359 			continue;
1360 		}
1361 		}
1362 
1363 		adapt_infop = (HBA_ADAPTER_INFO *)
1364 		    calloc(1, sizeof (HBA_ADAPTER_INFO));
1365 		if (adapt_infop == NULL) {
1366 #ifndef WIN32
1367 		(void) fprintf(stderr,
1368 		    "HBA_GetNumberOfAdapters: calloc failed"
1369 		    " on sizeof:%lu\n",
1370 		    (unsigned long)(sizeof (HBA_ADAPTER_INFO)));
1371 #endif
1372 		RELEASE_MUTEX(&_hbaapi_AL_mutex);
1373 		RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex,
1374 		    _hbaapi_total_adapter_count);
1375 		}
1376 		if ((adapt_infop->GNstatus = status) == HBA_STATUS_OK) {
1377 		adapt_infop->name = strdup(adaptername);
1378 		} else {
1379 		char dummyname[512];
1380 		(void) sprintf(dummyname, "NULLADAPTER-%255s-%03d",
1381 		    lib_infop->LibraryPath, _hbaapi_total_adapter_count);
1382 		dummyname[511] = '\0';
1383 		adapt_infop->name = strdup(dummyname);
1384 		}
1385 		lib_infop->numOfAdapters++;
1386 		adapt_infop->library = lib_infop;
1387 		adapt_infop->next = _hbaapi_adapterlist;
1388 		adapt_infop->index = _hbaapi_total_adapter_count;
1389 		_hbaapi_adapterlist = adapt_infop;
1390 		_hbaapi_total_adapter_count++;
1391 	}
1392 	}
1393 	RELEASE_MUTEX(&_hbaapi_AL_mutex);
1394 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, _hbaapi_total_adapter_count);
1395 }
1396 
1397 HBA_STATUS
1398 HBA_GetAdapterName(
1399     HBA_UINT32 adapterindex,
1400     char *adaptername)
1401 {
1402 	HBA_ADAPTER_INFO	*adapt_infop;
1403 	HBA_STATUS		ret = HBA_STATUS_ERROR_ILLEGAL_INDEX;
1404 
1405 	if (adaptername == NULL) {
1406 		DEBUG(1, "HBA_GetAdapterName: NULL pointer adaptername",
1407 		    0, 0, 0);
1408 		return (HBA_STATUS_ERROR_ARG);
1409 	}
1410 
1411 	/*
1412 	 * The adapter index is from old code, but we have
1413 	 * to support it.  Go down the list looking for
1414 	 * the adapter
1415 	 */
1416 	ARE_WE_INITED();
1417 	GRAB_MUTEX(&_hbaapi_AL_mutex);
1418 	*adaptername = '\0';
1419 	for (adapt_infop = _hbaapi_adapterlist;
1420 	    adapt_infop != NULL;
1421 	    adapt_infop = adapt_infop->next) {
1422 
1423 	if (adapt_infop->index == adapterindex) {
1424 		if (adapt_infop->name != NULL &&
1425 		    adapt_infop->GNstatus == HBA_STATUS_OK) {
1426 		(void) strcpy(adaptername, adapt_infop->name);
1427 		} else {
1428 		*adaptername = '\0';
1429 		}
1430 		ret = adapt_infop->GNstatus;
1431 		break;
1432 	}
1433 	}
1434 	DEBUG(2, "GetAdapterName for index:%d ->%s",
1435 	    adapterindex, adaptername, 0);
1436 	RELEASE_MUTEX_RETURN(&_hbaapi_AL_mutex, ret);
1437 }
1438 
1439 HBA_HANDLE
1440 HBA_OpenAdapter(char *adaptername)
1441 {
1442 	HBA_HANDLE		handle;
1443 	HBAOpenAdapterFunc	OpenAdapterFunc;
1444 	HBA_ADAPTER_INFO	*adapt_infop;
1445 	HBA_LIBRARY_INFO	*lib_infop;
1446 
1447 	DEBUG(2, "OpenAdapter: %s", adaptername, 0, 0);
1448 
1449 	handle = HBA_HANDLE_INVALID;
1450 	if (_hbaapi_librarylist == NULL) {
1451 		return (handle);
1452 	}
1453 	if (adaptername == NULL) {
1454 		DEBUG(1, "HBA_OpenAdapter: NULL pointer adaptername",
1455 		    0, 0, 0);
1456 		return (handle);
1457 	}
1458 	GRAB_MUTEX(&_hbaapi_AL_mutex);
1459 	for (adapt_infop = _hbaapi_adapterlist;
1460 	    adapt_infop != NULL;
1461 	    adapt_infop = adapt_infop->next) {
1462 	if (strcmp(adaptername, adapt_infop->name) != 0) {
1463 		continue;
1464 	}
1465 	lib_infop = adapt_infop->library;
1466 	OpenAdapterFunc = FUNCCOMMON(lib_infop, OpenAdapterHandler);
1467 
1468 	if (OpenAdapterFunc != NULL) {
1469 	    /* retrieve the vendor handle */
1470 		handle = (OpenAdapterFunc)(adaptername);
1471 		if (handle != 0) {
1472 		/* or this with the library index to get the common handle */
1473 		handle = HBA_HANDLE_FROM_LOCAL(lib_infop->index, handle);
1474 		}
1475 	}
1476 	break;
1477 	}
1478 	RELEASE_MUTEX_RETURN(&_hbaapi_AL_mutex, handle);
1479 }
1480 
1481 /*
1482  * Finding an adapter with matching WWN.
1483  */
1484 HBA_STATUS
1485 HBA_OpenAdapterByWWN(HBA_HANDLE *phandle, HBA_WWN nodeWWN) {
1486     HBA_HANDLE		handle;
1487     HBA_LIBRARY_INFO	*lib_infop;
1488     HBAGetNumberOfAdaptersFunc
1489 			GetNumberOfAdaptersFunc;
1490     HBAOpenAdapterByWWNFunc
1491 			OpenAdapterFunc;
1492     HBA_STATUS		status;
1493 
1494     DEBUG(2, "OpenAdapterByWWN: %s", WWN2STR1(&nodeWWN), 0, 0);
1495     ARE_WE_INITED();
1496 
1497 	*phandle = HBA_HANDLE_INVALID;
1498 
1499     GRAB_MUTEX(&_hbaapi_LL_mutex);
1500     for (lib_infop = _hbaapi_librarylist;
1501 	    lib_infop != NULL;
1502 	    lib_infop = lib_infop->next) {
1503 
1504 	status = HBA_STATUS_ERROR_ILLEGAL_WWN;
1505 
1506 	if (lib_infop->status != HBA_LIBRARY_LOADED) {
1507 	    continue;
1508 	}
1509 
1510 	/* only for HBAAPIV2 */
1511 	if (lib_infop->version != HBAAPIV2) {
1512 	    continue;
1513 	}
1514 
1515 	GetNumberOfAdaptersFunc =
1516 		FUNCCOMMON(lib_infop, GetNumberOfAdaptersHandler);
1517 	if (GetNumberOfAdaptersFunc == NULL)  {
1518 	    continue;
1519 	}
1520 
1521 	/* look for new hardware */
1522 	(void) ((GetNumberOfAdaptersFunc)());
1523 
1524 	OpenAdapterFunc =
1525 	    lib_infop->ftable.functionTable.OpenAdapterByWWNHandler;
1526 	if (OpenAdapterFunc == NULL) {
1527 	    continue;
1528 	}
1529 	/*
1530 	 * We do not know if the WWN is known by this vendor,
1531 	 * just try it
1532 	 */
1533 	if ((status = (OpenAdapterFunc)(&handle, nodeWWN)) != HBA_STATUS_OK) {
1534 	    continue;
1535 	}
1536 	/* OK, make a vendor non-specific handle */
1537 	*phandle = HBA_HANDLE_FROM_LOCAL(lib_infop->index, handle);
1538 	status = HBA_STATUS_OK;
1539 	break;
1540 	}
1541     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
1542 }
1543 
1544 void
1545 HBA_RefreshAdapterConfiguration() {
1546     DEBUG(2, "HBA_RefreshAdapterConfiguration", 0, 0, 0);
1547 	(void) HBA_GetNumberOfAdapters();
1548 }
1549 
1550 HBA_UINT32
1551 HBA_GetVersion() {
1552     DEBUG(2, "HBA_GetVersion", 0, 0, 0);
1553 	return (HBA_LIBVERSION);
1554 }
1555 
1556 /*
1557  * This function is VERY OS dependent.  Wing it as best you can.
1558  */
1559 HBA_UINT32
1560 HBA_GetWrapperLibraryAttributes(
1561     HBA_LIBRARYATTRIBUTES *attributes)
1562 {
1563 
1564 	DEBUG(2, "HBA_GetWrapperLibraryAttributes", 0, 0, 0);
1565 
1566 	if (attributes == NULL) {
1567 		DEBUG(1, "HBA_GetWrapperLibraryAttributes:"
1568 		    "NULL pointer attributes",
1569 		    0, 0, 0);
1570 		return (HBA_STATUS_ERROR_ARG);
1571 	}
1572 
1573 	(void) memset(attributes, 0, sizeof (HBA_LIBRARYATTRIBUTES));
1574 
1575 #if defined(SOLARIS)
1576 	if ((handle = dlopen("libHBAAPI.so", RTLD_NOW)) != NULL) {
1577 	if (dlinfo(handle, RTLD_DI_LINKMAP, &map) >= 0) {
1578 		for (mp = map; mp != NULL; mp = mp->l_next) {
1579 		if (strlen(map->l_name) < 256) {
1580 			(void) strcpy(attributes->LibPath, map->l_name);
1581 		}
1582 		}
1583 	}
1584 	}
1585 #elif defined(WIN32)
1586 	HMODULE module;
1587 
1588 	/* No need to do anything with the module handle */
1589 	/* It wasn't alloocated so it doesn't need to be freed */
1590 	module = GetModuleHandle("HBAAPI");
1591 	if (module != NULL) {
1592 		if (GetModuleFileName(module, attributes->LibPath,
1593 		    sizeof (attributes->LibPath)) == 0) {
1594 			attributes->LibPath[0] = '\0';
1595 		}
1596 	}
1597 #endif
1598 #if defined(VENDOR)
1599 	(void) strcpy(attributes->VName, VENDOR);
1600 #else
1601 	attributes->VName[0] = '\0';
1602 #endif
1603 #if defined(VERSION)
1604 	(void) strcpy(attributes->VVersion, VERSION);
1605 #else
1606 	attributes->VVersion[0] = '\0';
1607 #endif
1608 #if defined(BUILD_DATE)
1609 #if defined(WIN32)
1610 	int matchCount;
1611 	matchCount = sscanf(BUILD_DATE, "%u/%u/%u %u:%u:%u",
1612 	    &attributes->build_date.tm_year,
1613 	    &attributes->build_date.tm_mon,
1614 	    &attributes->build_date.tm_mday,
1615 	    &attributes->build_date.tm_hour,
1616 	    &attributes->build_date.tm_min,
1617 	    &attributes->build_date.tm_sec);
1618 
1619 	if (matchCount != 6) {
1620 		memset(&attributes->build_date, 0, sizeof (struct tm));
1621 	} else {
1622 		attributes->build_date.tm_year -= 1900;
1623 		attributes->build_date.tm_isdst = -1;
1624 	}
1625 #else
1626 	if (strptime(BUILD_DATE,
1627 	    "%Y/%m/%d %T %Z", &(attributes->build_date)) == NULL) {
1628 		(void) memset(&attributes->build_date, 0, sizeof (struct tm));
1629 	}
1630 #endif
1631 #else
1632 	(void) memset(&attributes->build_date, 0, sizeof (struct tm));
1633 #endif
1634 	return (2);
1635 }
1636 
1637 /*
1638  * Callback registation and handling
1639  */
1640 HBA_STATUS
1641 HBA_RemoveCallback(HBA_CALLBACKHANDLE cbhandle) {
1642     HBA_STATUS	status;
1643 
1644     DEBUG(2, "HBA_RemoveCallback", 0, 0, 0);
1645     ARE_WE_INITED();
1646 
1647     GRAB_MUTEX(&_hbaapi_LL_mutex);
1648     status = local_remove_callback(cbhandle);
1649     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
1650 }
1651 
1652 /* Adapter Add Events ************************************************* */
1653 static void
1654 /* LINTED E_FUNC_ARG_UNUSED */
1655 adapteraddevents_callback(void *data, HBA_WWN PortWWN, HBA_UINT32 eventType) {
1656     HBA_ALLADAPTERSCALLBACK_ELEM	*cbp;
1657 
1658     DEBUG(3, "AddAdapterEvent, port: %s", WWN2STR1(&PortWWN), 0, 0);
1659 
1660     GRAB_MUTEX(&_hbaapi_AAE_mutex);
1661     for (cbp = _hbaapi_adapteraddevents_callback_list;
1662 	    cbp != NULL;
1663 	    cbp = cbp->next) {
1664 	(*cbp->callback)(data, PortWWN, HBA_EVENT_ADAPTER_ADD);
1665 	}
1666     RELEASE_MUTEX(&_hbaapi_AAE_mutex);
1667 
1668 }
1669 
1670 HBA_STATUS
1671 HBA_RegisterForAdapterAddEvents(
1672     void		(*callback)(
1673 	void		*data,
1674 	HBA_WWN		PortWWN,
1675 	HBA_UINT32	eventType),
1676 	void		*userData,
1677     HBA_CALLBACKHANDLE *callbackHandle) {
1678 
1679     HBA_ALLADAPTERSCALLBACK_ELEM	*cbp;
1680     HBA_VENDORCALLBACK_ELEM		*vcbp;
1681     HBA_VENDORCALLBACK_ELEM		*vendorhandlelist;
1682     HBARegisterForAdapterAddEventsFunc	registeredfunc;
1683     HBA_STATUS				status = HBA_STATUS_OK;
1684     HBA_STATUS				failure = HBA_STATUS_OK;
1685     HBA_LIBRARY_INFO			*lib_infop;
1686     int					registered_cnt = 0;
1687     int					vendor_cnt = 0;
1688     int					not_supported_cnt = 0;
1689     int					status_OK_bar_cnt = 0;
1690     int					status_OK_cnt = 0;
1691 
1692     DEBUG(2, "HBA_RegisterForAdapterAddEvents", 0, 0, 0);
1693     ARE_WE_INITED();
1694 
1695     cbp = (HBA_ALLADAPTERSCALLBACK_ELEM *)
1696 	calloc(1, sizeof (HBA_ALLADAPTERSCALLBACK_ELEM));
1697 	*callbackHandle = (HBA_CALLBACKHANDLE) cbp;
1698 	if (cbp == NULL) {
1699 #ifndef WIN32
1700 	(void) fprintf(stderr,
1701 		"HBA_RegisterForAdapterAddEvents: calloc failed "
1702 		"for %lu bytes\n",
1703 		(unsigned long)(sizeof (HBA_ALLADAPTERSCALLBACK_ELEM)));
1704 #endif
1705 	return (HBA_STATUS_ERROR);
1706 	}
1707 
1708     GRAB_MUTEX(&_hbaapi_LL_mutex);
1709     GRAB_MUTEX(&_hbaapi_AAE_mutex);
1710     cbp->callback = callback;
1711     cbp->next = _hbaapi_adapteraddevents_callback_list;
1712     _hbaapi_adapteraddevents_callback_list = cbp;
1713 	/*
1714 	 * Need to release the mutex now incase the vendor function invokes the
1715 	 * callback.  We will grap the mutex later to attach the vendor handle
1716 	 * list to the callback structure
1717 	 */
1718     RELEASE_MUTEX(&_hbaapi_AAE_mutex);
1719 
1720 	/*
1721 	 * now create a list of vendors (vendor libraryies, NOT ADAPTERS)
1722 	 * that have successfully registerred
1723 	 */
1724 	vendorhandlelist = NULL;
1725     for (lib_infop = _hbaapi_librarylist;
1726 	    lib_infop != NULL;
1727 	    lib_infop = lib_infop->next) {
1728 
1729 	/* only for HBAAPI V2 */
1730 	if ((lib_infop->version != HBAAPIV2)) {
1731 	    continue;
1732 	} else {
1733 	    vendor_cnt++;
1734 	}
1735 
1736 	registeredfunc =
1737 	    lib_infop->ftable.functionTable.RegisterForAdapterAddEventsHandler;
1738 	if (registeredfunc == NULL) {
1739 	    continue;
1740 	}
1741 
1742 	vcbp = (HBA_VENDORCALLBACK_ELEM *)
1743 	    calloc(1, sizeof (HBA_VENDORCALLBACK_ELEM));
1744 	if (vcbp == NULL) {
1745 #ifndef WIN32
1746 	    (void) fprintf(stderr,
1747 		    "HBA_RegisterForAdapterAddEvents: "
1748 		    "calloc failed for %lu bytes\n",
1749 		    (unsigned long)(sizeof (HBA_VENDORCALLBACK_ELEM)));
1750 #endif
1751 	    freevendorhandlelist(vendorhandlelist);
1752 	    status = HBA_STATUS_ERROR;
1753 	    break;
1754 	}
1755 
1756 	registered_cnt++;
1757 	status = (registeredfunc)(adapteraddevents_callback,
1758 	    userData, &vcbp->vendorcbhandle);
1759 	if (status == HBA_STATUS_ERROR_NOT_SUPPORTED) {
1760 	    not_supported_cnt++;
1761 	    free(vcbp);
1762 	    continue;
1763 	} else if (status != HBA_STATUS_OK) {
1764 	    status_OK_bar_cnt++;
1765 	    DEBUG(1,
1766 		    "HBA_RegisterForAdapterAddEvents: Library->%s, Error->%d",
1767 		    lib_infop->LibraryPath, status, 0);
1768 #ifndef WIN32
1769 	    (void) fprintf(stderr,
1770 		    "HBA_RegisterForAdapterAddEvents: Library->%s, Error->%d",
1771 		    lib_infop->LibraryPath, status);
1772 #endif
1773 	    failure = status;
1774 	    free(vcbp);
1775 	    continue;
1776 	} else {
1777 	    status_OK_cnt++;
1778 	}
1779 	vcbp->lib_info = lib_infop;
1780 	vcbp->next = vendorhandlelist;
1781 	vendorhandlelist = vcbp;
1782 	}
1783     if (vendor_cnt == 0) {
1784 	/* no HBAAPIV2 is deteced.  should be okay? */
1785 	status = HBA_STATUS_ERROR;
1786 	} else if (registered_cnt == 0) {
1787 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
1788 	freevendorhandlelist(vendorhandlelist);
1789 	(void) local_remove_callback((HBA_CALLBACKHANDLE) cbp);
1790 	} else if (status_OK_cnt == 0 && not_supported_cnt != 0) {
1791 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
1792 	} else if (status_OK_cnt == 0) {
1793 	/*
1794 	 * At least one vendor library registered this function, but no
1795 	 * vendor call succeeded
1796 	 */
1797 	(void) local_remove_callback((HBA_CALLBACKHANDLE) cbp);
1798 	status = failure;
1799 	} else {
1800 	/* we have had atleast some success, now finish up */
1801 	GRAB_MUTEX(&_hbaapi_AAE_mutex);
1802 	/*
1803 	 * this seems silly, but what if another thread called
1804 	 * the callback remove
1805 	 */
1806 	for (cbp = _hbaapi_adapteraddevents_callback_list;
1807 	    cbp != NULL; cbp = cbp->next) {
1808 	    if ((HBA_CALLBACKHANDLE)cbp == *callbackHandle) {
1809 		/* yup, its still there, hooray */
1810 		cbp->vendorhandlelist = vendorhandlelist;
1811 		vendorhandlelist = NULL;
1812 		break;
1813 	    }
1814 	}
1815 	RELEASE_MUTEX(&_hbaapi_AAE_mutex);
1816 	if (vendorhandlelist != NULL) {
1817 		/*
1818 		 * bummer, somebody removed the callback before we finished
1819 		 * registration, probably will never happen
1820 		 */
1821 	    freevendorhandlelist(vendorhandlelist);
1822 	    DEBUG(1,
1823 		    "HBA_RegisterForAdapterAddEvents: HBA_RemoveCallback was "
1824 		    "called for a handle before registration was finished.",
1825 		    0, 0, 0);
1826 	    status = HBA_STATUS_ERROR;
1827 	} else {
1828 	    status = HBA_STATUS_OK;
1829 	}
1830 	}
1831     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
1832 }
1833 
1834 /* Adapter Events (other than add) ************************************** */
1835 static void
1836 adapterevents_callback(void *data,
1837 			HBA_WWN PortWWN,
1838 			HBA_UINT32 eventType) {
1839     HBA_ADAPTERCALLBACK_ELEM	*acbp;
1840 
1841     DEBUG(3, "AdapterEvent, port:%s, eventType:%d", WWN2STR1(&PortWWN),
1842 	    eventType, 0);
1843 
1844 	GRAB_MUTEX(&_hbaapi_AE_mutex);
1845 	for (acbp = _hbaapi_adapterevents_callback_list;
1846 	acbp != NULL;
1847 	acbp = acbp->next) {
1848 	if (data == (void *)acbp) {
1849 	    (*acbp->callback)(acbp->userdata, PortWWN, eventType);
1850 	    break;
1851 	}
1852 	}
1853     RELEASE_MUTEX(&_hbaapi_AE_mutex);
1854 }
1855 HBA_STATUS
1856 HBA_RegisterForAdapterEvents(
1857     void		(*callback) (
1858 	void		*data,
1859 	HBA_WWN		PortWWN,
1860 	HBA_UINT32	eventType),
1861     void		*userData,
1862     HBA_HANDLE		handle,
1863     HBA_CALLBACKHANDLE	*callbackHandle) {
1864 
1865     HBA_ADAPTERCALLBACK_ELEM		*acbp;
1866     HBARegisterForAdapterEventsFunc	registeredfunc;
1867     HBA_STATUS				status;
1868     HBA_LIBRARY_INFO			*lib_infop;
1869     HBA_HANDLE				vendorHandle;
1870 
1871     DEBUG(2, "HBA_RegisterForAdapterEvents", 0, 0, 0);
1872 
1873     CHECKLIBRARYANDVERSION(HBAAPIV2);
1874 
1875 	/* we now have the _hbaapi_LL_mutex */
1876 
1877     registeredfunc =
1878 	    lib_infop->ftable.functionTable.RegisterForAdapterEventsHandler;
1879     if (registeredfunc == NULL) {
1880 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
1881 	}
1882 
1883 	/*
1884 	 * that allocated memory is used both as the handle for the
1885 	 * caller, and as userdata to the vendor call so that on
1886 	 * callback the specific registration may be recalled
1887 	 */
1888     acbp = (HBA_ADAPTERCALLBACK_ELEM *)
1889 	calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
1890     if (acbp == NULL) {
1891 #ifndef WIN32
1892 	(void) fprintf(stderr,
1893 		"HBA_RegisterForAdapterEvents: calloc failed for %lu bytes\n",
1894 		(unsigned long)(sizeof (HBA_ADAPTERCALLBACK_ELEM)));
1895 #endif
1896 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
1897 	}
1898 	*callbackHandle = (HBA_CALLBACKHANDLE) acbp;
1899     acbp->callback = callback;
1900     acbp->userdata = userData;
1901     acbp->lib_info = lib_infop;
1902 
1903     status = (registeredfunc)(adapterevents_callback,
1904 	    (void *)acbp,
1905 	    vendorHandle,
1906 	    &acbp->vendorcbhandle);
1907     if (status != HBA_STATUS_OK) {
1908 	free(acbp);
1909 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
1910 	}
1911 
1912     GRAB_MUTEX(&_hbaapi_AE_mutex);
1913     acbp->next = _hbaapi_adapterevents_callback_list;
1914     _hbaapi_adapterevents_callback_list = acbp;
1915     RELEASE_MUTEX(&_hbaapi_AE_mutex);
1916 
1917     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
1918 }
1919 
1920 /* Adapter Port Events ************************************************** */
1921 static void
1922 adapterportevents_callback(void *data,
1923 			    HBA_WWN PortWWN,
1924 			    HBA_UINT32 eventType,
1925 			    HBA_UINT32 fabricPortID) {
1926     HBA_ADAPTERCALLBACK_ELEM	*acbp;
1927 
1928     DEBUG(3, "AdapterPortEvent, port:%s, eventType:%d fabricPortID:0X%06x",
1929 	    WWN2STR1(&PortWWN), eventType, fabricPortID);
1930 
1931     GRAB_MUTEX(&_hbaapi_APE_mutex);
1932 
1933     for (acbp = _hbaapi_adapterportevents_callback_list;
1934 	acbp != NULL;
1935 	acbp = acbp->next) {
1936 	if (data == (void *)acbp) {
1937 	    (*acbp->callback)(acbp->userdata, PortWWN, eventType, fabricPortID);
1938 	    break;
1939 	}
1940 	}
1941     RELEASE_MUTEX(&_hbaapi_APE_mutex);
1942 }
1943 
1944 HBA_STATUS
1945 HBA_RegisterForAdapterPortEvents(
1946     void		(*callback) (
1947 	void		*data,
1948 	HBA_WWN		PortWWN,
1949 	HBA_UINT32	eventType,
1950 	HBA_UINT32	fabricPortID),
1951     void		*userData,
1952     HBA_HANDLE		handle,
1953     HBA_WWN		PortWWN,
1954     HBA_CALLBACKHANDLE	*callbackHandle) {
1955 
1956     HBA_ADAPTERCALLBACK_ELEM		*acbp;
1957     HBARegisterForAdapterPortEventsFunc	registeredfunc;
1958     HBA_STATUS				status;
1959     HBA_LIBRARY_INFO			*lib_infop;
1960     HBA_HANDLE				vendorHandle;
1961 
1962     DEBUG(2, "HBA_RegisterForAdapterPortEvents for port: %s",
1963 	    WWN2STR1(&PortWWN), 0, 0);
1964 
1965     CHECKLIBRARYANDVERSION(HBAAPIV2);
1966 	/* we now have the _hbaapi_LL_mutex */
1967 
1968 	registeredfunc =
1969 	lib_infop->ftable.functionTable.RegisterForAdapterPortEventsHandler;
1970     if (registeredfunc == NULL) {
1971 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
1972 	}
1973 
1974 	/*
1975 	 * that allocated memory is used both as the handle for the
1976 	 * caller, and as userdata to the vendor call so that on
1977 	 * callback the specific registration may be recalled
1978 	 */
1979 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
1980 	calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
1981     if (acbp == NULL) {
1982 #ifndef WIN32
1983 	(void) fprintf(stderr,
1984 		"HBA_RegisterForAdapterPortEvents: "
1985 		"calloc failed for %lu bytes\n",
1986 		(unsigned long)(sizeof (HBA_ADAPTERCALLBACK_ELEM)));
1987 #endif
1988 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
1989 
1990 	}
1991 	*callbackHandle = (HBA_CALLBACKHANDLE) acbp;
1992     acbp->callback = callback;
1993     acbp->userdata = userData;
1994     acbp->lib_info = lib_infop;
1995 
1996     status = (registeredfunc)(adapterportevents_callback,
1997 	    (void *)acbp,
1998 	    vendorHandle,
1999 	    PortWWN,
2000 	    &acbp->vendorcbhandle);
2001     if (status != HBA_STATUS_OK) {
2002 	free(acbp);
2003 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2004 	}
2005 
2006     GRAB_MUTEX(&_hbaapi_APE_mutex);
2007     acbp->next = _hbaapi_adapterportevents_callback_list;
2008     _hbaapi_adapterportevents_callback_list = acbp;
2009     RELEASE_MUTEX(&_hbaapi_APE_mutex);
2010 
2011     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
2012 }
2013 
2014 /* Adapter State Events ************************************************ */
2015 static void
2016 adapterportstatevents_callback(void *data,
2017 				HBA_WWN PortWWN,
2018 				HBA_UINT32 eventType) {
2019     HBA_ADAPTERCALLBACK_ELEM	*acbp;
2020 
2021 	DEBUG(3, "AdapterPortStatEvent, port:%s, eventType:%d",
2022 	    WWN2STR1(&PortWWN),
2023 	    eventType, 0);
2024 
2025     GRAB_MUTEX(&_hbaapi_APSE_mutex);
2026     for (acbp = _hbaapi_adapterportstatevents_callback_list;
2027 	acbp != NULL;
2028 	acbp = acbp->next) {
2029 	if (data == (void *)acbp) {
2030 	    (*acbp->callback)(acbp->userdata, PortWWN, eventType);
2031 	    return;
2032 	}
2033 	}
2034     RELEASE_MUTEX(&_hbaapi_APSE_mutex);
2035 }
2036 HBA_STATUS
2037 HBA_RegisterForAdapterPortStatEvents(
2038     void		(*callback) (
2039 	void		*data,
2040 	HBA_WWN		PortWWN,
2041 	HBA_UINT32	eventType),
2042     void		*userData,
2043     HBA_HANDLE		handle,
2044     HBA_WWN		PortWWN,
2045     HBA_PORTSTATISTICS	stats,
2046     HBA_UINT32		statType,
2047     HBA_CALLBACKHANDLE	*callbackHandle) {
2048 
2049     HBA_ADAPTERCALLBACK_ELEM	*acbp;
2050     HBARegisterForAdapterPortStatEventsFunc
2051 				registeredfunc;
2052     HBA_STATUS			status;
2053     HBA_LIBRARY_INFO		*lib_infop;
2054     HBA_HANDLE			vendorHandle;
2055 
2056     DEBUG(2, "HBA_RegisterForAdapterPortStatEvents for port: %s",
2057 	    WWN2STR1(&PortWWN), 0, 0);
2058 
2059     CHECKLIBRARYANDVERSION(HBAAPIV2);
2060 	/* we now have the _hbaapi_LL_mutex */
2061 
2062     registeredfunc =
2063 	lib_infop->ftable.functionTable.RegisterForAdapterPortStatEventsHandler;
2064     if (registeredfunc == NULL) {
2065 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
2066 	}
2067 
2068 	/*
2069 	 * that allocated memory is used both as the handle for the
2070 	 * caller, and as userdata to the vendor call so that on
2071 	 * callback the specific registration may be recalled
2072 	 */
2073     acbp = (HBA_ADAPTERCALLBACK_ELEM *)
2074 	calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
2075     if (acbp == NULL) {
2076 #ifndef WIN32
2077 	(void) fprintf(stderr,
2078 		"HBA_RegisterForAdapterPortStatEvents: "
2079 		"calloc failed for %lu bytes\n",
2080 		(unsigned long)(sizeof (HBA_ADAPTERCALLBACK_ELEM)));
2081 #endif
2082 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
2083 	}
2084 	*callbackHandle = (HBA_CALLBACKHANDLE) acbp;
2085     acbp->callback = callback;
2086     acbp->userdata = userData;
2087     acbp->lib_info = lib_infop;
2088 
2089     status = (registeredfunc)(adapterportstatevents_callback,
2090 	    (void *)acbp,
2091 	    vendorHandle,
2092 	    PortWWN,
2093 	    stats,
2094 	    statType,
2095 	    &acbp->vendorcbhandle);
2096     if (status != HBA_STATUS_OK) {
2097 	free(acbp);
2098 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2099 	}
2100 
2101     GRAB_MUTEX(&_hbaapi_APSE_mutex);
2102     acbp->next = _hbaapi_adapterportstatevents_callback_list;
2103     _hbaapi_adapterportstatevents_callback_list = acbp;
2104     RELEASE_MUTEX(&_hbaapi_APSE_mutex);
2105 
2106     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
2107 }
2108 
2109 /* Target Events ******************************************************* */
2110 static void
2111 targetevents_callback(void *data,
2112     HBA_WWN hbaPortWWN,
2113     HBA_WWN discoveredPortWWN,
2114     HBA_UINT32 eventType) {
2115 
2116 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
2117 
2118     DEBUG(3, "TargetEvent, hbaPort:%s, discoveredPort:%s eventType:%d",
2119 	    WWN2STR1(&hbaPortWWN), WWN2STR2(&discoveredPortWWN), eventType);
2120 
2121     GRAB_MUTEX(&_hbaapi_TE_mutex);
2122     for (acbp = _hbaapi_targetevents_callback_list;
2123 	acbp != NULL;
2124 	acbp = acbp->next) {
2125 	if (data == (void *)acbp) {
2126 	    (*acbp->callback)(acbp->userdata, hbaPortWWN,
2127 	    discoveredPortWWN, eventType);
2128 	    break;
2129 	}
2130 	}
2131     RELEASE_MUTEX(&_hbaapi_TE_mutex);
2132 }
2133 
2134 HBA_STATUS
2135 HBA_RegisterForTargetEvents(
2136     void		(*callback) (
2137 	void		*data,
2138 	HBA_WWN		hbaPortWWN,
2139 	HBA_WWN		discoveredPortWWN,
2140 	HBA_UINT32	eventType),
2141     void		*userData,
2142     HBA_HANDLE		handle,
2143     HBA_WWN		hbaPortWWN,
2144     HBA_WWN		discoveredPortWWN,
2145     HBA_CALLBACKHANDLE	*callbackHandle,
2146     HBA_UINT32		allTargets) {
2147 
2148     HBA_ADAPTERCALLBACK_ELEM
2149 			*acbp;
2150     HBARegisterForTargetEventsFunc
2151 			registeredfunc;
2152     HBA_STATUS		status;
2153     HBA_LIBRARY_INFO	*lib_infop;
2154     HBA_HANDLE		vendorHandle;
2155 
2156     DEBUG(2, "HBA_RegisterForTargetEvents, hbaPort: %s, discoveredPort: %s",
2157 	    WWN2STR1(&hbaPortWWN), WWN2STR2(&discoveredPortWWN), 0);
2158 
2159     CHECKLIBRARYANDVERSION(HBAAPIV2);
2160 	/* we now have the _hbaapi_LL_mutex */
2161 
2162     registeredfunc =
2163 	    lib_infop->ftable.functionTable.RegisterForTargetEventsHandler;
2164     if (registeredfunc == NULL) {
2165 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
2166 	}
2167 
2168 	/*
2169 	 * that allocated memory is used both as the handle for the
2170 	 * caller, and as userdata to the vendor call so that on
2171 	 * callback the specific registration may be recalled
2172 	 */
2173 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
2174 	calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
2175     if (acbp == NULL) {
2176 #ifndef WIN32
2177 	(void) fprintf(stderr,
2178 		"HBA_RegisterForTargetEvents: calloc failed for %lu bytes\n",
2179 		(unsigned long)(sizeof (HBA_ADAPTERCALLBACK_ELEM)));
2180 #endif
2181 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
2182 	}
2183 	*callbackHandle = (HBA_CALLBACKHANDLE) acbp;
2184     acbp->callback = callback;
2185     acbp->userdata = userData;
2186     acbp->lib_info = lib_infop;
2187 
2188     status = (registeredfunc)(targetevents_callback,
2189 	    (void *)acbp,
2190 	    vendorHandle,
2191 	    hbaPortWWN,
2192 	    discoveredPortWWN,
2193 	    &acbp->vendorcbhandle,
2194 	    allTargets);
2195     if (status != HBA_STATUS_OK) {
2196 	free(acbp);
2197 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2198 	}
2199 
2200     GRAB_MUTEX(&_hbaapi_TE_mutex);
2201     acbp->next = _hbaapi_targetevents_callback_list;
2202     _hbaapi_targetevents_callback_list = acbp;
2203     RELEASE_MUTEX(&_hbaapi_TE_mutex);
2204 
2205     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
2206 }
2207 
2208 /* Link Events ********************************************************* */
2209 static void
2210 linkevents_callback(void *data,
2211     HBA_WWN adapterWWN,
2212     HBA_UINT32 eventType,
2213     void *pRLIRBuffer,
2214     HBA_UINT32 RLIRBufferSize) {
2215 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
2216 
2217     DEBUG(3, "LinkEvent, hbaWWN:%s, eventType:%d",
2218 	    WWN2STR1(&adapterWWN), eventType, 0);
2219 
2220     GRAB_MUTEX(&_hbaapi_LE_mutex);
2221     for (acbp = _hbaapi_linkevents_callback_list;
2222 	acbp != NULL;
2223 	acbp = acbp->next) {
2224 	if (data == (void *)acbp) {
2225 	    (*acbp->callback)(acbp->userdata, adapterWWN,
2226 		eventType, pRLIRBuffer, RLIRBufferSize);
2227 	    break;
2228 	}
2229 	}
2230     RELEASE_MUTEX(&_hbaapi_LE_mutex);
2231 }
2232 HBA_STATUS
2233 HBA_RegisterForLinkEvents(
2234     void		(*callback) (
2235 	void		*data,
2236 	HBA_WWN		adapterWWN,
2237 	HBA_UINT32	eventType,
2238 	void		*pRLIRBuffer,
2239 	HBA_UINT32	RLIRBufferSize),
2240     void		*userData,
2241     void		*pRLIRBuffer,
2242     HBA_UINT32		RLIRBufferSize,
2243     HBA_HANDLE		handle,
2244     HBA_CALLBACKHANDLE	*callbackHandle) {
2245 
2246     HBA_ADAPTERCALLBACK_ELEM	*acbp;
2247     HBARegisterForLinkEventsFunc
2248 				registeredfunc;
2249     HBA_STATUS			status;
2250     HBA_LIBRARY_INFO		*lib_infop;
2251     HBA_HANDLE			vendorHandle;
2252 
2253     DEBUG(2, "HBA_RegisterForLinkEvents", 0, 0, 0);
2254 
2255     CHECKLIBRARY();
2256 	/* we now have the _hbaapi_LL_mutex */
2257 
2258     registeredfunc = FUNCCOMMON(lib_infop, RegisterForLinkEventsHandler);
2259 
2260     if (registeredfunc == NULL) {
2261 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
2262 	}
2263 
2264 	/*
2265 	 * that allocated memory is used both as the handle for the
2266 	 * caller, and as userdata to the vendor call so that on
2267 	 * callback the specific registration may be recalled
2268 	 */
2269     acbp = (HBA_ADAPTERCALLBACK_ELEM *)
2270 	calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
2271     if (acbp == NULL) {
2272 #ifndef WIN32
2273 	(void) fprintf(stderr,
2274 		"HBA_RegisterForLinkEvents: calloc failed for %lu bytes\n",
2275 		(unsigned long)(sizeof (HBA_ADAPTERCALLBACK_ELEM)));
2276 #endif
2277 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
2278 	}
2279 	*callbackHandle = (HBA_CALLBACKHANDLE) acbp;
2280     acbp->callback = callback;
2281     acbp->userdata = userData;
2282     acbp->lib_info = lib_infop;
2283 
2284     status = (registeredfunc)(linkevents_callback,
2285 	    (void *)acbp,
2286 	    pRLIRBuffer,
2287 	    RLIRBufferSize,
2288 	    vendorHandle,
2289 	    &acbp->vendorcbhandle);
2290     if (status != HBA_STATUS_OK) {
2291 	free(acbp);
2292 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2293 	}
2294 
2295     GRAB_MUTEX(&_hbaapi_LE_mutex);
2296     acbp->next = _hbaapi_linkevents_callback_list;
2297     _hbaapi_linkevents_callback_list = acbp;
2298     RELEASE_MUTEX(&_hbaapi_LE_mutex);
2299 
2300     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
2301 }
2302 
2303 /*
2304  * All of the functions below are almost passthru functions to the
2305  * vendor specific function
2306  */
2307 
2308 void
2309 HBA_CloseAdapter(HBA_HANDLE handle) {
2310     HBA_STATUS		status;
2311     HBA_LIBRARY_INFO	*lib_infop;
2312     HBA_HANDLE		vendorHandle;
2313     HBACloseAdapterFunc CloseAdapterFunc;
2314 
2315     DEBUG(2, "HBA_CloseAdapter", 0, 0, 0);
2316 
2317     status = HBA_CheckLibrary(handle, &lib_infop, &vendorHandle);
2318     if (status == HBA_STATUS_OK) {
2319 	CloseAdapterFunc = FUNCCOMMON(lib_infop, CloseAdapterHandler);
2320 	if (CloseAdapterFunc != NULL) {
2321 	    ((CloseAdapterFunc)(vendorHandle));
2322 	}
2323 	RELEASE_MUTEX(&_hbaapi_LL_mutex);
2324 	}
2325 }
2326 
2327 HBA_STATUS
2328 HBA_GetAdapterAttributes(
2329     HBA_HANDLE		handle,
2330     HBA_ADAPTERATTRIBUTES
2331 			*hbaattributes)
2332 {
2333 	HBA_STATUS		status;
2334 	HBA_LIBRARY_INFO	*lib_infop;
2335 	HBA_HANDLE		vendorHandle;
2336 	HBAGetAdapterAttributesFunc GetAdapterAttributesFunc;
2337 
2338 	DEBUG(2, "HBA_GetAdapterAttributes", 0, 0, 0);
2339 
2340 	CHECKLIBRARY();
2341 
2342 	if (lib_infop->version == SMHBA) {
2343 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2344 	}
2345 
2346 	GetAdapterAttributesFunc =
2347 	    lib_infop->ftable.functionTable.GetAdapterAttributesHandler;
2348 	if (GetAdapterAttributesFunc != NULL) {
2349 	status = ((GetAdapterAttributesFunc)(vendorHandle, hbaattributes));
2350 	} else {
2351 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2352 	}
2353 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2354 }
2355 
2356 HBA_STATUS
2357 HBA_GetAdapterPortAttributes(
2358     HBA_HANDLE		handle,
2359     HBA_UINT32		portindex,
2360     HBA_PORTATTRIBUTES	*portattributes)
2361 {
2362 	HBA_STATUS		status;
2363 	HBA_LIBRARY_INFO	*lib_infop;
2364 	HBA_HANDLE		vendorHandle;
2365 	HBAGetAdapterPortAttributesFunc
2366 	    GetAdapterPortAttributesFunc;
2367 
2368 	DEBUG(2, "HBA_GetAdapterPortAttributes", 0, 0, 0);
2369 
2370 	CHECKLIBRARY();
2371 	if (lib_infop->version == SMHBA) {
2372 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2373 	}
2374 
2375 	GetAdapterPortAttributesFunc =
2376 	    lib_infop->ftable.functionTable.GetAdapterPortAttributesHandler;
2377 	if (GetAdapterPortAttributesFunc != NULL) {
2378 	status = ((GetAdapterPortAttributesFunc)
2379 	    (vendorHandle, portindex, portattributes));
2380 	} else {
2381 		status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2382 	}
2383 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2384 }
2385 
2386 HBA_STATUS
2387 HBA_GetPortStatistics(
2388     HBA_HANDLE		handle,
2389     HBA_UINT32		portindex,
2390     HBA_PORTSTATISTICS	*portstatistics)
2391 {
2392 	HBA_STATUS		status;
2393 	HBA_LIBRARY_INFO	*lib_infop;
2394 	HBA_HANDLE		vendorHandle;
2395 	HBAGetPortStatisticsFunc
2396 	    GetPortStatisticsFunc;
2397 
2398 	DEBUG(2, "HBA_GetPortStatistics", 0, 0, 0);
2399 
2400 	CHECKLIBRARY();
2401 	if (lib_infop->version == SMHBA) {
2402 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2403 	}
2404 
2405 	GetPortStatisticsFunc =
2406 	    lib_infop->ftable.functionTable.GetPortStatisticsHandler;
2407 	if (GetPortStatisticsFunc != NULL) {
2408 	status = ((GetPortStatisticsFunc)
2409 	    (vendorHandle, portindex, portstatistics));
2410 	} else {
2411 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2412 	}
2413 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2414 }
2415 
2416 HBA_STATUS
2417 HBA_GetDiscoveredPortAttributes(
2418     HBA_HANDLE		handle,
2419     HBA_UINT32		portindex,
2420     HBA_UINT32		discoveredportindex,
2421     HBA_PORTATTRIBUTES	*portattributes)
2422 {
2423 	HBA_STATUS		status;
2424 	HBA_LIBRARY_INFO	*lib_infop;
2425 	HBA_HANDLE		vendorHandle;
2426 	HBAGetDiscoveredPortAttributesFunc
2427 	    GetDiscoveredPortAttributesFunc;
2428 
2429 	DEBUG(2, "HBA_GetDiscoveredPortAttributes", 0, 0, 0);
2430 
2431 	CHECKLIBRARY();
2432 	if (lib_infop->version == SMHBA) {
2433 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2434 	}
2435 
2436 	GetDiscoveredPortAttributesFunc =
2437 	    lib_infop->ftable.functionTable.GetDiscoveredPortAttributesHandler;
2438 	if (GetDiscoveredPortAttributesFunc != NULL)  {
2439 	status = ((GetDiscoveredPortAttributesFunc)
2440 	    (vendorHandle, portindex, discoveredportindex,
2441 	    portattributes));
2442 	} else {
2443 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2444 	}
2445 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2446 }
2447 
2448 HBA_STATUS
2449 HBA_GetPortAttributesByWWN(
2450     HBA_HANDLE		handle,
2451     HBA_WWN		PortWWN,
2452     HBA_PORTATTRIBUTES	*portattributes)
2453 {
2454 	HBA_STATUS		status;
2455 	HBA_LIBRARY_INFO	*lib_infop;
2456 	HBA_HANDLE		vendorHandle;
2457 	HBAGetPortAttributesByWWNFunc
2458 	    GetPortAttributesByWWNFunc;
2459 
2460 	DEBUG(2, "HBA_GetPortAttributesByWWN: %s", WWN2STR1(&PortWWN), 0, 0);
2461 
2462 	CHECKLIBRARY();
2463 	if (lib_infop->version == SMHBA) {
2464 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2465 	}
2466 
2467 	GetPortAttributesByWWNFunc =
2468 	    lib_infop->ftable.functionTable.GetPortAttributesByWWNHandler;
2469 	if (GetPortAttributesByWWNFunc != NULL) {
2470 	status = ((GetPortAttributesByWWNFunc)
2471 	    (vendorHandle, PortWWN, portattributes));
2472 	} else {
2473 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2474 	}
2475 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2476 }
2477 
2478 HBA_STATUS
2479 HBA_SendCTPassThru(
2480     HBA_HANDLE		handle,
2481     void		*pReqBuffer,
2482     HBA_UINT32		ReqBufferSize,
2483     void		*pRspBuffer,
2484     HBA_UINT32		RspBufferSize)
2485 {
2486 	HBA_STATUS		status;
2487 	HBA_LIBRARY_INFO	*lib_infop;
2488 	HBA_HANDLE		vendorHandle;
2489 	HBASendCTPassThruFunc
2490 	    SendCTPassThruFunc;
2491 
2492 	DEBUG(2, "HBA_SendCTPassThru", 0, 0, 0);
2493 
2494 	CHECKLIBRARY();
2495 	if (lib_infop->version == SMHBA) {
2496 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2497 	}
2498 
2499 	SendCTPassThruFunc =
2500 	    lib_infop->ftable.functionTable.SendCTPassThruHandler;
2501 	if (SendCTPassThruFunc != NULL) {
2502 	status = (SendCTPassThruFunc)
2503 	    (vendorHandle,
2504 	    pReqBuffer, ReqBufferSize,
2505 	    pRspBuffer, RspBufferSize);
2506 	} else {
2507 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2508 	}
2509 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2510 }
2511 
2512 HBA_STATUS
2513 HBA_SendCTPassThruV2(
2514     HBA_HANDLE		handle,
2515     HBA_WWN		hbaPortWWN,
2516     void		*pReqBuffer,
2517     HBA_UINT32		ReqBufferSize,
2518     void		*pRspBuffer,
2519     HBA_UINT32		*pRspBufferSize)
2520 {
2521 	HBA_STATUS		status;
2522 	HBA_LIBRARY_INFO	*lib_infop;
2523 	HBA_HANDLE		vendorHandle;
2524 	HBASendCTPassThruV2Func
2525 	    registeredfunc;
2526 
2527 	DEBUG(2, "HBA_SendCTPassThruV2m hbaPortWWN: %s",
2528 	    WWN2STR1(&hbaPortWWN), 0, 0);
2529 
2530 	CHECKLIBRARYANDVERSION(HBAAPIV2);
2531 	registeredfunc = FUNCCOMMON(lib_infop, SendCTPassThruV2Handler);
2532 	if (registeredfunc != NULL) {
2533 	status = (registeredfunc)
2534 	    (vendorHandle, hbaPortWWN,
2535 	    pReqBuffer, ReqBufferSize,
2536 	    pRspBuffer, pRspBufferSize);
2537 	} else {
2538 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2539 	}
2540 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2541 }
2542 
2543 HBA_STATUS
2544 HBA_GetEventBuffer(
2545     HBA_HANDLE		handle,
2546     PHBA_EVENTINFO	EventBuffer,
2547     HBA_UINT32		*EventBufferCount)
2548 {
2549 	HBA_STATUS		status;
2550 	HBA_LIBRARY_INFO	*lib_infop;
2551 	HBA_HANDLE		vendorHandle;
2552 	HBAGetEventBufferFunc
2553 	    GetEventBufferFunc;
2554 
2555 	DEBUG(2, "HBA_GetEventBuffer", 0, 0, 0);
2556 
2557 	CHECKLIBRARY();
2558 	if (lib_infop->version == SMHBA) {
2559 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2560 	}
2561 
2562 	GetEventBufferFunc =
2563 	    lib_infop->ftable.functionTable.GetEventBufferHandler;
2564 	if (GetEventBufferFunc != NULL) {
2565 	status = (GetEventBufferFunc)
2566 	    (vendorHandle, EventBuffer, EventBufferCount);
2567 	} else {
2568 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2569 	}
2570 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2571 }
2572 
2573 HBA_STATUS
2574 HBA_SetRNIDMgmtInfo(HBA_HANDLE handle, HBA_MGMTINFO Info) {
2575     HBA_STATUS		status;
2576     HBA_LIBRARY_INFO	*lib_infop;
2577     HBA_HANDLE		vendorHandle;
2578     HBASetRNIDMgmtInfoFunc
2579 			SetRNIDMgmtInfoFunc;
2580 
2581     DEBUG(2, "HBA_SetRNIDMgmtInfo", 0, 0, 0);
2582 
2583     CHECKLIBRARY();
2584     SetRNIDMgmtInfoFunc = FUNCCOMMON(lib_infop, SetRNIDMgmtInfoHandler);
2585     if (SetRNIDMgmtInfoFunc != NULL) {
2586 	status = (SetRNIDMgmtInfoFunc)(vendorHandle, Info);
2587 	} else {
2588 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2589 	}
2590     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2591 }
2592 
2593 HBA_STATUS
2594 HBA_GetRNIDMgmtInfo(HBA_HANDLE handle, HBA_MGMTINFO *pInfo) {
2595     HBA_STATUS		status;
2596     HBA_LIBRARY_INFO	*lib_infop;
2597     HBA_HANDLE		vendorHandle;
2598     HBAGetRNIDMgmtInfoFunc
2599 	    GetRNIDMgmtInfoFunc;
2600 
2601     DEBUG(2, "HBA_GetRNIDMgmtInfo", 0, 0, 0);
2602 
2603     CHECKLIBRARY();
2604     GetRNIDMgmtInfoFunc = FUNCCOMMON(lib_infop, GetRNIDMgmtInfoHandler);
2605     if (GetRNIDMgmtInfoFunc != NULL) {
2606 	status = (GetRNIDMgmtInfoFunc)(vendorHandle, pInfo);
2607 	} else {
2608 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2609 	}
2610     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2611 }
2612 
2613 HBA_STATUS
2614 HBA_SendRNID(
2615     HBA_HANDLE		handle,
2616     HBA_WWN		wwn,
2617     HBA_WWNTYPE		wwntype,
2618     void		*pRspBuffer,
2619     HBA_UINT32		*pRspBufferSize)
2620 {
2621 	HBA_STATUS		status;
2622 	HBA_LIBRARY_INFO	*lib_infop;
2623 	HBA_HANDLE		vendorHandle;
2624 	HBASendRNIDFunc	SendRNIDFunc;
2625 
2626 	DEBUG(2, "HBA_SendRNID for wwn: %s", WWN2STR1(&wwn), 0, 0);
2627 
2628 	CHECKLIBRARY();
2629 	if (lib_infop->version == SMHBA) {
2630 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2631 	}
2632 
2633 	SendRNIDFunc = lib_infop->ftable.functionTable.SendRNIDHandler;
2634 	if (SendRNIDFunc != NULL) {
2635 	status = ((SendRNIDFunc)(vendorHandle, wwn, wwntype,
2636 	    pRspBuffer, pRspBufferSize));
2637 	} else {
2638 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2639 	}
2640 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2641 }
2642 
2643 HBA_STATUS
2644 HBA_SendRNIDV2(
2645     HBA_HANDLE		handle,
2646     HBA_WWN		hbaPortWWN,
2647     HBA_WWN		destWWN,
2648     HBA_UINT32		destFCID,
2649     HBA_UINT32		NodeIdDataFormat,
2650     void		*pRspBuffer,
2651     HBA_UINT32		*pRspBufferSize)
2652 {
2653 	HBA_STATUS		status;
2654 	HBA_LIBRARY_INFO	*lib_infop;
2655 	HBA_HANDLE		vendorHandle;
2656 	HBASendRNIDV2Func	registeredfunc;
2657 
2658 	DEBUG(2, "HBA_SendRNIDV2, hbaPortWWN: %s", WWN2STR1(&hbaPortWWN), 0, 0);
2659 
2660 	CHECKLIBRARY();
2661 	registeredfunc = FUNCCOMMON(lib_infop, SendRNIDV2Handler);
2662 	if (registeredfunc != NULL) {
2663 	status = (registeredfunc)
2664 	    (vendorHandle, hbaPortWWN, destWWN, destFCID, NodeIdDataFormat,
2665 	    pRspBuffer, pRspBufferSize);
2666 	} else {
2667 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2668 	}
2669 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2670 }
2671 
2672 void
2673 HBA_RefreshInformation(HBA_HANDLE handle) {
2674     HBA_STATUS		status;
2675     HBA_LIBRARY_INFO	*lib_infop;
2676     HBA_HANDLE		vendorHandle;
2677     HBARefreshInformationFunc
2678 	    RefreshInformationFunc;
2679 
2680 	DEBUG(2, "HBA_RefreshInformation", 0, 0, 0);
2681 
2682 	status = HBA_CheckLibrary(handle, &lib_infop, &vendorHandle);
2683 	if (status == HBA_STATUS_OK) {
2684 	RefreshInformationFunc =
2685 	    FUNCCOMMON(lib_infop, RefreshInformationHandler);
2686 	if (RefreshInformationFunc != NULL) {
2687 	    ((RefreshInformationFunc)(vendorHandle));
2688 	}
2689 	RELEASE_MUTEX(&_hbaapi_LL_mutex);
2690 	}
2691 }
2692 
2693 void
2694 HBA_ResetStatistics(HBA_HANDLE handle, HBA_UINT32 portindex) {
2695     HBA_STATUS		status;
2696     HBA_LIBRARY_INFO	*lib_infop;
2697     HBA_HANDLE		vendorHandle;
2698     HBAResetStatisticsFunc
2699 			ResetStatisticsFunc;
2700 
2701     DEBUG(2, "HBA_ResetStatistics", 0, 0, 0);
2702 
2703     status = HBA_CheckLibrary(handle, &lib_infop, &vendorHandle);
2704     if (status == HBA_STATUS_OK) {
2705 	if (lib_infop->version == SMHBA) {
2706 		RELEASE_MUTEX(&_hbaapi_LL_mutex);
2707 	}
2708 
2709 	ResetStatisticsFunc =
2710 	    lib_infop->ftable.functionTable.ResetStatisticsHandler;
2711 	if (ResetStatisticsFunc != NULL) {
2712 	    ((ResetStatisticsFunc)(vendorHandle, portindex));
2713 	}
2714 	RELEASE_MUTEX(&_hbaapi_LL_mutex);
2715 	}
2716 }
2717 
2718 HBA_STATUS
2719 HBA_GetFcpTargetMapping(HBA_HANDLE handle, PHBA_FCPTARGETMAPPING mapping) {
2720     HBA_STATUS		status;
2721     HBA_LIBRARY_INFO	*lib_infop;
2722     HBA_HANDLE		vendorHandle;
2723     HBAGetFcpTargetMappingFunc GetFcpTargetMappingFunc;
2724 
2725     DEBUG(2, "HBA_GetFcpTargetMapping", 0, 0, 0);
2726 
2727     CHECKLIBRARY();
2728     if (lib_infop->version == SMHBA) {
2729 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2730 	}
2731 
2732     GetFcpTargetMappingFunc =
2733 	lib_infop->ftable.functionTable.GetFcpTargetMappingHandler;
2734     if (GetFcpTargetMappingFunc != NULL) {
2735 	status = ((GetFcpTargetMappingFunc)(vendorHandle, mapping));
2736 	} else {
2737 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2738 	}
2739     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2740 }
2741 
2742 HBA_STATUS
2743 HBA_GetFcpTargetMappingV2(
2744     HBA_HANDLE		handle,
2745     HBA_WWN		hbaPortWWN,
2746     HBA_FCPTARGETMAPPINGV2 *pmapping)
2747 {
2748 	HBA_STATUS		status;
2749 	HBA_LIBRARY_INFO	*lib_infop;
2750 	HBA_HANDLE		vendorHandle;
2751 	HBAGetFcpTargetMappingV2Func
2752 	    registeredfunc;
2753 
2754 	DEBUG(2, "HBA_GetFcpTargetMapping", 0, 0, 0);
2755 
2756 	CHECKLIBRARYANDVERSION(HBAAPIV2);
2757 
2758 	registeredfunc =
2759 	    lib_infop->ftable.functionTable.GetFcpTargetMappingV2Handler;
2760 	if (registeredfunc != NULL) {
2761 	status = ((registeredfunc)(vendorHandle, hbaPortWWN, pmapping));
2762 	} else {
2763 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2764 	}
2765 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2766 }
2767 
2768 HBA_STATUS
2769 HBA_GetFcpPersistentBinding(HBA_HANDLE handle, PHBA_FCPBINDING binding) {
2770     HBA_STATUS		status;
2771     HBA_LIBRARY_INFO	*lib_infop;
2772     HBA_HANDLE		vendorHandle;
2773     HBAGetFcpPersistentBindingFunc
2774 	    GetFcpPersistentBindingFunc;
2775 
2776 	DEBUG(2, "HBA_GetFcpPersistentBinding", 0, 0, 0);
2777 
2778 	CHECKLIBRARY();
2779 	if (lib_infop->version == SMHBA) {
2780 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2781 	}
2782 
2783 	GetFcpPersistentBindingFunc =
2784 	    lib_infop->ftable.functionTable.GetFcpPersistentBindingHandler;
2785 	if (GetFcpPersistentBindingFunc != NULL) {
2786 	status = ((GetFcpPersistentBindingFunc)(vendorHandle, binding));
2787 	} else {
2788 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2789 	}
2790 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2791 }
2792 
2793 HBA_STATUS
2794 HBA_ScsiInquiryV2(
2795     HBA_HANDLE	handle,
2796     HBA_WWN	hbaPortWWN,
2797     HBA_WWN	discoveredPortWWN,
2798     HBA_UINT64	fcLUN,
2799     HBA_UINT8	CDB_Byte1,
2800     HBA_UINT8	CDB_Byte2,
2801     void	*pRspBuffer,
2802     HBA_UINT32	*pRspBufferSize,
2803     HBA_UINT8	*pScsiStatus,
2804     void	*pSenseBuffer,
2805     HBA_UINT32	*pSenseBufferSize)
2806 {
2807 	HBA_STATUS		status;
2808 	HBA_LIBRARY_INFO	*lib_infop;
2809 	HBA_HANDLE		vendorHandle;
2810 	HBAScsiInquiryV2Func ScsiInquiryV2Func;
2811 
2812 	DEBUG(2, "HBA_ScsiInquiryV2 to discoveredPortWWN: %s",
2813 	    WWN2STR1(&discoveredPortWWN), 0, 0);
2814 
2815 	CHECKLIBRARYANDVERSION(HBAAPIV2);
2816 
2817 	ScsiInquiryV2Func =
2818 	    lib_infop->ftable.functionTable.ScsiInquiryV2Handler;
2819 	if (ScsiInquiryV2Func != NULL) {
2820 	status = ((ScsiInquiryV2Func)(
2821 	    vendorHandle, hbaPortWWN, discoveredPortWWN, fcLUN, CDB_Byte1,
2822 	    CDB_Byte2, pRspBuffer, pRspBufferSize, pScsiStatus,
2823 	    pSenseBuffer, pSenseBufferSize));
2824 	} else {
2825 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2826 	}
2827 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2828 }
2829 
2830 HBA_STATUS
2831 HBA_SendScsiInquiry(
2832     HBA_HANDLE	handle,
2833     HBA_WWN	PortWWN,
2834     HBA_UINT64	fcLUN,
2835     HBA_UINT8	EVPD,
2836     HBA_UINT32	PageCode,
2837     void	*pRspBuffer,
2838     HBA_UINT32	RspBufferSize,
2839     void	*pSenseBuffer,
2840     HBA_UINT32	SenseBufferSize)
2841 {
2842 	HBA_STATUS		status;
2843 	HBA_LIBRARY_INFO	*lib_infop;
2844 	HBA_HANDLE		vendorHandle;
2845 	HBASendScsiInquiryFunc SendScsiInquiryFunc;
2846 
2847 	DEBUG(2, "HBA_SendScsiInquiry to PortWWN: %s",
2848 	    WWN2STR1(&PortWWN), 0, 0);
2849 
2850 	CHECKLIBRARY();
2851 	if (lib_infop->version == SMHBA) {
2852 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2853 	}
2854 
2855 	SendScsiInquiryFunc =
2856 	    lib_infop->ftable.functionTable.ScsiInquiryHandler;
2857 	if (SendScsiInquiryFunc != NULL) {
2858 	status = ((SendScsiInquiryFunc)(
2859 	    vendorHandle, PortWWN, fcLUN, EVPD, PageCode, pRspBuffer,
2860 	    RspBufferSize, pSenseBuffer, SenseBufferSize));
2861 	} else {
2862 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2863 	}
2864 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2865 }
2866 
2867 HBA_STATUS
2868 HBA_ScsiReportLUNsV2(
2869     HBA_HANDLE		handle,
2870     HBA_WWN		hbaPortWWN,
2871     HBA_WWN		discoveredPortWWN,
2872     void		*pRespBuffer,
2873     HBA_UINT32		*pRespBufferSize,
2874     HBA_UINT8		*pScsiStatus,
2875     void		*pSenseBuffer,
2876     HBA_UINT32		*pSenseBufferSize)
2877 {
2878 	HBA_STATUS		status;
2879 	HBA_LIBRARY_INFO	*lib_infop;
2880 	HBA_HANDLE		vendorHandle;
2881 	HBAScsiReportLUNsV2Func ScsiReportLUNsV2Func;
2882 
2883 	DEBUG(2, "HBA_ScsiReportLUNsV2 to discoveredPortWWN: %s",
2884 	    WWN2STR1(&discoveredPortWWN), 0, 0);
2885 
2886 	CHECKLIBRARYANDVERSION(HBAAPIV2);
2887 
2888 	ScsiReportLUNsV2Func =
2889 	    lib_infop->ftable.functionTable.ScsiReportLUNsV2Handler;
2890 	if (ScsiReportLUNsV2Func != NULL) {
2891 	status = ((ScsiReportLUNsV2Func)(
2892 	    vendorHandle, hbaPortWWN, discoveredPortWWN,
2893 	    pRespBuffer, pRespBufferSize,
2894 	    pScsiStatus,
2895 	    pSenseBuffer, pSenseBufferSize));
2896 	} else {
2897 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2898 	}
2899 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2900 }
2901 
2902 HBA_STATUS
2903 HBA_SendReportLUNs(
2904     HBA_HANDLE handle,
2905     HBA_WWN portWWN,
2906     void *pRspBuffer,
2907     HBA_UINT32 RspBufferSize,
2908     void *pSenseBuffer,
2909     HBA_UINT32 SenseBufferSize)
2910 {
2911 	HBA_STATUS		status;
2912 	HBA_LIBRARY_INFO	*lib_infop;
2913 	HBA_HANDLE		vendorHandle;
2914 	HBASendReportLUNsFunc SendReportLUNsFunc;
2915 
2916 	DEBUG(2, "HBA_SendReportLUNs to PortWWN: %s", WWN2STR1(&portWWN), 0, 0);
2917 
2918 	CHECKLIBRARY();
2919 	if (lib_infop->version == SMHBA) {
2920 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2921 	}
2922 
2923 	SendReportLUNsFunc = lib_infop->ftable.functionTable.ReportLUNsHandler;
2924 	if (SendReportLUNsFunc != NULL) {
2925 	status = ((SendReportLUNsFunc)(
2926 	    vendorHandle, portWWN, pRspBuffer,
2927 	    RspBufferSize, pSenseBuffer, SenseBufferSize));
2928 	} else {
2929 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2930 	}
2931 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2932 }
2933 
2934 HBA_STATUS
2935 HBA_ScsiReadCapacityV2(
2936     HBA_HANDLE		handle,
2937     HBA_WWN		hbaPortWWN,
2938     HBA_WWN		discoveredPortWWN,
2939     HBA_UINT64		fcLUN,
2940     void		*pRspBuffer,
2941     HBA_UINT32		*pRspBufferSize,
2942     HBA_UINT8		*pScsiStatus,
2943     void		*pSenseBuffer,
2944     HBA_UINT32		*SenseBufferSize)
2945 {
2946 	HBA_STATUS		status;
2947 	HBA_LIBRARY_INFO	*lib_infop;
2948 	HBA_HANDLE		vendorHandle;
2949 	HBAScsiReadCapacityV2Func ScsiReadCapacityV2Func;
2950 
2951 	DEBUG(2, "HBA_ScsiReadCapacityV2 to discoveredPortWWN: %s",
2952 	    WWN2STR1(&discoveredPortWWN), 0, 0);
2953 
2954 	CHECKLIBRARYANDVERSION(HBAAPIV2);
2955 
2956 	ScsiReadCapacityV2Func =
2957 	    lib_infop->ftable.functionTable.ScsiReadCapacityV2Handler;
2958 	if (ScsiReadCapacityV2Func != NULL) {
2959 	status = ((ScsiReadCapacityV2Func)(
2960 	    vendorHandle, hbaPortWWN, discoveredPortWWN, fcLUN,
2961 	    pRspBuffer, pRspBufferSize,
2962 	    pScsiStatus,
2963 	    pSenseBuffer, SenseBufferSize));
2964 	} else {
2965 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
2966 	}
2967 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
2968 }
2969 
2970 HBA_STATUS
2971 HBA_SendReadCapacity(
2972     HBA_HANDLE handle,
2973     HBA_WWN portWWN,
2974     HBA_UINT64 fcLUN,
2975     void *pRspBuffer,
2976     HBA_UINT32 RspBufferSize,
2977     void *pSenseBuffer,
2978     HBA_UINT32 SenseBufferSize)
2979 {
2980 	HBA_STATUS		status;
2981 	HBA_LIBRARY_INFO	*lib_infop;
2982 	HBA_HANDLE		vendorHandle;
2983 	HBASendReadCapacityFunc SendReadCapacityFunc;
2984 
2985 	DEBUG(2, "HBA_SendReadCapacity to portWWN: %s",
2986 	    WWN2STR1(&portWWN), 0, 0);
2987 
2988 	CHECKLIBRARY();
2989 	if (lib_infop->version == SMHBA) {
2990 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_INCOMPATIBLE);
2991 	}
2992 
2993 	SendReadCapacityFunc =
2994 	    lib_infop->ftable.functionTable.ReadCapacityHandler;
2995 	if (SendReadCapacityFunc != NULL) {
2996 	status = ((SendReadCapacityFunc)
2997 	    (vendorHandle, portWWN, fcLUN, pRspBuffer,
2998 	    RspBufferSize, pSenseBuffer, SenseBufferSize));
2999 	} else {
3000 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3001 	}
3002 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3003 }
3004 
3005 HBA_STATUS
3006 HBA_SendRPL(
3007     HBA_HANDLE		handle,
3008     HBA_WWN		hbaPortWWN,
3009     HBA_WWN		agent_wwn,
3010     HBA_UINT32		agent_domain,
3011     HBA_UINT32		portindex,
3012     void		*pRspBuffer,
3013     HBA_UINT32		*pRspBufferSize)
3014 {
3015 	HBA_STATUS		status;
3016 	HBA_LIBRARY_INFO	*lib_infop;
3017 	HBA_HANDLE		vendorHandle;
3018 	HBASendRPLFunc registeredfunc;
3019 
3020 	DEBUG(2, "HBA_SendRPL to agent_wwn: %s:%d",
3021 	    WWN2STR1(&agent_wwn), agent_domain, 0);
3022 
3023 	CHECKLIBRARY();
3024 	registeredfunc = FUNCCOMMON(lib_infop, SendRPLHandler);
3025 	if (registeredfunc != NULL) {
3026 	status = (registeredfunc)(
3027 	    vendorHandle, hbaPortWWN, agent_wwn, agent_domain, portindex,
3028 	    pRspBuffer, pRspBufferSize);
3029 	} else {
3030 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3031 	}
3032 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3033 }
3034 
3035 HBA_STATUS
3036 HBA_SendRPS(
3037     HBA_HANDLE		handle,
3038     HBA_WWN		hbaPortWWN,
3039     HBA_WWN		agent_wwn,
3040     HBA_UINT32		agent_domain,
3041     HBA_WWN		object_wwn,
3042     HBA_UINT32		object_port_number,
3043     void		*pRspBuffer,
3044     HBA_UINT32		*pRspBufferSize)
3045 {
3046 	HBA_STATUS		status;
3047 	HBA_LIBRARY_INFO	*lib_infop;
3048 	HBA_HANDLE		vendorHandle;
3049 	HBASendRPSFunc registeredfunc;
3050 
3051 	DEBUG(2, "HBA_SendRPS  to agent_wwn: %s:%d",
3052 	    WWN2STR1(&agent_wwn), agent_domain, 0);
3053 
3054 	CHECKLIBRARY();
3055 	registeredfunc = FUNCCOMMON(lib_infop, SendRPSHandler);
3056 	if (registeredfunc != NULL) {
3057 	status = (registeredfunc)(
3058 	    vendorHandle, hbaPortWWN, agent_wwn, agent_domain,
3059 	    object_wwn, object_port_number,
3060 	    pRspBuffer, pRspBufferSize);
3061 	} else {
3062 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3063 	}
3064 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3065 }
3066 
3067 HBA_STATUS
3068 HBA_SendSRL(
3069     HBA_HANDLE		handle,
3070     HBA_WWN		hbaPortWWN,
3071     HBA_WWN		wwn,
3072     HBA_UINT32		domain,
3073     void		*pRspBuffer,
3074     HBA_UINT32		*pRspBufferSize)
3075 {
3076 	HBA_STATUS		status;
3077 	HBA_LIBRARY_INFO	*lib_infop;
3078 	HBA_HANDLE		vendorHandle;
3079 	HBASendSRLFunc registeredfunc;
3080 
3081 	DEBUG(2, "HBA_SendSRL to wwn:%s domain:%d", WWN2STR1(&wwn), domain, 0);
3082 
3083 	CHECKLIBRARY();
3084 	registeredfunc = FUNCCOMMON(lib_infop, SendSRLHandler);
3085 	if (registeredfunc != NULL) {
3086 	status = (registeredfunc)(
3087 	    vendorHandle, hbaPortWWN, wwn, domain,
3088 	    pRspBuffer, pRspBufferSize);
3089 	} else {
3090 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3091 	}
3092 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3093 }
3094 HBA_STATUS
3095 HBA_SendRLS(
3096     HBA_HANDLE		handle,
3097     HBA_WWN		hbaPortWWN,
3098     HBA_WWN		destWWN,
3099     void		*pRspBuffer,
3100     HBA_UINT32		*pRspBufferSize)
3101 {
3102 	HBA_STATUS		status;
3103 	HBA_LIBRARY_INFO	*lib_infop;
3104 	HBA_HANDLE		vendorHandle;
3105 	HBASendRLSFunc registeredfunc;
3106 
3107 	DEBUG(2, "HBA_SendRLS dest_wwn: %s",
3108 	    WWN2STR1(&destWWN), 0, 0);
3109 
3110 	CHECKLIBRARY();
3111 	registeredfunc = FUNCCOMMON(lib_infop, SendRLSHandler);
3112 	if (registeredfunc != NULL) {
3113 	status = (registeredfunc)(
3114 	    vendorHandle, hbaPortWWN, destWWN,
3115 	    pRspBuffer, pRspBufferSize);
3116 	} else {
3117 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3118 	}
3119 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3120 }
3121 
3122 HBA_STATUS
3123 HBA_SendLIRR(
3124     HBA_HANDLE		handle,
3125     HBA_WWN		sourceWWN,
3126     HBA_WWN		destWWN,
3127     HBA_UINT8		function,
3128     HBA_UINT8		type,
3129     void		*pRspBuffer,
3130     HBA_UINT32		*pRspBufferSize)
3131 {
3132 	HBA_STATUS		status;
3133 	HBA_LIBRARY_INFO	*lib_infop;
3134 	HBA_HANDLE		vendorHandle;
3135 	HBASendLIRRFunc registeredfunc;
3136 
3137 	DEBUG(2, "HBA_SendLIRR destWWN:%s", WWN2STR1(&destWWN), 0, 0);
3138 
3139 	CHECKLIBRARY();
3140 	registeredfunc = FUNCCOMMON(lib_infop, SendLIRRHandler);
3141 	if (registeredfunc != NULL) {
3142 	status = (registeredfunc)(
3143 	    vendorHandle, sourceWWN, destWWN, function, type,
3144 	    pRspBuffer, pRspBufferSize);
3145 	} else {
3146 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3147 	}
3148 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3149 }
3150 
3151 HBA_STATUS
3152 HBA_GetBindingCapability(
3153     HBA_HANDLE		handle,
3154     HBA_WWN		hbaPortWWN,
3155     HBA_BIND_CAPABILITY *pcapability)
3156 {
3157 	HBA_STATUS		status;
3158 	HBA_LIBRARY_INFO	*lib_infop;
3159 	HBA_HANDLE		vendorHandle;
3160 	HBAGetBindingCapabilityFunc
3161 	    registeredfunc;
3162 
3163 	DEBUG(2, "HBA_GetBindingCapability", 0, 0, 0);
3164 
3165 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3166 
3167 	registeredfunc =
3168 	    lib_infop->ftable.functionTable.GetBindingCapabilityHandler;
3169 	if (registeredfunc != NULL) {
3170 	status = (registeredfunc)(vendorHandle, hbaPortWWN, pcapability);
3171 	} else {
3172 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3173 	}
3174 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3175 }
3176 
3177 HBA_STATUS
3178 HBA_GetBindingSupport(
3179     HBA_HANDLE		handle,
3180     HBA_WWN		hbaPortWWN,
3181     HBA_BIND_CAPABILITY *pcapability)
3182 {
3183 	HBA_STATUS		status;
3184 	HBA_LIBRARY_INFO	*lib_infop;
3185 	HBA_HANDLE		vendorHandle;
3186 	HBAGetBindingSupportFunc
3187 	    registeredfunc;
3188 
3189 	DEBUG(2, "HBA_GetBindingSupport", 0, 0, 0);
3190 
3191 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3192 
3193 	registeredfunc =
3194 	    lib_infop->ftable.functionTable.GetBindingSupportHandler;
3195 	if (registeredfunc != NULL) {
3196 	status = (registeredfunc)(vendorHandle, hbaPortWWN, pcapability);
3197 	} else {
3198 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3199 	}
3200 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3201 }
3202 
3203 HBA_STATUS
3204 HBA_SetBindingSupport(
3205     HBA_HANDLE		handle,
3206     HBA_WWN		hbaPortWWN,
3207     HBA_BIND_CAPABILITY capability)
3208 {
3209 	HBA_STATUS		status;
3210 	HBA_LIBRARY_INFO	*lib_infop;
3211 	HBA_HANDLE		vendorHandle;
3212 	HBASetBindingSupportFunc
3213 	    registeredfunc;
3214 
3215 	DEBUG(2, "HBA_SetBindingSupport", 0, 0, 0);
3216 
3217 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3218 
3219 	registeredfunc =
3220 	    lib_infop->ftable.functionTable.SetBindingSupportHandler;
3221 	if (registeredfunc != NULL) {
3222 	status = (registeredfunc)(vendorHandle, hbaPortWWN, capability);
3223 	} else {
3224 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3225 	}
3226 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3227 }
3228 
3229 HBA_STATUS
3230 HBA_SetPersistentBindingV2(
3231     HBA_HANDLE		handle,
3232     HBA_WWN		hbaPortWWN,
3233     const HBA_FCPBINDING2 *pbinding)
3234 {
3235 	HBA_STATUS		status;
3236 	HBA_LIBRARY_INFO	*lib_infop;
3237 	HBA_HANDLE		vendorHandle;
3238 	HBASetPersistentBindingV2Func
3239 	    registeredfunc;
3240 
3241 	DEBUG(2, "HBA_SetPersistentBindingV2 port: %s",
3242 	    WWN2STR1(&hbaPortWWN), 0, 0);
3243 
3244 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3245 
3246 	registeredfunc =
3247 	    lib_infop->ftable.functionTable.SetPersistentBindingV2Handler;
3248 	if (registeredfunc != NULL) {
3249 	status = (registeredfunc)(vendorHandle, hbaPortWWN, pbinding);
3250 	} else {
3251 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3252 	}
3253 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3254 }
3255 
3256 HBA_STATUS
3257 HBA_GetPersistentBindingV2(
3258     HBA_HANDLE		handle,
3259     HBA_WWN		hbaPortWWN,
3260     HBA_FCPBINDING2	*pbinding)
3261 {
3262 	HBA_STATUS		status;
3263 	HBA_LIBRARY_INFO	*lib_infop;
3264 	HBA_HANDLE		vendorHandle;
3265 	HBAGetPersistentBindingV2Func
3266 	    registeredfunc;
3267 
3268 	DEBUG(2, "HBA_GetPersistentBindingV2 port: %s",
3269 	    WWN2STR1(&hbaPortWWN), 0, 0);
3270 
3271 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3272 
3273 	registeredfunc =
3274 	    lib_infop->ftable.functionTable.GetPersistentBindingV2Handler;
3275 	if (registeredfunc != NULL) {
3276 	status = (registeredfunc)(vendorHandle, hbaPortWWN, pbinding);
3277 	} else {
3278 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3279 	}
3280 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3281 }
3282 
3283 HBA_STATUS
3284 HBA_RemovePersistentBinding(
3285     HBA_HANDLE		handle,
3286     HBA_WWN		hbaPortWWN,
3287     const HBA_FCPBINDING2
3288 			*pbinding)
3289 {
3290 	HBA_STATUS		status;
3291 	HBA_LIBRARY_INFO	*lib_infop;
3292 	HBA_HANDLE		vendorHandle;
3293 	HBARemovePersistentBindingFunc
3294 	    registeredfunc;
3295 
3296 	DEBUG(2, "HBA_RemovePersistentBinding", 0, 0, 0);
3297 
3298 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3299 
3300 	registeredfunc =
3301 	    lib_infop->ftable.functionTable.RemovePersistentBindingHandler;
3302 	if (registeredfunc != NULL) {
3303 	status = (registeredfunc)(vendorHandle, hbaPortWWN, pbinding);
3304 	} else {
3305 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3306 	}
3307 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3308 }
3309 
3310 HBA_STATUS
3311 HBA_RemoveAllPersistentBindings(
3312     HBA_HANDLE		handle,
3313     HBA_WWN		hbaPortWWN)
3314 {
3315 	HBA_STATUS		status;
3316 	HBA_LIBRARY_INFO	*lib_infop;
3317 	HBA_HANDLE		vendorHandle;
3318 	HBARemoveAllPersistentBindingsFunc
3319 	    registeredfunc;
3320 
3321 	DEBUG(2, "HBA_RemoveAllPersistentBindings", 0, 0, 0);
3322 
3323 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3324 
3325 	registeredfunc =
3326 	    lib_infop->ftable.functionTable.RemoveAllPersistentBindingsHandler;
3327 	if (registeredfunc != NULL) {
3328 	status = (registeredfunc)(vendorHandle, hbaPortWWN);
3329 	} else {
3330 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3331 	}
3332 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3333 }
3334 
3335 HBA_STATUS
3336 HBA_GetFC4Statistics(
3337     HBA_HANDLE		handle,
3338     HBA_WWN		portWWN,
3339     HBA_UINT8		FC4type,
3340     HBA_FC4STATISTICS	*pstatistics)
3341 {
3342 	HBA_STATUS		status;
3343 	HBA_LIBRARY_INFO	*lib_infop;
3344 	HBA_HANDLE		vendorHandle;
3345 	HBAGetFC4StatisticsFunc
3346 	    registeredfunc;
3347 
3348 	DEBUG(2, "HBA_GetFC4Statistics port: %s", WWN2STR1(&portWWN), 0, 0);
3349 
3350 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3351 
3352 	registeredfunc =
3353 	    lib_infop->ftable.functionTable.GetFC4StatisticsHandler;
3354 	if (registeredfunc != NULL) {
3355 	status = (registeredfunc)
3356 	    (vendorHandle, portWWN, FC4type, pstatistics);
3357 	} else {
3358 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3359 	}
3360 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3361 }
3362 
3363 HBA_STATUS
3364 HBA_GetFCPStatistics(
3365     HBA_HANDLE		handle,
3366     const HBA_SCSIID	*lunit,
3367     HBA_FC4STATISTICS	*pstatistics)
3368 {
3369 	HBA_STATUS		status;
3370 	HBA_LIBRARY_INFO	*lib_infop;
3371 	HBA_HANDLE		vendorHandle;
3372 	HBAGetFCPStatisticsFunc
3373 	    registeredfunc;
3374 
3375 	DEBUG(2, "HBA_GetFCPStatistics", 0, 0, 0);
3376 
3377 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3378 
3379 	registeredfunc =
3380 	    lib_infop->ftable.functionTable.GetFCPStatisticsHandler;
3381 	if (registeredfunc != NULL) {
3382 	status = (registeredfunc)(vendorHandle, lunit, pstatistics);
3383 	} else {
3384 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3385 	}
3386 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3387 }
3388 
3389 HBA_UINT32
3390 HBA_GetVendorLibraryAttributes(
3391     HBA_UINT32 adapter_index,
3392     HBA_LIBRARYATTRIBUTES *attributes)
3393 {
3394 	HBA_ADAPTER_INFO	*adapt_infop;
3395 	HBAGetVendorLibraryAttributesFunc
3396 	    registeredfunc;
3397 	HBA_UINT32		ret = 0;
3398 
3399 	DEBUG(2, "HBA_GetVendorLibraryAttributes adapterindex:%d",
3400 	    adapter_index, 0, 0);
3401 	if (_hbaapi_librarylist == NULL) {
3402 	DEBUG(1, "HBAAPI not loaded yet.", 0, 0, 0);
3403 	return (0);
3404 	}
3405 
3406 	if (attributes == NULL) {
3407 		DEBUG(1,
3408 		    "HBA_GetVendorLibraryAttributes: NULL pointer attributes",
3409 		    0, 0, 0);
3410 		return (HBA_STATUS_ERROR_ARG);
3411 	}
3412 
3413 	(void) memset(attributes, 0, sizeof (HBA_LIBRARYATTRIBUTES));
3414 
3415 	GRAB_MUTEX(&_hbaapi_LL_mutex);
3416 	GRAB_MUTEX(&_hbaapi_AL_mutex);
3417 	for (adapt_infop = _hbaapi_adapterlist;
3418 	    adapt_infop != NULL;
3419 	    adapt_infop = adapt_infop->next) {
3420 
3421 	if (adapt_infop->index == adapter_index) {
3422 
3423 		if (adapt_infop->library->version == SMHBA) {
3424 		RELEASE_MUTEX(&_hbaapi_AL_mutex);
3425 		RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex,
3426 		    HBA_STATUS_ERROR_INCOMPATIBLE);
3427 		}
3428 
3429 		registeredfunc = adapt_infop->library->
3430 		    ftable.functionTable.GetVendorLibraryAttributesHandler;
3431 		if (registeredfunc != NULL) {
3432 		ret = (registeredfunc)(attributes);
3433 		} else {
3434 		/* Version 1 libary? */
3435 		HBAGetVersionFunc	GetVersionFunc;
3436 		GetVersionFunc = adapt_infop->library->
3437 		    ftable.functionTable.GetVersionHandler;
3438 		if (GetVersionFunc != NULL) {
3439 			ret = ((GetVersionFunc)());
3440 		}
3441 #ifdef NOTDEF
3442 		else {
3443 		    /* This should not happen, dont think its going to */
3444 		}
3445 #endif
3446 		}
3447 		if (attributes->LibPath[0] == '\0') {
3448 		if (strlen(adapt_infop->library->LibraryPath) < 256) {
3449 			(void) strcpy(attributes->LibPath,
3450 			    adapt_infop->library->LibraryPath);
3451 		}
3452 		}
3453 		break;
3454 	}
3455 	}
3456 	RELEASE_MUTEX(&_hbaapi_AL_mutex);
3457 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, ret);
3458 }
3459 
3460 
3461 /*
3462  * This function returns SM-HBA version that the warpper library implemented.
3463  */
3464 HBA_UINT32
3465 SMHBA_GetVersion() {
3466     DEBUG(2, "SMHBA_GetVersion", 0, 0, 0);
3467     return (SMHBA_LIBVERSION);
3468 }
3469 
3470 /*
3471  * This function returns the attributes for the warpper library.
3472  */
3473 HBA_UINT32
3474 SMHBA_GetWrapperLibraryAttributes(
3475     SMHBA_LIBRARYATTRIBUTES *attributes)
3476 {
3477 
3478 	struct timeval tv;
3479 	struct tm tp;
3480 
3481 	DEBUG(2, "SMHBA_GetWrapperLibraryAttributes", 0, 0, 0);
3482 
3483 	if (attributes == NULL) {
3484 		DEBUG(1, "SMHBA_GetWrapperLibraryAttributes: "
3485 		    "NULL pointer attributes",
3486 		    0, 0, 0);
3487 		return (HBA_STATUS_ERROR_ARG);
3488 	}
3489 
3490 	(void) memset(attributes, 0, sizeof (SMHBA_LIBRARYATTRIBUTES));
3491 
3492 #if defined(SOLARIS)
3493 	if ((handle = dlopen("libSMHBAAPI.so", RTLD_NOW)) != NULL) {
3494 	if (dlinfo(handle, RTLD_DI_LINKMAP, &map) >= 0) {
3495 		for (mp = map; mp != NULL; mp = mp->l_next) {
3496 		if (strlen(map->l_name) < 256) {
3497 			(void) strcpy(attributes->LibPath, map->l_name);
3498 		}
3499 		}
3500 	}
3501 	}
3502 
3503 #endif
3504 
3505 #if defined(VENDOR)
3506 	(void) strcpy(attributes->VName, VENDOR);
3507 #else
3508 	attributes->VName[0] = '\0';
3509 #endif
3510 #if	defined(VERSION)
3511 	(void) strcpy(attributes->VVersion, VERSION);
3512 #else
3513 	attributes->VVersion[0] = '\0';
3514 #endif
3515 
3516 	if (gettimeofday(&tv, (void *)0) == 0) {
3517 	if (localtime_r(&tv.tv_sec, &tp) != NULL) {
3518 		attributes->build_date.tm_mday = tp.tm_mday;
3519 		attributes->build_date.tm_mon = tp.tm_mon;
3520 		attributes->build_date.tm_year = tp.tm_year;
3521 	} else {
3522 		(void) memset(&attributes->build_date, 0,
3523 		    sizeof (attributes->build_date));
3524 	}
3525 	(void) memset(&attributes->build_date, 0,
3526 	    sizeof (attributes->build_date));
3527 	}
3528 
3529 	return (1);
3530 }
3531 
3532 /*
3533  * This function returns the attributes for the warpper library.
3534  */
3535 HBA_UINT32
3536 SMHBA_GetVendorLibraryAttributes(
3537     HBA_UINT32 adapter_index,
3538     SMHBA_LIBRARYATTRIBUTES *attributes)
3539 {
3540 	HBA_ADAPTER_INFO	*adapt_infop;
3541 	SMHBAGetVendorLibraryAttributesFunc
3542 	    registeredfunc;
3543 	HBA_UINT32		ret = 0;
3544 
3545 	DEBUG(2, "SMHBA_GetVendorLibraryAttributes adapterindex:%d",
3546 	    adapter_index, 0, 0);
3547 	if (_hbaapi_librarylist == NULL) {
3548 	DEBUG(1, "SMHBAAPI not loaded yet.", 0, 0, 0);
3549 	return (0);
3550 	}
3551 
3552 	if (attributes == NULL) {
3553 		DEBUG(1, "SMHBA_GetVendorLibraryAttributes: "
3554 		    "NULL pointer attributes",
3555 		    0, 0, 0);
3556 		return (HBA_STATUS_ERROR_ARG);
3557 	}
3558 
3559 	(void) memset(attributes, 0, sizeof (SMHBA_LIBRARYATTRIBUTES));
3560 
3561 	GRAB_MUTEX(&_hbaapi_LL_mutex);
3562 	GRAB_MUTEX(&_hbaapi_AL_mutex);
3563 	for (adapt_infop = _hbaapi_adapterlist;
3564 	    adapt_infop != NULL;
3565 	    adapt_infop = adapt_infop->next) {
3566 
3567 	if (adapt_infop->index == adapter_index) {
3568 
3569 		if (adapt_infop->library->version != SMHBA) {
3570 		RELEASE_MUTEX(&_hbaapi_AL_mutex);
3571 		RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex,
3572 		    HBA_STATUS_ERROR_INCOMPATIBLE);
3573 		}
3574 
3575 		registeredfunc = adapt_infop->library->
3576 		    ftable.smhbafunctionTable.GetVendorLibraryAttributesHandler;
3577 		if (registeredfunc != NULL) {
3578 		ret = (registeredfunc)(attributes);
3579 #ifdef NOTDEF
3580 		} else {
3581 		/* This should not happen since the VSL is already loaded. */
3582 #endif
3583 		}
3584 		if (attributes->LibPath[0] == '\0') {
3585 		if (strlen(adapt_infop->library->LibraryPath) < 256) {
3586 			(void) strcpy(attributes->LibPath,
3587 			    adapt_infop->library->LibraryPath);
3588 		}
3589 		}
3590 		break;
3591 	}
3592 	}
3593 	RELEASE_MUTEX(&_hbaapi_AL_mutex);
3594 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, ret);
3595 }
3596 
3597 HBA_STATUS
3598 SMHBA_GetAdapterAttributes(
3599     HBA_HANDLE		handle,
3600     SMHBA_ADAPTERATTRIBUTES *hbaattributes)
3601 {
3602 	HBA_STATUS		status;
3603 	HBA_LIBRARY_INFO	*lib_infop;
3604 	HBA_HANDLE		vendorHandle;
3605 	SMHBAGetAdapterAttributesFunc GetAdapterAttributesFunc;
3606 
3607 	DEBUG(2, "SMHBA_GetAdapterAttributes", 0, 0, 0);
3608 
3609 	CHECKLIBRARYANDVERSION(SMHBA);
3610 
3611 	GetAdapterAttributesFunc =
3612 	    lib_infop->ftable.smhbafunctionTable.GetAdapterAttributesHandler;
3613 	if (GetAdapterAttributesFunc != NULL) {
3614 	status = ((GetAdapterAttributesFunc)(vendorHandle, hbaattributes));
3615 	} else {
3616 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3617 	}
3618 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3619 }
3620 
3621 HBA_STATUS
3622 SMHBA_GetNumberOfPorts(
3623     HBA_HANDLE		handle,
3624     HBA_UINT32		*numberofports)
3625 {
3626 	HBA_STATUS		status;
3627 	HBA_LIBRARY_INFO	*lib_infop;
3628 	HBA_HANDLE		vendorHandle;
3629 	SMHBAGetNumberOfPortsFunc GetNumberOfPortsFunc;
3630 
3631 	DEBUG(2, "SMHBA_GetAdapterAttributes", 0, 0, 0);
3632 
3633 	CHECKLIBRARYANDVERSION(SMHBA);
3634 
3635 	GetNumberOfPortsFunc =
3636 	    lib_infop->ftable.smhbafunctionTable.GetNumberOfPortsHandler;
3637 	if (GetNumberOfPortsFunc != NULL) {
3638 	status = ((GetNumberOfPortsFunc)(vendorHandle, numberofports));
3639 	} else {
3640 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3641 	}
3642 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3643 }
3644 
3645 HBA_STATUS
3646 SMHBA_GetPortType(
3647     HBA_HANDLE		handle,
3648     HBA_UINT32		portindex,
3649     HBA_PORTTYPE	*porttype)
3650 {
3651 	HBA_STATUS		status;
3652 	HBA_LIBRARY_INFO	*lib_infop;
3653 	HBA_HANDLE		vendorHandle;
3654 	SMHBAGetPortTypeFunc GetPortTypeFunc;
3655 
3656 	DEBUG(2, "SMHBA_GetAdapterAttributes", 0, 0, 0);
3657 
3658 	CHECKLIBRARYANDVERSION(SMHBA);
3659 
3660 	GetPortTypeFunc =
3661 	    lib_infop->ftable.smhbafunctionTable.GetPortTypeHandler;
3662 	if (GetPortTypeFunc != NULL) {
3663 	status = ((GetPortTypeFunc)(vendorHandle, portindex, porttype));
3664 	} else {
3665 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3666 	}
3667 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3668 }
3669 
3670 HBA_STATUS
3671 SMHBA_GetAdapterPortAttributes(
3672     HBA_HANDLE		handle,
3673     HBA_UINT32		portindex,
3674     SMHBA_PORTATTRIBUTES	*portattributes)
3675 {
3676 	HBA_STATUS		status;
3677 	HBA_LIBRARY_INFO	*lib_infop;
3678 	HBA_HANDLE		vendorHandle;
3679 	SMHBAGetAdapterPortAttributesFunc
3680 	    GetAdapterPortAttributesFunc;
3681 
3682 	DEBUG(2, "SMHBA_GetAdapterPortAttributes", 0, 0, 0);
3683 
3684 	CHECKLIBRARYANDVERSION(SMHBA);
3685 
3686 	GetAdapterPortAttributesFunc =
3687 	    lib_infop->ftable.smhbafunctionTable.\
3688 	    GetAdapterPortAttributesHandler;
3689 	if (GetAdapterPortAttributesFunc != NULL) {
3690 	status = ((GetAdapterPortAttributesFunc)
3691 	    (vendorHandle, portindex, portattributes));
3692 	} else {
3693 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3694 	}
3695 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3696 }
3697 
3698 HBA_STATUS
3699 SMHBA_GetDiscoveredPortAttributes(
3700     HBA_HANDLE		handle,
3701     HBA_UINT32		portindex,
3702     HBA_UINT32		discoveredportindex,
3703     SMHBA_PORTATTRIBUTES	*portattributes)
3704 {
3705 	HBA_STATUS		status;
3706 	HBA_LIBRARY_INFO	*lib_infop;
3707 	HBA_HANDLE		vendorHandle;
3708 	SMHBAGetDiscoveredPortAttributesFunc
3709 	    GetDiscoveredPortAttributesFunc;
3710 
3711 	DEBUG(2, "SMHBA_GetDiscoveredPortAttributes", 0, 0, 0);
3712 
3713 	CHECKLIBRARYANDVERSION(SMHBA);
3714 
3715 	GetDiscoveredPortAttributesFunc =
3716 	    lib_infop->ftable.smhbafunctionTable.\
3717 	    GetDiscoveredPortAttributesHandler;
3718 	if (GetDiscoveredPortAttributesFunc != NULL)  {
3719 	status = ((GetDiscoveredPortAttributesFunc)
3720 	    (vendorHandle, portindex, discoveredportindex,
3721 	    portattributes));
3722 	} else {
3723 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3724 	}
3725 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3726 }
3727 
3728 HBA_STATUS
3729 SMHBA_GetPortAttributesByWWN(
3730     HBA_HANDLE		handle,
3731     HBA_WWN		portWWN,
3732     HBA_WWN		domainPortWWN,
3733     SMHBA_PORTATTRIBUTES	*portattributes)
3734 {
3735 	HBA_STATUS		status;
3736 	HBA_LIBRARY_INFO	*lib_infop;
3737 	HBA_HANDLE		vendorHandle;
3738 	SMHBAGetPortAttributesByWWNFunc
3739 	    GetPortAttributesByWWNFunc;
3740 
3741 	DEBUG(2, "SMHBA_GetPortAttributesByWWN: %s", WWN2STR1(&portWWN), 0, 0);
3742 
3743 	CHECKLIBRARYANDVERSION(SMHBA);
3744 
3745 	GetPortAttributesByWWNFunc =
3746 	    lib_infop->ftable.smhbafunctionTable.GetPortAttributesByWWNHandler;
3747 	if (GetPortAttributesByWWNFunc != NULL) {
3748 	status = ((GetPortAttributesByWWNFunc)
3749 	    (vendorHandle, portWWN, domainPortWWN, portattributes));
3750 	} else {
3751 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3752 	}
3753 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3754 }
3755 
3756 HBA_STATUS
3757 SMHBA_GetFCPhyAttributes(
3758     HBA_HANDLE		handle,
3759     HBA_UINT32		portindex,
3760     HBA_UINT32		phyindex,
3761     SMHBA_FC_PHY	*phytype)
3762 {
3763 	HBA_STATUS		status;
3764 	HBA_LIBRARY_INFO	*lib_infop;
3765 	HBA_HANDLE		vendorHandle;
3766 	SMHBAGetFCPhyAttributesFunc GetFCPhyAttributesFunc;
3767 
3768 	DEBUG(2, "SMHBA_GetFCPhyAttributesByWWN", 0, 0, 0);
3769 
3770 	CHECKLIBRARYANDVERSION(SMHBA);
3771 
3772 	GetFCPhyAttributesFunc =
3773 	    lib_infop->ftable.smhbafunctionTable.GetFCPhyAttributesHandler;
3774 	if (GetFCPhyAttributesFunc != NULL) {
3775 	status = ((GetFCPhyAttributesFunc)
3776 	    (vendorHandle, portindex, phyindex, phytype));
3777 	} else {
3778 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3779 	}
3780 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3781 }
3782 
3783 HBA_STATUS
3784 SMHBA_GetSASPhyAttributes(
3785     HBA_HANDLE		handle,
3786     HBA_UINT32		portindex,
3787     HBA_UINT32		phyindex,
3788     SMHBA_SAS_PHY	*phytype)
3789 {
3790 	HBA_STATUS		status;
3791 	HBA_LIBRARY_INFO	*lib_infop;
3792 	HBA_HANDLE		vendorHandle;
3793 	SMHBAGetSASPhyAttributesFunc GetSASPhyAttributesFunc;
3794 
3795 	DEBUG(2, "SMHBA_GetFCPhyAttributesByWWN", 0, 0, 0);
3796 
3797 	CHECKLIBRARYANDVERSION(SMHBA);
3798 
3799 	GetSASPhyAttributesFunc =
3800 	    lib_infop->ftable.smhbafunctionTable.GetSASPhyAttributesHandler;
3801 	if (GetSASPhyAttributesFunc != NULL) {
3802 	status = ((GetSASPhyAttributesFunc)
3803 	    (vendorHandle, portindex, phyindex, phytype));
3804 	} else {
3805 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3806 	}
3807 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3808 }
3809 
3810 HBA_STATUS
3811 SMHBA_GetProtocolStatistics(
3812     HBA_HANDLE		handle,
3813     HBA_UINT32		portindex,
3814     HBA_UINT32		protocoltype,
3815     SMHBA_PROTOCOLSTATISTICS *pProtocolStatistics)
3816 {
3817 	HBA_STATUS		status;
3818 	HBA_LIBRARY_INFO	*lib_infop;
3819 	HBA_HANDLE		vendorHandle;
3820 	SMHBAGetProtocolStatisticsFunc
3821 	    GetProtocolStatisticsFunc;
3822 
3823 	DEBUG(2, "SMHBA_GetProtocolStatistics port index: %d protocol type: %d",
3824 	    portindex, protocoltype, 0);
3825 
3826 	CHECKLIBRARYANDVERSION(SMHBA);
3827 
3828 	GetProtocolStatisticsFunc =
3829 	    lib_infop->ftable.smhbafunctionTable.GetProtocolStatisticsHandler;
3830 	if (GetProtocolStatisticsFunc != NULL) {
3831 	status = (GetProtocolStatisticsFunc)
3832 	    (vendorHandle, portindex, protocoltype, pProtocolStatistics);
3833 	} else {
3834 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3835 	}
3836 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3837 }
3838 
3839 HBA_STATUS
3840 SMHBA_GetPhyStatistics(
3841     HBA_HANDLE		handle,
3842     HBA_UINT32		portindex,
3843     HBA_UINT32		phyindex,
3844     SMHBA_PHYSTATISTICS *pPhyStatistics)
3845 {
3846 	HBA_STATUS		status;
3847 	HBA_LIBRARY_INFO	*lib_infop;
3848 	HBA_HANDLE		vendorHandle;
3849 	SMHBAGetPhyStatisticsFunc
3850 	    GetPhyStatisticsFunc;
3851 
3852 	DEBUG(2, "SMHBA_GetPhyStatistics port index: %d phy idex: %d",
3853 	    portindex, phyindex, 0);
3854 
3855 	CHECKLIBRARYANDVERSION(SMHBA);
3856 
3857 	GetPhyStatisticsFunc =
3858 	    lib_infop->ftable.smhbafunctionTable.GetPhyStatisticsHandler;
3859 	if (GetPhyStatisticsFunc != NULL) {
3860 	status = (GetPhyStatisticsFunc)
3861 	    (vendorHandle, portindex, phyindex, pPhyStatistics);
3862 	} else {
3863 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3864 	}
3865 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3866 }
3867 
3868 HBA_STATUS
3869 SMHBA_GetBindingCapability(
3870     HBA_HANDLE		handle,
3871     HBA_WWN		hbaPortWWN,
3872     HBA_WWN		domainPortWWN,
3873     SMHBA_BIND_CAPABILITY *pFlags)
3874 {
3875 	HBA_STATUS		status;
3876 	HBA_LIBRARY_INFO	*lib_infop;
3877 	HBA_HANDLE		vendorHandle;
3878 	SMHBAGetBindingCapabilityFunc GetBindingCapabilityFunc;
3879 
3880 	DEBUG(2, "HBA_GetBindingCapability", 0, 0, 0);
3881 
3882 	CHECKLIBRARYANDVERSION(SMHBA);
3883 
3884 	GetBindingCapabilityFunc =
3885 	    lib_infop->ftable.smhbafunctionTable.GetBindingCapabilityHandler;
3886 	if (GetBindingCapabilityFunc != NULL) {
3887 	status = (GetBindingCapabilityFunc)(vendorHandle, hbaPortWWN,
3888 	    domainPortWWN, pFlags);
3889 	} else {
3890 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3891 	}
3892 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3893 }
3894 
3895 HBA_STATUS
3896 SMHBA_GetBindingSupport(
3897     HBA_HANDLE		handle,
3898     HBA_WWN		hbaPortWWN,
3899     HBA_WWN		domainPortWWN,
3900     SMHBA_BIND_CAPABILITY *pFlags)
3901 {
3902 	HBA_STATUS		status;
3903 	HBA_LIBRARY_INFO	*lib_infop;
3904 	HBA_HANDLE		vendorHandle;
3905 	SMHBAGetBindingSupportFunc
3906 	    GetBindingSupporFunc;
3907 
3908 	DEBUG(2, "SMHBA_GetBindingSupport port: %s",
3909 	    WWN2STR1(&hbaPortWWN), 0, 0);
3910 
3911 	CHECKLIBRARYANDVERSION(SMHBA);
3912 
3913 	GetBindingSupporFunc =
3914 	    lib_infop->ftable.smhbafunctionTable.GetBindingSupportHandler;
3915 	if (GetBindingSupporFunc != NULL) {
3916 	status = (GetBindingSupporFunc)(vendorHandle,
3917 	    hbaPortWWN, domainPortWWN, pFlags);
3918 	} else {
3919 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3920 	}
3921 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3922 }
3923 
3924 HBA_STATUS
3925 SMHBA_SetBindingSupport(
3926     HBA_HANDLE		handle,
3927     HBA_WWN		hbaPortWWN,
3928     HBA_WWN		domainPortWWN,
3929     SMHBA_BIND_CAPABILITY flags)
3930 {
3931 	HBA_STATUS		status;
3932 	HBA_LIBRARY_INFO	*lib_infop;
3933 	HBA_HANDLE		vendorHandle;
3934 	SMHBASetBindingSupportFunc
3935 	    SetBindingSupporFunc;
3936 
3937 	DEBUG(2, "SMHBA_GetBindingSupport port: %s",
3938 	    WWN2STR1(&hbaPortWWN), 0, 0);
3939 
3940 	CHECKLIBRARYANDVERSION(HBAAPIV2);
3941 
3942 	SetBindingSupporFunc =
3943 	    lib_infop->ftable.smhbafunctionTable.SetBindingSupportHandler;
3944 	if (SetBindingSupporFunc != NULL) {
3945 	status = (SetBindingSupporFunc)
3946 	    (vendorHandle, hbaPortWWN, domainPortWWN, flags);
3947 	} else {
3948 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3949 	}
3950 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3951 }
3952 
3953 HBA_STATUS
3954 SMHBA_GetTargetMapping(
3955     HBA_HANDLE		handle,
3956     HBA_WWN		hbaPortWWN,
3957     HBA_WWN		domainPortWWN,
3958     SMHBA_TARGETMAPPING *pMapping)
3959 {
3960 	HBA_STATUS		status;
3961 	HBA_LIBRARY_INFO	*lib_infop;
3962 	HBA_HANDLE		vendorHandle;
3963 	SMHBAGetTargetMappingFunc GetTargetMappingFunc;
3964 
3965 	DEBUG(2, "SMHBA_GetTargetMapping port WWN: %s",
3966 	    WWN2STR1(&hbaPortWWN), 0, 0);
3967 
3968 	CHECKLIBRARYANDVERSION(SMHBA);
3969 
3970 	GetTargetMappingFunc =
3971 	    lib_infop->ftable.smhbafunctionTable.GetTargetMappingHandler;
3972 	if (GetTargetMappingFunc != NULL) {
3973 	status = ((GetTargetMappingFunc)(vendorHandle,
3974 	    hbaPortWWN, domainPortWWN, pMapping));
3975 	} else {
3976 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
3977 	}
3978 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
3979 }
3980 
3981 HBA_STATUS
3982 SMHBA_GetPersistentBinding(
3983     HBA_HANDLE handle,
3984     HBA_WWN	hbaPortWWN,
3985     HBA_WWN	domainPortWWN,
3986     SMHBA_BINDING *binding)
3987 {
3988 	HBA_STATUS		status;
3989 	HBA_LIBRARY_INFO	*lib_infop;
3990 	HBA_HANDLE		vendorHandle;
3991 	SMHBAGetPersistentBindingFunc
3992 	    GetPersistentBindingFunc;
3993 
3994 	DEBUG(2, "SMHBA_GetPersistentBinding port WWN: %s",
3995 	    WWN2STR1(&hbaPortWWN), 0, 0);
3996 
3997 	CHECKLIBRARYANDVERSION(SMHBA);
3998 
3999 	GetPersistentBindingFunc =
4000 	    lib_infop->ftable.smhbafunctionTable.GetPersistentBindingHandler;
4001 	if (GetPersistentBindingFunc != NULL) {
4002 	status = ((GetPersistentBindingFunc)(vendorHandle,
4003 	    hbaPortWWN, domainPortWWN, binding));
4004 	} else {
4005 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4006 	}
4007 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4008 }
4009 
4010 HBA_STATUS
4011 SMHBA_SetPersistentBinding(
4012     HBA_HANDLE handle,
4013     HBA_WWN	hbaPortWWN,
4014     HBA_WWN	domainPortWWN,
4015     const SMHBA_BINDING *binding)
4016 {
4017 	HBA_STATUS		status;
4018 	HBA_LIBRARY_INFO	*lib_infop;
4019 	HBA_HANDLE		vendorHandle;
4020 	SMHBASetPersistentBindingFunc
4021 	    SetPersistentBindingFunc;
4022 
4023 	DEBUG(2, "SMHBA_SetPersistentBinding port WWN: %s",
4024 	    WWN2STR1(&hbaPortWWN), 0, 0);
4025 
4026 	CHECKLIBRARYANDVERSION(SMHBA);
4027 
4028 	SetPersistentBindingFunc =
4029 	    lib_infop->ftable.smhbafunctionTable.SetPersistentBindingHandler;
4030 	if (SetPersistentBindingFunc != NULL) {
4031 	status = ((SetPersistentBindingFunc)(vendorHandle,
4032 	    hbaPortWWN, domainPortWWN, binding));
4033 	} else {
4034 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4035 	}
4036 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4037 }
4038 
4039 HBA_STATUS
4040 SMHBA_RemovePersistentBinding(
4041     HBA_HANDLE handle,
4042     HBA_WWN	hbaPortWWN,
4043     HBA_WWN	domainPortWWN,
4044     const SMHBA_BINDING *binding)
4045 {
4046 	HBA_STATUS		status;
4047 	HBA_LIBRARY_INFO	*lib_infop;
4048 	HBA_HANDLE		vendorHandle;
4049 	SMHBARemovePersistentBindingFunc
4050 	    RemovePersistentBindingFunc;
4051 
4052 	DEBUG(2, "SMHBA_RemovePersistentBinding port WWN: %s",
4053 	    WWN2STR1(&hbaPortWWN), 0, 0);
4054 
4055 	CHECKLIBRARYANDVERSION(SMHBA);
4056 
4057 	RemovePersistentBindingFunc =
4058 	    lib_infop->ftable.smhbafunctionTable.RemovePersistentBindingHandler;
4059 	if (RemovePersistentBindingFunc != NULL) {
4060 	status = ((RemovePersistentBindingFunc)(vendorHandle,
4061 	    hbaPortWWN, domainPortWWN, binding));
4062 	} else {
4063 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4064 	}
4065 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4066 }
4067 
4068 HBA_STATUS
4069 SMHBA_RemoveAllPersistentBindings(
4070     HBA_HANDLE handle,
4071     HBA_WWN	hbaPortWWN,
4072     HBA_WWN	domainPortWWN)
4073 {
4074 	HBA_STATUS		status;
4075 	HBA_LIBRARY_INFO	*lib_infop;
4076 	HBA_HANDLE		vendorHandle;
4077 	SMHBARemoveAllPersistentBindingsFunc
4078 	    RemoveAllPersistentBindingsFunc;
4079 
4080 	DEBUG(2, "SMHBA_RemoveAllPersistentBinding port WWN: %s",
4081 	    WWN2STR1(&hbaPortWWN), 0, 0);
4082 
4083 	CHECKLIBRARYANDVERSION(SMHBA);
4084 
4085 	RemoveAllPersistentBindingsFunc =
4086 	    lib_infop->ftable.smhbafunctionTable.\
4087 	    RemoveAllPersistentBindingsHandler;
4088 	if (RemoveAllPersistentBindingsFunc != NULL) {
4089 	status = ((RemoveAllPersistentBindingsFunc)(vendorHandle,
4090 	    hbaPortWWN, domainPortWWN));
4091 	} else {
4092 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4093 	}
4094 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4095 }
4096 
4097 HBA_STATUS
4098 SMHBA_GetLUNStatistics(
4099     HBA_HANDLE handle,
4100     const HBA_SCSIID *lunit,
4101     SMHBA_PROTOCOLSTATISTICS *statistics)
4102 {
4103 	HBA_STATUS		status;
4104 	HBA_LIBRARY_INFO	*lib_infop;
4105 	HBA_HANDLE		vendorHandle;
4106 	SMHBAGetLUNStatisticsFunc GetLUNStatisticsFunc;
4107 
4108 	DEBUG(2, "SMHBA_GetLUNStatistics", 0, 0, 0);
4109 
4110 	CHECKLIBRARYANDVERSION(SMHBA);
4111 
4112 	GetLUNStatisticsFunc =
4113 	    lib_infop->ftable.smhbafunctionTable.GetLUNStatisticsHandler;
4114 	if (GetLUNStatisticsFunc != NULL) {
4115 	status = ((GetLUNStatisticsFunc)(vendorHandle, lunit, statistics));
4116 	} else {
4117 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4118 	}
4119 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4120 }
4121 
4122 HBA_STATUS
4123 SMHBA_ScsiInquiry(
4124     HBA_HANDLE	handle,
4125     HBA_WWN	hbaPortWWN,
4126     HBA_WWN	discoveredPortWWN,
4127     HBA_WWN	domainPortWWN,
4128     SMHBA_SCSILUN	smhbaLUN,
4129     HBA_UINT8	CDB_Byte1,
4130     HBA_UINT8	CDB_Byte2,
4131     void	*pRspBuffer,
4132     HBA_UINT32	*pRspBufferSize,
4133     HBA_UINT8	*pScsiStatus,
4134     void	*pSenseBuffer,
4135     HBA_UINT32	*pSenseBufferSize)
4136 {
4137 	HBA_STATUS		status;
4138 	HBA_LIBRARY_INFO	*lib_infop;
4139 	HBA_HANDLE		vendorHandle;
4140 	SMHBAScsiInquiryFunc ScsiInquiryFunc;
4141 
4142 	DEBUG(2, "SMHBA_ScsiInquiry to hba port: %s discoveredPortWWN: %s",
4143 	    WWN2STR1(&hbaPortWWN), WWN2STR1(&discoveredPortWWN), 0);
4144 
4145 	CHECKLIBRARYANDVERSION(SMHBA);
4146 
4147 	ScsiInquiryFunc =
4148 	    lib_infop->ftable.smhbafunctionTable.ScsiInquiryHandler;
4149 	if (ScsiInquiryFunc != NULL) {
4150 	status = ((ScsiInquiryFunc)(
4151 	    vendorHandle, hbaPortWWN, discoveredPortWWN, domainPortWWN,
4152 	    smhbaLUN, CDB_Byte1, CDB_Byte2, pRspBuffer, pRspBufferSize,
4153 	    pScsiStatus, pSenseBuffer, pSenseBufferSize));
4154 	} else {
4155 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4156 	}
4157 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4158 }
4159 
4160 HBA_STATUS
4161 SMHBA_ScsiReportLUNs(
4162     HBA_HANDLE	handle,
4163     HBA_WWN	hbaPortWWN,
4164     HBA_WWN	discoveredPortWWN,
4165     HBA_WWN	domainPortWWN,
4166     void	*pRspBuffer,
4167     HBA_UINT32	*pRspBufferSize,
4168     HBA_UINT8	*pScsiStatus,
4169     void	*pSenseBuffer,
4170     HBA_UINT32	*pSenseBufferSize)
4171 {
4172 	HBA_STATUS		status;
4173 	HBA_LIBRARY_INFO	*lib_infop;
4174 	HBA_HANDLE		vendorHandle;
4175 	SMHBAScsiReportLUNsFunc ScsiReportLUNsFunc;
4176 
4177 	DEBUG(2, "SMHBA_ScsiReportLuns to hba port: %s discoveredPortWWN: %s",
4178 	    WWN2STR1(&hbaPortWWN), WWN2STR1(&discoveredPortWWN), 0);
4179 
4180 	CHECKLIBRARYANDVERSION(SMHBA);
4181 
4182 	ScsiReportLUNsFunc =
4183 	    lib_infop->ftable.smhbafunctionTable.ScsiReportLUNsHandler;
4184 	if (ScsiReportLUNsFunc != NULL) {
4185 	status = ((ScsiReportLUNsFunc)(
4186 	    vendorHandle, hbaPortWWN, discoveredPortWWN, domainPortWWN,
4187 	    pRspBuffer, pRspBufferSize, pScsiStatus, pSenseBuffer,
4188 	    pSenseBufferSize));
4189 	} else {
4190 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4191 	}
4192 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4193 }
4194 
4195 HBA_STATUS
4196 SMHBA_ScsiReadCapacity(
4197     HBA_HANDLE	handle,
4198     HBA_WWN	hbaPortWWN,
4199     HBA_WWN	discoveredPortWWN,
4200     HBA_WWN	domainPortWWN,
4201     SMHBA_SCSILUN	smhbaLUN,
4202     void	*pRspBuffer,
4203     HBA_UINT32	*pRspBufferSize,
4204     HBA_UINT8	*pScsiStatus,
4205     void	*pSenseBuffer,
4206     HBA_UINT32	*pSenseBufferSize)
4207 {
4208 	HBA_STATUS		status;
4209 	HBA_LIBRARY_INFO	*lib_infop;
4210 	HBA_HANDLE		vendorHandle;
4211 	SMHBAScsiReadCapacityFunc ScsiReadCapacityFunc;
4212 
4213 	DEBUG(2, "SMHBA_ScsiReadCapacity to hba port: %s discoveredPortWWN: %s",
4214 	    WWN2STR1(&hbaPortWWN), WWN2STR1(&discoveredPortWWN), 0);
4215 
4216 	CHECKLIBRARYANDVERSION(SMHBA);
4217 
4218 	ScsiReadCapacityFunc =
4219 	    lib_infop->ftable.smhbafunctionTable.ScsiReadCapacityHandler;
4220 	if (ScsiReadCapacityFunc != NULL) {
4221 	status = ((ScsiReadCapacityFunc)(
4222 	    vendorHandle, hbaPortWWN, discoveredPortWWN, domainPortWWN,
4223 	    smhbaLUN, pRspBuffer, pRspBufferSize, pScsiStatus, pSenseBuffer,
4224 	    pSenseBufferSize));
4225 	} else {
4226 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4227 	}
4228 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4229 }
4230 
4231 HBA_STATUS
4232 SMHBA_SendTEST(
4233     HBA_HANDLE		handle,
4234     HBA_WWN		hbaPortWWN,
4235     HBA_WWN		destWWN,
4236     HBA_UINT32		destFCID,
4237     void		*pRspBuffer,
4238     HBA_UINT32		pRspBufferSize)
4239 {
4240 	HBA_STATUS		status;
4241 	HBA_LIBRARY_INFO	*lib_infop;
4242 	HBA_HANDLE		vendorHandle;
4243 	SMHBASendTESTFunc	SendTESTFunc;
4244 
4245 	DEBUG(2, "SMHBA_SendTEST, hbaPortWWN: %s destWWN",
4246 	    WWN2STR1(&hbaPortWWN),
4247 	    WWN2STR1(&destWWN), 0);
4248 
4249 	CHECKLIBRARYANDVERSION(SMHBA);
4250 
4251 	SendTESTFunc = lib_infop->ftable.smhbafunctionTable.SendTESTHandler;
4252 	if (SendTESTFunc != NULL) {
4253 	status = (SendTESTFunc)
4254 	    (vendorHandle, hbaPortWWN, destWWN, destFCID,
4255 	    pRspBuffer, pRspBufferSize);
4256 	} else {
4257 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4258 	}
4259 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4260 }
4261 
4262 HBA_STATUS
4263 SMHBA_SendECHO(
4264     HBA_HANDLE		handle,
4265     HBA_WWN		hbaPortWWN,
4266     HBA_WWN		destWWN,
4267     HBA_UINT32		destFCID,
4268     void		*pReqBuffer,
4269     HBA_UINT32		ReqBufferSize,
4270     void		*pRspBuffer,
4271     HBA_UINT32		*pRspBufferSize)
4272 {
4273 	HBA_STATUS		status;
4274 	HBA_LIBRARY_INFO	*lib_infop;
4275 	HBA_HANDLE		vendorHandle;
4276 	SMHBASendECHOFunc	SendECHOFunc;
4277 
4278 	DEBUG(2, "SMHBA_SendECHO, hbaPortWWN: %s destWWN",
4279 	    WWN2STR1(&hbaPortWWN), WWN2STR1(&destWWN), 0);
4280 
4281 	CHECKLIBRARYANDVERSION(SMHBA);
4282 
4283 	SendECHOFunc = lib_infop->ftable.smhbafunctionTable.SendECHOHandler;
4284 	if (SendECHOFunc != NULL) {
4285 	status = (SendECHOFunc)
4286 	    (vendorHandle, hbaPortWWN, destWWN, destFCID,
4287 	    pReqBuffer, ReqBufferSize, pRspBuffer, pRspBufferSize);
4288 	} else {
4289 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4290 	}
4291 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4292 }
4293 
4294 HBA_STATUS
4295 SMHBA_SendSMPPassThru(
4296     HBA_HANDLE		handle,
4297     HBA_WWN		hbaPortWWN,
4298     HBA_WWN		destWWN,
4299     HBA_WWN		domainPortWWN,
4300     void		*pReqBuffer,
4301     HBA_UINT32		ReqBufferSize,
4302     void		*pRspBuffer,
4303     HBA_UINT32		*pRspBufferSize)
4304 {
4305 	HBA_STATUS		status;
4306 	HBA_LIBRARY_INFO	*lib_infop;
4307 	HBA_HANDLE		vendorHandle;
4308 	SMHBASendSMPPassThruFunc	SendSMPPassThruFunc;
4309 
4310 	DEBUG(2, "SMHBA_SendSMPPassThru, hbaPortWWN: %s destWWN: %s",
4311 	    WWN2STR1(&hbaPortWWN), WWN2STR1(&destWWN), 0);
4312 
4313 	CHECKLIBRARYANDVERSION(SMHBA);
4314 
4315 	SendSMPPassThruFunc = lib_infop->ftable.\
4316 	    smhbafunctionTable.SendSMPPassThruHandler;
4317 
4318 	if (SendSMPPassThruFunc != NULL) {
4319 	status = (SendSMPPassThruFunc)
4320 	    (vendorHandle, hbaPortWWN, destWWN, domainPortWWN,
4321 	    pReqBuffer, ReqBufferSize, pRspBuffer, pRspBufferSize);
4322 	} else {
4323 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4324 	}
4325 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4326 }
4327 
4328 /*
4329  * Following the similar logic of HBAAPI addaspterevents_callback.
4330  *
4331  * Unlike other events Adapter Add Event is not limited to a specific
4332  * adapter(i.e. no adapter handle is passed for registration) so
4333  * the event should be passed to all registrants.  The routine below
4334  * is passed to the VSLs as a callback and when Adapter Add event is detected
4335  * by VSL it will call smhba_adapteraddevents_callback() which in turn check
4336  * if the passed userdata ptr matches with the one stored in the callback list
4337  * and calls the stored callback.
4338  *
4339  * For the situation that multiple clients are registered for Adapter Add event
4340  * each registration is passed to VSLs so VSL may call
4341  * smhba_adapteraddevents_callback() multiple times or it may call only once
4342  * since the callback function is same.  For this implemneation, the userdata
4343  * is stored in HBA_ALLADAPTERSCALLBACK_ELEM so it is expected that VSL call
4344  * smhba_adapteraddevents_callback() only once and
4345  * smhba_adapteraddevents_callback() will call the client callback with proper
4346  * userdata.
4347  */
4348 static void
4349 smhba_adapteraddevents_callback(
4350 /* LINTED E_FUNC_ARG_UNUSED */
4351     void *data,
4352     HBA_WWN PortWWN,
4353 /* LINTED E_FUNC_ARG_UNUSED */
4354     HBA_UINT32 eventType)
4355 {
4356 	HBA_ALLADAPTERSCALLBACK_ELEM	*cbp;
4357 
4358 	DEBUG(3, "AddAdapterEvent, port:%s", WWN2STR1(&PortWWN), 0, 0);
4359 
4360 	GRAB_MUTEX(&_smhba_AAE_mutex);
4361 	for (cbp = _smhba_adapteraddevents_callback_list;
4362 	    cbp != NULL;
4363 	    cbp = cbp->next) {
4364 	(*cbp->callback)(cbp->userdata, PortWWN, HBA_EVENT_ADAPTER_ADD);
4365 	}
4366 	RELEASE_MUTEX(&_smhba_AAE_mutex);
4367 
4368 }
4369 
4370 HBA_STATUS
4371 SMHBA_RegisterForAdapterAddEvents(
4372     void		(*pCallback) (
4373 	void		*data,
4374 	HBA_WWN		PortWWN,
4375 	HBA_UINT32	eventType),
4376     void		*pUserData,
4377     HBA_CALLBACKHANDLE  *pCallbackHandle) {
4378 
4379     HBA_ALLADAPTERSCALLBACK_ELEM	*cbp;
4380     HBA_VENDORCALLBACK_ELEM		*vcbp;
4381     HBA_VENDORCALLBACK_ELEM		*vendorhandlelist;
4382     SMHBARegisterForAdapterAddEventsFunc	registeredfunc;
4383     HBA_STATUS				status = HBA_STATUS_OK;
4384     HBA_STATUS				failure = HBA_STATUS_OK;
4385     HBA_LIBRARY_INFO			*lib_infop;
4386     int					registered_cnt = 0;
4387     int					vendor_cnt = 0;
4388     int					not_supported_cnt = 0;
4389     int					status_OK_bar_cnt = 0;
4390     int					status_OK_cnt = 0;
4391 
4392     DEBUG(2, "SMHBA_RegisterForAdapterAddEvents", 0, 0, 0);
4393     ARE_WE_INITED();
4394 
4395     cbp = (HBA_ALLADAPTERSCALLBACK_ELEM *)
4396 	    calloc(1, sizeof (HBA_ALLADAPTERSCALLBACK_ELEM));
4397 	*pCallbackHandle = (HBA_CALLBACKHANDLE) cbp;
4398     if (cbp == NULL) {
4399 	return (HBA_STATUS_ERROR);
4400 	}
4401 
4402     GRAB_MUTEX(&_hbaapi_LL_mutex);
4403     GRAB_MUTEX(&_smhba_AAE_mutex);
4404     cbp->callback = pCallback;
4405     cbp->userdata = pUserData;
4406     cbp->next = _smhba_adapteraddevents_callback_list;
4407     _smhba_adapteraddevents_callback_list = cbp;
4408 
4409 	/*
4410 	 * Need to release the mutex now incase the vendor function invokes the
4411 	 * callback.  We will grap the mutex later to attach the vendor handle
4412 	 * list to the callback structure
4413 	 */
4414 	RELEASE_MUTEX(&_smhba_AAE_mutex);
4415 
4416 
4417 	/*
4418 	 * now create a list of vendors (vendor libraryies, NOT ADAPTERS)
4419 	 * that have successfully registerred
4420 	 */
4421     vendorhandlelist = NULL;
4422     for (lib_infop = _hbaapi_librarylist;
4423 	lib_infop != NULL;
4424 	lib_infop = lib_infop->next) {
4425 
4426 	/* only for HBAAPI V2 */
4427 	if (lib_infop->version != SMHBA) {
4428 	    continue;
4429 	} else {
4430 	    vendor_cnt++;
4431 	}
4432 
4433 	registeredfunc =
4434 	    lib_infop->ftable.smhbafunctionTable.\
4435 	    RegisterForAdapterAddEventsHandler;
4436 	if (registeredfunc == NULL) {
4437 	    continue;
4438 	}
4439 
4440 	vcbp = (HBA_VENDORCALLBACK_ELEM *)
4441 	    calloc(1, sizeof (HBA_VENDORCALLBACK_ELEM));
4442 	if (vcbp == NULL) {
4443 	    freevendorhandlelist(vendorhandlelist);
4444 	    status = HBA_STATUS_ERROR;
4445 	    break;
4446 	}
4447 
4448 	registered_cnt++;
4449 	status = (registeredfunc)(smhba_adapteraddevents_callback,
4450 	    pUserData, &vcbp->vendorcbhandle);
4451 	if (status == HBA_STATUS_ERROR_NOT_SUPPORTED) {
4452 	    not_supported_cnt++;
4453 	    free(vcbp);
4454 	    continue;
4455 	} else if (status != HBA_STATUS_OK) {
4456 	    status_OK_bar_cnt++;
4457 	    DEBUG(1,
4458 		    "SMHBA_RegisterForAdapterAddEvents: Library->%s, Error->%d",
4459 		    lib_infop->LibraryPath, status, 0);
4460 	    failure = status;
4461 	    free(vcbp);
4462 	    continue;
4463 	} else {
4464 	    status_OK_cnt++;
4465 	}
4466 	vcbp->lib_info = lib_infop;
4467 	vcbp->next = vendorhandlelist;
4468 	vendorhandlelist = vcbp;
4469 	}
4470 
4471     if (vendor_cnt == 0) {
4472 	/* no SMHBA VSL found.  Should be okay?? */
4473 	status = HBA_STATUS_ERROR;
4474 	} else if (registered_cnt == 0) {
4475 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4476 	freevendorhandlelist(vendorhandlelist);
4477 	(void) local_remove_callback((HBA_CALLBACKHANDLE) cbp);
4478 	} else if (status_OK_cnt == 0 && not_supported_cnt != 0) {
4479 	status = HBA_STATUS_ERROR_NOT_SUPPORTED;
4480 	} else if (status_OK_cnt == 0) {
4481 	/*
4482 	 * At least one vendor library registered this function, but no
4483 	 * vendor call succeeded
4484 	 */
4485 	(void) local_remove_callback((HBA_CALLBACKHANDLE) cbp);
4486 	status = failure;
4487 	} else {
4488 	/* we have had atleast some success, now finish up */
4489 	GRAB_MUTEX(&_smhba_AAE_mutex);
4490 	/*
4491 	 * this seems silly, but what if another thread called
4492 	 * the callback remove
4493 	 */
4494 	for (cbp = _smhba_adapteraddevents_callback_list;
4495 	    cbp != NULL; cbp = cbp->next) {
4496 	    if ((HBA_CALLBACKHANDLE)cbp == *pCallbackHandle) {
4497 		/* yup, its still there, hooray */
4498 		cbp->vendorhandlelist = vendorhandlelist;
4499 		vendorhandlelist = NULL;
4500 		break;
4501 	    }
4502 	}
4503 	RELEASE_MUTEX(&_smhba_AAE_mutex);
4504 	if (vendorhandlelist != NULL) {
4505 		/*
4506 		 * bummer, somebody removed the callback before we finished
4507 		 * registration, probably will never happen
4508 		 */
4509 	    freevendorhandlelist(vendorhandlelist);
4510 	    DEBUG(1,
4511 		    "HBA_RegisterForAdapterAddEvents: HBA_RemoveCallback was "
4512 		    "called for a handle before registration was finished.",
4513 		    0, 0, 0);
4514 	    status = HBA_STATUS_ERROR;
4515 	} else {
4516 	    status = HBA_STATUS_OK;
4517 	}
4518 	}
4519     RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4520 }
4521 
4522 /* SMHBA Adapter Events (other than add) ******************************** */
4523 static void
4524 smhba_adapterevents_callback(void *data,
4525 			HBA_WWN PortWWN,
4526 			HBA_UINT32 eventType)
4527 {
4528 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4529 
4530 	DEBUG(3, "AdapterEvent, port:%s, eventType:%d", WWN2STR1(&PortWWN),
4531 	    eventType, 0);
4532 
4533 	GRAB_MUTEX(&_hbaapi_AE_mutex);
4534 	for (acbp = _smhba_adapterevents_callback_list;
4535 	    acbp != NULL;
4536 	    acbp = acbp->next) {
4537 	if (data == (void *)acbp) {
4538 		(*acbp->callback)(acbp->userdata, PortWWN, eventType);
4539 		break;
4540 	}
4541 	}
4542 	RELEASE_MUTEX(&_hbaapi_AE_mutex);
4543 }
4544 
4545 HBA_STATUS
4546 SMHBA_RegisterForAdapterEvents(
4547     void		(*pCallback) (
4548 	void		*data,
4549 	HBA_WWN		PortWWN,
4550 	HBA_UINT32	eventType),
4551     void		*pUserData,
4552     HBA_HANDLE		handle,
4553     HBA_CALLBACKHANDLE	*pCallbackHandle) {
4554 
4555 	HBA_ADAPTERCALLBACK_ELEM		*acbp;
4556 	SMHBARegisterForAdapterEventsFunc	registeredfunc;
4557 	HBA_STATUS				status;
4558 	HBA_LIBRARY_INFO			*lib_infop;
4559 	HBA_HANDLE				vendorHandle;
4560 
4561 	DEBUG(2, "SMHBA_RegisterForAdapterEvents", 0, 0, 0);
4562 
4563 	CHECKLIBRARYANDVERSION(SMHBA);
4564 
4565 	/* we now have the _hbaapi_LL_mutex */
4566 
4567 	registeredfunc = lib_infop->ftable.smhbafunctionTable.\
4568 	    RegisterForAdapterEventsHandler;
4569     if (registeredfunc == NULL) {
4570 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
4571 	}
4572 
4573 	/*
4574 	 * that allocated memory is used both as the handle for the
4575 	 * caller, and as userdata to the vendor call so that on
4576 	 * callback the specific registration may be recalled
4577 	 */
4578 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
4579 	    calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
4580 	if (acbp == NULL) {
4581 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
4582 	}
4583 	*pCallbackHandle = (HBA_CALLBACKHANDLE) acbp;
4584 	acbp->callback = pCallback;
4585 	acbp->userdata = pUserData;
4586 	acbp->lib_info = lib_infop;
4587 
4588 	status = (registeredfunc)(smhba_adapterevents_callback,
4589 	    (void *)acbp,
4590 	    vendorHandle,
4591 	    &acbp->vendorcbhandle);
4592     if (status != HBA_STATUS_OK) {
4593 	free(acbp);
4594 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4595 	}
4596 
4597 	GRAB_MUTEX(&_smhba_AE_mutex);
4598 	acbp->next = _smhba_adapterevents_callback_list;
4599 	    _hbaapi_adapterevents_callback_list = acbp;
4600 
4601 	RELEASE_MUTEX(&_smhba_AE_mutex);
4602 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
4603 }
4604 
4605 /* Adapter Port Events *********************************************** */
4606 static void
4607 smhba_adapterportevents_callback(void *data,
4608 			    HBA_WWN PortWWN,
4609 			    HBA_UINT32 eventType,
4610 			    HBA_UINT32 fabricPortID)
4611 {
4612 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4613 
4614 	DEBUG(3,
4615 	    "SMHBA_AdapterPortEvent, port:%s, eventType:%d fabricPortID:0X%06x",
4616 	    WWN2STR1(&PortWWN), eventType, fabricPortID);
4617 
4618 	GRAB_MUTEX(&_smhba_APE_mutex);
4619 
4620 	for (acbp = _smhba_adapterportevents_callback_list;
4621 	    acbp != NULL;
4622 	    acbp = acbp->next) {
4623 	if (data == (void *)acbp) {
4624 		(*acbp->callback)(acbp->userdata, PortWWN,
4625 		    eventType, fabricPortID);
4626 		break;
4627 	}
4628 	}
4629 	RELEASE_MUTEX(&_smhba_APE_mutex);
4630 }
4631 
4632 HBA_STATUS
4633 SMHBA_RegisterForAdapterPortEvents(
4634     void		(*pCallback) (
4635 	void		*pData,
4636 	HBA_WWN		PortWWN,
4637 	HBA_UINT32	eventType,
4638 	HBA_UINT32	fabricPortID),
4639     void		*pUserData,
4640     HBA_HANDLE		handle,
4641     HBA_WWN		portWWN,
4642     HBA_UINT32		specificEventType,
4643     HBA_CALLBACKHANDLE	*pCallbackHandle) {
4644 
4645 	HBA_ADAPTERCALLBACK_ELEM		*acbp;
4646 	SMHBARegisterForAdapterPortEventsFunc	registeredfunc;
4647 	HBA_STATUS				status;
4648 	HBA_LIBRARY_INFO			*lib_infop;
4649 	HBA_HANDLE				vendorHandle;
4650 
4651 	DEBUG(2, "SMHBA_RegisterForAdapterPortEvents for port: %s",
4652 	    WWN2STR1(&portWWN), 0, 0);
4653 
4654 	CHECKLIBRARYANDVERSION(SMHBA);
4655 	/* we now have the _hbaapi_LL_mutex */
4656 
4657 	registeredfunc =
4658 	    lib_infop->ftable.smhbafunctionTable.\
4659 	    RegisterForAdapterPortEventsHandler;
4660 	if (registeredfunc == NULL) {
4661 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
4662 	}
4663 
4664 	/*
4665 	 * that allocated memory is used both as the handle for the
4666 	 * caller, and as userdata to the vendor call so that on
4667 	 * callback the specific registration may be recalled
4668 	 */
4669 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
4670 	    calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
4671 	if (acbp == NULL) {
4672 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
4673 	}
4674 	*pCallbackHandle = (HBA_CALLBACKHANDLE) acbp;
4675 	acbp->callback = pCallback;
4676 	acbp->userdata = pUserData;
4677 	acbp->lib_info = lib_infop;
4678 
4679 	status = (registeredfunc)(smhba_adapterportevents_callback,
4680 	    (void *)acbp,
4681 	    vendorHandle,
4682 	    portWWN,
4683 	    specificEventType,
4684 	    &acbp->vendorcbhandle);
4685 	if (status != HBA_STATUS_OK) {
4686 	free(acbp);
4687 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4688 	}
4689 
4690 	GRAB_MUTEX(&_smhba_APE_mutex);
4691 	acbp->next = _smhba_adapterportevents_callback_list;
4692 	_smhba_adapterportevents_callback_list = acbp;
4693 
4694 	RELEASE_MUTEX(&_smhba_APE_mutex);
4695 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
4696 }
4697 
4698 /* SMHBA Adapter Port Stat Events ******************************** */
4699 static void
4700 smhba_adapterportstatevents_callback(void *data,
4701 				HBA_WWN portWWN,
4702 				HBA_UINT32 protocolType,
4703 				HBA_UINT32 eventType)
4704 {
4705 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4706 
4707 	DEBUG(3,
4708 	    "SMBA_AdapterPortStateEvent, port:%s, eventType:%d",
4709 	    WWN2STR1(&portWWN), eventType, 0);
4710 
4711 	GRAB_MUTEX(&_smhba_APSE_mutex);
4712 	for (acbp = _smhba_adapterportstatevents_callback_list;
4713 	    acbp != NULL;
4714 	    acbp = acbp->next) {
4715 	if (data == (void *)acbp) {
4716 		(*acbp->callback)(acbp->userdata, portWWN,
4717 		    protocolType, eventType);
4718 		return;
4719 	}
4720 	}
4721 	RELEASE_MUTEX(&_smhba_APSE_mutex);
4722 }
4723 
4724 HBA_STATUS
4725 SMHBA_RegisterForAdapterPortStatEvents(
4726     void		(*pCallback) (
4727 	void		*pData,
4728 	HBA_WWN		portWWN,
4729 	HBA_UINT32	protocolType,
4730 	HBA_UINT32	eventType),
4731     void		*pUserData,
4732     HBA_HANDLE		handle,
4733     HBA_WWN		portWWN,
4734     HBA_UINT32		protocolType,
4735     SMHBA_PROTOCOLSTATISTICS	stats,
4736     HBA_UINT32		statType,
4737     HBA_CALLBACKHANDLE	*pCallbackHandle) {
4738 
4739 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4740 	SMHBARegisterForAdapterPortStatEventsFunc
4741 	    registeredfunc;
4742 	HBA_STATUS			status;
4743 	HBA_LIBRARY_INFO		*lib_infop;
4744 	HBA_HANDLE			vendorHandle;
4745 
4746 	DEBUG(2, "SMHBA_RegisterForAdapterPortStatEvents for port: %s",
4747 	    WWN2STR1(&portWWN), 0, 0);
4748 
4749 	CHECKLIBRARYANDVERSION(SMHBA);
4750 	/* we now have the _hbaapi_LL_mutex */
4751 
4752 	registeredfunc =
4753 	    lib_infop->ftable.smhbafunctionTable.\
4754 	    RegisterForAdapterPortStatEventsHandler;
4755 	if (registeredfunc == NULL) {
4756 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
4757 	}
4758 
4759 	/*
4760 	 * that allocated memory is used both as the handle for the
4761 	 * caller, and as userdata to the vendor call so that on
4762 	 * callback the specific registration may be recalled
4763 	 */
4764 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
4765 	    calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
4766 	if (acbp == NULL) {
4767 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
4768 	}
4769 	*pCallbackHandle = (HBA_CALLBACKHANDLE) acbp;
4770 	acbp->callback = pCallback;
4771 	acbp->userdata = pUserData;
4772 	acbp->lib_info = lib_infop;
4773 
4774 	status = (registeredfunc)(smhba_adapterportstatevents_callback,
4775 	    (void *)acbp,
4776 	    vendorHandle,
4777 	    portWWN,
4778 	    protocolType,
4779 	    stats,
4780 	    statType,
4781 	    &acbp->vendorcbhandle);
4782 	if (status != HBA_STATUS_OK) {
4783 	free(acbp);
4784 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4785 	}
4786 
4787 	GRAB_MUTEX(&_smhba_APSE_mutex);
4788 	acbp->next = _smhba_adapterportstatevents_callback_list;
4789 	_smhba_adapterportstatevents_callback_list = acbp;
4790 
4791 	RELEASE_MUTEX(&_smhba_APSE_mutex);
4792 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
4793 }
4794 
4795 /* SMHBA Adapter Port Phy Stat Events ************************************ */
4796 static void
4797 smhba_adapterphystatevents_callback(void *data,
4798 				HBA_WWN portWWN,
4799 				HBA_UINT32 phyIndex,
4800 				HBA_UINT32 eventType)
4801 {
4802 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4803 
4804 	DEBUG(3,
4805 	    "SMBA_AdapterPortStateEvent, port:%s, eventType:%d",
4806 	    WWN2STR1(&portWWN), eventType, 0);
4807 
4808 	GRAB_MUTEX(&_smhba_APHYSE_mutex);
4809 	for (acbp = _smhba_adapterphystatevents_callback_list;
4810 	    acbp != NULL;
4811 	    acbp = acbp->next) {
4812 	if (data == (void *)acbp) {
4813 		(*acbp->callback)(acbp->userdata, portWWN, phyIndex, eventType);
4814 		return;
4815 	}
4816 	}
4817 	RELEASE_MUTEX(&_smhba_APHYSE_mutex);
4818 }
4819 
4820 HBA_STATUS
4821 SMHBA_RegisterForAdapterPhyStatEvents(
4822     void		(*pCallback) (
4823 	void		*pData,
4824 	HBA_WWN		portWWN,
4825 	HBA_UINT32	phyIndex,
4826 	HBA_UINT32	eventType),
4827     void		*pUserData,
4828     HBA_HANDLE		handle,
4829     HBA_WWN		portWWN,
4830     HBA_UINT32		phyIndex,
4831     SMHBA_PHYSTATISTICS	stats,
4832     HBA_UINT32		statType,
4833     HBA_CALLBACKHANDLE	*pCallbackHandle) {
4834 
4835 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4836 	SMHBARegisterForAdapterPhyStatEventsFunc
4837 	    registeredfunc;
4838 	HBA_STATUS			status;
4839 	HBA_LIBRARY_INFO		*lib_infop;
4840 	HBA_HANDLE			vendorHandle;
4841 
4842 	DEBUG(2, "SMHBA_RegisterForAdapterPhyStatEvents for port: %s",
4843 	    WWN2STR1(&portWWN), 0, 0);
4844 
4845 	CHECKLIBRARYANDVERSION(SMHBA);
4846 	/* we now have the _hbaapi_LL_mutex */
4847 
4848 	registeredfunc =
4849 	    lib_infop->ftable.smhbafunctionTable.\
4850 	    RegisterForAdapterPhyStatEventsHandler;
4851 	if (registeredfunc == NULL) {
4852 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
4853 	}
4854 
4855 	/*
4856 	 * that allocated memory is used both as the handle for the
4857 	 * caller, and as userdata to the vendor call so that on
4858 	 * callback the specific registration may be recalled
4859 	 */
4860 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
4861 	    calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
4862 	if (acbp == NULL) {
4863 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
4864 	}
4865 	*pCallbackHandle = (HBA_CALLBACKHANDLE) acbp;
4866 	acbp->callback = pCallback;
4867 	acbp->userdata = pUserData;
4868 	acbp->lib_info = lib_infop;
4869 
4870 	status = (registeredfunc)(smhba_adapterphystatevents_callback,
4871 	    (void *)acbp,
4872 	    vendorHandle,
4873 	    portWWN,
4874 	    phyIndex,
4875 	    stats,
4876 	    statType,
4877 	    &acbp->vendorcbhandle);
4878 	if (status != HBA_STATUS_OK) {
4879 	free(acbp);
4880 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4881 	}
4882 
4883 	GRAB_MUTEX(&_smhba_APHYSE_mutex);
4884 	acbp->next = _smhba_adapterphystatevents_callback_list;
4885 	_smhba_adapterphystatevents_callback_list = acbp;
4886 
4887 	RELEASE_MUTEX(&_smhba_APHYSE_mutex);
4888 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
4889 }
4890 
4891 /* SMHBA Target Events ********************************************* */
4892 static void
4893 smhba_targetevents_callback(void *data,
4894 	HBA_WWN hbaPortWWN,
4895 	HBA_WWN discoveredPortWWN,
4896 	HBA_WWN domainPortWWN,
4897 	HBA_UINT32 eventType)
4898 {
4899 	HBA_ADAPTERCALLBACK_ELEM	*acbp;
4900 
4901 	DEBUG(3, "TargetEvent, hbaPort:%s, discoveredPort:%s eventType:%d",
4902 	    WWN2STR1(&hbaPortWWN), WWN2STR2(&discoveredPortWWN), eventType);
4903 
4904 	GRAB_MUTEX(&_smhba_TE_mutex);
4905 	for (acbp = _smhba_targetevents_callback_list;
4906 	    acbp != NULL;
4907 	    acbp = acbp->next) {
4908 	if (data == (void *)acbp) {
4909 		(*acbp->callback)(acbp->userdata, hbaPortWWN,
4910 		    discoveredPortWWN, domainPortWWN, eventType);
4911 		break;
4912 	}
4913 	}
4914 	RELEASE_MUTEX(&_smhba_TE_mutex);
4915 }
4916 
4917 HBA_STATUS
4918 SMHBA_RegisterForTargetEvents(
4919     void		(*pCallback) (
4920 	void		*pData,
4921 	HBA_WWN		hbaPortWWN,
4922 	HBA_WWN		discoveredPortWWN,
4923 	HBA_WWN		domainPortWWN,
4924 	HBA_UINT32	eventType),
4925     void		*pUserData,
4926     HBA_HANDLE		handle,
4927     HBA_WWN		hbaPortWWN,
4928     HBA_WWN		discoveredPortWWN,
4929     HBA_WWN		domainPortWWN,
4930     HBA_CALLBACKHANDLE	*pCallbackHandle,
4931     HBA_UINT32		allTargets) {
4932 
4933 	HBA_ADAPTERCALLBACK_ELEM *acbp;
4934 	SMHBARegisterForTargetEventsFunc
4935 	    registeredfunc;
4936 	HBA_STATUS		status;
4937 	HBA_LIBRARY_INFO	*lib_infop;
4938 	HBA_HANDLE		vendorHandle;
4939 
4940 	DEBUG(2, "SMHBA_RegisterForTargetEvents, hbaPort:"
4941 	    "%s, discoveredPort: %s",
4942 	    WWN2STR1(&hbaPortWWN), WWN2STR2(&discoveredPortWWN), 0);
4943 
4944 	CHECKLIBRARYANDVERSION(SMHBA);
4945 	/* we now have the _hbaapi_LL_mutex */
4946 
4947 	registeredfunc = lib_infop->ftable.smhbafunctionTable.\
4948 	    RegisterForTargetEventsHandler;
4949 
4950 	if (registeredfunc == NULL) {
4951 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR_NOT_SUPPORTED);
4952 	}
4953 
4954 	/*
4955 	 * that allocated memory is used both as the handle for the
4956 	 * caller, and as userdata to the vendor call so that on
4957 	 * callback the specific registration may be recalled
4958 	 */
4959 	acbp = (HBA_ADAPTERCALLBACK_ELEM *)
4960 	    calloc(1, sizeof (HBA_ADAPTERCALLBACK_ELEM));
4961 	if (acbp == NULL) {
4962 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_ERROR);
4963 	}
4964 	*pCallbackHandle = (HBA_CALLBACKHANDLE) acbp;
4965 	acbp->callback = pCallback;
4966 	acbp->userdata = pUserData;
4967 	acbp->lib_info = lib_infop;
4968 
4969 	status = (registeredfunc)(smhba_targetevents_callback,
4970 	    (void *)acbp,
4971 	    vendorHandle,
4972 	    hbaPortWWN,
4973 	    discoveredPortWWN,
4974 	    domainPortWWN,
4975 	    &acbp->vendorcbhandle,
4976 	    allTargets);
4977 	if (status != HBA_STATUS_OK) {
4978 	free(acbp);
4979 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, status);
4980 	}
4981 
4982 	GRAB_MUTEX(&_smhba_TE_mutex);
4983 	acbp->next = _smhba_targetevents_callback_list;
4984 	_smhba_targetevents_callback_list = acbp;
4985 
4986 	RELEASE_MUTEX(&_smhba_TE_mutex);
4987 	RELEASE_MUTEX_RETURN(&_hbaapi_LL_mutex, HBA_STATUS_OK);
4988 }
4989