Blob Blame History Raw
/* BEGIN_ICS_COPYRIGHT4 ****************************************

Copyright (c) 2015, Intel Corporation

Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:

    * Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
    * Redistributions in binary form must reproduce the above copyright
      notice, this list of conditions and the following disclaimer in the
     documentation and/or other materials provided with the distribution.
    * Neither the name of Intel Corporation nor the names of its contributors
      may be used to endorse or promote products derived from this software
      without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

** END_ICS_COPYRIGHT4   ****************************************/

#include "ibnotify.h"

//
// Globals
//
IBT_NOTIFY_GROUP	NotifyGroup;


//
// Initialize the Notify Group
//
//INPUT:
// 	Callback - when a new User is added to the Notify group, this routine
//		is called to distribute initial events to User
void
IbtInitNotifyGroup (
   IN	IB_ROOT_CALLBACK		Callback
   )
{
	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, IbtInitNotifyGroup );

	ASSERT(! NotifyGroup.Initialized);

	MapInitState( &NotifyGroup.Lists );
	SpinLockInitState( &NotifyGroup.Lock );

	if (MapInit( &NotifyGroup.Lists ) != FSUCCESS)
	{
		_DBG_ERROR (("Failed to Initialize Notification Map\n"));
		goto done;
	}
	if (SpinLockInit( &NotifyGroup.Lock ) == FALSE)
	{
		_DBG_ERROR (("Failed to Initialize Notification lock\n"));
		goto faillock;
	}

	// Add root callback for new users
	ASSERT ( Callback );
	NotifyGroup.RootCallback = Callback;

	NotifyGroup.Initialized = TRUE;

done:
	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
	return;

faillock:
	MapDestroy ( &NotifyGroup.Lists );
	goto done;
}


//
// Cleanup for shutdown
void
IbtDestroyNotifyGroup ( void )
{
	QUICK_LIST				*pList;

	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, IbtDestroyNotifyGroup );

	if (! NotifyGroup.Initialized)
		goto done;

	// Empty list contents
	SpinLockAcquire( &NotifyGroup.Lock );

	while (NULL != (pList = (QUICK_LIST*)MapRemoveHead(&NotifyGroup.Lists)))
	{
		LIST_ITEM				*pListItem;

		while (NULL != (pListItem = QListRemoveHead(pList)))
		{
			IBT_NOTIFY_USER_BLOCK	*pUser = (IBT_NOTIFY_USER_BLOCK *)pListItem;
			MemoryDeallocate( pUser );
		}	
		QListDestroy(pList);
		MemoryDeallocate(pList);
	}
	NotifyGroup.Initialized = FALSE;

	SpinLockRelease( &NotifyGroup.Lock );

	MapDestroy( &NotifyGroup.Lists );
	SpinLockDestroy( &NotifyGroup.Lock );

done:
	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
}

// Notify all users in pList of pRec event
void
IbtNotifyList(
	QUICK_LIST				*pList,
	IB_NOTIFY_RECORD		*pRec
	 )
{
	LIST_ITEM				*pListFirst;
	IBT_NOTIFY_USER_BLOCK	*pUser;

	pListFirst = QListHead( pList );
			
	while (pListFirst)
	{
		// If event has been subscribed by user, notify
		if ( pRec->EventType & ((IBT_NOTIFY_USER_BLOCK *)pListFirst)->EventMask)
		{
			pUser = (IBT_NOTIFY_USER_BLOCK *)pListFirst;
			pRec->Context = pUser->Context;
			if (pUser->Locking == IB_NOTIFY_ASYNC)
			{
				pUser->RefCount++;
				SpinLockRelease( &NotifyGroup.Lock );
				( pUser->Callback )( *pRec );
				SpinLockAcquire( &NotifyGroup.Lock );
				pListFirst = QListNext( pList, pListFirst );
				if (--(pUser->RefCount) == 0)
				{
					QListRemoveItem ( pList, &pUser->ListItem );
					MemoryDeallocate( pUser );
				}
				continue;
			} else {
				ASSERT(pUser->RefCount);
				( pUser->Callback )( *pRec );
			}
		}
		pListFirst = QListNext( pList, pListFirst );
	}
}

// Notify all registered clients of an event against the given Guid
void
IbtNotifyGroup (
	IN	EUI64				Guid,			// Port/Node GUID
	IN	IB_NOTIFY_TYPE		EventType		// Type of event being reported
	)
{
	IB_NOTIFY_RECORD		Rec;
	QUICK_LIST*				pList;

	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, IbtNotifyGroup );

	_DBG_PRINT( _DBG_LVL_IBT_API,(
				"NotifyGroup\n"
				"\tEventType...:%"PRIxN"\n"
				"\tGUID........:x%016"PRIx64"\n",
				EventType, 
				Guid ));

	if (! NotifyGroup.Initialized)
		goto done;

	// Fill in common record info
	Rec.Guid = Guid;
	Rec.EventType = EventType;

	SpinLockAcquire( &NotifyGroup.Lock );
		
	// iterate on List in Map for Guid, then map for Guid==0
	pList = (QUICK_LIST*)MapGet(&NotifyGroup.Lists, Guid);
	if (pList != NULL)
	{
		IbtNotifyList(pList, &Rec);
	}

	pList = (QUICK_LIST*)MapGet(&NotifyGroup.Lists, 0);
	if (pList != NULL)
	{
		IbtNotifyList(pList, &Rec);
	}

	SpinLockRelease( &NotifyGroup.Lock );

done:
	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
}

// Send a notification to a single registered notify client
// used as part of RootCallback to send initial events to a newly
// registered client
// returns FALSE if client has deregistered itself in callback
boolean
IbtNotifyUserEvent (
	IN	void				*UserContext,
	IN	EUI64				Guid,			// Port/Node GUID
	IN	IB_NOTIFY_TYPE		EventType		// Type of event being reported
	)
{
	IB_NOTIFY_RECORD		Rec;
	IBT_NOTIFY_USER_BLOCK	*pUser = (IBT_NOTIFY_USER_BLOCK*)UserContext;

	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, IbtNotifyUserEvent );

	_DBG_PRINT( _DBG_LVL_IBT_API,(
			"NotifyGroup\n"
			"\tEventType...:%"PRIxN"\n"
			"\tGUID........:x%016"PRIx64"\n",
			EventType, 
			Guid ));

	if (! NotifyGroup.Initialized)
	{
		pUser = NULL;	// no more notifications needed
		goto done;
	}

	// Fill in common record info
	Rec.Guid = Guid;
	Rec.EventType = EventType;

	SpinLockAcquire( &NotifyGroup.Lock );

	// If event has been subscribed by user, notify
	if ( ( EventType & pUser->EventMask )
		&& (pUser->Guid == 0 || pUser->Guid == Guid))
	{
		Rec.Context = pUser->Context;
		if (pUser->Locking == IB_NOTIFY_ASYNC)
		{
			pUser->RefCount++;
			SpinLockRelease( &NotifyGroup.Lock );
			( pUser->Callback )( Rec );
			SpinLockAcquire( &NotifyGroup.Lock );
			if (--(pUser->RefCount) == 0)
			{
				QUICK_LIST*				pList;

				pList = (QUICK_LIST*)MapGet(&NotifyGroup.Lists, pUser->Guid);
				if (pList != NULL)
					QListRemoveItem ( pList, &pUser->ListItem );
				MemoryDeallocate( pUser );
				pUser = NULL;
			}
		} else {
			ASSERT(pUser->RefCount);
			( pUser->Callback )( Rec );
		}
	} 

	SpinLockRelease( &NotifyGroup.Lock );

done:
	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
	return (pUser != NULL);
}



//
// iba_register_notify
//
//	The caller uses this API to register for notifications of IB type events
//
//
// INPUTS:
//
//	Callback		-Callback routine to be registered for notifications
//
//	Context			-User specified context to return in callbacks
//	
//	EventMask		-Mask to subscribe to event types
//
//	Locking			-locking model for callback
//
//
//
// OUTPUTS:
//
//	NotifyHandle	-Notification handle on successful registration
//
//
// RETURNS:
//
//	FSUCCESS
//	FINSUFFICIENT_MEMORY
//
//
FSTATUS
iba_register_notify (
	IN	IB_NOTIFY_CALLBACK	Callback,
	IN	void				*Context,
	IN	IB_NOTIFY_TYPE		EventMask,
	IN	IB_NOTIFY_LOCKING	Locking,
	OUT	IB_HANDLE			*NotifyHandle
	)
{
	FSTATUS					status	= FSUCCESS;
	
	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, iba_register_notify );
	status = iba_register_guid_notify(0, Callback, Context, EventMask, Locking, NotifyHandle);
	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
	return status;
}


//
// iba_register_guid_notify
//
//	The caller uses this API to register for notifications of IB type events
//
//
// INPUTS:
//
//	Guid			-Guid user is interested in
//
//	Callback		-Callback routine to be registered for notifications
//
//	Context			-User specified context to return in callbacks
//	
//	EventMask		-Mask to subscribe to event types
//
//	Locking			-locking model for callback
//
//
//
// OUTPUTS:
//
//	NotifyHandle	-Notification handle on successful registration
//
//
// RETURNS:
//
//	FSUCCESS
//	FINSUFFICIENT_MEMORY
//
//
FSTATUS
iba_register_guid_notify (
	IN  EUI64				Guid,
	IN	IB_NOTIFY_CALLBACK	Callback,
	IN	void				*Context,
	IN	IB_NOTIFY_TYPE		EventMask,
	IN	IB_NOTIFY_LOCKING	Locking,
	OUT	IB_HANDLE			*NotifyHandle
	)
{
	FSTATUS					status	= FSUCCESS;
	IBT_NOTIFY_USER_BLOCK	*pUser;
	QUICK_LIST *pList;
	
	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, iba_register_notify );

	if (! NotifyGroup.Initialized)
	{
		status = FINVALID_STATE;
		goto done;
	}

	// Allocate memory to hold user registration
	pUser = (IBT_NOTIFY_USER_BLOCK*)MemoryAllocateAndClear(
										sizeof(IBT_NOTIFY_USER_BLOCK), FALSE,
										IBT_NOTIFY_TAG );
	if ( !pUser )
	{
		status = FINSUFFICIENT_MEMORY;
		_DBG_ERROR(("%s\n", _DBG_PTR(FSTATUS_MSG(status))));
		goto done;
	}

	// Fill in user details
	pUser->Guid = Guid;	// all Guids
	pUser->Callback = Callback;
	pUser->Context = Context;
	pUser->RefCount = 1;
	pUser->EventMask = EventMask;
	pUser->Locking = Locking;

	// Save user list in group
	SpinLockAcquire( &NotifyGroup.Lock );
	pList = (QUICK_LIST*)MapGet(&NotifyGroup.Lists, Guid);
	if (pList == NULL)
	{
		pList = (QUICK_LIST*)MemoryAllocateAndClear( sizeof(QUICK_LIST), FALSE,
										IBT_NOTIFY_TAG );
		QListInitState(pList);
		if (! QListInit(pList))
		{
			status = FINSUFFICIENT_RESOURCES;
			goto faillist;
		}
		status = MapInsert(&NotifyGroup.Lists, Guid, pList);
		if (status != FSUCCESS)
		{
			goto failmap;
		}
	}
	QListInsertHead( pList, &pUser->ListItem );
	SpinLockRelease( &NotifyGroup.Lock );

	*NotifyHandle = pUser;

	_DBG_PRINT(_DBG_LVL_IBT_API,("Created entry x%p\n", _DBG_PTR(pUser)));

	if (EventMask & IB_NOTIFY_ON_REGISTER)
	{
		// Notify Root to pass user initial events
		ASSERT (( NotifyGroup.RootCallback ));
		( NotifyGroup.RootCallback )( pUser );
	}

done:
	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
	return status;

failmap:
	QListDestroy(pList);
faillist:
	MemoryDeallocate(pList);
	goto done;
}



//
// iba_deregister_notify 
//
//	The caller uses this API to Deregister previously registered notifications
//
//
// INPUTS:
//
//	NotifyHandle		-Handle returned on successful registration
//
//
//
// OUTPUTS:
//
//
//
// RETURNS:
//
//	FSUCCESS
//	FINVALID_PARAMETER
//	FINSUFFICIENT_MEMORY
//
//
FSTATUS
iba_deregister_notify (
	IN 	IB_HANDLE			NotifyHandle
	)
{
	FSTATUS					status	= FSUCCESS;
	IBT_NOTIFY_USER_BLOCK	*pUser	= (IBT_NOTIFY_USER_BLOCK*)NotifyHandle;
	QUICK_LIST				*pList;


	_DBG_ENTER_LVL( _DBG_LVL_IBT_API, iba_deregister_notify );

	if (! NotifyGroup.Initialized)
	{
		// could have never registered if we didn't initialize
		status = FINVALID_PARAMETER;
		goto done;
	}
	SpinLockAcquire( &NotifyGroup.Lock );
	pList = (QUICK_LIST*)MapGet(&NotifyGroup.Lists, pUser->Guid);
	ASSERT(pList);
	if ( !QListIsItemInList( pList, &pUser->ListItem ) )
	{
		status = FINVALID_PARAMETER;
		_DBG_ERROR(("Could not deregister. Notification is not in List!!!\n"));
		goto done;
	}
	if (pUser->Locking == IB_NOTIFY_ASYNC)
	{
		if (--(pUser->RefCount) != 0)
		{
			// in use in callback
			goto done;
		}
	} else {
		ASSERT(pUser->RefCount == 1);
	}

	QListRemoveItem ( pList, &pUser->ListItem );
	MemoryDeallocate( pUser );

done:
	SpinLockRelease( &NotifyGroup.Lock );

	_DBG_LEAVE_LVL(_DBG_LVL_IBT_API);
	return status;
}